source: dev/trunk/abdev/BasicCompiler_Common/include/CodeGenerator.h@ 237

Last change on this file since 237 was 237, checked in by dai_9181, 17 years ago
File size: 12.3 KB
Line 
1#pragma once
2
3#include <NativeCode.h>
4
5#ifdef _AMD64_
6#include "../../BasicCompiler64/MachineFixed.h"
7#else
8#include "../../BasicCompiler32/MachineFixed.h"
9#endif
10
11
12void ReallocNativeCodeBuffer();
13
14class CodeGenerator
15{
16 NativeCode *pNativeCode;
17
18public:
19 class PertialSchedule
20 {
21 int codePos;
22 int typeSize;
23
24 int _obpOld;
25 public:
26 PertialSchedule( int codePos, int typeSize )
27 : codePos( codePos )
28 , typeSize( typeSize )
29 {
30 extern int obp;
31 _obpOld = obp-typeSize;
32 }
33 ~PertialSchedule()
34 {
35 }
36
37 int GetCodePos() const
38 {
39 return codePos;
40 }
41 int GetTypeSize() const
42 {
43 return typeSize;
44 }
45 int GetObpOld() const
46 {
47 return _obpOld;
48 }
49 };
50
51private:
52 typedef std::vector<PertialSchedule> PertialSchedules;
53 PertialSchedules pertialSchedules;
54
55public:
56
57 CodeGenerator()
58 : pNativeCode( 0 )
59 {
60 }
61 ~CodeGenerator()
62 {
63 if( pNativeCode )
64 {
65 CheckUnresolveSchedule();
66 }
67 }
68
69 void Select( NativeCode &nativeCode )
70 {
71 if( pNativeCode )
72 {
73 CheckUnresolveSchedule();
74 }
75 pNativeCode = &nativeCode;
76 }
77
78 void CheckUnresolveSchedule();
79
80 void opfix_JmpPertialSchedule( const PertialSchedule *pPertialSchedule )
81 {
82 PertialSchedules::iterator it = pertialSchedules.begin();
83 while( it != pertialSchedules.end() )
84 {
85 if( &(*it) == pPertialSchedule )
86 {
87 pNativeCode->Overwrite( pPertialSchedule->GetCodePos(), (char)(pNativeCode->GetSize() - pPertialSchedule->GetCodePos()) );
88
89 // 未完成
90 extern int obp;
91 pNativeCode->OverwriteOld( pPertialSchedule->GetObpOld(), obp-pPertialSchedule->GetObpOld() );
92
93 pertialSchedules.erase( it );
94 }
95 else
96 {
97 it++;
98 }
99 }
100 }
101
102
103 /////////////////////////////////////////////////////////////////
104 // 32bit/64bit共通 機械語生成
105 /////////////////////////////////////////////////////////////////
106
107private:
108 PertialSchedule *__jmp_op_format( char opcode, long offset, int op_size, bool isPertialSchedule = false );
109public:
110 void op_jle( long offset, int op_size = sizeof(char) );
111 void op_jbe( long offset, int op_size = sizeof(char) );
112 void op_jge( long offset, int op_size = sizeof(char) );
113 void op_jae( long offset, int op_size = sizeof(char) );
114 void op_jl( long offset, int op_size = sizeof(char) );
115 void op_jb( long offset, int op_size = sizeof(char) );
116 void op_jg( long offset, int op_size = sizeof(char) );
117 PertialSchedule *op_ja( long offset, int op_size = sizeof(char), bool isPertialSchedule = false );
118 PertialSchedule *op_jne( long offset, int op_size = sizeof(char), bool isPertialSchedule = false );
119 void op_je( long offset, int op_size = sizeof(char) );
120 void op_jmp( long offset, int op_size = sizeof(char) );
121
122
123#ifdef _AMD64_
124 /////////////////////////////////////////////////////////////////
125 // 64ビット機械語生成
126 /////////////////////////////////////////////////////////////////
127private:
128 void set_rex(int op_size,int reg,int index_reg,int base_reg);
129 void set_mod_rm_sib_disp(char mod,int reg,int scale,int index_reg,int base_reg,long disp, Schedule::Type scheduleType = Schedule::None );
130 void __op_format(int op_size,char op_prefix,char opcode1,char opcode2,int reg,int base_reg,long offset,char mod, Schedule::Type scheduleType = Schedule::None );
131public:
132 void op_mov_RV (int op_size,int reg,long i32data, Schedule::Type scheduleType = Schedule::None );
133 void op_mov_RV64 (int reg,_int64 i64data);
134 void op_mov_RM (int op_size,int reg,int base_reg,long offset,char mod, Schedule::Type scheduleType = Schedule::None );
135 void op_mov_RM_ex (int op_size,int reg,int base_reg1,int base_reg2,long offset,BOOL bUseOffset, Schedule::Type scheduleType = Schedule::None );
136 void op_mov_MR (int op_size,int reg,int base_reg,long offset,char mod, Schedule::Type scheduleType = Schedule::None );
137 void op_mov_MR_ex (int op_size,int reg,int base_reg1,int base_reg2,long offset,BOOL bUseOffset, Schedule::Type scheduleType = Schedule::None );
138 void op_mov_MV (int op_size,int base_reg,int offset,BOOL bUseOffset,long i32data);
139 void op_mov_RR (int reg1,int reg2);
140 void op_movsxd (int reg64,int reg32);
141 void op_movsx64_FromReg16 (int reg64,int reg16);
142 void op_movsx64_FromReg8 (int reg64,int reg8);
143 void op_movsx32_FromReg16 (int reg32,int reg16);
144 void op_movsx32_FromReg8 (int reg32,int reg8);
145 void op_movsx16_FromReg8 (int reg32,int reg8);
146 void op_cqo ();
147 void op_inc (int reg);
148 void op_dec (int reg);
149 void op_add_RM (int op_size,int reg,int base_reg,int offset,char mod, Schedule::Type scheduleType = Schedule::None );
150 void op_add_RV (int reg,long offset, Schedule::Type scheduleType = Schedule::None );
151 void op_add_RR (int reg1,int reg2);
152 void op_add32_reg (int reg1,int reg2);
153 void op_sub_RV (int op_size,int reg,long i32data);
154 void op_sub64_reg (int reg1,int reg2);
155 void op_sub32_reg (int reg1,int reg2);
156 void op_sbb_RR ( int op_size, int reg1, int reg2 );
157 void op_imul_RR (int op_size,int reg1,int reg2);
158 void op_imul_RV (int op_size,int reg,long i32data);
159 void op_div64_reg (int reg);
160 void op_idiv64_reg (int reg);
161 void op_shl_reg (int op_size,int reg);
162 void op_sar_reg (int op_size,int reg);
163 void op_shr_reg (int op_size,int reg);
164 void op_and_reg (int op_size,int reg1,int reg2);
165 void op_and64_value (int reg,long offset);
166 void op_and32_value (int reg,long offset);
167 void op_or_reg (int op_size,int reg1,int reg2);
168 void op_xor_reg (int op_size,int reg1,int reg2);
169 void op_not_reg (int op_size,int reg);
170 void op_neg ( int reg );
171 void op_test (int reg1,int reg2);
172 void op_cmp_reg (int op_size,int reg1,int reg2);
173 void op_cmp_value (int op_size,int reg,char byte_data);
174 void op_setne (int reg);
175 void op_movlpd_MR (int xmm_reg,int base_reg,int offset,char mod, Schedule::Type scheduleType = Schedule::None );
176 void op_movlpd_RM (int xmm_reg,int base_reg,int offset,char mod, Schedule::Type scheduleType = Schedule::None );
177 void op_movsd_RR (int xmm_reg1,int xmm_reg2);
178 void op_movsd_MR (int xmm_reg,int base_reg,int offset,char mod, Schedule::Type scheduleType = Schedule::None );
179 void op_movss_RR (int xmm_reg1,int xmm_reg2);
180 void op_movss_RM (int xmm_reg,int base_reg,int offset,char mod, Schedule::Type scheduleType = Schedule::None );
181 void op_movss_MR (int xmm_reg,int base_reg,int offset,char mod, Schedule::Type scheduleType = Schedule::None );
182 void op_movd_RX (int reg,int xmm_reg);
183 void op_cvtsd2ss (int xmm_reg1,int xmm_reg2);
184 void op_cvtss2sd (int xmm_reg1,int xmm_reg2);
185 void op_cvttsd2si_xmm (int op_size,int reg,int xmm_reg);
186 void op_cvttss2si_xmm (int op_size,int reg,int xmm_reg);
187 void op_cvtsi2sd_reg (int op_size,int xmm_reg,int reg);
188 void op_cvtsi2ss_reg (int op_size,int xmm_reg,int reg);
189 void op_comisd (int xmm_reg1,int xmm_reg2);
190 void op_comiss (int xmm_reg1,int xmm_reg2);
191 void op_rep_movs (int op_size);
192 void op_add_rsp(long num);
193 void op_sub_rsp(long num);
194 void op_add_esp(long num);
195 void op_sub_esp(long num);
196 void op_fld_ptr_esp(int type);
197 void op_zero_reg(int reg);
198 void op_call( const UserProc *pUserProc );
199 void op_call( const DllProc *pDllProc );
200 void op_ret();
201
202#else
203 /////////////////////////////////////////////////////////////////
204 // 32ビット機械語生成
205 /////////////////////////////////////////////////////////////////
206private:
207 void set_mod_rm_sib_disp(char mod,int reg,int scale,int index_reg,int base_reg,long disp, Schedule::Type scheduleType = Schedule::None );
208 void __op_format(char op_prefix,char opcode,int reg);
209 void __op_format(char op_prefix,char opcode1,char opcode2,int reg,int base_reg,int offset,char mod, Schedule::Type scheduleType = Schedule::None );
210public:
211 void op_mov_MV ( int op_size, int base_reg, long offset, Schedule::Type offsetScheduleType, long value, Schedule::Type valueScheduleType = Schedule::None );
212 void op_mov_RV (int reg,long offset, Schedule::Type scheduleType = Schedule::None );
213 void op_mov_RR (int reg1,int reg2);
214 void op_mov_RM (int op_size,int reg,int base_reg,int offset,char mod, Schedule::Type scheduleType = Schedule::None );
215 void op_mov_RM_ex (int op_size,int reg,int base_reg1,int base_reg2,long offset,BOOL bUseOffset, Schedule::Type scheduleType = Schedule::None );
216 void op_mov_MR (int op_size,int reg,int base_reg,int offset,char mod, Schedule::Type scheduleType = Schedule::None );
217 void op_mov_MR_ex (int op_size,int reg,int base_reg1,int base_reg2,long offset,BOOL bUseOffset, Schedule::Type scheduleType = Schedule::None );
218 void op_movsx_R32R16 (int reg32,int reg16 = REG_NON);
219 void op_movsx_R32R8 (int reg32,int reg8 = REG_NON);
220 void op_movsx_R16R8 (int reg16,int reg8 = REG_NON);
221 void op_lea_RM ( int reg, int base_reg, long offset, char mod, Schedule::Type scheduleType = Schedule::None );
222 void op_inc (int reg);
223 void op_dec (int reg);
224 void op_add_RV8 (int reg,char cValue);
225 void op_add_RV ( int reg, long offset, Schedule::Type scheduleType = Schedule::None );
226 void op_add_RR ( int reg1, int reg2 );
227 void op_add_RM (int op_size,int reg,int base_reg,int offset,char mod, Schedule::Type scheduleType = Schedule::None );
228 void op_adc_RV8 (int reg,char cValue);
229 void op_adc_RR ( int reg1, int reg2 );
230 void op_sub_RV8 (int reg,char cValue);
231 void op_sub_RR ( int reg1, int reg2 );
232 void op_sbb_RV8 (int reg,char cValue);
233 void op_sbb_RR ( int reg1, int reg2 );
234 void op_imul_RR (int reg1,int reg2);
235 void op_imul_RV (int reg,long i32data);
236 void op_imul_RV8 (int reg,char cValue);
237 void op_div_R ( int reg );
238 void op_idiv_R ( int reg );
239 void op_and_RV (int reg,long value);
240 void op_and_RR ( int reg1, int reg2 );
241 void op_or_RR ( int op_size, int reg1, int reg2 );
242 void op_xor_RR ( int reg1, int reg2 = REG_NON );
243 void op_neg ( int reg );
244 void op_cdq ();
245
246 void op_rep_movs (int op_size);
247
248 void op_push(int reg);
249 void op_push_V( long data, Schedule::Type scheduleType = Schedule::None );
250 void op_push_M( int base_reg );
251 void op_push_M( int base_reg, long offset, Schedule::Type scheduleType = Schedule::None );
252 void op_pop(int reg = REG_NON);
253 void op_add_esp(long num);
254 void op_sub_esp(long num);
255 void op_cmp_RR( int reg1, int reg2 );
256 void op_cmp_value(int op_size,int reg,char byte_data);
257 void op_setne( int reg );
258 void op_test(int reg1,int reg2);
259 void op_test_ah( char cValue );
260 void op_fld_ptr_esp(int type);
261 void op_fld_basereg (int type,int base_reg);
262 void op_fld_base_offset (int type,int base_reg,long offset, Schedule::Type scheduleType = Schedule::None );
263 void op_fld_base_offset_ex (int type,int base_reg1,int base_reg2,long offset,BOOL bUseOffset, Schedule::Type scheduleType = Schedule::None);
264 void op_fstp_basereg (int type,int base_reg);
265 void op_fstp_base_offset (int type,int base_reg,long offset, Schedule::Type scheduleType = Schedule::None);
266 void op_fstp_base_offset_ex (int type,int base_reg1,int base_reg2,long offset,BOOL bUseOffset, Schedule::Type scheduleType = Schedule::None);
267 void op_fistp_ptr_esp ( int typeSize );
268 void op_fstp_push ( Type &type );
269 void op_fcompp();
270 void op_fnstsw_ax();
271 void op_zero_reg(int reg);
272 void fpu_cast();
273 void fpu_cast_end();
274
275 void op_call_R( int reg );
276 void op_call(const UserProc *pUserProc);
277 void op_ret();
278#endif
279
280
281
282
283 void PutOld( char c1, char c2 )
284 {
285 pNativeCode->Put( c1 );
286 pNativeCode->Put( c2 );
287 }
288 void PutOld( char c1, char c2, char c3 )
289 {
290 pNativeCode->Put( c1 );
291 pNativeCode->Put( c2 );
292 pNativeCode->Put( c3 );
293 }
294 void PutOld( char c1, char c2, char c3, char c4 )
295 {
296 pNativeCode->Put( c1 );
297 pNativeCode->Put( c2 );
298 pNativeCode->Put( c3 );
299 pNativeCode->Put( c4 );
300 }
301 void PutOld( char c1, char c2, char c3, long l )
302 {
303 pNativeCode->Put( c1 );
304 pNativeCode->Put( c2 );
305 pNativeCode->Put( c3 );
306 pNativeCode->Put( l );
307 }
308 void PutOld( char c1, char c2, char c3, char c4, char c5 )
309 {
310 pNativeCode->Put( c1 );
311 pNativeCode->Put( c2 );
312 pNativeCode->Put( c3 );
313 pNativeCode->Put( c4 );
314 pNativeCode->Put( c5 );
315 }
316 void PutOld( char c1, char c2, char c3, char c4, char c5, char c6 )
317 {
318 pNativeCode->Put( c1 );
319 pNativeCode->Put( c2 );
320 pNativeCode->Put( c3 );
321 pNativeCode->Put( c4 );
322 pNativeCode->Put( c5 );
323 pNativeCode->Put( c6 );
324 }
325};
Note: See TracBrowser for help on using the repository browser.