8241434: x86: Fix Assembler::emit_operand asserts for XMM registers
Reviewed-by: kvn, redestad
This commit is contained in:
parent
536e062a56
commit
d0a672292f
@ -470,97 +470,143 @@ bool Assembler::emit_compressed_disp_byte(int &disp) {
|
|||||||
return is8bit(disp);
|
return is8bit(disp);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool is_valid_encoding(int reg_enc) {
|
||||||
|
return reg_enc >= 0;
|
||||||
|
}
|
||||||
|
|
||||||
void Assembler::emit_operand(Register reg, Register base, Register index,
|
static int raw_encode(Register reg) {
|
||||||
Address::ScaleFactor scale, int disp,
|
assert(reg == noreg || reg->is_valid(), "sanity");
|
||||||
RelocationHolder const& rspec,
|
int reg_enc = (intptr_t)reg;
|
||||||
int rip_relative_correction) {
|
assert(reg_enc == -1 || is_valid_encoding(reg_enc), "sanity");
|
||||||
|
return reg_enc;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int raw_encode(XMMRegister xmmreg) {
|
||||||
|
assert(xmmreg == xnoreg || xmmreg->is_valid(), "sanity");
|
||||||
|
int xmmreg_enc = (intptr_t)xmmreg;
|
||||||
|
assert(xmmreg_enc == -1 || is_valid_encoding(xmmreg_enc), "sanity");
|
||||||
|
return xmmreg_enc;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int modrm_encoding(int mod, int dst_enc, int src_enc) {
|
||||||
|
return (mod & 3) << 6 | (dst_enc & 7) << 3 | (src_enc & 7);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int sib_encoding(Address::ScaleFactor scale, int index_enc, int base_enc) {
|
||||||
|
return (scale & 3) << 6 | (index_enc & 7) << 3 | (base_enc & 7);
|
||||||
|
}
|
||||||
|
|
||||||
|
inline void Assembler::emit_modrm(int mod, int dst_enc, int src_enc) {
|
||||||
|
assert((mod & 3) != 0b11, "forbidden");
|
||||||
|
int modrm = modrm_encoding(mod, dst_enc, src_enc);
|
||||||
|
emit_int8(modrm);
|
||||||
|
}
|
||||||
|
|
||||||
|
inline void Assembler::emit_modrm_disp8(int mod, int dst_enc, int src_enc,
|
||||||
|
int disp) {
|
||||||
|
int modrm = modrm_encoding(mod, dst_enc, src_enc);
|
||||||
|
emit_int16(modrm, disp & 0xFF);
|
||||||
|
}
|
||||||
|
|
||||||
|
inline void Assembler::emit_modrm_sib(int mod, int dst_enc, int src_enc,
|
||||||
|
Address::ScaleFactor scale, int index_enc, int base_enc) {
|
||||||
|
int modrm = modrm_encoding(mod, dst_enc, src_enc);
|
||||||
|
int sib = sib_encoding(scale, index_enc, base_enc);
|
||||||
|
emit_int16(modrm, sib);
|
||||||
|
}
|
||||||
|
|
||||||
|
inline void Assembler::emit_modrm_sib_disp8(int mod, int dst_enc, int src_enc,
|
||||||
|
Address::ScaleFactor scale, int index_enc, int base_enc,
|
||||||
|
int disp) {
|
||||||
|
int modrm = modrm_encoding(mod, dst_enc, src_enc);
|
||||||
|
int sib = sib_encoding(scale, index_enc, base_enc);
|
||||||
|
emit_int24(modrm, sib, disp & 0xFF);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Assembler::emit_operand_helper(int reg_enc, int base_enc, int index_enc,
|
||||||
|
Address::ScaleFactor scale, int disp,
|
||||||
|
RelocationHolder const& rspec,
|
||||||
|
int rip_relative_correction) {
|
||||||
bool no_relocation = (rspec.type() == relocInfo::none);
|
bool no_relocation = (rspec.type() == relocInfo::none);
|
||||||
|
|
||||||
// Encode the registers as needed in the fields they are used in
|
if (is_valid_encoding(base_enc)) {
|
||||||
int regenc = encode(reg) << 3;
|
if (is_valid_encoding(index_enc)) {
|
||||||
if (base->is_valid()) {
|
|
||||||
int baseenc = encode(base);
|
|
||||||
if (index->is_valid()) {
|
|
||||||
assert(scale != Address::no_scale, "inconsistent address");
|
assert(scale != Address::no_scale, "inconsistent address");
|
||||||
// [base + index*scale + disp]
|
// [base + index*scale + disp]
|
||||||
int indexenc = encode(index) << 3;
|
|
||||||
if (disp == 0 && no_relocation &&
|
if (disp == 0 && no_relocation &&
|
||||||
base != rbp LP64_ONLY(&& base != r13)) {
|
base_enc != rbp->encoding() LP64_ONLY(&& base_enc != r13->encoding())) {
|
||||||
// [base + index*scale]
|
// [base + index*scale]
|
||||||
// [00 reg 100][ss index base]
|
// [00 reg 100][ss index base]
|
||||||
assert(index != rsp, "illegal addressing mode");
|
emit_modrm_sib(0b00, reg_enc, 0b100,
|
||||||
emit_int16((0x04 | regenc),
|
scale, index_enc, base_enc);
|
||||||
(scale << 6 | indexenc | baseenc));
|
|
||||||
} else if (emit_compressed_disp_byte(disp) && no_relocation) {
|
} else if (emit_compressed_disp_byte(disp) && no_relocation) {
|
||||||
// [base + index*scale + imm8]
|
// [base + index*scale + imm8]
|
||||||
// [01 reg 100][ss index base] imm8
|
// [01 reg 100][ss index base] imm8
|
||||||
assert(index != rsp, "illegal addressing mode");
|
emit_modrm_sib_disp8(0b01, reg_enc, 0b100,
|
||||||
emit_int24(0x44 | regenc,
|
scale, index_enc, base_enc,
|
||||||
scale << 6 | indexenc | baseenc,
|
disp);
|
||||||
disp & 0xFF);
|
|
||||||
} else {
|
} else {
|
||||||
// [base + index*scale + disp32]
|
// [base + index*scale + disp32]
|
||||||
// [10 reg 100][ss index base] disp32
|
// [10 reg 100][ss index base] disp32
|
||||||
assert(index != rsp, "illegal addressing mode");
|
emit_modrm_sib(0b10, reg_enc, 0b100,
|
||||||
emit_int16(0x84 | regenc,
|
scale, index_enc, base_enc);
|
||||||
scale << 6 | indexenc | baseenc);
|
|
||||||
emit_data(disp, rspec, disp32_operand);
|
emit_data(disp, rspec, disp32_operand);
|
||||||
}
|
}
|
||||||
} else if (base == rsp LP64_ONLY(|| base == r12)) {
|
} else if (base_enc == rsp->encoding() LP64_ONLY(|| base_enc == r12->encoding())) {
|
||||||
// [rsp + disp]
|
// [rsp + disp]
|
||||||
if (disp == 0 && no_relocation) {
|
if (disp == 0 && no_relocation) {
|
||||||
// [rsp]
|
// [rsp]
|
||||||
// [00 reg 100][00 100 100]
|
// [00 reg 100][00 100 100]
|
||||||
emit_int16(0x04 | regenc,
|
emit_modrm_sib(0b00, reg_enc, 0b100,
|
||||||
0x24);
|
Address::times_1, 0b100, 0b100);
|
||||||
} else if (emit_compressed_disp_byte(disp) && no_relocation) {
|
} else if (emit_compressed_disp_byte(disp) && no_relocation) {
|
||||||
// [rsp + imm8]
|
// [rsp + imm8]
|
||||||
// [01 reg 100][00 100 100] disp8
|
// [01 reg 100][00 100 100] disp8
|
||||||
emit_int24(0x44 | regenc,
|
emit_modrm_sib_disp8(0b01, reg_enc, 0b100,
|
||||||
0x24,
|
Address::times_1, 0b100, 0b100,
|
||||||
disp & 0xFF);
|
disp);
|
||||||
} else {
|
} else {
|
||||||
// [rsp + imm32]
|
// [rsp + imm32]
|
||||||
// [10 reg 100][00 100 100] disp32
|
// [10 reg 100][00 100 100] disp32
|
||||||
emit_int16(0x84 | regenc,
|
emit_modrm_sib(0b10, reg_enc, 0b100,
|
||||||
0x24);
|
Address::times_1, 0b100, 0b100);
|
||||||
emit_data(disp, rspec, disp32_operand);
|
emit_data(disp, rspec, disp32_operand);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// [base + disp]
|
// [base + disp]
|
||||||
assert(base != rsp LP64_ONLY(&& base != r12), "illegal addressing mode");
|
assert(base_enc != rsp->encoding() LP64_ONLY(&& base_enc != r12->encoding()), "illegal addressing mode");
|
||||||
if (disp == 0 && no_relocation &&
|
if (disp == 0 && no_relocation &&
|
||||||
base != rbp LP64_ONLY(&& base != r13)) {
|
base_enc != rbp->encoding() LP64_ONLY(&& base_enc != r13->encoding())) {
|
||||||
// [base]
|
// [base]
|
||||||
// [00 reg base]
|
// [00 reg base]
|
||||||
emit_int8(0x00 | regenc | baseenc);
|
emit_modrm(0, reg_enc, base_enc);
|
||||||
} else if (emit_compressed_disp_byte(disp) && no_relocation) {
|
} else if (emit_compressed_disp_byte(disp) && no_relocation) {
|
||||||
// [base + disp8]
|
// [base + disp8]
|
||||||
// [01 reg base] disp8
|
// [01 reg base] disp8
|
||||||
emit_int16(0x40 | regenc | baseenc,
|
emit_modrm_disp8(0b01, reg_enc, base_enc,
|
||||||
disp & 0xFF);
|
disp);
|
||||||
} else {
|
} else {
|
||||||
// [base + disp32]
|
// [base + disp32]
|
||||||
// [10 reg base] disp32
|
// [10 reg base] disp32
|
||||||
emit_int8(0x80 | regenc | baseenc);
|
emit_modrm(0b10, reg_enc, base_enc);
|
||||||
emit_data(disp, rspec, disp32_operand);
|
emit_data(disp, rspec, disp32_operand);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (index->is_valid()) {
|
if (is_valid_encoding(index_enc)) {
|
||||||
assert(scale != Address::no_scale, "inconsistent address");
|
assert(scale != Address::no_scale, "inconsistent address");
|
||||||
|
// base == noreg
|
||||||
// [index*scale + disp]
|
// [index*scale + disp]
|
||||||
// [00 reg 100][ss index 101] disp32
|
// [00 reg 100][ss index 101] disp32
|
||||||
assert(index != rsp, "illegal addressing mode");
|
emit_modrm_sib(0b00, reg_enc, 0b100,
|
||||||
emit_int16(0x04 | regenc,
|
scale, index_enc, 0b101 /* no base */);
|
||||||
scale << 6 | (encode(index) << 3) | 0x05);
|
|
||||||
emit_data(disp, rspec, disp32_operand);
|
emit_data(disp, rspec, disp32_operand);
|
||||||
} else if (!no_relocation) {
|
} else if (!no_relocation) {
|
||||||
|
// base == noreg, index == noreg
|
||||||
// [disp] (64bit) RIP-RELATIVE (32bit) abs
|
// [disp] (64bit) RIP-RELATIVE (32bit) abs
|
||||||
// [00 000 101] disp32
|
// [00 reg 101] disp32
|
||||||
|
|
||||||
emit_int8(0x05 | regenc);
|
emit_modrm(0b00, reg_enc, 0b101 /* no base */);
|
||||||
// Note that the RIP-rel. correction applies to the generated
|
// Note that the RIP-rel. correction applies to the generated
|
||||||
// disp field, but _not_ to the target address in the rspec.
|
// disp field, but _not_ to the target address in the rspec.
|
||||||
|
|
||||||
@ -577,45 +623,44 @@ void Assembler::emit_operand(Register reg, Register base, Register index,
|
|||||||
emit_data((int32_t) adjusted, rspec, disp32_operand);
|
emit_data((int32_t) adjusted, rspec, disp32_operand);
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
|
// base == noreg, index == noreg, no_relocation == true
|
||||||
// 32bit never did this, did everything as the rip-rel/disp code above
|
// 32bit never did this, did everything as the rip-rel/disp code above
|
||||||
// [disp] ABSOLUTE
|
// [disp] ABSOLUTE
|
||||||
// [00 reg 100][00 100 101] disp32
|
// [00 reg 100][00 100 101] disp32
|
||||||
emit_int16(0x04 | regenc,
|
emit_modrm_sib(0b00, reg_enc, 0b100 /* no base */,
|
||||||
0x25);
|
Address::times_1, 0b100, 0b101);
|
||||||
emit_data(disp, rspec, disp32_operand);
|
emit_data(disp, rspec, disp32_operand);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void Assembler::emit_operand(XMMRegister reg, Register base, Register index,
|
void Assembler::emit_operand(Register reg, Register base, Register index,
|
||||||
|
Address::ScaleFactor scale, int disp,
|
||||||
|
RelocationHolder const& rspec,
|
||||||
|
int rip_relative_correction) {
|
||||||
|
assert(!index->is_valid() || index != rsp, "illegal addressing mode");
|
||||||
|
emit_operand_helper(raw_encode(reg), raw_encode(base), raw_encode(index),
|
||||||
|
scale, disp, rspec, rip_relative_correction);
|
||||||
|
|
||||||
|
}
|
||||||
|
void Assembler::emit_operand(XMMRegister xmmreg, Register base, Register index,
|
||||||
Address::ScaleFactor scale, int disp,
|
Address::ScaleFactor scale, int disp,
|
||||||
RelocationHolder const& rspec) {
|
RelocationHolder const& rspec) {
|
||||||
if (UseAVX > 2) {
|
assert(!index->is_valid() || index != rsp, "illegal addressing mode");
|
||||||
int xreg_enc = reg->encoding();
|
assert(xmmreg->encoding() < 16 || UseAVX > 2, "not supported");
|
||||||
if (xreg_enc > 15) {
|
emit_operand_helper(raw_encode(xmmreg), raw_encode(base), raw_encode(index),
|
||||||
XMMRegister new_reg = as_XMMRegister(xreg_enc & 0xf);
|
scale, disp, rspec);
|
||||||
emit_operand((Register)new_reg, base, index, scale, disp, rspec);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
emit_operand((Register)reg, base, index, scale, disp, rspec);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Assembler::emit_operand(XMMRegister reg, Register base, XMMRegister index,
|
void Assembler::emit_operand(XMMRegister xmmreg, Register base, XMMRegister xmmindex,
|
||||||
Address::ScaleFactor scale, int disp,
|
Address::ScaleFactor scale, int disp,
|
||||||
RelocationHolder const& rspec) {
|
RelocationHolder const& rspec) {
|
||||||
if (UseAVX > 2) {
|
assert(xmmreg->encoding() < 16 || UseAVX > 2, "not supported");
|
||||||
int xreg_enc = reg->encoding();
|
assert(xmmindex->encoding() < 16 || UseAVX > 2, "not supported");
|
||||||
int xmmindex_enc = index->encoding();
|
emit_operand_helper(raw_encode(xmmreg), raw_encode(base), raw_encode(xmmindex),
|
||||||
XMMRegister new_reg = as_XMMRegister(xreg_enc & 0xf);
|
scale, disp, rspec, /* rip_relative_correction */ 0);
|
||||||
XMMRegister new_index = as_XMMRegister(xmmindex_enc & 0xf);
|
|
||||||
emit_operand((Register)new_reg, base, (Register)new_index, scale, disp, rspec);
|
|
||||||
} else {
|
|
||||||
emit_operand((Register)reg, base, (Register)index, scale, disp, rspec);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// Secret local extension to Assembler::WhichOperand:
|
// Secret local extension to Assembler::WhichOperand:
|
||||||
#define end_pc_operand (_WhichOperand_limit)
|
#define end_pc_operand (_WhichOperand_limit)
|
||||||
|
|
||||||
@ -1102,13 +1147,6 @@ void Assembler::check_relocation(RelocationHolder const& rspec, int format) {
|
|||||||
}
|
}
|
||||||
#endif // ASSERT
|
#endif // ASSERT
|
||||||
|
|
||||||
void Assembler::emit_operand32(Register reg, Address adr) {
|
|
||||||
assert(reg->encoding() < 8, "no extended registers");
|
|
||||||
assert(!adr.base_needs_rex() && !adr.index_needs_rex(), "no extended registers");
|
|
||||||
emit_operand(reg, adr._base, adr._index, adr._scale, adr._disp,
|
|
||||||
adr._rspec);
|
|
||||||
}
|
|
||||||
|
|
||||||
void Assembler::emit_operand(Register reg, Address adr,
|
void Assembler::emit_operand(Register reg, Address adr,
|
||||||
int rip_relative_correction) {
|
int rip_relative_correction) {
|
||||||
emit_operand(reg, adr._base, adr._index, adr._scale, adr._disp,
|
emit_operand(reg, adr._base, adr._index, adr._scale, adr._disp,
|
||||||
@ -1138,13 +1176,6 @@ void Assembler::emit_operand(Address adr, MMXRegister reg) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void Assembler::emit_farith(int b1, int b2, int i) {
|
|
||||||
assert(isByte(b1) && isByte(b2), "wrong opcode");
|
|
||||||
assert(0 <= i && i < 8, "illegal stack offset");
|
|
||||||
emit_int16(b1, b2 + i);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// Now the Assembler instructions (identical for 32/64 bits)
|
// Now the Assembler instructions (identical for 32/64 bits)
|
||||||
|
|
||||||
void Assembler::adcl(Address dst, int32_t imm32) {
|
void Assembler::adcl(Address dst, int32_t imm32) {
|
||||||
@ -6865,6 +6896,19 @@ void Assembler::decl(Register dst) {
|
|||||||
|
|
||||||
// 64bit doesn't use the x87
|
// 64bit doesn't use the x87
|
||||||
|
|
||||||
|
void Assembler::emit_operand32(Register reg, Address adr) {
|
||||||
|
assert(reg->encoding() < 8, "no extended registers");
|
||||||
|
assert(!adr.base_needs_rex() && !adr.index_needs_rex(), "no extended registers");
|
||||||
|
emit_operand(reg, adr._base, adr._index, adr._scale, adr._disp,
|
||||||
|
adr._rspec);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Assembler::emit_farith(int b1, int b2, int i) {
|
||||||
|
assert(isByte(b1) && isByte(b2), "wrong opcode");
|
||||||
|
assert(0 <= i && i < 8, "illegal stack offset");
|
||||||
|
emit_int16(b1, b2 + i);
|
||||||
|
}
|
||||||
|
|
||||||
void Assembler::fabs() {
|
void Assembler::fabs() {
|
||||||
emit_int16((unsigned char)0xD9, (unsigned char)0xE1);
|
emit_int16((unsigned char)0xD9, (unsigned char)0xE1);
|
||||||
}
|
}
|
||||||
|
@ -725,20 +725,39 @@ private:
|
|||||||
|
|
||||||
bool emit_compressed_disp_byte(int &disp);
|
bool emit_compressed_disp_byte(int &disp);
|
||||||
|
|
||||||
|
void emit_modrm(int mod, int dst_enc, int src_enc);
|
||||||
|
void emit_modrm_disp8(int mod, int dst_enc, int src_enc,
|
||||||
|
int disp);
|
||||||
|
void emit_modrm_sib(int mod, int dst_enc, int src_enc,
|
||||||
|
Address::ScaleFactor scale, int index_enc, int base_enc);
|
||||||
|
void emit_modrm_sib_disp8(int mod, int dst_enc, int src_enc,
|
||||||
|
Address::ScaleFactor scale, int index_enc, int base_enc,
|
||||||
|
int disp);
|
||||||
|
|
||||||
|
void emit_operand_helper(int reg_enc,
|
||||||
|
int base_enc, int index_enc, Address::ScaleFactor scale,
|
||||||
|
int disp,
|
||||||
|
RelocationHolder const& rspec,
|
||||||
|
int rip_relative_correction = 0);
|
||||||
|
|
||||||
void emit_operand(Register reg,
|
void emit_operand(Register reg,
|
||||||
Register base, Register index, Address::ScaleFactor scale,
|
Register base, Register index, Address::ScaleFactor scale,
|
||||||
int disp,
|
int disp,
|
||||||
RelocationHolder const& rspec,
|
RelocationHolder const& rspec,
|
||||||
int rip_relative_correction = 0);
|
int rip_relative_correction = 0);
|
||||||
|
|
||||||
void emit_operand(XMMRegister reg, Register base, XMMRegister index,
|
void emit_operand(Register reg,
|
||||||
Address::ScaleFactor scale,
|
Register base, XMMRegister index, Address::ScaleFactor scale,
|
||||||
int disp, RelocationHolder const& rspec);
|
int disp,
|
||||||
|
RelocationHolder const& rspec);
|
||||||
|
|
||||||
void emit_operand(Register reg, Address adr, int rip_relative_correction = 0);
|
void emit_operand(XMMRegister xreg,
|
||||||
|
Register base, XMMRegister xindex, Address::ScaleFactor scale,
|
||||||
|
int disp,
|
||||||
|
RelocationHolder const& rspec);
|
||||||
|
|
||||||
// operands that only take the original 32bit registers
|
void emit_operand(Register reg, Address adr,
|
||||||
void emit_operand32(Register reg, Address adr);
|
int rip_relative_correction = 0);
|
||||||
|
|
||||||
void emit_operand(XMMRegister reg,
|
void emit_operand(XMMRegister reg,
|
||||||
Register base, Register index, Address::ScaleFactor scale,
|
Register base, Register index, Address::ScaleFactor scale,
|
||||||
@ -752,13 +771,9 @@ private:
|
|||||||
// workaround gcc (3.2.1-7) bug
|
// workaround gcc (3.2.1-7) bug
|
||||||
void emit_operand(Address adr, MMXRegister reg);
|
void emit_operand(Address adr, MMXRegister reg);
|
||||||
|
|
||||||
|
|
||||||
// Immediate-to-memory forms
|
// Immediate-to-memory forms
|
||||||
void emit_arith_operand(int op1, Register rm, Address adr, int32_t imm32);
|
void emit_arith_operand(int op1, Register rm, Address adr, int32_t imm32);
|
||||||
|
|
||||||
void emit_farith(int b1, int b2, int i);
|
|
||||||
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
#ifdef ASSERT
|
#ifdef ASSERT
|
||||||
void check_relocation(RelocationHolder const& rspec, int format);
|
void check_relocation(RelocationHolder const& rspec, int format);
|
||||||
@ -1156,6 +1171,13 @@ private:
|
|||||||
void emms();
|
void emms();
|
||||||
|
|
||||||
#ifndef _LP64
|
#ifndef _LP64
|
||||||
|
private:
|
||||||
|
// operands that only take the original 32bit registers
|
||||||
|
void emit_operand32(Register reg, Address adr);
|
||||||
|
|
||||||
|
void emit_farith(int b1, int b2, int i);
|
||||||
|
|
||||||
|
public:
|
||||||
void fabs();
|
void fabs();
|
||||||
|
|
||||||
void fadd(int i);
|
void fadd(int i);
|
||||||
|
Loading…
Reference in New Issue
Block a user