Commit bffe143153b1d4efa472dd24e10c0ae8c3c3cbc1
1 parent
2ae72bce
Restore AREG0 after calls
git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@5018 c046a42c-6fe2-441c-8c8c-71466251a162
Showing
1 changed file
with
64 additions
and
73 deletions
tcg/sparc/tcg-target.c
| ... | ... | @@ -491,11 +491,28 @@ static const void * const qemu_st_helpers[4] = { |
| 491 | 491 | }; |
| 492 | 492 | #endif |
| 493 | 493 | |
| 494 | +#if TARGET_LONG_BITS == 32 | |
| 495 | +#define TARGET_LD_OP LDUW | |
| 496 | +#else | |
| 497 | +#define TARGET_LD_OP LDX | |
| 498 | +#endif | |
| 499 | + | |
| 500 | +#ifdef __arch64__ | |
| 501 | +#define HOST_LD_OP LDX | |
| 502 | +#define HOST_ST_OP STX | |
| 503 | +#define HOST_SLL_OP SHIFT_SLLX | |
| 504 | +#define HOST_SRA_OP SHIFT_SRAX | |
| 505 | +#else | |
| 506 | +#define HOST_LD_OP LDUW | |
| 507 | +#define HOST_ST_OP STW | |
| 508 | +#define HOST_SLL_OP SHIFT_SLL | |
| 509 | +#define HOST_SRA_OP SHIFT_SRA | |
| 510 | +#endif | |
| 511 | + | |
| 494 | 512 | static void tcg_out_qemu_ld(TCGContext *s, const TCGArg *args, |
| 495 | 513 | int opc) |
| 496 | 514 | { |
| 497 | 515 | int addr_reg, data_reg, arg0, arg1, arg2, mem_index, s_bits; |
| 498 | - int target_ld_op, host_ld_op, sll_op, sra_op; | |
| 499 | 516 | #if defined(CONFIG_SOFTMMU) |
| 500 | 517 | uint32_t *label1_ptr, *label2_ptr; |
| 501 | 518 | #endif |
| ... | ... | @@ -509,23 +526,6 @@ static void tcg_out_qemu_ld(TCGContext *s, const TCGArg *args, |
| 509 | 526 | arg1 = TCG_REG_O1; |
| 510 | 527 | arg2 = TCG_REG_O2; |
| 511 | 528 | |
| 512 | -#if TARGET_LONG_BITS == 32 | |
| 513 | - target_ld_op = LDUW; | |
| 514 | -#else | |
| 515 | - target_ld_op = LDX; | |
| 516 | -#endif | |
| 517 | - | |
| 518 | -#ifdef __arch64__ | |
| 519 | - host_ld_op = LDX; | |
| 520 | - sll_op = SHIFT_SLLX; | |
| 521 | - sra_op = SHIFT_SRAX; | |
| 522 | -#else | |
| 523 | - host_ld_op = LDUW; | |
| 524 | - sll_op = SHIFT_SLL; | |
| 525 | - sra_op = SHIFT_SRA; | |
| 526 | -#endif | |
| 527 | - | |
| 528 | - | |
| 529 | 529 | #if defined(CONFIG_SOFTMMU) |
| 530 | 530 | /* srl addr_reg, x, arg1 */ |
| 531 | 531 | tcg_out_arithi(s, arg1, addr_reg, TARGET_PAGE_BITS - CPU_TLB_ENTRY_BITS, |
| ... | ... | @@ -545,7 +545,7 @@ static void tcg_out_qemu_ld(TCGContext *s, const TCGArg *args, |
| 545 | 545 | tcg_out_arith(s, arg1, TCG_AREG0, arg1, ARITH_ADD); |
| 546 | 546 | |
| 547 | 547 | /* ld [arg1], arg2 */ |
| 548 | - tcg_out32(s, target_ld_op | INSN_RD(arg2) | INSN_RS1(arg1) | | |
| 548 | + tcg_out32(s, TARGET_LD_OP | INSN_RD(arg2) | INSN_RS1(arg1) | | |
| 549 | 549 | INSN_RS2(TCG_REG_G0)); |
| 550 | 550 | |
| 551 | 551 | /* subcc arg0, arg2, %g0 */ |
| ... | ... | @@ -559,37 +559,45 @@ static void tcg_out_qemu_ld(TCGContext *s, const TCGArg *args, |
| 559 | 559 | /* mov (delay slot) */ |
| 560 | 560 | tcg_out_mov(s, arg0, addr_reg); |
| 561 | 561 | |
| 562 | + /* mov */ | |
| 563 | + tcg_out_movi(s, TCG_TYPE_I32, arg1, mem_index); | |
| 564 | + | |
| 562 | 565 | /* XXX: move that code at the end of the TB */ |
| 563 | 566 | /* qemu_ld_helper[s_bits](arg0, arg1) */ |
| 564 | 567 | tcg_out32(s, CALL | ((((tcg_target_ulong)qemu_ld_helpers[s_bits] |
| 565 | 568 | - (tcg_target_ulong)s->code_ptr) >> 2) |
| 566 | 569 | & 0x3fffffff)); |
| 567 | - /* mov (delay slot) */ | |
| 568 | - tcg_out_movi(s, TCG_TYPE_I32, arg1, mem_index); | |
| 570 | + /* Store AREG0 in stack to avoid ugly glibc bugs that mangle | |
| 571 | + global registers */ | |
| 572 | + // delay slot | |
| 573 | + tcg_out_ldst(s, TCG_AREG0, TCG_REG_CALL_STACK, | |
| 574 | + TCG_TARGET_CALL_STACK_OFFSET - sizeof(long), HOST_ST_OP); | |
| 575 | + tcg_out_ldst(s, TCG_AREG0, TCG_REG_CALL_STACK, | |
| 576 | + TCG_TARGET_CALL_STACK_OFFSET - sizeof(long), HOST_LD_OP); | |
| 569 | 577 | |
| 570 | 578 | /* data_reg = sign_extend(arg0) */ |
| 571 | 579 | switch(opc) { |
| 572 | 580 | case 0 | 4: |
| 573 | 581 | /* sll arg0, 24/56, data_reg */ |
| 574 | 582 | tcg_out_arithi(s, data_reg, arg0, (int)sizeof(tcg_target_long) * 8 - 8, |
| 575 | - sll_op); | |
| 583 | + HOST_SLL_OP); | |
| 576 | 584 | /* sra data_reg, 24/56, data_reg */ |
| 577 | 585 | tcg_out_arithi(s, data_reg, data_reg, |
| 578 | - (int)sizeof(tcg_target_long) * 8 - 8, sra_op); | |
| 586 | + (int)sizeof(tcg_target_long) * 8 - 8, HOST_SRA_OP); | |
| 579 | 587 | break; |
| 580 | 588 | case 1 | 4: |
| 581 | 589 | /* sll arg0, 16/48, data_reg */ |
| 582 | 590 | tcg_out_arithi(s, data_reg, arg0, |
| 583 | - (int)sizeof(tcg_target_long) * 8 - 16, sll_op); | |
| 591 | + (int)sizeof(tcg_target_long) * 8 - 16, HOST_SLL_OP); | |
| 584 | 592 | /* sra data_reg, 16/48, data_reg */ |
| 585 | 593 | tcg_out_arithi(s, data_reg, data_reg, |
| 586 | - (int)sizeof(tcg_target_long) * 8 - 16, sra_op); | |
| 594 | + (int)sizeof(tcg_target_long) * 8 - 16, HOST_SRA_OP); | |
| 587 | 595 | break; |
| 588 | 596 | case 2 | 4: |
| 589 | 597 | /* sll arg0, 32, data_reg */ |
| 590 | - tcg_out_arithi(s, data_reg, arg0, 32, sll_op); | |
| 598 | + tcg_out_arithi(s, data_reg, arg0, 32, HOST_SLL_OP); | |
| 591 | 599 | /* sra data_reg, 32, data_reg */ |
| 592 | - tcg_out_arithi(s, data_reg, data_reg, 32, sra_op); | |
| 600 | + tcg_out_arithi(s, data_reg, data_reg, 32, HOST_SRA_OP); | |
| 593 | 601 | break; |
| 594 | 602 | case 0: |
| 595 | 603 | case 1: |
| ... | ... | @@ -616,7 +624,7 @@ static void tcg_out_qemu_ld(TCGContext *s, const TCGArg *args, |
| 616 | 624 | |
| 617 | 625 | /* ld [arg1 + x], arg1 */ |
| 618 | 626 | tcg_out_ldst(s, arg1, arg1, offsetof(CPUTLBEntry, addend) - |
| 619 | - offsetof(CPUTLBEntry, addr_read), host_ld_op); | |
| 627 | + offsetof(CPUTLBEntry, addr_read), HOST_LD_OP); | |
| 620 | 628 | /* add addr_reg, arg1, arg0 */ |
| 621 | 629 | tcg_out_arith(s, arg0, addr_reg, arg1, ARITH_ADD); |
| 622 | 630 | #else |
| ... | ... | @@ -693,7 +701,6 @@ static void tcg_out_qemu_st(TCGContext *s, const TCGArg *args, |
| 693 | 701 | int opc) |
| 694 | 702 | { |
| 695 | 703 | int addr_reg, data_reg, arg0, arg1, arg2, mem_index, s_bits; |
| 696 | - int target_ld_op, host_ld_op; | |
| 697 | 704 | #if defined(CONFIG_SOFTMMU) |
| 698 | 705 | uint32_t *label1_ptr, *label2_ptr; |
| 699 | 706 | #endif |
| ... | ... | @@ -708,18 +715,6 @@ static void tcg_out_qemu_st(TCGContext *s, const TCGArg *args, |
| 708 | 715 | arg1 = TCG_REG_O1; |
| 709 | 716 | arg2 = TCG_REG_O2; |
| 710 | 717 | |
| 711 | -#if TARGET_LONG_BITS == 32 | |
| 712 | - target_ld_op = LDUW; | |
| 713 | -#else | |
| 714 | - target_ld_op = LDX; | |
| 715 | -#endif | |
| 716 | - | |
| 717 | -#ifdef __arch64__ | |
| 718 | - host_ld_op = LDX; | |
| 719 | -#else | |
| 720 | - host_ld_op = LDUW; | |
| 721 | -#endif | |
| 722 | - | |
| 723 | 718 | #if defined(CONFIG_SOFTMMU) |
| 724 | 719 | /* srl addr_reg, x, arg1 */ |
| 725 | 720 | tcg_out_arithi(s, arg1, addr_reg, TARGET_PAGE_BITS - CPU_TLB_ENTRY_BITS, |
| ... | ... | @@ -740,7 +735,7 @@ static void tcg_out_qemu_st(TCGContext *s, const TCGArg *args, |
| 740 | 735 | tcg_out_arith(s, arg1, TCG_AREG0, arg1, ARITH_ADD); |
| 741 | 736 | |
| 742 | 737 | /* ld [arg1], arg2 */ |
| 743 | - tcg_out32(s, target_ld_op | INSN_RD(arg2) | INSN_RS1(arg1) | | |
| 738 | + tcg_out32(s, TARGET_LD_OP | INSN_RD(arg2) | INSN_RS1(arg1) | | |
| 744 | 739 | INSN_RS2(TCG_REG_G0)); |
| 745 | 740 | |
| 746 | 741 | /* subcc arg0, arg2, %g0 */ |
| ... | ... | @@ -757,13 +752,21 @@ static void tcg_out_qemu_st(TCGContext *s, const TCGArg *args, |
| 757 | 752 | /* mov */ |
| 758 | 753 | tcg_out_mov(s, arg1, data_reg); |
| 759 | 754 | |
| 755 | + /* mov */ | |
| 756 | + tcg_out_movi(s, TCG_TYPE_I32, arg2, mem_index); | |
| 757 | + | |
| 760 | 758 | /* XXX: move that code at the end of the TB */ |
| 761 | 759 | /* qemu_st_helper[s_bits](arg0, arg1, arg2) */ |
| 762 | 760 | tcg_out32(s, CALL | ((((tcg_target_ulong)qemu_st_helpers[s_bits] |
| 763 | 761 | - (tcg_target_ulong)s->code_ptr) >> 2) |
| 764 | 762 | & 0x3fffffff)); |
| 765 | - /* mov (delay slot) */ | |
| 766 | - tcg_out_movi(s, TCG_TYPE_I32, arg2, mem_index); | |
| 763 | + /* Store AREG0 in stack to avoid ugly glibc bugs that mangle | |
| 764 | + global registers */ | |
| 765 | + // delay slot | |
| 766 | + tcg_out_ldst(s, TCG_AREG0, TCG_REG_CALL_STACK, | |
| 767 | + TCG_TARGET_CALL_STACK_OFFSET - sizeof(long), HOST_ST_OP); | |
| 768 | + tcg_out_ldst(s, TCG_AREG0, TCG_REG_CALL_STACK, | |
| 769 | + TCG_TARGET_CALL_STACK_OFFSET - sizeof(long), HOST_LD_OP); | |
| 767 | 770 | |
| 768 | 771 | /* will become: |
| 769 | 772 | ba label2 */ |
| ... | ... | @@ -780,7 +783,7 @@ static void tcg_out_qemu_st(TCGContext *s, const TCGArg *args, |
| 780 | 783 | |
| 781 | 784 | /* ld [arg1 + x], arg1 */ |
| 782 | 785 | tcg_out_ldst(s, arg1, arg1, offsetof(CPUTLBEntry, addend) - |
| 783 | - offsetof(CPUTLBEntry, addr_write), host_ld_op); | |
| 786 | + offsetof(CPUTLBEntry, addr_write), HOST_LD_OP); | |
| 784 | 787 | |
| 785 | 788 | /* add addr_reg, arg1, arg0 */ |
| 786 | 789 | tcg_out_arith(s, arg0, addr_reg, arg1, ARITH_ADD); |
| ... | ... | @@ -862,35 +865,23 @@ static inline void tcg_out_op(TCGContext *s, int opc, const TCGArg *args, |
| 862 | 865 | s->tb_next_offset[args[0]] = s->code_ptr - s->code_buf; |
| 863 | 866 | break; |
| 864 | 867 | case INDEX_op_call: |
| 865 | - { | |
| 866 | - unsigned int st_op, ld_op; | |
| 867 | - | |
| 868 | -#ifdef __arch64__ | |
| 869 | - st_op = STX; | |
| 870 | - ld_op = LDX; | |
| 871 | -#else | |
| 872 | - st_op = STW; | |
| 873 | - ld_op = LDUW; | |
| 874 | -#endif | |
| 875 | - if (const_args[0]) | |
| 876 | - tcg_out32(s, CALL | ((((tcg_target_ulong)args[0] | |
| 877 | - - (tcg_target_ulong)s->code_ptr) >> 2) | |
| 878 | - & 0x3fffffff)); | |
| 879 | - else { | |
| 880 | - tcg_out_ld_ptr(s, TCG_REG_I5, | |
| 881 | - (tcg_target_long)(s->tb_next + args[0])); | |
| 882 | - tcg_out32(s, JMPL | INSN_RD(TCG_REG_O7) | INSN_RS1(TCG_REG_I5) | | |
| 883 | - INSN_RS2(TCG_REG_G0)); | |
| 884 | - } | |
| 885 | - /* Store AREG0 in stack to avoid ugly glibc bugs that mangle | |
| 886 | - global registers */ | |
| 887 | - tcg_out_ldst(s, TCG_AREG0, TCG_REG_CALL_STACK, | |
| 888 | - TCG_TARGET_CALL_STACK_OFFSET - sizeof(long), | |
| 889 | - st_op); // delay slot | |
| 890 | - tcg_out_ldst(s, TCG_AREG0, TCG_REG_CALL_STACK, | |
| 891 | - TCG_TARGET_CALL_STACK_OFFSET - sizeof(long), | |
| 892 | - ld_op); | |
| 868 | + if (const_args[0]) | |
| 869 | + tcg_out32(s, CALL | ((((tcg_target_ulong)args[0] | |
| 870 | + - (tcg_target_ulong)s->code_ptr) >> 2) | |
| 871 | + & 0x3fffffff)); | |
| 872 | + else { | |
| 873 | + tcg_out_ld_ptr(s, TCG_REG_I5, | |
| 874 | + (tcg_target_long)(s->tb_next + args[0])); | |
| 875 | + tcg_out32(s, JMPL | INSN_RD(TCG_REG_O7) | INSN_RS1(TCG_REG_I5) | | |
| 876 | + INSN_RS2(TCG_REG_G0)); | |
| 893 | 877 | } |
| 878 | + /* Store AREG0 in stack to avoid ugly glibc bugs that mangle | |
| 879 | + global registers */ | |
| 880 | + // delay slot | |
| 881 | + tcg_out_ldst(s, TCG_AREG0, TCG_REG_CALL_STACK, | |
| 882 | + TCG_TARGET_CALL_STACK_OFFSET - sizeof(long), HOST_ST_OP); | |
| 883 | + tcg_out_ldst(s, TCG_AREG0, TCG_REG_CALL_STACK, | |
| 884 | + TCG_TARGET_CALL_STACK_OFFSET - sizeof(long), HOST_LD_OP); | |
| 894 | 885 | break; |
| 895 | 886 | case INDEX_op_jmp: |
| 896 | 887 | case INDEX_op_br: | ... | ... |