diff options
author | H.J. Lu <hjl.tools@gmail.com> | 2006-12-29 06:02:04 +0000 |
---|---|---|
committer | H.J. Lu <hjl.tools@gmail.com> | 2006-12-29 06:02:04 +0000 |
commit | cab737b91fedd56af3b24b5bdd3c3c2efb1426f2 (patch) | |
tree | 8f6456d8b674fdd0a02c01d66e0153a647765cb8 /gas/config | |
parent | 994d1171c0efce1e6cefd6024abd4f965b41dffc (diff) | |
download | gdb-cab737b91fedd56af3b24b5bdd3c3c2efb1426f2.zip gdb-cab737b91fedd56af3b24b5bdd3c3c2efb1426f2.tar.gz gdb-cab737b91fedd56af3b24b5bdd3c3c2efb1426f2.tar.bz2 |
2006-12-28 H.J. Lu <hongjiu.lu@intel.com>
* config/tc-i386.c (process_operands): Check i.reg_operands
and increment i.operands when adding a register operand.
(build_modrm_byte): Fix 4 operand instruction handling.
Diffstat (limited to 'gas/config')
-rw-r--r-- | gas/config/tc-i386.c | 38 |
1 files changed, 27 insertions, 11 deletions
diff --git a/gas/config/tc-i386.c b/gas/config/tc-i386.c index 856e953..079e9d8 100644 --- a/gas/config/tc-i386.c +++ b/gas/config/tc-i386.c @@ -3335,10 +3335,12 @@ process_operands () { unsigned int first_reg_op = (i.types[0] & Reg) ? 0 : 1; /* Pretend we saw the extra register operand. */ - assert (i.op[first_reg_op + 1].regs == 0); + assert (i.reg_operands == 1 + && i.op[first_reg_op + 1].regs == 0); i.op[first_reg_op + 1].regs = i.op[first_reg_op].regs; i.types[first_reg_op + 1] = i.types[first_reg_op]; - i.reg_operands = 2; + i.operands++; + i.reg_operands++; } if (i.tm.opcode_modifier & ShortForm) @@ -3427,16 +3429,30 @@ build_modrm_byte () if (i.reg_operands == 2) { unsigned int source, dest; - source = ((i.types[0] - & (Reg | RegMMX | RegXMM - | SReg2 | SReg3 - | Control | Debug | Test)) - ? 0 : 1); - /* In 4 operands instructions with 2 immediate operands, the first - two are immediate bytes and hence source operand will be in the - next byte after the immediates */ - if ((i.operands == 4)&&(i.imm_operands=2)) source++; + switch (i.operands) + { + case 2: + source = 0; + break; + case 3: + /* When there are 3 operands, one of them must be immediate, + which may be the first or the last operand. */ + assert (i.imm_operands == 1); + source = (i.types[0] & Imm) ? 1 : 0; + break; + case 4: + /* When there are 4 operands, the first two must be immediate + operands. The source operand will be the 3rd one. */ + assert (i.imm_operands == 2 + && (i.types[0] & Imm) + && (i.types[1] & Imm)); + source = 2; + break; + default: + abort (); + } + dest = source + 1; i.rm.mode = 3; |