8282414: x86: Enhance the assembler to generate more compact instructions

Reviewed-by: thartmann, sviswanathan
This commit is contained in:
Quan Anh Mai 2022-03-16 01:10:22 +00:00 committed by Sandhya Viswanathan
parent 1465ea98b7
commit bacfaa3ee1
3 changed files with 81 additions and 23 deletions

View File

@ -300,12 +300,24 @@ void Assembler::emit_arith_b(int op1, int op2, Register dst, int imm8) {
void Assembler::emit_arith(int op1, int op2, Register dst, int32_t imm32) { void Assembler::emit_arith(int op1, int op2, Register dst, int32_t imm32) {
assert(isByte(op1) && isByte(op2), "wrong opcode"); assert(isByte(op1) && isByte(op2), "wrong opcode");
assert((op1 & 0x01) == 1, "should be 32bit operation"); assert(op1 == 0x81, "Unexpected opcode");
assert((op1 & 0x02) == 0, "sign-extension bit should not be set");
if (is8bit(imm32)) { if (is8bit(imm32)) {
emit_int24(op1 | 0x02, // set sign bit emit_int24(op1 | 0x02, // set sign bit
op2 | encode(dst), op2 | encode(dst),
imm32 & 0xFF); imm32 & 0xFF);
} else if (dst == rax) {
switch (op2) {
case 0xD0: emit_int8(0x15); break; // adc
case 0xC0: emit_int8(0x05); break; // add
case 0xE0: emit_int8(0x25); break; // and
case 0xF8: emit_int8(0x3D); break; // cmp
case 0xC8: emit_int8(0x0D); break; // or
case 0xD8: emit_int8(0x1D); break; // sbb
case 0xE8: emit_int8(0x2D); break; // sub
case 0xF0: emit_int8(0x35); break; // xor
default: ShouldNotReachHere();
}
emit_int32(imm32);
} else { } else {
emit_int16(op1, (op2 | encode(dst))); emit_int16(op1, (op2 | encode(dst)));
emit_int32(imm32); emit_int32(imm32);
@ -929,6 +941,16 @@ address Assembler::locate_operand(address inst, WhichOperand which) {
tail_size = 1; tail_size = 1;
break; break;
case 0x15: // adc rax, #32
case 0x05: // add rax, #32
case 0x25: // and rax, #32
case 0x3D: // cmp rax, #32
case 0x0D: // or rax, #32
case 0x1D: // sbb rax, #32
case 0x2D: // sub rax, #32
case 0x35: // xor rax, #32
return which == end_pc_operand ? ip + 4 : ip;
case 0x9B: case 0x9B:
switch (0xFF & *ip++) { switch (0xFF & *ip++) {
case 0xD9: // fnstcw a case 0xD9: // fnstcw a
@ -954,6 +976,11 @@ address Assembler::locate_operand(address inst, WhichOperand which) {
debug_only(has_disp32 = true); // has both kinds of operands! debug_only(has_disp32 = true); // has both kinds of operands!
break; break;
case 0xA8: // testb rax, #8
return which == end_pc_operand ? ip + 1 : ip;
case 0xA9: // testl/testq rax, #32
return which == end_pc_operand ? ip + 4 : ip;
case 0xC1: // sal a, #8; sar a, #8; shl a, #8; shr a, #8 case 0xC1: // sal a, #8; sar a, #8; shl a, #8; shr a, #8
case 0xC6: // movb a, #8 case 0xC6: // movb a, #8
case 0x80: // cmpb a, #8 case 0x80: // cmpb a, #8
@ -1683,12 +1710,6 @@ void Assembler::cmpl(Address dst, int32_t imm32) {
emit_int32(imm32); emit_int32(imm32);
} }
void Assembler::cmp(Register dst, int32_t imm32) {
prefix(dst);
emit_int8((unsigned char)0x3D);
emit_int32(imm32);
}
void Assembler::cmpl(Register dst, int32_t imm32) { void Assembler::cmpl(Register dst, int32_t imm32) {
prefix(dst); prefix(dst);
emit_arith(0x81, 0xF8, dst, imm32); emit_arith(0x81, 0xF8, dst, imm32);
@ -5775,8 +5796,13 @@ void Assembler::subss(XMMRegister dst, Address src) {
void Assembler::testb(Register dst, int imm8) { void Assembler::testb(Register dst, int imm8) {
NOT_LP64(assert(dst->has_byte_register(), "must have byte register")); NOT_LP64(assert(dst->has_byte_register(), "must have byte register"));
(void) prefix_and_encode(dst->encoding(), true); if (dst == rax) {
emit_arith_b(0xF6, 0xC0, dst, imm8); emit_int8((unsigned char)0xA8);
emit_int8(imm8);
} else {
(void) prefix_and_encode(dst->encoding(), true);
emit_arith_b(0xF6, 0xC0, dst, imm8);
}
} }
void Assembler::testb(Address dst, int imm8) { void Assembler::testb(Address dst, int imm8) {
@ -5787,14 +5813,34 @@ void Assembler::testb(Address dst, int imm8) {
emit_int8(imm8); emit_int8(imm8);
} }
void Assembler::testl(Address dst, int32_t imm32) {
if (imm32 >= 0 && is8bit(imm32)) {
testb(dst, imm32);
return;
}
InstructionMark im(this);
emit_int8((unsigned char)0xF7);
emit_operand(as_Register(0), dst);
emit_int32(imm32);
}
void Assembler::testl(Register dst, int32_t imm32) { void Assembler::testl(Register dst, int32_t imm32) {
if (imm32 >= 0 && is8bit(imm32) && dst->has_byte_register()) {
testb(dst, imm32);
return;
}
// not using emit_arith because test // not using emit_arith because test
// doesn't support sign-extension of // doesn't support sign-extension of
// 8bit operands // 8bit operands
int encode = dst->encoding(); if (dst == rax) {
encode = prefix_and_encode(encode); emit_int8((unsigned char)0xA9);
emit_int16((unsigned char)0xF7, (0xC0 | encode)); emit_int32(imm32);
emit_int32(imm32); } else {
int encode = dst->encoding();
encode = prefix_and_encode(encode);
emit_int16((unsigned char)0xF7, (0xC0 | encode));
emit_int32(imm32);
}
} }
void Assembler::testl(Register dst, Register src) { void Assembler::testl(Register dst, Register src) {
@ -13013,6 +13059,10 @@ void Assembler::subq(Register dst, Register src) {
} }
void Assembler::testq(Address dst, int32_t imm32) { void Assembler::testq(Address dst, int32_t imm32) {
if (imm32 >= 0) {
testl(dst, imm32);
return;
}
InstructionMark im(this); InstructionMark im(this);
emit_int16(get_prefixq(dst), (unsigned char)0xF7); emit_int16(get_prefixq(dst), (unsigned char)0xF7);
emit_operand(as_Register(0), dst); emit_operand(as_Register(0), dst);
@ -13020,13 +13070,23 @@ void Assembler::testq(Address dst, int32_t imm32) {
} }
void Assembler::testq(Register dst, int32_t imm32) { void Assembler::testq(Register dst, int32_t imm32) {
if (imm32 >= 0) {
testl(dst, imm32);
return;
}
// not using emit_arith because test // not using emit_arith because test
// doesn't support sign-extension of // doesn't support sign-extension of
// 8bit operands // 8bit operands
int encode = dst->encoding(); if (dst == rax) {
encode = prefixq_and_encode(encode); prefix(REX_W);
emit_int16((unsigned char)0xF7, (0xC0 | encode)); emit_int8((unsigned char)0xA9);
emit_int32(imm32); emit_int32(imm32);
} else {
int encode = dst->encoding();
encode = prefixq_and_encode(encode);
emit_int16((unsigned char)0xF7, (0xC0 | encode));
emit_int32(imm32);
}
} }
void Assembler::testq(Register dst, Register src) { void Assembler::testq(Register dst, Register src) {

View File

@ -1081,15 +1081,12 @@ private:
void cmpb(Address dst, int imm8); void cmpb(Address dst, int imm8);
void cmpl(Address dst, int32_t imm32); void cmpl(Address dst, int32_t imm32);
void cmp(Register dst, int32_t imm32);
void cmpl(Register dst, int32_t imm32); void cmpl(Register dst, int32_t imm32);
void cmpl(Register dst, Register src); void cmpl(Register dst, Register src);
void cmpl(Register dst, Address src); void cmpl(Register dst, Address src);
void cmpq(Address dst, int32_t imm32); void cmpq(Address dst, int32_t imm32);
void cmpq(Address dst, Register src); void cmpq(Address dst, Register src);
void cmpq(Register dst, int32_t imm32); void cmpq(Register dst, int32_t imm32);
void cmpq(Register dst, Register src); void cmpq(Register dst, Register src);
void cmpq(Register dst, Address src); void cmpq(Register dst, Address src);
@ -2099,9 +2096,10 @@ private:
void subss(XMMRegister dst, Address src); void subss(XMMRegister dst, Address src);
void subss(XMMRegister dst, XMMRegister src); void subss(XMMRegister dst, XMMRegister src);
void testb(Register dst, int imm8);
void testb(Address dst, int imm8); void testb(Address dst, int imm8);
void testb(Register dst, int imm8);
void testl(Address dst, int32_t imm32);
void testl(Register dst, int32_t imm32); void testl(Register dst, int32_t imm32);
void testl(Register dst, Register src); void testl(Register dst, Register src);
void testl(Register dst, Address src); void testl(Register dst, Address src);

View File

@ -1926,7 +1926,7 @@ encode %{
Label done; Label done;
// cmp $0x80000000,%eax // cmp $0x80000000,%eax
__ cmp(as_Register(RAX_enc), 0x80000000); __ cmpl(as_Register(RAX_enc), 0x80000000);
// jne e <normal> // jne e <normal>
__ jccb(Assembler::notEqual, normal); __ jccb(Assembler::notEqual, normal);