2 * Copyright (C) 2009-2011 STMicroelectronics. All rights reserved.
3 * Copyright (C) 2008 Apple Inc. All rights reserved.
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
14 * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
15 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
17 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
18 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
19 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
20 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
21 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
22 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
23 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
24 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 #ifndef SH4Assembler_h
28 #define SH4Assembler_h
30 #if ENABLE(ASSEMBLER) && CPU(SH4)
32 #include "AssemblerBuffer.h"
33 #include "AssemblerBufferWithConstantPool.h"
36 #include <wtf/Assertions.h>
37 #include <wtf/Vector.h>
40 #define SH4_ASSEMBLER_TRACING
44 typedef uint16_t SH4Word
;
47 INVALID_OPCODE
= 0xffff,
49 ADDIMM_OPCODE
= 0x7000,
53 ANDIMM_OPCODE
= 0xc900,
68 CMPEQ_OPCODE
= 0x3000,
69 CMPEQIMM_OPCODE
= 0x8800,
70 CMPGE_OPCODE
= 0x3003,
71 CMPGT_OPCODE
= 0x3007,
72 CMPHI_OPCODE
= 0x3006,
73 CMPHS_OPCODE
= 0x3002,
74 CMPPL_OPCODE
= 0x4015,
75 CMPPZ_OPCODE
= 0x4011,
76 CMPSTR_OPCODE
= 0x200c,
78 FCMPEQ_OPCODE
= 0xf004,
79 FCMPGT_OPCODE
= 0xf005,
88 LDSPR_OPCODE
= 0x402a,
89 LDSLPR_OPCODE
= 0x4026,
91 MOVIMM_OPCODE
= 0xe000,
92 MOVB_WRITE_RN_OPCODE
= 0x2000,
93 MOVB_WRITE_RNDEC_OPCODE
= 0x2004,
94 MOVB_WRITE_R0RN_OPCODE
= 0x0004,
95 MOVB_WRITE_OFFGBR_OPCODE
= 0xc000,
96 MOVB_WRITE_OFFRN_OPCODE
= 0x8000,
97 MOVB_READ_RM_OPCODE
= 0x6000,
98 MOVB_READ_RMINC_OPCODE
= 0x6004,
99 MOVB_READ_R0RM_OPCODE
= 0x000c,
100 MOVB_READ_OFFGBR_OPCODE
= 0xc400,
101 MOVB_READ_OFFRM_OPCODE
= 0x8400,
102 MOVL_WRITE_RN_OPCODE
= 0x2002,
103 MOVL_WRITE_RNDEC_OPCODE
= 0x2006,
104 MOVL_WRITE_R0RN_OPCODE
= 0x0006,
105 MOVL_WRITE_OFFGBR_OPCODE
= 0xc200,
106 MOVL_WRITE_OFFRN_OPCODE
= 0x1000,
107 MOVL_READ_RM_OPCODE
= 0x6002,
108 MOVL_READ_RMINC_OPCODE
= 0x6006,
109 MOVL_READ_R0RM_OPCODE
= 0x000e,
110 MOVL_READ_OFFGBR_OPCODE
= 0xc600,
111 MOVL_READ_OFFPC_OPCODE
= 0xd000,
112 MOVL_READ_OFFRM_OPCODE
= 0x5000,
113 MOVW_WRITE_RN_OPCODE
= 0x2001,
114 MOVW_READ_RM_OPCODE
= 0x6001,
115 MOVW_READ_R0RM_OPCODE
= 0x000d,
116 MOVW_READ_OFFRM_OPCODE
= 0x8500,
117 MOVW_READ_OFFPC_OPCODE
= 0x9000,
118 MOVA_READ_OFFPC_OPCODE
= 0xc700,
119 MOVT_OPCODE
= 0x0029,
120 MULL_OPCODE
= 0x0007,
121 DMULL_L_OPCODE
= 0x3005,
122 STSMACL_OPCODE
= 0x001a,
123 STSMACH_OPCODE
= 0x000a,
124 DMULSL_OPCODE
= 0x300d,
126 NEGC_OPCODE
= 0x600a,
129 ORIMM_OPCODE
= 0xcb00,
130 ORBIMM_OPCODE
= 0xcf00,
131 SETS_OPCODE
= 0x0058,
132 SETT_OPCODE
= 0x0018,
133 SHAD_OPCODE
= 0x400c,
134 SHAL_OPCODE
= 0x4020,
135 SHAR_OPCODE
= 0x4021,
136 SHLD_OPCODE
= 0x400d,
137 SHLL_OPCODE
= 0x4000,
138 SHLL2_OPCODE
= 0x4008,
139 SHLL8_OPCODE
= 0x4018,
140 SHLL16_OPCODE
= 0x4028,
141 SHLR_OPCODE
= 0x4001,
142 SHLR2_OPCODE
= 0x4009,
143 SHLR8_OPCODE
= 0x4019,
144 SHLR16_OPCODE
= 0x4029,
145 STSPR_OPCODE
= 0x002a,
146 STSLPR_OPCODE
= 0x4022,
147 FLOAT_OPCODE
= 0xf02d,
149 SUBC_OPCODE
= 0x300a,
150 SUBV_OPCODE
= 0x300b,
152 TSTIMM_OPCODE
= 0xc800,
153 TSTB_OPCODE
= 0xcc00,
154 EXTUW_OPCODE
= 0x600d,
156 XORIMM_OPCODE
= 0xca00,
157 XORB_OPCODE
= 0xce00,
158 FMOVS_READ_RM_INC_OPCODE
= 0xf009,
159 FMOVS_READ_RM_OPCODE
= 0xf008,
160 FMOVS_READ_R0RM_OPCODE
= 0xf006,
161 FMOVS_WRITE_RN_OPCODE
= 0xf00a,
162 FMOVS_WRITE_RN_DEC_OPCODE
= 0xf00b,
163 FMOVS_WRITE_R0RN_OPCODE
= 0xf007,
164 FCNVDS_DRM_FPUL_OPCODE
= 0xf0bd,
165 LDS_RM_FPUL_OPCODE
= 0x405a,
166 FLDS_FRM_FPUL_OPCODE
= 0xf01d,
167 STS_FPUL_RN_OPCODE
= 0x005a,
168 FSTS_FPUL_FRN_OPCODE
= 0xF00d,
169 LDSFPSCR_OPCODE
= 0x406a,
170 STSFPSCR_OPCODE
= 0x006a,
171 LDSRMFPUL_OPCODE
= 0x405a,
172 FSTSFPULFRN_OPCODE
= 0xf00d,
173 FSQRT_OPCODE
= 0xf06d,
174 FSCHG_OPCODE
= 0xf3fd,
178 namespace SH4Registers
{
220 inline uint16_t getOpcodeGroup1(uint16_t opc
, int rm
, int rn
)
222 return (opc
| ((rm
& 0xf) << 8) | ((rn
& 0xf) << 4));
225 inline uint16_t getOpcodeGroup2(uint16_t opc
, int rm
)
227 return (opc
| ((rm
& 0xf) << 8));
230 inline uint16_t getOpcodeGroup3(uint16_t opc
, int rm
, int rn
)
232 return (opc
| ((rm
& 0xf) << 8) | (rn
& 0xff));
235 inline uint16_t getOpcodeGroup4(uint16_t opc
, int rm
, int rn
, int offset
)
237 return (opc
| ((rm
& 0xf) << 8) | ((rn
& 0xf) << 4) | (offset
& 0xf));
240 inline uint16_t getOpcodeGroup5(uint16_t opc
, int rm
)
242 return (opc
| (rm
& 0xff));
245 inline uint16_t getOpcodeGroup6(uint16_t opc
, int rm
)
247 return (opc
| (rm
& 0xfff));
250 inline uint16_t getOpcodeGroup7(uint16_t opc
, int rm
)
252 return (opc
| ((rm
& 0x7) << 9));
255 inline uint16_t getOpcodeGroup8(uint16_t opc
, int rm
, int rn
)
257 return (opc
| ((rm
& 0x7) << 9) | ((rn
& 0x7) << 5));
260 inline uint16_t getOpcodeGroup9(uint16_t opc
, int rm
, int rn
)
262 return (opc
| ((rm
& 0xf) << 8) | ((rn
& 0x7) << 5));
265 inline uint16_t getOpcodeGroup10(uint16_t opc
, int rm
, int rn
)
267 return (opc
| ((rm
& 0x7) << 9) | ((rn
& 0xf) << 4));
270 inline uint16_t getOpcodeGroup11(uint16_t opc
, int rm
, int rn
)
272 return (opc
| ((rm
& 0xf) << 4) | (rn
& 0xf));
275 inline uint16_t getRn(uint16_t x
)
277 return ((x
& 0xf00) >> 8);
280 inline uint16_t getRm(uint16_t x
)
282 return ((x
& 0xf0) >> 4);
285 inline uint16_t getDisp(uint16_t x
)
290 inline uint16_t getImm8(uint16_t x
)
295 inline uint16_t getImm12(uint16_t x
)
300 inline uint16_t getDRn(uint16_t x
)
302 return ((x
& 0xe00) >> 9);
305 inline uint16_t getDRm(uint16_t x
)
307 return ((x
& 0xe0) >> 5);
312 typedef SH4Registers::RegisterID RegisterID
;
313 typedef SH4Registers::FPRegisterID FPRegisterID
;
314 typedef AssemblerBufferWithConstantPool
<512, 4, 2, SH4Assembler
> SH4Buffer
;
315 static const RegisterID scratchReg1
= SH4Registers::r3
;
316 static const RegisterID scratchReg2
= SH4Registers::r11
;
317 static const uint32_t maxInstructionSize
= 16;
321 padForAlign16
= 0x0009,
322 padForAlign32
= 0x00090009,
327 m_claimscratchReg
= 0x0;
330 // SH4 condition codes
333 NE
= 0x1, // Not Equal
334 HS
= 0x2, // Unsigend Greater Than equal
335 HI
= 0x3, // Unsigend Greater Than
336 LS
= 0x4, // Unsigend Lower or Same
337 LI
= 0x5, // Unsigend Lower
338 GE
= 0x6, // Greater or Equal
339 LT
= 0x7, // Less Than
340 GT
= 0x8, // Greater Than
341 LE
= 0x9, // Less or Equal
342 OF
= 0xa, // OverFlow
344 EQU
= 0xc, // Equal or unordered(NaN)
352 // Opaque label types
354 bool isImmediate(int constant
)
356 return ((constant
<= 127) && (constant
>= -128));
359 RegisterID
claimScratch()
361 ASSERT((m_claimscratchReg
!= 0x3));
363 if (!(m_claimscratchReg
& 0x1)) {
364 m_claimscratchReg
= (m_claimscratchReg
| 0x1);
368 m_claimscratchReg
= (m_claimscratchReg
| 0x2);
372 void releaseScratch(RegisterID scratchR
)
374 if (scratchR
== scratchReg1
)
375 m_claimscratchReg
= (m_claimscratchReg
& 0x2);
377 m_claimscratchReg
= (m_claimscratchReg
& 0x1);
382 void pushReg(RegisterID reg
)
384 if (reg
== SH4Registers::pr
) {
385 oneShortOp(getOpcodeGroup2(STSLPR_OPCODE
, SH4Registers::sp
));
389 oneShortOp(getOpcodeGroup1(MOVL_WRITE_RNDEC_OPCODE
, SH4Registers::sp
, reg
));
392 void popReg(RegisterID reg
)
394 if (reg
== SH4Registers::pr
) {
395 oneShortOp(getOpcodeGroup2(LDSLPR_OPCODE
, SH4Registers::sp
));
399 oneShortOp(getOpcodeGroup1(MOVL_READ_RMINC_OPCODE
, reg
, SH4Registers::sp
));
402 void movt(RegisterID dst
)
404 uint16_t opc
= getOpcodeGroup2(MOVT_OPCODE
, dst
);
408 // Arithmetic operations
410 void addlRegReg(RegisterID src
, RegisterID dst
)
412 uint16_t opc
= getOpcodeGroup1(ADD_OPCODE
, dst
, src
);
416 void addclRegReg(RegisterID src
, RegisterID dst
)
418 uint16_t opc
= getOpcodeGroup1(ADDC_OPCODE
, dst
, src
);
422 void addvlRegReg(RegisterID src
, RegisterID dst
)
424 uint16_t opc
= getOpcodeGroup1(ADDV_OPCODE
, dst
, src
);
428 void addlImm8r(int imm8
, RegisterID dst
)
430 ASSERT((imm8
<= 127) && (imm8
>= -128));
432 uint16_t opc
= getOpcodeGroup3(ADDIMM_OPCODE
, dst
, imm8
);
436 void andlRegReg(RegisterID src
, RegisterID dst
)
438 uint16_t opc
= getOpcodeGroup1(AND_OPCODE
, dst
, src
);
442 void andlImm8r(int imm8
, RegisterID dst
)
444 ASSERT((imm8
<= 255) && (imm8
>= 0));
445 ASSERT(dst
== SH4Registers::r0
);
447 uint16_t opc
= getOpcodeGroup5(ANDIMM_OPCODE
, imm8
);
451 void div1lRegReg(RegisterID src
, RegisterID dst
)
453 uint16_t opc
= getOpcodeGroup1(DIV1_OPCODE
, dst
, src
);
457 void div0lRegReg(RegisterID src
, RegisterID dst
)
459 uint16_t opc
= getOpcodeGroup1(DIV0_OPCODE
, dst
, src
);
463 void notlReg(RegisterID src
, RegisterID dst
)
465 uint16_t opc
= getOpcodeGroup1(NOT_OPCODE
, dst
, src
);
469 void orlRegReg(RegisterID src
, RegisterID dst
)
471 uint16_t opc
= getOpcodeGroup1(OR_OPCODE
, dst
, src
);
475 void orlImm8r(int imm8
, RegisterID dst
)
477 ASSERT((imm8
<= 255) && (imm8
>= 0));
478 ASSERT(dst
== SH4Registers::r0
);
480 uint16_t opc
= getOpcodeGroup5(ORIMM_OPCODE
, imm8
);
484 void sublRegReg(RegisterID src
, RegisterID dst
)
486 uint16_t opc
= getOpcodeGroup1(SUB_OPCODE
, dst
, src
);
490 void subvlRegReg(RegisterID src
, RegisterID dst
)
492 uint16_t opc
= getOpcodeGroup1(SUBV_OPCODE
, dst
, src
);
496 void xorlRegReg(RegisterID src
, RegisterID dst
)
498 uint16_t opc
= getOpcodeGroup1(XOR_OPCODE
, dst
, src
);
502 void xorlImm8r(int imm8
, RegisterID dst
)
504 ASSERT((imm8
<= 255) && (imm8
>= 0));
505 ASSERT(dst
== SH4Registers::r0
);
507 uint16_t opc
= getOpcodeGroup5(XORIMM_OPCODE
, imm8
);
511 void shllImm8r(int imm
, RegisterID dst
)
515 oneShortOp(getOpcodeGroup2(SHLL_OPCODE
, dst
));
518 oneShortOp(getOpcodeGroup2(SHLL2_OPCODE
, dst
));
521 oneShortOp(getOpcodeGroup2(SHLL8_OPCODE
, dst
));
524 oneShortOp(getOpcodeGroup2(SHLL16_OPCODE
, dst
));
527 ASSERT_NOT_REACHED();
531 void neg(RegisterID dst
, RegisterID src
)
533 uint16_t opc
= getOpcodeGroup1(NEG_OPCODE
, dst
, src
);
537 void shllRegReg(RegisterID dst
, RegisterID rShift
)
539 uint16_t opc
= getOpcodeGroup1(SHLD_OPCODE
, dst
, rShift
);
543 void shlrRegReg(RegisterID dst
, RegisterID rShift
)
546 shllRegReg(dst
, rShift
);
549 void sharRegReg(RegisterID dst
, RegisterID rShift
)
552 shaRegReg(dst
, rShift
);
555 void shaRegReg(RegisterID dst
, RegisterID rShift
)
557 uint16_t opc
= getOpcodeGroup1(SHAD_OPCODE
, dst
, rShift
);
561 void shlrImm8r(int imm
, RegisterID dst
)
565 oneShortOp(getOpcodeGroup2(SHLR_OPCODE
, dst
));
568 oneShortOp(getOpcodeGroup2(SHLR2_OPCODE
, dst
));
571 oneShortOp(getOpcodeGroup2(SHLR8_OPCODE
, dst
));
574 oneShortOp(getOpcodeGroup2(SHLR16_OPCODE
, dst
));
577 ASSERT_NOT_REACHED();
581 void imullRegReg(RegisterID src
, RegisterID dst
)
583 uint16_t opc
= getOpcodeGroup1(MULL_OPCODE
, dst
, src
);
587 void dmullRegReg(RegisterID src
, RegisterID dst
)
589 uint16_t opc
= getOpcodeGroup1(DMULL_L_OPCODE
, dst
, src
);
593 void dmulslRegReg(RegisterID src
, RegisterID dst
)
595 uint16_t opc
= getOpcodeGroup1(DMULSL_OPCODE
, dst
, src
);
599 void stsmacl(RegisterID reg
)
601 uint16_t opc
= getOpcodeGroup2(STSMACL_OPCODE
, reg
);
605 void stsmach(RegisterID reg
)
607 uint16_t opc
= getOpcodeGroup2(STSMACH_OPCODE
, reg
);
613 void cmplRegReg(RegisterID left
, RegisterID right
, Condition cond
)
617 oneShortOp(getOpcodeGroup1(CMPEQ_OPCODE
, right
, left
));
620 oneShortOp(getOpcodeGroup1(CMPGT_OPCODE
, right
, left
));
623 oneShortOp(getOpcodeGroup1(CMPEQ_OPCODE
, right
, left
));
626 oneShortOp(getOpcodeGroup1(CMPGE_OPCODE
, right
, left
));
629 oneShortOp(getOpcodeGroup1(CMPHS_OPCODE
, right
, left
));
632 oneShortOp(getOpcodeGroup1(CMPHI_OPCODE
, right
, left
));
635 oneShortOp(getOpcodeGroup1(CMPHI_OPCODE
, left
, right
));
638 oneShortOp(getOpcodeGroup1(CMPHS_OPCODE
, left
, right
));
641 oneShortOp(getOpcodeGroup1(CMPGE_OPCODE
, left
, right
));
644 oneShortOp(getOpcodeGroup1(CMPGT_OPCODE
, left
, right
));
647 ASSERT_NOT_REACHED();
651 void cmppl(RegisterID reg
)
653 uint16_t opc
= getOpcodeGroup2(CMPPL_OPCODE
, reg
);
657 void cmppz(RegisterID reg
)
659 uint16_t opc
= getOpcodeGroup2(CMPPZ_OPCODE
, reg
);
663 void cmpEqImmR0(int imm
, RegisterID dst
)
665 uint16_t opc
= getOpcodeGroup5(CMPEQIMM_OPCODE
, imm
);
669 void testlRegReg(RegisterID src
, RegisterID dst
)
671 uint16_t opc
= getOpcodeGroup1(TST_OPCODE
, dst
, src
);
675 void testlImm8r(int imm
, RegisterID dst
)
677 ASSERT((dst
== SH4Registers::r0
) && (imm
<= 255) && (imm
>= 0));
679 uint16_t opc
= getOpcodeGroup5(TSTIMM_OPCODE
, imm
);
685 oneShortOp(NOP_OPCODE
, false);
690 oneShortOp(SETT_OPCODE
);
695 oneShortOp(CLRT_OPCODE
);
700 oneShortOp(FSCHG_OPCODE
);
705 oneShortOp(BRK_OPCODE
, false);
708 void branch(uint16_t opc
, int label
)
712 ASSERT((label
<= 127) && (label
>= -128));
713 oneShortOp(getOpcodeGroup5(BT_OPCODE
, label
));
716 ASSERT((label
<= 2047) && (label
>= -2048));
717 oneShortOp(getOpcodeGroup6(BRA_OPCODE
, label
));
720 ASSERT((label
<= 127) && (label
>= -128));
721 oneShortOp(getOpcodeGroup5(BF_OPCODE
, label
));
724 ASSERT_NOT_REACHED();
728 void branch(uint16_t opc
, RegisterID reg
)
732 oneShortOp(getOpcodeGroup2(BRAF_OPCODE
, reg
));
735 oneShortOp(getOpcodeGroup2(JMP_OPCODE
, reg
));
738 oneShortOp(getOpcodeGroup2(JSR_OPCODE
, reg
));
741 oneShortOp(getOpcodeGroup2(BSRF_OPCODE
, reg
));
744 ASSERT_NOT_REACHED();
748 void ldspr(RegisterID reg
)
750 uint16_t opc
= getOpcodeGroup2(LDSPR_OPCODE
, reg
);
754 void stspr(RegisterID reg
)
756 uint16_t opc
= getOpcodeGroup2(STSPR_OPCODE
, reg
);
760 void extuw(RegisterID src
, RegisterID dst
)
762 uint16_t opc
= getOpcodeGroup1(EXTUW_OPCODE
, dst
, src
);
768 void ldsrmfpul(RegisterID src
)
770 uint16_t opc
= getOpcodeGroup2(LDS_RM_FPUL_OPCODE
, src
);
774 void fneg(FPRegisterID dst
)
776 uint16_t opc
= getOpcodeGroup2(FNEG_OPCODE
, dst
);
777 oneShortOp(opc
, true, false);
780 void fsqrt(FPRegisterID dst
)
782 uint16_t opc
= getOpcodeGroup2(FSQRT_OPCODE
, dst
);
783 oneShortOp(opc
, true, false);
786 void stsfpulReg(RegisterID src
)
788 uint16_t opc
= getOpcodeGroup2(STS_FPUL_RN_OPCODE
, src
);
792 void floatfpulfrn(RegisterID src
)
794 uint16_t opc
= getOpcodeGroup2(FLOAT_OPCODE
, src
);
795 oneShortOp(opc
, true, false);
798 void fmull(FPRegisterID src
, FPRegisterID dst
)
800 uint16_t opc
= getOpcodeGroup1(FMUL_OPCODE
, dst
, src
);
801 oneShortOp(opc
, true, false);
804 void fmovsReadrm(RegisterID src
, FPRegisterID dst
)
806 uint16_t opc
= getOpcodeGroup1(FMOVS_READ_RM_OPCODE
, dst
, src
);
807 oneShortOp(opc
, true, false);
810 void fmovsWriterm(FPRegisterID src
, RegisterID dst
)
812 uint16_t opc
= getOpcodeGroup1(FMOVS_WRITE_RN_OPCODE
, dst
, src
);
813 oneShortOp(opc
, true, false);
816 void fmovsWriter0r(FPRegisterID src
, RegisterID dst
)
818 uint16_t opc
= getOpcodeGroup1(FMOVS_WRITE_R0RN_OPCODE
, dst
, src
);
819 oneShortOp(opc
, true, false);
822 void fmovsReadr0r(RegisterID src
, FPRegisterID dst
)
824 uint16_t opc
= getOpcodeGroup1(FMOVS_READ_R0RM_OPCODE
, dst
, src
);
825 oneShortOp(opc
, true, false);
828 void fmovsReadrminc(RegisterID src
, FPRegisterID dst
)
830 uint16_t opc
= getOpcodeGroup1(FMOVS_READ_RM_INC_OPCODE
, dst
, src
);
831 oneShortOp(opc
, true, false);
834 void fmovsWriterndec(FPRegisterID src
, RegisterID dst
)
836 uint16_t opc
= getOpcodeGroup1(FMOVS_WRITE_RN_DEC_OPCODE
, dst
, src
);
837 oneShortOp(opc
, true, false);
840 void ftrcRegfpul(FPRegisterID src
)
842 uint16_t opc
= getOpcodeGroup2(FTRC_OPCODE
, src
);
843 oneShortOp(opc
, true, false);
846 void fldsfpul(RegisterID src
)
848 uint16_t opc
= getOpcodeGroup2(FLDS_FRM_FPUL_OPCODE
, src
);
852 void fstsfpul(RegisterID src
)
854 uint16_t opc
= getOpcodeGroup2(FSTS_FPUL_FRN_OPCODE
, src
);
858 void ldsfpscr(RegisterID reg
)
860 uint16_t opc
= getOpcodeGroup2(LDSFPSCR_OPCODE
, reg
);
864 void stsfpscr(RegisterID reg
)
866 uint16_t opc
= getOpcodeGroup2(STSFPSCR_OPCODE
, reg
);
872 void dcnvds(FPRegisterID src
)
874 uint16_t opc
= getOpcodeGroup7(FCNVDS_DRM_FPUL_OPCODE
, src
>> 1);
878 void dcmppeq(FPRegisterID src
, FPRegisterID dst
)
880 uint16_t opc
= getOpcodeGroup8(FCMPEQ_OPCODE
, dst
>> 1, src
>> 1);
884 void dcmppgt(FPRegisterID src
, FPRegisterID dst
)
886 uint16_t opc
= getOpcodeGroup8(FCMPGT_OPCODE
, dst
>> 1, src
>> 1);
890 void dmulRegReg(FPRegisterID src
, FPRegisterID dst
)
892 uint16_t opc
= getOpcodeGroup8(FMUL_OPCODE
, dst
>> 1, src
>> 1);
896 void dsubRegReg(FPRegisterID src
, FPRegisterID dst
)
898 uint16_t opc
= getOpcodeGroup8(FSUB_OPCODE
, dst
>> 1, src
>> 1);
902 void daddRegReg(FPRegisterID src
, FPRegisterID dst
)
904 uint16_t opc
= getOpcodeGroup8(FADD_OPCODE
, dst
>> 1, src
>> 1);
908 void dmovRegReg(FPRegisterID src
, FPRegisterID dst
)
910 uint16_t opc
= getOpcodeGroup8(FMOV_OPCODE
, dst
>> 1, src
>> 1);
914 void ddivRegReg(FPRegisterID src
, FPRegisterID dst
)
916 uint16_t opc
= getOpcodeGroup8(FDIV_OPCODE
, dst
>> 1, src
>> 1);
920 void dsqrt(FPRegisterID dst
)
922 uint16_t opc
= getOpcodeGroup7(FSQRT_OPCODE
, dst
>> 1);
926 void dneg(FPRegisterID dst
)
928 uint16_t opc
= getOpcodeGroup7(FNEG_OPCODE
, dst
>> 1);
932 void fmovReadrm(RegisterID src
, FPRegisterID dst
)
934 uint16_t opc
= getOpcodeGroup10(FMOVS_READ_RM_OPCODE
, dst
>> 1, src
);
938 void fmovWriterm(FPRegisterID src
, RegisterID dst
)
940 uint16_t opc
= getOpcodeGroup9(FMOVS_WRITE_RN_OPCODE
, dst
, src
>> 1);
944 void fmovWriter0r(FPRegisterID src
, RegisterID dst
)
946 uint16_t opc
= getOpcodeGroup9(FMOVS_WRITE_R0RN_OPCODE
, dst
, src
>> 1);
950 void fmovReadr0r(RegisterID src
, FPRegisterID dst
)
952 uint16_t opc
= getOpcodeGroup10(FMOVS_READ_R0RM_OPCODE
, dst
>> 1, src
);
956 void fmovReadrminc(RegisterID src
, FPRegisterID dst
)
958 uint16_t opc
= getOpcodeGroup10(FMOVS_READ_RM_INC_OPCODE
, dst
>> 1, src
);
962 void fmovWriterndec(FPRegisterID src
, RegisterID dst
)
964 uint16_t opc
= getOpcodeGroup9(FMOVS_WRITE_RN_DEC_OPCODE
, dst
, src
>> 1);
968 void floatfpulDreg(FPRegisterID src
)
970 uint16_t opc
= getOpcodeGroup7(FLOAT_OPCODE
, src
>> 1);
974 void ftrcdrmfpul(FPRegisterID src
)
976 uint16_t opc
= getOpcodeGroup7(FTRC_OPCODE
, src
>> 1);
982 void movImm8(int imm8
, RegisterID dst
)
984 ASSERT((imm8
<= 127) && (imm8
>= -128));
986 uint16_t opc
= getOpcodeGroup3(MOVIMM_OPCODE
, dst
, imm8
);
990 void movlRegReg(RegisterID src
, RegisterID dst
)
992 uint16_t opc
= getOpcodeGroup1(MOV_OPCODE
, dst
, src
);
996 void movwRegMem(RegisterID src
, RegisterID dst
)
998 uint16_t opc
= getOpcodeGroup1(MOVW_WRITE_RN_OPCODE
, dst
, src
);
1002 void movwMemReg(RegisterID src
, RegisterID dst
)
1004 uint16_t opc
= getOpcodeGroup1(MOVW_READ_RM_OPCODE
, dst
, src
);
1008 void movwPCReg(int offset
, RegisterID base
, RegisterID dst
)
1010 ASSERT(base
== SH4Registers::pc
);
1011 ASSERT((offset
<= 255) && (offset
>= 0));
1013 uint16_t opc
= getOpcodeGroup3(MOVW_READ_OFFPC_OPCODE
, dst
, offset
);
1017 void movwMemReg(int offset
, RegisterID base
, RegisterID dst
)
1019 ASSERT(dst
== SH4Registers::r0
);
1021 uint16_t opc
= getOpcodeGroup11(MOVW_READ_OFFRM_OPCODE
, base
, offset
);
1025 void movwR0mr(RegisterID src
, RegisterID dst
)
1027 uint16_t opc
= getOpcodeGroup1(MOVW_READ_R0RM_OPCODE
, dst
, src
);
1031 void movlRegMem(RegisterID src
, int offset
, RegisterID base
)
1033 ASSERT((offset
<= 15) && (offset
>= 0));
1036 oneShortOp(getOpcodeGroup1(MOVL_WRITE_RN_OPCODE
, base
, src
));
1040 oneShortOp(getOpcodeGroup4(MOVL_WRITE_OFFRN_OPCODE
, base
, src
, offset
));
1043 void movlRegMem(RegisterID src
, RegisterID base
)
1045 uint16_t opc
= getOpcodeGroup1(MOVL_WRITE_RN_OPCODE
, base
, src
);
1049 void movlMemReg(int offset
, RegisterID base
, RegisterID dst
)
1051 if (base
== SH4Registers::pc
) {
1052 ASSERT((offset
<= 255) && (offset
>= 0));
1053 oneShortOp(getOpcodeGroup3(MOVL_READ_OFFPC_OPCODE
, dst
, offset
));
1057 ASSERT((offset
<= 15) && (offset
>= 0));
1059 oneShortOp(getOpcodeGroup1(MOVL_READ_RM_OPCODE
, dst
, base
));
1063 oneShortOp(getOpcodeGroup4(MOVL_READ_OFFRM_OPCODE
, dst
, base
, offset
));
1066 void movbMemReg(int offset
, RegisterID base
, RegisterID dst
)
1068 ASSERT(dst
== SH4Registers::r0
);
1070 uint16_t opc
= getOpcodeGroup11(MOVB_READ_OFFRM_OPCODE
, base
, offset
);
1074 void movbR0mr(RegisterID src
, RegisterID dst
)
1076 uint16_t opc
= getOpcodeGroup1(MOVB_READ_R0RM_OPCODE
, dst
, src
);
1080 void movbMemReg(RegisterID src
, RegisterID dst
)
1082 uint16_t opc
= getOpcodeGroup1(MOVB_READ_RM_OPCODE
, dst
, src
);
1086 void movlMemReg(RegisterID base
, RegisterID dst
)
1088 uint16_t opc
= getOpcodeGroup1(MOVL_READ_RM_OPCODE
, dst
, base
);
1092 void movlMemRegIn(RegisterID base
, RegisterID dst
)
1094 uint16_t opc
= getOpcodeGroup1(MOVL_READ_RMINC_OPCODE
, dst
, base
);
1098 void movlR0mr(RegisterID src
, RegisterID dst
)
1100 uint16_t opc
= getOpcodeGroup1(MOVL_READ_R0RM_OPCODE
, dst
, src
);
1104 void movlRegMemr0(RegisterID src
, RegisterID dst
)
1106 uint16_t opc
= getOpcodeGroup1(MOVL_WRITE_R0RN_OPCODE
, dst
, src
);
1110 void movlImm8r(int imm8
, RegisterID dst
)
1112 ASSERT((imm8
<= 127) && (imm8
>= -128));
1114 uint16_t opc
= getOpcodeGroup3(MOVIMM_OPCODE
, dst
, imm8
);
1118 void loadConstant(uint32_t constant
, RegisterID dst
)
1120 if (((int)constant
<= 0x7f) && ((int)constant
>= -0x80)) {
1121 movImm8(constant
, dst
);
1125 uint16_t opc
= getOpcodeGroup3(MOVIMM_OPCODE
, dst
, 0);
1127 m_buffer
.ensureSpace(maxInstructionSize
, sizeof(uint32_t));
1128 printInstr(getOpcodeGroup3(MOVIMM_OPCODE
, dst
, constant
), m_buffer
.codeSize());
1129 m_buffer
.putShortWithConstantInt(opc
, constant
, true);
1132 void loadConstantUnReusable(uint32_t constant
, RegisterID dst
, bool ensureSpace
= false)
1134 uint16_t opc
= getOpcodeGroup3(MOVIMM_OPCODE
, dst
, 0);
1137 m_buffer
.ensureSpace(maxInstructionSize
, sizeof(uint32_t));
1139 printInstr(getOpcodeGroup3(MOVIMM_OPCODE
, dst
, constant
), m_buffer
.codeSize());
1140 m_buffer
.putShortWithConstantInt(opc
, constant
);
1145 AssemblerLabel
call()
1147 RegisterID scr
= claimScratch();
1148 m_buffer
.ensureSpace(maxInstructionSize
+ 4, sizeof(uint32_t));
1149 loadConstantUnReusable(0x0, scr
);
1150 branch(JSR_OPCODE
, scr
);
1152 releaseScratch(scr
);
1153 return m_buffer
.label();
1156 AssemblerLabel
call(RegisterID dst
)
1158 m_buffer
.ensureSpace(maxInstructionSize
+ 2);
1159 branch(JSR_OPCODE
, dst
);
1161 return m_buffer
.label();
1164 AssemblerLabel
jmp()
1166 RegisterID scr
= claimScratch();
1167 m_buffer
.ensureSpace(maxInstructionSize
+ 4, sizeof(uint32_t));
1168 AssemblerLabel label
= m_buffer
.label();
1169 loadConstantUnReusable(0x0, scr
);
1170 branch(BRAF_OPCODE
, scr
);
1172 releaseScratch(scr
);
1176 AssemblerLabel
jmp(RegisterID dst
)
1179 return m_buffer
.label();
1182 void jmpReg(RegisterID dst
)
1184 m_buffer
.ensureSpace(maxInstructionSize
+ 2);
1185 branch(JMP_OPCODE
, dst
);
1189 AssemblerLabel
jne()
1191 AssemblerLabel label
= m_buffer
.label();
1192 branch(BF_OPCODE
, 0);
1198 AssemblerLabel label
= m_buffer
.label();
1199 branch(BT_OPCODE
, 0);
1205 m_buffer
.ensureSpace(maxInstructionSize
+ 2);
1206 oneShortOp(RTS_OPCODE
, false);
1209 AssemblerLabel
label()
1211 m_buffer
.ensureSpaceForAnyOneInstruction();
1212 return m_buffer
.label();
1215 int sizeOfConstantPool()
1217 return m_buffer
.sizeOfConstantPool();
1220 AssemblerLabel
align(int alignment
)
1222 m_buffer
.ensureSpace(maxInstructionSize
+ 2);
1223 while (!m_buffer
.isAligned(alignment
)) {
1225 m_buffer
.ensureSpace(maxInstructionSize
+ 2);
1230 static void changePCrelativeAddress(int offset
, uint16_t* instructionPtr
, uint32_t newAddress
)
1232 uint32_t address
= (offset
<< 2) + ((reinterpret_cast<uint32_t>(instructionPtr
) + 4) &(~0x3));
1233 *reinterpret_cast<uint32_t*>(address
) = newAddress
;
1236 static uint32_t readPCrelativeAddress(int offset
, uint16_t* instructionPtr
)
1238 uint32_t address
= (offset
<< 2) + ((reinterpret_cast<uint32_t>(instructionPtr
) + 4) &(~0x3));
1239 return *reinterpret_cast<uint32_t*>(address
);
1242 static uint16_t* getInstructionPtr(void* code
, int offset
)
1244 return reinterpret_cast<uint16_t*> (reinterpret_cast<uint32_t>(code
) + offset
);
1247 static void linkJump(void* code
, AssemblerLabel from
, void* to
)
1249 ASSERT(from
.isSet());
1251 uint16_t* instructionPtr
= getInstructionPtr(code
, from
.m_offset
);
1252 uint16_t instruction
= *instructionPtr
;
1253 int offsetBits
= (reinterpret_cast<uint32_t>(to
) - reinterpret_cast<uint32_t>(code
)) - from
.m_offset
;
1255 if (((instruction
& 0xff00) == BT_OPCODE
) || ((instruction
& 0xff00) == BF_OPCODE
)) {
1256 /* BT label ==> BF 2
1262 instruction
^= 0x0202;
1263 *instructionPtr
++ = instruction
;
1264 changePCrelativeAddress((*instructionPtr
& 0xff), instructionPtr
, offsetBits
);
1265 instruction
= (BRAF_OPCODE
| (*instructionPtr
++ & 0xf00));
1266 *instructionPtr
= instruction
;
1267 printBlockInstr(instructionPtr
- 2, from
.m_offset
, 3);
1271 /* MOV #imm, reg => LDR reg
1275 ASSERT((*(instructionPtr
+ 1) & BRAF_OPCODE
) == BRAF_OPCODE
);
1278 if (offsetBits
>= -4096 && offsetBits
<= 4094) {
1279 *instructionPtr
= getOpcodeGroup6(BRA_OPCODE
, offsetBits
>> 1);
1280 *(++instructionPtr
) = NOP_OPCODE
;
1281 printBlockInstr(instructionPtr
- 1, from
.m_offset
, 2);
1285 changePCrelativeAddress((*instructionPtr
& 0xff), instructionPtr
, offsetBits
- 2);
1286 printInstr(*instructionPtr
, from
.m_offset
+ 2);
1289 static void linkCall(void* code
, AssemblerLabel from
, void* to
)
1291 uint16_t* instructionPtr
= getInstructionPtr(code
, from
.m_offset
);
1292 instructionPtr
-= 3;
1293 changePCrelativeAddress((*instructionPtr
& 0xff), instructionPtr
, reinterpret_cast<uint32_t>(to
));
1296 static void linkPointer(void* code
, AssemblerLabel where
, void* value
)
1298 uint16_t* instructionPtr
= getInstructionPtr(code
, where
.m_offset
);
1299 changePCrelativeAddress((*instructionPtr
& 0xff), instructionPtr
, reinterpret_cast<uint32_t>(value
));
1302 static unsigned getCallReturnOffset(AssemblerLabel call
)
1304 ASSERT(call
.isSet());
1305 return call
.m_offset
;
1308 static uint32_t* getLdrImmAddressOnPool(SH4Word
* insn
, uint32_t* constPool
)
1310 return (constPool
+ (*insn
& 0xff));
1313 static SH4Word
patchConstantPoolLoad(SH4Word load
, int value
)
1315 return ((load
& ~0xff) | value
);
1318 static SH4Buffer::TwoShorts
placeConstantPoolBarrier(int offset
)
1320 ASSERT(((offset
>> 1) <=2047) && ((offset
>> 1) >= -2048));
1322 SH4Buffer::TwoShorts m_barrier
;
1323 m_barrier
.high
= (BRA_OPCODE
| (offset
>> 1));
1324 m_barrier
.low
= NOP_OPCODE
;
1325 printInstr(((BRA_OPCODE
| (offset
>> 1))), 0);
1326 printInstr(NOP_OPCODE
, 0);
1330 static void patchConstantPoolLoad(void* loadAddr
, void* constPoolAddr
)
1332 SH4Word
* instructionPtr
= reinterpret_cast<SH4Word
*>(loadAddr
);
1333 SH4Word instruction
= *instructionPtr
;
1334 SH4Word index
= instruction
& 0xff;
1336 if ((instruction
& 0xf000) != MOVIMM_OPCODE
)
1339 ASSERT((((reinterpret_cast<uint32_t>(constPoolAddr
) - reinterpret_cast<uint32_t>(loadAddr
)) + index
* 4)) < 1024);
1341 int offset
= reinterpret_cast<uint32_t>(constPoolAddr
) + (index
* 4) - ((reinterpret_cast<uint32_t>(instructionPtr
) & ~0x03) + 4);
1342 instruction
&=0xf00;
1343 instruction
|= 0xd000;
1345 instruction
|= (offset
>> 2);
1346 *instructionPtr
= instruction
;
1347 printInstr(instruction
, reinterpret_cast<uint32_t>(loadAddr
));
1350 static void repatchPointer(void* where
, void* value
)
1352 patchPointer(where
, value
);
1355 static void* readPointer(void* code
)
1357 return static_cast<void*>(readInt32(code
));
1360 static void repatchInt32(void* where
, int32_t value
)
1362 uint16_t* instructionPtr
= reinterpret_cast<uint16_t*>(where
);
1363 changePCrelativeAddress((*instructionPtr
& 0xff), instructionPtr
, value
);
1366 static void repatchCompact(void* where
, int32_t value
)
1368 repatchInt32(where
, value
);
1371 static void relinkCall(void* from
, void* to
)
1373 uint16_t* instructionPtr
= reinterpret_cast<uint16_t*>(from
);
1374 instructionPtr
-= 3;
1375 changePCrelativeAddress((*instructionPtr
& 0xff), instructionPtr
, reinterpret_cast<uint32_t>(to
));
1378 static void relinkJump(void* from
, void* to
)
1380 uint16_t* instructionPtr
= reinterpret_cast<uint16_t*> (from
);
1381 uint16_t instruction
= *instructionPtr
;
1382 int32_t offsetBits
= (reinterpret_cast<uint32_t>(to
) - reinterpret_cast<uint32_t>(from
));
1384 if (((*instructionPtr
& 0xff00) == BT_OPCODE
) || ((*instructionPtr
& 0xff00) == BF_OPCODE
)) {
1387 changePCrelativeAddress((*instructionPtr
& 0xff), instructionPtr
, offsetBits
);
1388 instruction
= (BRAF_OPCODE
| (*instructionPtr
++ & 0xf00));
1389 *instructionPtr
= instruction
;
1390 printBlockInstr(instructionPtr
, reinterpret_cast<uint32_t>(from
) + 1, 3);
1394 ASSERT((*(instructionPtr
+ 1) & BRAF_OPCODE
) == BRAF_OPCODE
);
1396 if (offsetBits
>= -4096 && offsetBits
<= 4094) {
1397 *instructionPtr
= getOpcodeGroup6(BRA_OPCODE
, offsetBits
>> 1);
1398 *(++instructionPtr
) = NOP_OPCODE
;
1399 printBlockInstr(instructionPtr
- 2, reinterpret_cast<uint32_t>(from
), 2);
1403 changePCrelativeAddress((*instructionPtr
& 0xff), instructionPtr
, offsetBits
- 2);
1404 printInstr(*instructionPtr
, reinterpret_cast<uint32_t>(from
));
1407 // Linking & patching
1409 void linkJump(AssemblerLabel from
, AssemblerLabel to
)
1412 ASSERT(from
.isSet());
1414 uint16_t* instructionPtr
= getInstructionPtr(data(), from
.m_offset
);
1415 uint16_t instruction
= *instructionPtr
;
1418 if (((instruction
& 0xff00) == BT_OPCODE
) || ((instruction
& 0xff00) == BF_OPCODE
)) {
1424 offsetBits
= (to
.m_offset
- from
.m_offset
) - 8;
1425 instruction
^= 0x0202;
1426 *instructionPtr
++ = instruction
;
1427 if ((*instructionPtr
& 0xf000) == 0xe000) {
1428 uint32_t* addr
= getLdrImmAddressOnPool(instructionPtr
, m_buffer
.poolAddress());
1431 changePCrelativeAddress((*instructionPtr
& 0xff), instructionPtr
, offsetBits
);
1432 instruction
= (BRAF_OPCODE
| (*instructionPtr
++ & 0xf00));
1433 *instructionPtr
= instruction
;
1434 printBlockInstr(instructionPtr
- 2, from
.m_offset
, 3);
1438 /* MOV # imm, reg => LDR reg
1442 ASSERT((*(instructionPtr
+ 1) & BRAF_OPCODE
) == BRAF_OPCODE
);
1443 offsetBits
= (to
.m_offset
- from
.m_offset
) - 4;
1444 if (offsetBits
>= -4096 && offsetBits
<= 4094) {
1445 *instructionPtr
= getOpcodeGroup6(BRA_OPCODE
, offsetBits
>> 1);
1446 *(++instructionPtr
) = NOP_OPCODE
;
1447 printBlockInstr(instructionPtr
- 1, from
.m_offset
, 2);
1451 instruction
= *instructionPtr
;
1452 if ((instruction
& 0xf000) == 0xe000) {
1453 uint32_t* addr
= getLdrImmAddressOnPool(instructionPtr
, m_buffer
.poolAddress());
1454 *addr
= offsetBits
- 2;
1455 printInstr(*instructionPtr
, from
.m_offset
+ 2);
1459 changePCrelativeAddress((*instructionPtr
& 0xff), instructionPtr
, offsetBits
- 2);
1460 printInstr(*instructionPtr
, from
.m_offset
+ 2);
1463 static void* getRelocatedAddress(void* code
, AssemblerLabel label
)
1465 return reinterpret_cast<void*>(reinterpret_cast<char*>(code
) + label
.m_offset
);
1468 static int getDifferenceBetweenLabels(AssemblerLabel a
, AssemblerLabel b
)
1470 return b
.m_offset
- a
.m_offset
;
1473 static void patchPointer(void* code
, AssemblerLabel where
, void* value
)
1475 patchPointer(reinterpret_cast<uint32_t*>(code
) + where
.m_offset
, value
);
1478 static void patchPointer(void* code
, void* value
)
1480 patchInt32(code
, reinterpret_cast<uint32_t>(value
));
1483 static void patchInt32(void* code
, uint32_t value
)
1485 changePCrelativeAddress((*(reinterpret_cast<uint16_t*>(code
)) & 0xff), reinterpret_cast<uint16_t*>(code
), value
);
1488 static uint32_t readInt32(void* code
)
1490 return readPCrelativeAddress((*(reinterpret_cast<uint16_t*>(code
)) & 0xff), reinterpret_cast<uint16_t*>(code
));
1493 void* executableCopy(JSGlobalData
& globalData
, ExecutablePool
* allocator
)
1495 return m_buffer
.executableCopy(globalData
, allocator
);
1498 void prefix(uint16_t pre
)
1500 m_buffer
.putByte(pre
);
1503 void oneShortOp(uint16_t opcode
, bool checksize
= true, bool isDouble
= true)
1505 printInstr(opcode
, m_buffer
.codeSize(), isDouble
);
1507 m_buffer
.ensureSpace(maxInstructionSize
);
1508 m_buffer
.putShortUnchecked(opcode
);
1511 void ensureSpace(int space
)
1513 m_buffer
.ensureSpace(space
);
1516 void ensureSpace(int insnSpace
, int constSpace
)
1518 m_buffer
.ensureSpace(insnSpace
, constSpace
);
1521 // Administrative methods
1523 void* data() const { return m_buffer
.data(); }
1524 size_t codeSize() const { return m_buffer
.codeSize(); }
1526 #ifdef SH4_ASSEMBLER_TRACING
1527 static void printInstr(uint16_t opc
, unsigned int size
, bool isdoubleInst
= true)
1529 if (!getenv("JavaScriptCoreDumpJIT"))
1532 const char *format
= 0;
1533 printfStdoutInstr("offset: 0x%8.8x\t", size
);
1554 format
= " FSCHG\n";
1558 printfStdoutInstr(format
);
1561 switch (opc
& 0xf0ff) {
1563 format
= " *BRAF R%d\n";
1566 format
= " DT R%d\n";
1569 format
= " CMP/PL R%d\n";
1572 format
= " CMP/PZ R%d\n";
1575 format
= " *JMP @R%d\n";
1578 format
= " *JSR @R%d\n";
1581 format
= " LDS R%d, PR\n";
1584 format
= " LDS.L @R%d+, PR\n";
1587 format
= " MOVT R%d\n";
1590 format
= " SHAL R%d\n";
1593 format
= " SHAR R%d\n";
1596 format
= " SHLL R%d\n";
1599 format
= " SHLL2 R%d\n";
1602 format
= " SHLL8 R%d\n";
1605 format
= " SHLL16 R%d\n";
1608 format
= " SHLR R%d\n";
1611 format
= " SHLR2 R%d\n";
1614 format
= " SHLR8 R%d\n";
1617 format
= " SHLR16 R%d\n";
1620 format
= " STS PR, R%d\n";
1623 format
= " STS.L PR, @-R%d\n";
1625 case LDS_RM_FPUL_OPCODE
:
1626 format
= " LDS R%d, FPUL\n";
1628 case STS_FPUL_RN_OPCODE
:
1629 format
= " STS FPUL, R%d \n";
1631 case FLDS_FRM_FPUL_OPCODE
:
1632 format
= " FLDS FR%d, FPUL\n";
1634 case FSTS_FPUL_FRN_OPCODE
:
1635 format
= " FSTS FPUL, R%d \n";
1637 case LDSFPSCR_OPCODE
:
1638 format
= " LDS R%d, FPSCR \n";
1640 case STSFPSCR_OPCODE
:
1641 format
= " STS FPSCR, R%d \n";
1643 case STSMACL_OPCODE
:
1644 format
= " STS MACL, R%d \n";
1646 case STSMACH_OPCODE
:
1647 format
= " STS MACH, R%d \n";
1650 format
= " *BSRF R%d";
1653 format
= " FTRC FR%d, FPUL\n";
1657 printfStdoutInstr(format
, getRn(opc
));
1660 switch (opc
& 0xf0ff) {
1662 format
= " FNEG DR%d\n";
1665 format
= " FLOAT DR%d\n";
1668 format
= " FTRC FR%d, FPUL\n";
1671 format
= " FSQRT FR%d\n";
1673 case FCNVDS_DRM_FPUL_OPCODE
:
1674 format
= " FCNVDS FR%d, FPUL\n";
1679 printfStdoutInstr(format
, getDRn(opc
) << 1);
1681 printfStdoutInstr(format
, getRn(opc
));
1684 switch (opc
& 0xf00f) {
1686 format
= " ADD R%d, R%d\n";
1689 format
= " ADDC R%d, R%d\n";
1692 format
= " ADDV R%d, R%d\n";
1695 format
= " AND R%d, R%d\n";
1698 format
= " DIV1 R%d, R%d\n";
1701 format
= " CMP/EQ R%d, R%d\n";
1704 format
= " CMP/GE R%d, R%d\n";
1707 format
= " CMP/GT R%d, R%d\n";
1710 format
= " CMP/HI R%d, R%d\n";
1713 format
= " CMP/HS R%d, R%d\n";
1716 format
= " MOV R%d, R%d\n";
1718 case MOVB_WRITE_RN_OPCODE
:
1719 format
= " MOV.B R%d, @R%d\n";
1721 case MOVB_WRITE_RNDEC_OPCODE
:
1722 format
= " MOV.B R%d, @-R%d\n";
1724 case MOVB_WRITE_R0RN_OPCODE
:
1725 format
= " MOV.B R%d, @(R0, R%d)\n";
1727 case MOVB_READ_RM_OPCODE
:
1728 format
= " MOV.B @R%d, R%d\n";
1730 case MOVB_READ_RMINC_OPCODE
:
1731 format
= " MOV.B @R%d+, R%d\n";
1733 case MOVB_READ_R0RM_OPCODE
:
1734 format
= " MOV.B @(R0, R%d), R%d\n";
1736 case MOVL_WRITE_RN_OPCODE
:
1737 format
= " MOV.L R%d, @R%d\n";
1739 case MOVL_WRITE_RNDEC_OPCODE
:
1740 format
= " MOV.L R%d, @-R%d\n";
1742 case MOVL_WRITE_R0RN_OPCODE
:
1743 format
= " MOV.L R%d, @(R0, R%d)\n";
1745 case MOVL_READ_RM_OPCODE
:
1746 format
= " MOV.L @R%d, R%d\n";
1748 case MOVL_READ_RMINC_OPCODE
:
1749 format
= " MOV.L @R%d+, R%d\n";
1751 case MOVL_READ_R0RM_OPCODE
:
1752 format
= " MOV.L @(R0, R%d), R%d\n";
1755 format
= " MUL.L R%d, R%d\n";
1757 case DMULL_L_OPCODE
:
1758 format
= " DMULU.L R%d, R%d\n";
1761 format
= " DMULS.L R%d, R%d\n";
1764 format
= " NEG R%d, R%d\n";
1767 format
= " NEGC R%d, R%d\n";
1770 format
= " NOT R%d, R%d\n";
1773 format
= " OR R%d, R%d\n";
1776 format
= " SHAD R%d, R%d\n";
1779 format
= " SHLD R%d, R%d\n";
1782 format
= " SUB R%d, R%d\n";
1785 format
= " SUBC R%d, R%d\n";
1788 format
= " SUBV R%d, R%d\n";
1791 format
= " TST R%d, R%d\n";
1794 format
= " XOR R%d, R%d\n";break;
1795 case MOVW_WRITE_RN_OPCODE
:
1796 format
= " MOV.W R%d, @R%d\n";
1798 case MOVW_READ_RM_OPCODE
:
1799 format
= " MOV.W @R%d, R%d\n";
1801 case MOVW_READ_R0RM_OPCODE
:
1802 format
= " MOV.W @(R0, R%d), R%d\n";
1805 format
= " EXTU.W R%d, R%d\n";
1809 printfStdoutInstr(format
, getRm(opc
), getRn(opc
));
1812 switch (opc
& 0xf00f) {
1814 format
= " FSUB FR%d, FR%d\n";
1817 format
= " FADD FR%d, FR%d\n";
1820 format
= " FDIV FR%d, FR%d\n";
1823 format
= " DMULL FR%d, FR%d\n";
1826 format
= " FMOV FR%d, FR%d\n";
1829 format
= " FCMP/EQ FR%d, FR%d\n";
1832 format
= " FCMP/GT FR%d, FR%d\n";
1837 printfStdoutInstr(format
, getDRm(opc
) << 1, getDRn(opc
) << 1);
1839 printfStdoutInstr(format
, getRm(opc
), getRn(opc
));
1842 switch (opc
& 0xf00f) {
1843 case FMOVS_WRITE_RN_DEC_OPCODE
:
1844 format
= " %s FR%d, @-R%d\n";
1846 case FMOVS_WRITE_RN_OPCODE
:
1847 format
= " %s FR%d, @R%d\n";
1849 case FMOVS_WRITE_R0RN_OPCODE
:
1850 format
= " %s FR%d, @(R0, R%d)\n";
1855 printfStdoutInstr(format
, "FMOV", getDRm(opc
) << 1, getDRn(opc
));
1857 printfStdoutInstr(format
, "FMOV.S", getRm(opc
), getRn(opc
));
1860 switch (opc
& 0xf00f) {
1861 case FMOVS_READ_RM_OPCODE
:
1862 format
= " %s @R%d, FR%d\n";
1864 case FMOVS_READ_RM_INC_OPCODE
:
1865 format
= " %s @R%d+, FR%d\n";
1867 case FMOVS_READ_R0RM_OPCODE
:
1868 format
= " %s @(R0, R%d), FR%d\n";
1873 printfStdoutInstr(format
, "FMOV", getDRm(opc
), getDRn(opc
) << 1);
1875 printfStdoutInstr(format
, "FMOV.S", getRm(opc
), getRn(opc
));
1878 switch (opc
& 0xff00) {
1880 format
= " BF %d\n";
1883 format
= " *BF/S %d\n";
1886 format
= " AND #%d, R0\n";
1889 format
= " BT %d\n";
1892 format
= " *BT/S %d\n";
1894 case CMPEQIMM_OPCODE
:
1895 format
= " CMP/EQ #%d, R0\n";
1897 case MOVB_WRITE_OFFGBR_OPCODE
:
1898 format
= " MOV.B R0, @(%d, GBR)\n";
1900 case MOVB_READ_OFFGBR_OPCODE
:
1901 format
= " MOV.B @(%d, GBR), R0\n";
1903 case MOVL_WRITE_OFFGBR_OPCODE
:
1904 format
= " MOV.L R0, @(%d, GBR)\n";
1906 case MOVL_READ_OFFGBR_OPCODE
:
1907 format
= " MOV.L @(%d, GBR), R0\n";
1909 case MOVA_READ_OFFPC_OPCODE
:
1910 format
= " MOVA @(%d, PC), R0\n";
1913 format
= " OR #%d, R0\n";
1916 format
= " OR.B #%d, @(R0, GBR)\n";
1919 format
= " TST #%d, R0\n";
1922 format
= " TST.B %d, @(R0, GBR)\n";
1925 format
= " XOR #%d, R0\n";
1928 format
= " XOR.B %d, @(R0, GBR)\n";
1932 printfStdoutInstr(format
, getImm8(opc
));
1935 switch (opc
& 0xff00) {
1936 case MOVB_WRITE_OFFRN_OPCODE
:
1937 format
= " MOV.B R0, @(%d, R%d)\n";
1939 case MOVB_READ_OFFRM_OPCODE
:
1940 format
= " MOV.B @(%d, R%d), R0\n";
1944 printfStdoutInstr(format
, getDisp(opc
), getRm(opc
));
1947 switch (opc
& 0xf000) {
1949 format
= " *BRA %d\n";
1952 format
= " *BSR %d\n";
1956 printfStdoutInstr(format
, getImm12(opc
));
1959 switch (opc
& 0xf000) {
1960 case MOVL_READ_OFFPC_OPCODE
:
1961 format
= " MOV.L @(%d, PC), R%d\n";
1964 format
= " ADD #%d, R%d\n";
1967 format
= " MOV #%d, R%d\n";
1969 case MOVW_READ_OFFPC_OPCODE
:
1970 format
= " MOV.W @(%d, PC), R%d\n";
1974 printfStdoutInstr(format
, getImm8(opc
), getRn(opc
));
1977 switch (opc
& 0xf000) {
1978 case MOVL_WRITE_OFFRN_OPCODE
:
1979 format
= " MOV.L R%d, @(%d, R%d)\n";
1980 printfStdoutInstr(format
, getRm(opc
), getDisp(opc
), getRn(opc
));
1982 case MOVL_READ_OFFRM_OPCODE
:
1983 format
= " MOV.L @(%d, R%d), R%d\n";
1984 printfStdoutInstr(format
, getDisp(opc
), getRm(opc
), getRn(opc
));
1989 static void printfStdoutInstr(const char* format
, ...)
1991 if (getenv("JavaScriptCoreDumpJIT")) {
1993 va_start(args
, format
);
1994 vprintfStdoutInstr(format
, args
);
1999 static void vprintfStdoutInstr(const char* format
, va_list args
)
2001 if (getenv("JavaScriptCoreDumpJIT"))
2002 vfprintf(stdout
, format
, args
);
2005 static void printBlockInstr(uint16_t* first
, unsigned int offset
, int nbInstr
)
2007 printfStdoutInstr(">> repatch instructions after link\n");
2008 for (int i
= 0; i
<= nbInstr
; i
++)
2009 printInstr(*(first
+ i
), offset
+ i
);
2010 printfStdoutInstr(">> end repatch\n");
2013 static void printInstr(uint16_t opc
, unsigned int size
, bool isdoubleInst
= true) {};
2014 static void printBlockInstr(uint16_t* first
, unsigned int offset
, int nbInstr
) {};
2019 int m_claimscratchReg
;
2024 #endif // ENABLE(ASSEMBLER) && CPU(SH4)
2026 #endif // SH4Assembler_h