1 /* ----------------------------------------------------------------------- *
3 * Copyright 1996-2017 The NASM Authors - All Rights Reserved
4 * See the file AUTHORS included with the NASM distribution for
5 * the specific copyright holders.
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following
11 * * Redistributions of source code must retain the above copyright
12 * notice, this list of conditions and the following disclaimer.
13 * * Redistributions in binary form must reproduce the above
14 * copyright notice, this list of conditions and the following
15 * disclaimer in the documentation and/or other materials provided
16 * with the distribution.
18 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
19 * CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
20 * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
21 * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
22 * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
23 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
25 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
26 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
27 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
28 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
29 * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
30 * EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32 * ----------------------------------------------------------------------- */
35 * parser.c source line parser for the Netwide Assembler
58 static int is_comma_next(void);
60 static struct tokenval tokval
;
62 static int prefix_slot(int prefix
)
101 nasm_panic(0, "Invalid value %d passed to prefix_slot()", prefix
);
106 static void process_size_override(insn
*result
, operand
*op
)
108 if (tasm_compatible_mode
) {
109 switch (tokval
.t_integer
) {
110 /* For TASM compatibility a size override inside the
111 * brackets changes the size of the operand, not the
112 * address type of the operand as it does in standard
113 * NASM syntax. Hence:
115 * mov eax,[DWORD val]
117 * is valid syntax in TASM compatibility mode. Note that
118 * you lose the ability to override the default address
119 * type for the instruction, but we never use anything
120 * but 32-bit flat model addressing in our code.
142 nasm_error(ERR_NONFATAL
,
143 "invalid operand size specification");
147 /* Standard NASM compatible syntax */
148 switch (tokval
.t_integer
) {
150 op
->eaflags
|= EAF_TIMESTWO
;
153 op
->eaflags
|= EAF_REL
;
156 op
->eaflags
|= EAF_ABS
;
160 op
->eaflags
|= EAF_BYTEOFFS
;
165 if (result
->prefixes
[PPS_ASIZE
] &&
166 result
->prefixes
[PPS_ASIZE
] != tokval
.t_integer
)
167 nasm_error(ERR_NONFATAL
,
168 "conflicting address size specifications");
170 result
->prefixes
[PPS_ASIZE
] = tokval
.t_integer
;
174 op
->eaflags
|= EAF_WORDOFFS
;
179 op
->eaflags
|= EAF_WORDOFFS
;
183 op
->eaflags
|= EAF_WORDOFFS
;
186 nasm_error(ERR_NONFATAL
, "invalid size specification in"
187 " effective address");
194 * Brace decorators are are parsed here. opmask and zeroing
195 * decorators can be placed in any order. e.g. zmm1 {k2}{z} or zmm2
196 * {z}{k3} decorator(s) are placed at the end of an operand.
198 static bool parse_braces(decoflags_t
*decoflags
)
207 if (*decoflags
& OPMASK_MASK
) {
208 nasm_error(ERR_NONFATAL
,
209 "opmask k%"PRIu64
" is already set",
210 *decoflags
& OPMASK_MASK
);
211 *decoflags
&= ~OPMASK_MASK
;
213 *decoflags
|= VAL_OPMASK(nasm_regvals
[tokval
.t_integer
]);
215 case TOKEN_DECORATOR
:
216 j
= tokval
.t_integer
;
219 *decoflags
|= Z_MASK
;
225 *decoflags
|= BRDCAST_MASK
| VAL_BRNUM(j
- BRC_1TO2
);
228 nasm_error(ERR_NONFATAL
,
229 "{%s} is not an expected decorator",
238 nasm_error(ERR_NONFATAL
,
239 "only a series of valid decorators expected");
242 i
= stdscan(NULL
, &tokval
);
246 static int parse_mref(operand
*op
, const expr
*e
)
248 int b
, i
, s
; /* basereg, indexreg, scale */
249 int64_t o
; /* offset */
253 op
->segment
= op
->wrt
= NO_SEG
;
255 if (e
->type
&& e
->type
<= EXPR_REG_END
) { /* this bit's a register */
256 bool is_gpr
= is_class(REG_GPR
,nasm_reg_flags
[e
->type
]);
258 if (is_gpr
&& e
->value
== 1)
259 b
= e
->type
; /* It can be basereg */
260 else /* No, it has to be indexreg */
261 i
= e
->type
, s
= e
->value
;
264 if (e
->type
&& e
->type
<= EXPR_REG_END
) { /* it's a 2nd register */
265 bool is_gpr
= is_class(REG_GPR
,nasm_reg_flags
[e
->type
]);
267 if (b
!= -1) /* If the first was the base, ... */
268 i
= e
->type
, s
= e
->value
; /* second has to be indexreg */
270 else if (!is_gpr
|| e
->value
!= 1) {
271 /* If both want to be index */
272 nasm_error(ERR_NONFATAL
,
273 "invalid effective address: two index registers");
280 if (e
->type
) { /* is there an offset? */
281 if (e
->type
<= EXPR_REG_END
) { /* in fact, is there an error? */
282 nasm_error(ERR_NONFATAL
,
283 "invalid effective address: impossible register");
286 if (e
->type
== EXPR_UNKNOWN
) {
287 op
->opflags
|= OPFLAG_UNKNOWN
;
288 o
= 0; /* doesn't matter what */
290 e
++; /* go to the end of the line */
292 if (e
->type
== EXPR_SIMPLE
) {
296 if (e
->type
== EXPR_WRT
) {
301 * Look for a segment base type.
303 for (; e
->type
; e
++) {
307 if (e
->type
<= EXPR_REG_END
) {
308 nasm_error(ERR_NONFATAL
,
309 "invalid effective address: too many registers");
311 } else if (e
->type
< EXPR_SEGBASE
) {
312 nasm_error(ERR_NONFATAL
,
313 "invalid effective address: bad subexpression type");
315 } else if (e
->value
== 1) {
316 if (op
->segment
!= NO_SEG
) {
317 nasm_error(ERR_NONFATAL
,
318 "invalid effective address: multiple base segments");
321 op
->segment
= e
->type
- EXPR_SEGBASE
;
322 } else if (e
->value
== -1 &&
323 e
->type
== location
.segment
+ EXPR_SEGBASE
&&
324 !(op
->opflags
& OPFLAG_RELATIVE
)) {
325 op
->opflags
|= OPFLAG_RELATIVE
;
327 nasm_error(ERR_NONFATAL
,
328 "invalid effective address: impossible segment base multiplier");
336 nasm_assert(!e
->type
); /* We should be at the end */
345 static void mref_set_optype(operand
*op
)
348 int i
= op
->indexreg
;
351 /* It is memory, but it can match any r/m operand */
352 op
->type
|= MEMORY_ANY
;
354 if (b
== -1 && (i
== -1 || s
== 0)) {
355 int is_rel
= globalbits
== 64 &&
356 !(op
->eaflags
& EAF_ABS
) &&
358 !(op
->eaflags
& EAF_FSGS
)) ||
359 (op
->eaflags
& EAF_REL
));
361 op
->type
|= is_rel
? IP_REL
: MEM_OFFS
;
365 opflags_t iclass
= nasm_reg_flags
[i
];
367 if (is_class(XMMREG
,iclass
))
369 else if (is_class(YMMREG
,iclass
))
371 else if (is_class(ZMMREG
,iclass
))
377 * Convert an expression vector returned from evaluate() into an
378 * extop structure. Return zero on success.
380 static int value_to_extop(expr
* vect
, extop
*eop
, int32_t myseg
)
382 eop
->type
= EOT_DB_NUMBER
;
384 eop
->segment
= eop
->wrt
= NO_SEG
;
385 eop
->relative
= false;
387 for (; vect
->type
; vect
++) {
388 if (!vect
->value
) /* zero term, safe to ignore */
391 if (vect
->type
<= EXPR_REG_END
) /* false if a register is present */
394 if (vect
->type
== EXPR_UNKNOWN
) /* something we can't resolve yet */
397 if (vect
->type
== EXPR_SIMPLE
) {
398 /* Simple number expression */
399 eop
->offset
+= vect
->value
;
402 if (eop
->wrt
== NO_SEG
&& !eop
->relative
&& vect
->type
== EXPR_WRT
) {
404 eop
->wrt
= vect
->value
;
408 if (!eop
->relative
&&
409 vect
->type
== EXPR_SEGBASE
+ myseg
&& vect
->value
== -1) {
410 /* Expression of the form: foo - $ */
411 eop
->relative
= true;
415 if (eop
->segment
== NO_SEG
&& vect
->type
>= EXPR_SEGBASE
&&
417 eop
->segment
= vect
->type
- EXPR_SEGBASE
;
421 /* Otherwise, badness */
425 /* We got to the end and it was all okay */
429 insn
*parse_line(int pass
, char *buffer
, insn
*result
, ldfunc ldef
)
431 bool insn_is_label
= false;
432 struct eval_hints hints
;
441 result
->forw_ref
= false;
445 i
= stdscan(NULL
, &tokval
);
447 result
->label
= NULL
; /* Assume no label */
448 result
->eops
= NULL
; /* must do this, whatever happens */
449 result
->operands
= 0; /* must initialize this */
450 result
->evex_rm
= 0; /* Ensure EVEX rounding mode is reset */
451 result
->evex_brerop
= -1; /* Reset EVEX broadcasting/ER op position */
453 /* Ignore blank lines */
460 (i
!= TOKEN_REG
|| !IS_SREG(tokval
.t_integer
))) {
461 nasm_error(ERR_NONFATAL
,
462 "label or instruction expected at start of line");
466 if (i
== TOKEN_ID
|| (insn_is_label
&& i
== TOKEN_INSN
)) {
467 /* there's a label here */
469 result
->label
= tokval
.t_charptr
;
470 i
= stdscan(NULL
, &tokval
);
471 if (i
== ':') { /* skip over the optional colon */
472 i
= stdscan(NULL
, &tokval
);
474 nasm_error(ERR_WARNING
| ERR_WARN_OL
| ERR_PASS1
,
475 "label alone on a line without a colon might be in error");
477 if (i
!= TOKEN_INSN
|| tokval
.t_integer
!= I_EQU
) {
479 * FIXME: location.segment could be NO_SEG, in which case
480 * it is possible we should be passing 'absolute.segment'. Look into this.
481 * Work out whether that is *really* what we should be doing.
482 * Generally fix things. I think this is right as it is, but
483 * am still not certain.
485 ldef(result
->label
, in_absolute
? absolute
.segment
: location
.segment
,
486 location
.offset
, NULL
, true, false);
490 /* Just a label here */
494 nasm_static_assert(P_none
== 0);
495 memset(result
->prefixes
, P_none
, sizeof(result
->prefixes
));
498 while (i
== TOKEN_PREFIX
||
499 (i
== TOKEN_REG
&& IS_SREG(tokval
.t_integer
))) {
503 * Handle special case: the TIMES prefix.
505 if (i
== TOKEN_PREFIX
&& tokval
.t_integer
== P_TIMES
) {
508 i
= stdscan(NULL
, &tokval
);
509 value
= evaluate(stdscan
, NULL
, &tokval
, NULL
, pass0
, NULL
);
511 if (!value
) /* Error in evaluator */
513 if (!is_simple(value
)) {
514 nasm_error(ERR_NONFATAL
,
515 "non-constant argument supplied to TIMES");
518 result
->times
= value
->value
;
519 if (value
->value
< 0 && pass0
== 2) {
520 nasm_error(ERR_NONFATAL
, "TIMES value %"PRId64
" is negative",
526 int slot
= prefix_slot(tokval
.t_integer
);
527 if (result
->prefixes
[slot
]) {
528 if (result
->prefixes
[slot
] == tokval
.t_integer
)
529 nasm_error(ERR_WARNING
| ERR_PASS1
,
530 "instruction has redundant prefixes");
532 nasm_error(ERR_NONFATAL
,
533 "instruction has conflicting prefixes");
535 result
->prefixes
[slot
] = tokval
.t_integer
;
536 i
= stdscan(NULL
, &tokval
);
540 if (i
!= TOKEN_INSN
) {
544 for (j
= 0; j
< MAXPREFIX
; j
++) {
545 if ((pfx
= result
->prefixes
[j
]) != P_none
)
549 if (i
== 0 && pfx
!= P_none
) {
551 * Instruction prefixes are present, but no actual
552 * instruction. This is allowed: at this point we
553 * invent a notional instruction of RESB 0.
555 result
->opcode
= I_RESB
;
556 result
->operands
= 1;
557 nasm_zero(result
->oprs
);
558 result
->oprs
[0].type
= IMMEDIATE
;
559 result
->oprs
[0].offset
= 0L;
560 result
->oprs
[0].segment
= result
->oprs
[0].wrt
= NO_SEG
;
563 nasm_error(ERR_NONFATAL
, "parser: instruction expected");
568 result
->opcode
= tokval
.t_integer
;
569 result
->condition
= tokval
.t_inttwo
;
572 * INCBIN cannot be satisfied with incorrectly
573 * evaluated operands, since the correct values _must_ be known
574 * on the first pass. Hence, even in pass one, we set the
575 * `critical' flag on calling evaluate(), so that it will bomb
576 * out on undefined symbols.
578 if (result
->opcode
== I_INCBIN
) {
579 critical
= (pass0
< 2 ? 1 : 2);
582 critical
= (pass
== 2 ? 2 : 0);
584 if (result
->opcode
== I_DB
|| result
->opcode
== I_DW
||
585 result
->opcode
== I_DD
|| result
->opcode
== I_DQ
||
586 result
->opcode
== I_DT
|| result
->opcode
== I_DO
||
587 result
->opcode
== I_DY
|| result
->opcode
== I_DZ
||
588 result
->opcode
== I_INCBIN
) {
589 extop
*eop
, **tail
= &result
->eops
, **fixptr
;
593 result
->eops_float
= false;
596 * Begin to read the DB/DW/DD/DQ/DT/DO/DY/DZ/INCBIN operands.
599 i
= stdscan(NULL
, &tokval
);
602 else if (first
&& i
== ':') {
603 insn_is_label
= true;
608 eop
= *tail
= nasm_malloc(sizeof(extop
));
611 eop
->type
= EOT_NOTHING
;
616 * is_comma_next() here is to distinguish this from
617 * a string used as part of an expression...
619 if (i
== TOKEN_STR
&& is_comma_next()) {
620 eop
->type
= EOT_DB_STRING
;
621 eop
->stringval
= tokval
.t_charptr
;
622 eop
->stringlen
= tokval
.t_inttwo
;
623 i
= stdscan(NULL
, &tokval
); /* eat the comma */
624 } else if (i
== TOKEN_STRFUNC
) {
626 const char *funcname
= tokval
.t_charptr
;
627 enum strfunc func
= tokval
.t_integer
;
628 i
= stdscan(NULL
, &tokval
);
631 i
= stdscan(NULL
, &tokval
);
633 if (i
!= TOKEN_STR
) {
634 nasm_error(ERR_NONFATAL
,
635 "%s must be followed by a string constant",
637 eop
->type
= EOT_NOTHING
;
639 eop
->type
= EOT_DB_STRING_FREE
;
641 string_transform(tokval
.t_charptr
, tokval
.t_inttwo
,
642 &eop
->stringval
, func
);
643 if (eop
->stringlen
== (size_t)-1) {
644 nasm_error(ERR_NONFATAL
, "invalid string for transform");
645 eop
->type
= EOT_NOTHING
;
648 if (parens
&& i
&& i
!= ')') {
649 i
= stdscan(NULL
, &tokval
);
651 nasm_error(ERR_NONFATAL
, "unterminated %s function",
656 i
= stdscan(NULL
, &tokval
);
657 } else if (i
== '-' || i
== '+') {
658 char *save
= stdscan_get();
660 sign
= (i
== '-') ? -1 : 1;
661 i
= stdscan(NULL
, &tokval
);
662 if (i
!= TOKEN_FLOAT
) {
664 i
= tokval
.t_type
= token
;
669 } else if (i
== TOKEN_FLOAT
) {
671 eop
->type
= EOT_DB_STRING
;
672 result
->eops_float
= true;
674 eop
->stringlen
= idata_bytes(result
->opcode
);
675 if (eop
->stringlen
> 16) {
676 nasm_error(ERR_NONFATAL
, "floating-point constant"
677 " encountered in DY or DZ instruction");
679 } else if (eop
->stringlen
< 1) {
680 nasm_error(ERR_NONFATAL
, "floating-point constant"
681 " encountered in unknown instruction");
683 * fix suggested by Pedro Gimeno... original line was:
684 * eop->type = EOT_NOTHING;
689 eop
= nasm_realloc(eop
, sizeof(extop
) + eop
->stringlen
);
692 eop
->stringval
= (char *)eop
+ sizeof(extop
);
693 if (!eop
->stringlen
||
694 !float_const(tokval
.t_charptr
, sign
,
695 (uint8_t *)eop
->stringval
, eop
->stringlen
))
696 eop
->type
= EOT_NOTHING
;
697 i
= stdscan(NULL
, &tokval
); /* eat the comma */
699 /* anything else, assume it is an expression */
703 value
= evaluate(stdscan
, NULL
, &tokval
, NULL
,
706 if (!value
) /* Error in evaluator */
708 if (value_to_extop(value
, eop
, location
.segment
)) {
709 nasm_error(ERR_NONFATAL
,
710 "operand %d: expression is not simple or relocatable",
716 * We're about to call stdscan(), which will eat the
717 * comma that we're currently sitting on between
718 * arguments. However, we'd better check first that it
721 if (i
== TOKEN_EOS
) /* also could be EOL */
724 nasm_error(ERR_NONFATAL
, "comma expected after operand %d",
730 if (result
->opcode
== I_INCBIN
) {
732 * Correct syntax for INCBIN is that there should be
733 * one string operand, followed by one or two numeric
736 if (!result
->eops
|| result
->eops
->type
!= EOT_DB_STRING
)
737 nasm_error(ERR_NONFATAL
, "`incbin' expects a file name");
738 else if (result
->eops
->next
&&
739 result
->eops
->next
->type
!= EOT_DB_NUMBER
)
740 nasm_error(ERR_NONFATAL
, "`incbin': second parameter is"
742 else if (result
->eops
->next
&& result
->eops
->next
->next
&&
743 result
->eops
->next
->next
->type
!= EOT_DB_NUMBER
)
744 nasm_error(ERR_NONFATAL
, "`incbin': third parameter is"
746 else if (result
->eops
->next
&& result
->eops
->next
->next
&&
747 result
->eops
->next
->next
->next
)
748 nasm_error(ERR_NONFATAL
,
749 "`incbin': more than three parameters");
753 * If we reach here, one of the above errors happened.
754 * Throw the instruction away.
757 } else /* DB ... */ if (oper_num
== 0)
758 nasm_error(ERR_WARNING
| ERR_PASS1
,
759 "no operand for data declaration");
761 result
->operands
= oper_num
;
767 * Now we begin to parse the operands. There may be up to four
768 * of these, separated by commas, and terminated by a zero token.
771 for (opnum
= 0; opnum
< MAX_OPERANDS
; opnum
++) {
772 operand
*op
= &result
->oprs
[opnum
];
773 expr
*value
; /* used most of the time */
774 bool mref
; /* is this going to be a memory ref? */
775 bool bracket
; /* is it a [] mref, or a & mref? */
776 bool mib
; /* compound (mib) mref? */
778 decoflags_t brace_flags
= 0; /* flags for decorators in braces */
780 op
->disp_size
= 0; /* have to zero this whatever */
781 op
->eaflags
= 0; /* and this */
785 i
= stdscan(NULL
, &tokval
);
787 break; /* end of operands: get out of here */
788 else if (first
&& i
== ':') {
789 insn_is_label
= true;
793 op
->type
= 0; /* so far, no override */
794 while (i
== TOKEN_SPECIAL
) { /* size specifiers */
795 switch (tokval
.t_integer
) {
797 if (!setsize
) /* we want to use only the first */
853 nasm_error(ERR_NONFATAL
, "invalid operand size specification");
855 i
= stdscan(NULL
, &tokval
);
858 if (i
== '[' || i
== '&') { /* memory reference */
860 bracket
= (i
== '[');
861 i
= stdscan(NULL
, &tokval
); /* then skip the colon */
862 while (i
== TOKEN_SPECIAL
|| i
== TOKEN_PREFIX
) {
863 process_size_override(result
, op
);
864 i
= stdscan(NULL
, &tokval
);
866 /* when a comma follows an opening bracket - [ , eax*4] */
868 /* treat as if there is a zero displacement virtually */
869 tokval
.t_type
= TOKEN_NUM
;
870 tokval
.t_integer
= 0;
871 stdscan_set(stdscan_get() - 1); /* rewind the comma */
873 } else { /* immediate operand, or register */
875 bracket
= false; /* placate optimisers */
878 if ((op
->type
& FAR
) && !mref
&&
879 result
->opcode
!= I_JMP
&& result
->opcode
!= I_CALL
) {
880 nasm_error(ERR_NONFATAL
, "invalid use of FAR operand specifier");
883 value
= evaluate(stdscan
, NULL
, &tokval
,
884 &op
->opflags
, critical
, &hints
);
886 if (op
->opflags
& OPFLAG_FORWARD
) {
887 result
->forw_ref
= true;
889 if (!value
) /* Error in evaluator */
891 if (i
== ':' && mref
) { /* it was seg:offset */
893 * Process the segment override.
895 if (value
[1].type
!= 0 ||
897 !IS_SREG(value
->type
))
898 nasm_error(ERR_NONFATAL
, "invalid segment override");
899 else if (result
->prefixes
[PPS_SEG
])
900 nasm_error(ERR_NONFATAL
,
901 "instruction has conflicting segment overrides");
903 result
->prefixes
[PPS_SEG
] = value
->type
;
904 if (IS_FSGS(value
->type
))
905 op
->eaflags
|= EAF_FSGS
;
908 i
= stdscan(NULL
, &tokval
); /* then skip the colon */
909 while (i
== TOKEN_SPECIAL
|| i
== TOKEN_PREFIX
) {
910 process_size_override(result
, op
);
911 i
= stdscan(NULL
, &tokval
);
913 value
= evaluate(stdscan
, NULL
, &tokval
,
914 &op
->opflags
, critical
, &hints
);
916 if (op
->opflags
& OPFLAG_FORWARD
) {
917 result
->forw_ref
= true;
919 /* and get the offset */
920 if (!value
) /* Error in evaluator */
925 if (mref
&& bracket
&& i
== ',') {
926 /* [seg:base+offset,index*scale] syntax (mib) */
928 operand o1
, o2
; /* Partial operands */
930 if (parse_mref(&o1
, value
))
933 i
= stdscan(NULL
, &tokval
); /* Eat comma */
934 value
= evaluate(stdscan
, NULL
, &tokval
, &op
->opflags
,
940 if (parse_mref(&o2
, value
))
943 if (o2
.basereg
!= -1 && o2
.indexreg
== -1) {
944 o2
.indexreg
= o2
.basereg
;
949 if (o1
.indexreg
!= -1 || o2
.basereg
!= -1 || o2
.offset
!= 0 ||
950 o2
.segment
!= NO_SEG
|| o2
.wrt
!= NO_SEG
) {
951 nasm_error(ERR_NONFATAL
, "invalid mib expression");
955 op
->basereg
= o1
.basereg
;
956 op
->indexreg
= o2
.indexreg
;
957 op
->scale
= o2
.scale
;
958 op
->offset
= o1
.offset
;
959 op
->segment
= o1
.segment
;
962 if (op
->basereg
!= -1) {
963 op
->hintbase
= op
->basereg
;
964 op
->hinttype
= EAH_MAKEBASE
;
965 } else if (op
->indexreg
!= -1) {
966 op
->hintbase
= op
->indexreg
;
967 op
->hinttype
= EAH_NOTBASE
;
970 op
->hinttype
= EAH_NOHINT
;
977 if (mref
&& bracket
) { /* find ] at the end */
979 nasm_error(ERR_NONFATAL
, "parser: expecting ]");
981 } else { /* we got the required ] */
982 i
= stdscan(NULL
, &tokval
);
983 if (i
== TOKEN_DECORATOR
|| i
== TOKEN_OPMASK
) {
984 /* parse opmask (and zeroing) after an operand */
985 recover
= parse_braces(&brace_flags
);
988 if (i
!= 0 && i
!= ',') {
989 nasm_error(ERR_NONFATAL
, "comma or end of line expected");
993 } else { /* immediate operand */
994 if (i
!= 0 && i
!= ',' && i
!= ':' &&
995 i
!= TOKEN_DECORATOR
&& i
!= TOKEN_OPMASK
) {
996 nasm_error(ERR_NONFATAL
, "comma, colon, decorator or end of "
997 "line expected after operand");
999 } else if (i
== ':') {
1001 } else if (i
== TOKEN_DECORATOR
|| i
== TOKEN_OPMASK
) {
1002 /* parse opmask (and zeroing) after an operand */
1003 recover
= parse_braces(&brace_flags
);
1007 do { /* error recovery */
1008 i
= stdscan(NULL
, &tokval
);
1009 } while (i
!= 0 && i
!= ',');
1013 * now convert the exprs returned from evaluate()
1014 * into operand descriptions...
1016 op
->decoflags
|= brace_flags
;
1018 if (mref
) { /* it's a memory reference */
1019 /* A mib reference was fully parsed already */
1021 if (parse_mref(op
, value
))
1023 op
->hintbase
= hints
.base
;
1024 op
->hinttype
= hints
.type
;
1026 mref_set_optype(op
);
1027 } else { /* it's not a memory reference */
1028 if (is_just_unknown(value
)) { /* it's immediate but unknown */
1029 op
->type
|= IMMEDIATE
;
1030 op
->opflags
|= OPFLAG_UNKNOWN
;
1031 op
->offset
= 0; /* don't care */
1032 op
->segment
= NO_SEG
; /* don't care again */
1033 op
->wrt
= NO_SEG
; /* still don't care */
1035 if(optimizing
>= 0 && !(op
->type
& STRICT
)) {
1038 UNITY
| SBYTEWORD
| SBYTEDWORD
| UDWORD
| SDWORD
;
1040 } else if (is_reloc(value
)) { /* it's immediate */
1041 uint64_t n
= reloc_value(value
);
1043 op
->type
|= IMMEDIATE
;
1045 op
->segment
= reloc_seg(value
);
1046 op
->wrt
= reloc_wrt(value
);
1047 op
->opflags
|= is_self_relative(value
) ? OPFLAG_RELATIVE
: 0;
1049 if (is_simple(value
)) {
1052 if (optimizing
>= 0 && !(op
->type
& STRICT
)) {
1053 if ((uint32_t) (n
+ 128) <= 255)
1054 op
->type
|= SBYTEDWORD
;
1055 if ((uint16_t) (n
+ 128) <= 255)
1056 op
->type
|= SBYTEWORD
;
1057 if (n
<= UINT64_C(0xFFFFFFFF))
1059 if (n
+ UINT64_C(0x80000000) <= UINT64_C(0xFFFFFFFF))
1063 } else if (value
->type
== EXPR_RDSAE
) {
1065 * it's not an operand but a rounding or SAE decorator.
1066 * put the decorator information in the (opflag_t) type field
1067 * of previous operand.
1070 switch (value
->value
) {
1076 op
->decoflags
|= (value
->value
== BRC_SAE
? SAE
: ER
);
1077 result
->evex_rm
= value
->value
;
1080 nasm_error(ERR_NONFATAL
, "invalid decorator");
1083 } else { /* it's a register */
1086 if (value
->type
>= EXPR_SIMPLE
|| value
->value
!= 1) {
1087 nasm_error(ERR_NONFATAL
, "invalid operand type");
1092 * check that its only 1 register, not an expression...
1094 for (i
= 1; value
[i
].type
; i
++)
1095 if (value
[i
].value
) {
1096 nasm_error(ERR_NONFATAL
, "invalid operand type");
1100 /* clear overrides, except TO which applies to FPU regs */
1101 if (op
->type
& ~TO
) {
1103 * we want to produce a warning iff the specified size
1104 * is different from the register size
1106 rs
= op
->type
& SIZE_MASK
;
1111 op
->type
|= REGISTER
;
1112 op
->type
|= nasm_reg_flags
[value
->type
];
1113 op
->decoflags
|= brace_flags
;
1114 op
->basereg
= value
->type
;
1116 if (rs
&& (op
->type
& SIZE_MASK
) != rs
)
1117 nasm_error(ERR_WARNING
| ERR_PASS1
,
1118 "register size specification ignored");
1122 /* remember the position of operand having broadcasting/ER mode */
1123 if (op
->decoflags
& (BRDCAST_MASK
| ER
| SAE
))
1124 result
->evex_brerop
= opnum
;
1127 result
->operands
= opnum
; /* set operand count */
1129 /* clear remaining operands */
1130 while (opnum
< MAX_OPERANDS
)
1131 result
->oprs
[opnum
++].type
= 0;
1134 * Transform RESW, RESD, RESQ, REST, RESO, RESY, RESZ into RESB.
1136 switch (result
->opcode
) {
1138 result
->opcode
= I_RESB
;
1139 result
->oprs
[0].offset
*= 2;
1142 result
->opcode
= I_RESB
;
1143 result
->oprs
[0].offset
*= 4;
1146 result
->opcode
= I_RESB
;
1147 result
->oprs
[0].offset
*= 8;
1150 result
->opcode
= I_RESB
;
1151 result
->oprs
[0].offset
*= 10;
1154 result
->opcode
= I_RESB
;
1155 result
->oprs
[0].offset
*= 16;
1158 result
->opcode
= I_RESB
;
1159 result
->oprs
[0].offset
*= 32;
1162 result
->opcode
= I_RESB
;
1163 result
->oprs
[0].offset
*= 64;
1172 result
->opcode
= I_none
;
1176 static int is_comma_next(void)
1183 i
= stdscan(NULL
, &tv
);
1186 return (i
== ',' || i
== ';' || !i
);
1189 void cleanup_insn(insn
* i
)
1193 while ((e
= i
->eops
)) {
1195 if (e
->type
== EOT_DB_STRING_FREE
)
1196 nasm_free(e
->stringval
);