|
|
560544 |
From f730da53e59ab25c5b250d7730ead57b6f72fdcf Mon Sep 17 00:00:00 2001
|
|
|
560544 |
From: Julian Seward <jseward@acm.org>
|
|
|
560544 |
Date: Sun, 23 Dec 2018 21:03:08 +0100
|
|
|
560544 |
Subject: [PATCH] Implement Iop_Sar64 in the x86 back end.
|
|
|
560544 |
|
|
|
560544 |
---
|
|
|
560544 |
VEX/priv/host_x86_isel.c | 50 ++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
560544 |
1 file changed, 50 insertions(+)
|
|
|
560544 |
|
|
|
560544 |
diff --git a/VEX/priv/host_x86_isel.c b/VEX/priv/host_x86_isel.c
|
|
|
560544 |
index 45aafeb..9ca8a45 100644
|
|
|
560544 |
--- a/VEX/priv/host_x86_isel.c
|
|
|
560544 |
+++ b/VEX/priv/host_x86_isel.c
|
|
|
560544 |
@@ -2380,6 +2380,56 @@ static void iselInt64Expr_wrk ( HReg* rHi, HReg* rLo, ISelEnv* env,
|
|
|
560544 |
return;
|
|
|
560544 |
}
|
|
|
560544 |
|
|
|
560544 |
+ case Iop_Sar64: {
|
|
|
560544 |
+ /* gcc -O2 does the following. I don't know how it works, but it
|
|
|
560544 |
+ does work. Don't mess with it. This is hard to test because the
|
|
|
560544 |
+ x86 front end doesn't create Iop_Sar64 for any x86 instruction,
|
|
|
560544 |
+ so it's impossible to write a test program that feeds values
|
|
|
560544 |
+ through Iop_Sar64 and prints their results. The implementation
|
|
|
560544 |
+ here was tested by using psrlq on mmx registers -- that generates
|
|
|
560544 |
+ Iop_Shr64 -- and temporarily hacking the front end to generate
|
|
|
560544 |
+ Iop_Sar64 for that instruction instead.
|
|
|
560544 |
+
|
|
|
560544 |
+ movl %amount, %ecx
|
|
|
560544 |
+ movl %srcHi, %r1
|
|
|
560544 |
+ movl %srcLo, %r2
|
|
|
560544 |
+
|
|
|
560544 |
+ movl %r1, %r3
|
|
|
560544 |
+ sarl %cl, %r3
|
|
|
560544 |
+ movl %r2, %r4
|
|
|
560544 |
+ shrdl %cl, %r1, %r4
|
|
|
560544 |
+ movl %r3, %r2
|
|
|
560544 |
+ sarl $31, %r2
|
|
|
560544 |
+ andl $32, %ecx
|
|
|
560544 |
+ cmovne %r3, %r4 // = resLo
|
|
|
560544 |
+ cmovne %r2, %r3 // = resHi
|
|
|
560544 |
+ */
|
|
|
560544 |
+ HReg amount = iselIntExpr_R(env, e->Iex.Binop.arg2);
|
|
|
560544 |
+ HReg srcHi = INVALID_HREG, srcLo = INVALID_HREG;
|
|
|
560544 |
+ iselInt64Expr(&srcHi, &srcLo, env, e->Iex.Binop.arg1);
|
|
|
560544 |
+ HReg r1 = newVRegI(env);
|
|
|
560544 |
+ HReg r2 = newVRegI(env);
|
|
|
560544 |
+ HReg r3 = newVRegI(env);
|
|
|
560544 |
+ HReg r4 = newVRegI(env);
|
|
|
560544 |
+ addInstr(env, mk_iMOVsd_RR(amount, hregX86_ECX()));
|
|
|
560544 |
+ addInstr(env, mk_iMOVsd_RR(srcHi, r1));
|
|
|
560544 |
+ addInstr(env, mk_iMOVsd_RR(srcLo, r2));
|
|
|
560544 |
+
|
|
|
560544 |
+ addInstr(env, mk_iMOVsd_RR(r1, r3));
|
|
|
560544 |
+ addInstr(env, X86Instr_Sh32(Xsh_SAR, 0/*%cl*/, r3));
|
|
|
560544 |
+ addInstr(env, mk_iMOVsd_RR(r2, r4));
|
|
|
560544 |
+ addInstr(env, X86Instr_Sh3232(Xsh_SHR, 0/*%cl*/, r1, r4));
|
|
|
560544 |
+ addInstr(env, mk_iMOVsd_RR(r3, r2));
|
|
|
560544 |
+ addInstr(env, X86Instr_Sh32(Xsh_SAR, 31, r2));
|
|
|
560544 |
+ addInstr(env, X86Instr_Alu32R(Xalu_AND, X86RMI_Imm(32),
|
|
|
560544 |
+ hregX86_ECX()));
|
|
|
560544 |
+ addInstr(env, X86Instr_CMov32(Xcc_NZ, X86RM_Reg(r3), r4));
|
|
|
560544 |
+ addInstr(env, X86Instr_CMov32(Xcc_NZ, X86RM_Reg(r2), r3));
|
|
|
560544 |
+ *rHi = r3;
|
|
|
560544 |
+ *rLo = r4;
|
|
|
560544 |
+ return;
|
|
|
560544 |
+ }
|
|
|
560544 |
+
|
|
|
560544 |
/* F64 -> I64 */
|
|
|
560544 |
/* Sigh, this is an almost exact copy of the F64 -> I32/I16
|
|
|
560544 |
case. Unfortunately I see no easy way to avoid the
|
|
|
560544 |
--
|
|
|
560544 |
1.8.3.1
|
|
|
560544 |
|