aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--tcg/ppc/tcg-target.c118
1 files changed, 80 insertions, 38 deletions
diff --git a/tcg/ppc/tcg-target.c b/tcg/ppc/tcg-target.c
index 205cd2bada..9b5c6d622f 100644
--- a/tcg/ppc/tcg-target.c
+++ b/tcg/ppc/tcg-target.c
@@ -23,6 +23,8 @@
*/
static uint8_t *tb_ret_addr;
+static uint8_t *udiv_addr;
+static uint8_t *div_addr;
#define FAST_PATH
#if TARGET_PHYS_ADDR_BITS <= 32
@@ -118,7 +120,7 @@ static const int tcg_target_call_oarg_regs[2] = {
};
static const int tcg_target_callee_save_regs[] = {
- TCG_REG_R13, /* sould r13 be saved? */
+ TCG_REG_R13, /* should r13 be saved? */
TCG_REG_R14,
TCG_REG_R15,
TCG_REG_R16,
@@ -135,6 +137,22 @@ static const int tcg_target_callee_save_regs[] = {
TCG_REG_R31
};
+static const int div_save_regs[] = {
+ TCG_REG_R4,
+ TCG_REG_R5,
+ TCG_REG_R7,
+ TCG_REG_R8,
+ TCG_REG_R9,
+ TCG_REG_R10,
+ TCG_REG_R11,
+ TCG_REG_R12,
+ TCG_REG_R13, /* should r13 be saved? */
+ TCG_REG_R24,
+ TCG_REG_R25,
+ TCG_REG_R26,
+ TCG_REG_R27,
+};
+
static uint32_t reloc_pc24_val (void *pc, tcg_target_long target)
{
tcg_target_long disp;
@@ -799,9 +817,25 @@ static void tcg_out_qemu_st (TCGContext *s, const TCGArg *args, int opc)
#endif
}
+static uint64_t ppc_udiv_helper (uint64_t a, uint32_t b)
+{
+ uint64_t rem, quo;
+ quo = a / b;
+ rem = a % b;
+ return (rem << 32) | (uint32_t) quo;
+}
+
+static uint64_t ppc_div_helper (int64_t a, int32_t b)
+{
+ int64_t rem, quo;
+ quo = a / b;
+ rem = a % b;
+ return (rem << 32) | (uint32_t) quo;
+}
+
void tcg_target_qemu_prologue (TCGContext *s)
{
- int i, frame_size;
+ int i, j, frame_size;
frame_size = 0
+ 4 /* back chain */
@@ -837,6 +871,49 @@ void tcg_target_qemu_prologue (TCGContext *s)
tcg_out32 (s, MTSPR | RS (0) | LR);
tcg_out32 (s, ADDI | RT (1) | RA (1) | frame_size);
tcg_out32 (s, BCLR | BO_ALWAYS);
+
+ /* div trampolines */
+ for (j = 0; j < 2; ++j) {
+ tcg_target_long target;
+
+ frame_size = 8 + ARRAY_SIZE (div_save_regs) * 4;
+ frame_size = (frame_size + 15) & ~15;
+
+ if (j == 0) {
+ target = (tcg_target_long) ppc_udiv_helper;
+ udiv_addr = s->code_ptr;
+ }
+ else {
+ target = (tcg_target_long) ppc_div_helper;
+ div_addr = s->code_ptr;
+ }
+
+ tcg_out32 (s, MFSPR | RT (0) | LR);
+ tcg_out32 (s, STWU | RS (1) | RA (1) | (-frame_size & 0xffff));
+ for (i = 0; i < ARRAY_SIZE (div_save_regs); ++i)
+ tcg_out32 (s, (STW
+ | RS (div_save_regs[i])
+ | RA (1)
+ | (i * 4 + 8)
+ )
+ );
+ tcg_out32 (s, STW | RS (0) | RA (1) | (frame_size - 4));
+ tcg_out_mov (s, 4, 6);
+ tcg_out_b (s, LK, target);
+ tcg_out_mov (s, 6, 4);
+
+ for (i = 0; i < ARRAY_SIZE (div_save_regs); ++i)
+ tcg_out32 (s, (LWZ
+ | RT (div_save_regs[i])
+ | RA (1)
+ | (i * 4 + 8)
+ )
+ );
+ tcg_out32 (s, LWZ | RT (0) | RA (1) | (frame_size - 4));
+ tcg_out32 (s, MTSPR | RS (0) | LR);
+ tcg_out32 (s, ADDI | RT (1) | RA (1) | frame_size);
+ tcg_out32 (s, BCLR | BO_ALWAYS);
+ }
}
static void tcg_out_ld (TCGContext *s, TCGType type, int ret, int arg1,
@@ -1018,41 +1095,6 @@ static void tcg_out_brcond2(TCGContext *s,
tcg_out_label(s, label_next, (tcg_target_long)s->code_ptr);
}
-static uint64_t __attribute ((used)) ppc_udiv_helper (uint64_t a, uint32_t b)
-{
- uint64_t rem, quo;
- quo = a / b;
- rem = a % b;
- return (rem << 32) | (uint32_t) quo;
-}
-
-static uint64_t __attribute ((used)) ppc_div_helper (int64_t a, int32_t b)
-{
- int64_t rem, quo;
- quo = a / b;
- rem = a % b;
- return (rem << 32) | (uint32_t) quo;
-}
-
-#define MAKE_TRAMPOLINE(name) \
-extern void name##_trampoline (void); \
-asm (#name "_trampoline:\n" \
- " mflr 0\n" \
- " addi 1,1,-112\n" \
- " mr 4,6\n" \
- " stmw 7,0(1)\n" \
- " stw 0,108(0)\n" \
- " bl ppc_" #name "_helper\n" \
- " lmw 7,0(1)\n" \
- " lwz 0,108(0)\n" \
- " addi 1,1,112\n" \
- " mtlr 0\n" \
- " blr\n" \
- )
-
-MAKE_TRAMPOLINE (div);
-MAKE_TRAMPOLINE (udiv);
-
static void tcg_out_div2 (TCGContext *s, int uns)
{
void *label1_ptr, *label2_ptr;
@@ -1067,7 +1109,7 @@ static void tcg_out_div2 (TCGContext *s, int uns)
label1_ptr = s->code_ptr;
tcg_out32 (s, BC | BI (7, CR_EQ) | BO_COND_TRUE);
- tcg_out_b (s, LK, (tcg_target_long) (uns ? udiv_trampoline : div_trampoline));
+ tcg_out_b (s, LK, (tcg_target_long) (uns ? udiv_addr : div_addr));
label2_ptr = s->code_ptr;
tcg_out32 (s, B);