diff options
author | Jan Beulich <jbeulich@novell.com> | 2012-08-07 16:51:34 +0000 |
---|---|---|
committer | Jan Beulich <jbeulich@novell.com> | 2012-08-07 16:51:34 +0000 |
commit | 4c692bc7aa91109a1102fd0c6f3229e7efbe87d0 (patch) | |
tree | 24ebc9515994d6053b89455ca2baedba0cb07593 /gas/config | |
parent | dc1039df6354fe0931769191e1413a624ad5697d (diff) | |
download | gdb-4c692bc7aa91109a1102fd0c6f3229e7efbe87d0.zip gdb-4c692bc7aa91109a1102fd0c6f3229e7efbe87d0.tar.gz gdb-4c692bc7aa91109a1102fd0c6f3229e7efbe87d0.tar.bz2 |
There were several cases where the registers in the REX encoded range
got treated identically to the ones in the base range, due to not
paying attention to the fact that reg_entry's reg_num field doesn't
fully specify the register number (reg_flags also needs to be checked
for RegRex). This patch introduces and uses a new (inline) function to
obtain the full register number, and uses it to fix all those cases.
It additionally adds the missing operand checks for SVME instructions
(which match the monitor/mwait ones).
gas/
2012-08-07 Jan Beulich <jbeulich@suse.com>
* config/tc-i386.c (register_number): New function.
(build_vex_prefix, process_immext, process_operands,
build_modrm_byte, i386_index_check): Use it.
gas/testsuite/
2012-08-07 Jan Beulich <jbeulich@suse.com>
* gas/i386/x86-64-specific-reg.{s,l}: New.
* gas/i386/i386.exp: Run new test.
opcodes/
2012-08-07 Jan Beulich <jbeulich@suse.com>
* i386-opc.tbl: Remove "FIXME" comments from SVME instructions.
Diffstat (limited to 'gas/config')
-rw-r--r-- | gas/config/tc-i386.c | 48 |
1 files changed, 25 insertions, 23 deletions
diff --git a/gas/config/tc-i386.c b/gas/config/tc-i386.c index 49eb8c1..93500fe 100644 --- a/gas/config/tc-i386.c +++ b/gas/config/tc-i386.c @@ -1758,6 +1758,17 @@ operand_type_register_match (i386_operand_type m0, } static INLINE unsigned int +register_number (const reg_entry *r) +{ + unsigned int nr = r->reg_num; + + if (r->reg_flags & RegRex) + nr += 8; + + return nr; +} + +static INLINE unsigned int mode_from_disp_size (i386_operand_type t) { if (t.bitfield.disp8) @@ -2830,12 +2841,7 @@ build_vex_prefix (const insn_template *t) /* Check register specifier. */ if (i.vex.register_specifier) - { - register_specifier = i.vex.register_specifier->reg_num; - if ((i.vex.register_specifier->reg_flags & RegRex)) - register_specifier += 8; - register_specifier = ~register_specifier & 0xf; - } + register_specifier = ~register_number (i.vex.register_specifier) & 0xf; else register_specifier = 0xf; @@ -2974,16 +2980,17 @@ process_immext (void) { expressionS *exp; - if (i.tm.cpu_flags.bitfield.cpusse3 && i.operands > 0) + if ((i.tm.cpu_flags.bitfield.cpusse3 || i.tm.cpu_flags.bitfield.cpusvme) + && i.operands > 0) { - /* SSE3 Instructions have the fixed operands with an opcode - suffix which is coded in the same place as an 8-bit immediate - field would be. Here we check those operands and remove them - afterwards. */ + /* MONITOR/MWAIT as well as SVME instructions have fixed operands + with an opcode suffix which is coded in the same place as an + 8-bit immediate field would be. + Here we check those operands and remove them afterwards. */ unsigned int x; for (x = 0; x < i.operands; x++) - if (i.op[x].regs->reg_num != x) + if (register_number (i.op[x].regs) != x) as_bad (_("can't use register '%s%s' as operand %d in '%s'."), register_prefix, i.op[x].regs->reg_name, x + 1, i.tm.name); @@ -5071,7 +5078,7 @@ process_operands (void) { /* The first operand is implicit and must be xmm0. */ gas_assert (operand_type_equal (&i.types[0], ®xmm)); - if (i.op[0].regs->reg_num != 0) + if (register_number (i.op[0].regs) != 0) return bad_implicit_operand (1); if (i.tm.opcode_modifier.vexsources == VEX3SOURCES) @@ -5145,7 +5152,7 @@ duplicate: gas_assert (i.reg_operands && (operand_type_equal (&i.types[0], ®xmm) || operand_type_equal (&i.types[0], ®ymm))); - if (i.op[0].regs->reg_num != 0) + if (register_number (i.op[0].regs) != 0) return bad_implicit_operand (i.types[0].bitfield.regxmm); for (j = 1; j < i.operands; j++) @@ -5348,10 +5355,7 @@ build_modrm_byte (void) || operand_type_equal (&i.tm.operand_types[reg_slot], ®ymm)); exp->X_op = O_constant; - exp->X_add_number - = ((i.op[reg_slot].regs->reg_num - + ((i.op[reg_slot].regs->reg_flags & RegRex) ? 8 : 0)) - << 4); + exp->X_add_number = register_number (i.op[reg_slot].regs) << 4; } else { @@ -5395,9 +5399,7 @@ build_modrm_byte (void) || operand_type_equal (&i.tm.operand_types[reg_slot], ®ymm)); i.op[imm_slot].imms->X_add_number - |= ((i.op[reg_slot].regs->reg_num - + ((i.op[reg_slot].regs->reg_flags & RegRex) ? 8 : 0)) - << 4); + |= register_number (i.op[reg_slot].regs) << 4; } gas_assert (operand_type_equal (&i.tm.operand_types[nds], ®xmm) @@ -7413,7 +7415,7 @@ i386_index_check (const char *operand_string) ? i.base_reg->reg_type.bitfield.reg32 : i.base_reg->reg_type.bitfield.reg16)) ok = 0; - else if (i.base_reg->reg_num != expected) + else if (register_number (i.base_reg) != expected) ok = -1; if (ok < 0) @@ -7428,7 +7430,7 @@ i386_index_check (const char *operand_string) : (flag_code == CODE_16BIT) ^ !i.prefix[ADDR_PREFIX] ? i386_regtab[j].reg_type.bitfield.reg32 : i386_regtab[j].reg_type.bitfield.reg16) - && i386_regtab[j].reg_num == expected) + && register_number(i386_regtab + j) == expected) break; gas_assert (j < i386_regtab_size); as_warn (_("`%s' is not valid here (expected `%c%s%s%c')"), |