8282414: x86: Enhance the assembler to generate more compact instructions
Reviewed-by: thartmann, sviswanathan
This commit is contained in:
parent
1465ea98b7
commit
bacfaa3ee1
@ -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) {
|
||||||
|
@ -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);
|
||||||
|
@ -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);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user