source: dev/trunk/abdev/BasicCompiler64/Compile_ProcOp.cpp@ 169

Last change on this file since 169 was 169, checked in by dai_9181, 17 years ago

NamespaceScopesCollection::Importsをリファクタリング

File size: 25.6 KB
RevLine 
[169]1#include <Program.h>
2
[3]3#include "../BasicCompiler_Common/common.h"
4#include "Opcode.h"
5
[86]6void SystemProc( const UserProc &userProc ){
7 if( userProc.GetName() == "_System_GetEip" ){
[3]8 //mov rax,qword ptr[rsp]
9 op_mov_RM(sizeof(_int64),REG_RAX,REG_RSP,0,MOD_BASE);
10
11 //ret
[142]12 op_ret();
[3]13 }
[86]14 else if( userProc.GetName() == "_System_InitDllGlobalVariables" ){
[3]15 ////////////////////////////////////////
16 // DLLのグローバル領域をコンパイル
17 ////////////////////////////////////////
18 extern BOOL bDll;
19 if(!bDll){
20 //ret
[142]21 op_ret();
[3]22
23 return;
24 }
25
[75]26 UserProc *pBackUserProc;
27 pBackUserProc = &UserProc::CompilingUserProc();
28 UserProc::CompileStartForGlobalArea();
[3]29
30 int BackCp;
31 BackCp=cp;
32 cp=-1;
33
34 //sub rsp,スタックフレームサイズ
35 int StackFrameSchedule;
36 op_sub_rsp(0xFFFFFFFF);
37 StackFrameSchedule=obp-sizeof(long);
38
39 extern BOOL bDebugCompile;
40 if(bDebugCompile){
41 //デバッグ用の変数を定義
42 DebugVariable();
43 }
44
45 //GC用の変数を定義
46 InitGCVariables();
47
[129]48 //_System_StartupProgramの呼び出し
49 extern UserProc *pSub_System_StartupProgram;
50 op_call(pSub_System_StartupProgram);
51
[42]52 //クラスに属する静的メンバを定義
53 CMember::InitStaticMember();
54
[3]55 GetGlobalDataForDll();
56
57 //add rsp,スタックフレームサイズ
[64]58 op_add_RV(REG_RSP,pobj_sf->GetFrameSize());
[3]59
60 //スタックフレームスケジュール(subコマンドに渡す値)
61 *((long *)(OpBuffer+StackFrameSchedule))=pobj_sf->GetFrameSize();
62
[75]63 UserProc::CompileStartForUserProc( pBackUserProc );
[3]64 cp=BackCp;
65
66 //ret
[142]67 op_ret();
[3]68 }
[86]69 else if( userProc.GetName() == "_System_InitStaticLocalVariables" ){
[3]70 //静的ローカルオブジェクトのコンストラクタ呼び出し
71
72 //sub rsp,スタックフレームサイズ
73 int StackFrameSchedule;
74 op_sub_rsp(0xFFFFFFFF);
75 StackFrameSchedule=obp-sizeof(long);
76
[75]77 foreach( Variable *pVar, globalVars ){
78 if(memicmp(pVar->GetName().c_str(),"Static%",7)==0){
[3]79 //コンストラクタ呼び出し
[75]80 if( pVar->IsObject() ){
[3]81
82 //エラー用
[75]83 cp=pVar->source_code_address;
[3]84
[50]85 CallConstructor(
[75]86 pVar->GetName().c_str(),
87 pVar->GetSubScriptsPtr(),
88 *pVar,
89 pVar->paramStrForConstructor.c_str());
[3]90 }
91 }
92 }
93
94 //add rsp,スタックフレームサイズ
[64]95 op_add_RV(REG_RSP,pobj_sf->GetFrameSize());
[3]96
97 //スタックフレームスケジュール(subコマンドに渡す値)
98 *((long *)(OpBuffer+StackFrameSchedule))=pobj_sf->GetFrameSize();
99
100 //ret
[142]101 op_ret();
[3]102 }
[86]103 else if( userProc.GetName() == "_System_Call_Destructor_of_GlobalObject" ){
[3]104 //sub rsp,8(※RSPを16バイト境界にあわせるため)
105 op_sub_rsp(0x8);
106
107
[75]108 UserProc *pBackUserProc;
109 pBackUserProc = &UserProc::CompilingUserProc();
110 UserProc::CompileStartForGlobalArea();
[3]111
[34]112 obj_LexScopes.CallDestructorsOfScopeEnd();
[3]113
[75]114 UserProc::CompileStartForUserProc( pBackUserProc );
[3]115
116
117 //add rsp,8
[64]118 op_add_RV(REG_RSP,0x8);
[3]119
120 //ret
[142]121 op_ret();
[3]122 }
[86]123 else if( userProc.GetName() == "_System_GetSp" ){
[3]124 //mov rax,rsp
125 op_mov_RR(REG_RAX,REG_RSP);
126
127 //add rax,PTR_SIZE
[64]128 op_add_RV(REG_RAX,PTR_SIZE);
[3]129
130 //ret
[142]131 op_ret();
[3]132 }
[86]133 else if( userProc.GetName() == "_allrem" ){
[3]134 //乗除演算用の特殊関数(64ビット整数対応)
135 BYTE Buffer_allrem[]={
136 0x53,0x57,0x33,0xFF,0x8B,0x44,0x24,0x10,0x0B,0xC0,0x7D,0x14,0x47,0x8B,0x54,0x24,0x0C,0xF7,0xD8,0xF7,0xDA,0x83,0xD8,0x00,0x89,0x44,0x24,0x10,0x89,0x54,0x24,0x0C,0x8B,0x44,0x24,0x18,0x0B,0xC0,0x7D,0x13,0x8B,0x54,0x24,0x14,0xF7,0xD8,0xF7,0xDA,0x83,0xD8,0x00,0x89,0x44,0x24,0x18,0x89,0x54,0x24,0x14,0x0B,0xC0,0x75,0x1B,0x8B,0x4C,0x24,0x14,0x8B,0x44,0x24,0x10,0x33,0xD2,0xF7,0xF1,0x8B,0x44,0x24,0x0C,0xF7,0xF1,0x8B,0xC2,0x33,0xD2,0x4F,0x79,0x4E,0xEB,0x53,0x8B,0xD8,0x8B,0x4C,0x24,0x14,0x8B,0x54,0x24,0x10,0x8B,0x44,0x24,0x0C,0xD1,0xEB,0xD1,0xD9,0xD1,0xEA,0xD1,0xD8,0x0B,0xDB,0x75,0xF4,0xF7,0xF1,0x8B,0xC8,0xF7,0x64,0x24,0x18,0x91,0xF7,0x64,0x24,0x14,0x03,0xD1,0x72,0x0E,0x3B,0x54,0x24,0x10,0x77,0x08,0x72,0x0E,0x3B,0x44,0x24,0x0C,0x76,0x08,0x2B,0x44,0x24,0x14,0x1B,0x54,0x24,0x18,0x2B,0x44,0x24,0x0C,0x1B,0x54,0x24,0x10,0x4F,0x79,0x07,0xF7,0xDA,0xF7,0xD8,0x83,0xDA,0x00,0x5F,0x5B,0xC2,0x10,0x00
137 };
138
139 memcpy(OpBuffer+obp,Buffer_allrem,178);
140 obp+=178;
141 }
[86]142 else if( userProc.GetName() == "_allmul" ){
[3]143 //乗算用の特殊関数(64ビット整数対応)
144 BYTE Buffer_allmul[]={
145 0x8B,0x44,0x24,0x08,0x8B,0x4C,0x24,0x10,0x0B,0xC8,0x8B,0x4C,0x24,0x0C,0x75,0x09,0x8B,0x44,0x24,0x04,0xF7,0xE1,0xC2,0x10,0x00,0x53,0xF7,0xE1,0x8B,0xD8,0x8B,0x44,0x24,0x08,0xF7,0x64,0x24,0x14,0x03,0xD8,0x8B,0x44,0x24,0x08,0xF7,0xE1,0x03,0xD3,0x5B,0xC2,0x10,0x00
146 };
147
148 memcpy(OpBuffer+obp,Buffer_allmul,52);
149 obp+=52;
150 }
[86]151 else if( userProc.GetName() == "_alldiv" ){
[3]152 //除算用の特殊関数(64ビット整数対応)
153 BYTE Buffer_alldiv[]={
154 0x57,0x56,0x53,0x33,0xFF,0x8B,0x44,0x24,0x14,0x0B,0xC0,0x7D,0x14,0x47,0x8B,0x54,0x24,0x10,0xF7,0xD8,0xF7,0xDA,0x83,0xD8,0x00,0x89,0x44,0x24,0x14,0x89,0x54,0x24,0x10,0x8B,0x44,0x24,0x1C,0x0B,0xC0,0x7D,0x14,0x47,0x8B,0x54,0x24,0x18,0xF7,0xD8,0xF7,0xDA,0x83,0xD8,0x00,0x89,0x44,0x24,0x1C,0x89,0x54,0x24,0x18,0x0B,0xC0,0x75,0x18,0x8B,0x4C,0x24,0x18,0x8B,0x44,0x24,0x14,0x33,0xD2,0xF7,0xF1,0x8B,0xD8,0x8B,0x44,0x24,0x10,0xF7,0xF1,0x8B,0xD3,0xEB,0x41,0x8B,0xD8,0x8B,0x4C,0x24,0x18,0x8B,0x54,0x24,0x14,0x8B,0x44,0x24,0x10,0xD1,0xEB,0xD1,0xD9,0xD1,0xEA,0xD1,0xD8,0x0B,0xDB,0x75,0xF4,0xF7,0xF1,0x8B,0xF0,0xF7,0x64,0x24,0x1C,0x8B,0xC8,0x8B,0x44,0x24,0x18,0xF7,0xE6,0x03,0xD1,0x72,0x0E,0x3B,0x54,0x24,0x14,0x77,0x08,0x72,0x07,0x3B,0x44,0x24,0x10,0x76,0x01,0x4E,0x33,0xD2,0x8B,0xC6,0x4F,0x75,0x07,0xF7,0xDA,0xF7,0xD8,0x83,0xDA,0x00,0x5B,0x5E,0x5F,0xC2,0x10,0x00
155 };
156
157 memcpy(OpBuffer+obp,Buffer_alldiv,170);
158 obp+=170;
159 }
[86]160 else if( userProc.GetName() == "_allshl" ){
[3]161 //符号あり左ビットシフト用の特殊関数(64ビット整数対応)
162 BYTE Buffer_allshl[]={
163 0x80,0xF9,0x40,0x73,0x15,0x80,0xF9,0x20,0x73,0x06,0x0F,0xA5,0xC2,0xD3,0xE0,0xC3,0x8B,0xD0,0x33,0xC0,0x80,0xE1,0x1F,0xD3,0xE2,0xC3,0x33,0xC0,0x33,0xD2,0xC3
164 };
165
166 memcpy(OpBuffer+obp,Buffer_allshl,31);
167 obp+=31;
168 }
[86]169 else if( userProc.GetName() == "_allshr" ){
[3]170 //符号あり右ビットシフト用の特殊関数(64ビット整数対応)
171 BYTE Buffer_allshr[]={
172 0x80,0xF9,0x40,0x73,0x16,0x80,0xF9,0x20,0x73,0x06,0x0F,0xAD,0xD0,0xD3,0xFA,0xC3,0x8B,0xC2,0xC1,0xFA,0x1F,0x80,0xE1,0x1F,0xD3,0xF8,0xC3,0xC1,0xFA,0x1F,0x8B,0xC2,0xC3
173 };
174
175 memcpy(OpBuffer+obp,Buffer_allshr,33);
176 obp+=33;
177 }
[86]178 else if( userProc.GetName() == "_aullshr" ){
[3]179 //符号なし右ビットシフト用の特殊関数(64ビット整数対応)
180 BYTE Buffer_aullshr[]={
181 0x80,0xF9,0x40, //cmp cl,40h
182 0x73,0x15, //jae RETZERO (0040d71a)
183 0x80,0xF9,0x20, //cmp cl,20h
184 0x73,0x06, //jae MORE32 (0040d710)
185 0x0F,0xAD,0xD0, //shrd eax,edx,cl
186 0xD3,0xEA, //shr edx,cl
187 0xC3, //ret
188 //MORE32:
189 0x8B,0xC2, //mov eax,edx
190 0x33,0xD2, //xor edx,edx
191 0x80,0xE1,0x1F, //and cl,1Fh
192 0xD3,0xE8, //shr eax,cl
193 0xC3, //ret
194 //RETZERO:
195 0x33,0xC0, //xor eax,eax
196 0x33,0xD2, //xor edx,edx
197 0xC3 //ret
198 };
199
200 memcpy(OpBuffer+obp,Buffer_aullshr,31);
201 obp+=31;
202 }
[90]203 else{
204 SetError();
205 }
206}
207void AutoGeneration(UserProc &userProc){
208 if( userProc.GetName() == "InitializeUserTypes"
[89]209 && userProc.HasParentClass()
[131]210 && userProc.GetParentClass().GetName() == "_System_TypeBase" ){
[89]211
212 pobj_DBClass->Compile_System_InitializeUserTypes();
[86]213 }
[95]214 else if( userProc.GetName() == "RegisterGlobalRoots"
215 && userProc.HasParentClass()
[131]216 && userProc.GetParentClass().GetName() == "_System_CGarbageCollection" ){
[95]217
218 Compile_AddGlobalRootsForGc();
219 }
[86]220 else{
221 SetError();
222 }
[3]223}
[91]224void _compile_proc(UserProc *pUserProc){
[3]225 extern char *basbuf;
226 extern HANDLE hHeap;
[100]227 extern GlobalProc **ppSubHash;
[3]228 extern BOOL bDebugCompile;
[76]229 int i3,i4;
[3]230 char temporary[VN_SIZE];
231
[75]232 if( pUserProc->IsUsing() == false || pUserProc->IsCompiled() ) return;
[3]233
[76]234 if( pUserProc->localVars.size() ){
235 SetError();
236 return;
237 }
238
[75]239 pUserProc->CompleteCompile();
[3]240
241 extern BOOL bSystemProc;
[75]242 if(memcmp(pUserProc->GetName().c_str(),"_System_",8)==0) bSystemProc=1;
[3]243 else bSystemProc=0;
244
245 extern BOOL bDebugSupportProc;
[75]246 if(memcmp(pUserProc->GetName().c_str(),"_DebugSys_",10)==0){
[3]247 if(!bDebugCompile){
248 return;
249 }
250 bDebugSupportProc=1;
251 }
252 else bDebugSupportProc=0;
253
[75]254 pUserProc->beginOpAddress=obp;
[3]255
[89]256 //コンパイル中の関数が属するクラス
257 pobj_CompilingClass=pUserProc->GetParentClassPtr();
258
259 //コンパイルスタートをクラス管理クラスに追加
260 pobj_DBClass->StartCompile( pUserProc );
261
262 //コンパイル中の関数
263 UserProc::CompileStartForUserProc( pUserProc );
264
[101]265 // コンパイル中の関数が属する名前空間
266 Smoothie::Lexical::liveingNamespaceScopes = pUserProc->GetNamespaceScopes();
267
[108]268 // コンパイル中の関数でImportsされている名前空間
[143]269 Smoothie::Temp::importedNamespaces = pUserProc->GetImportedNamespaces();
[108]270
[75]271 if(pUserProc->IsSystem()){
[3]272 ////////////////////
273 // 特殊関数
274 ////////////////////
275
276 extern int AllLocalVarSize;
277 AllLocalVarSize=0;
278
279 //スタックフレーム管理用オブジェクトを初期化
280 extern CStackFrame *pobj_sf;
281 pobj_sf=new CStackFrame();
282
[86]283 SystemProc(*pUserProc);
[3]284
285 //スタックフレーム管理用オブジェクトを破棄
286 delete pobj_sf;
287 pobj_sf=0;
288
[75]289 pUserProc->endOpAddress=obp;
[3]290 return;
291 }
292
[75]293 cp=pUserProc->GetCodePos();
[3]294 for(;;cp++){
295 if(IsCommandDelimitation(basbuf[cp])) break;
296 }
297 cp--;
298
299 //プロシージャ抜け出しスケジュール(Exit Sub/Function)
300 extern DWORD *pExitSubSchedule;
301 extern int ExitSubScheduleNum;
302 pExitSubSchedule=(DWORD *)HeapAlloc(hHeap,0,1);
303 ExitSubScheduleNum=0;
304
305 //ラベル用のメモリを確保
306 extern LABEL *pLabelNames;
307 extern int MaxLabelNum;
308 pLabelNames=(LABEL *)HeapAlloc(hHeap,0,1);
309 MaxLabelNum=0;
310
311 //Gotoラベルスケジュール
312 extern GOTOLABELSCHEDULE *pGotoLabelSchedule;
313 extern int GotoLabelScheduleNum;
314 pGotoLabelSchedule=(GOTOLABELSCHEDULE *)HeapAlloc(hHeap,0,1);
315 GotoLabelScheduleNum=0;
316
317 //With情報のメモリを確保
318 extern WITHINFO WithInfo;
319 WithInfo.ppName=(char **)HeapAlloc(hHeap,0,1);
320 WithInfo.pWithCp=(int *)HeapAlloc(hHeap,0,1);
321 WithInfo.num=0;
322
323 //重複エラー情報管理のメモリを確保
324 extern char **SynonymErrorWords;
325 extern int SynonymErrorNum;
326 SynonymErrorNum=0;
327 SynonymErrorWords=(char **)HeapAlloc(hHeap,0,1);
328
329 //Continueアドレスを初期化
330 extern DWORD dwContinueAddress;
331 dwContinueAddress=-1;
332
333 //ローカル変数に関する情報
334 extern int AllLocalVarSize;
335 AllLocalVarSize=0;
336
337 //ローカル変数アドレススケジュール
338 extern DWORD *pLocalVarAddrSchedule;
339 extern int LocalVarAddrScheduleNum;
340 pLocalVarAddrSchedule=(DWORD *)HeapAlloc(hHeap,0,1);
341 LocalVarAddrScheduleNum=0;
342
343 //レキシカルスコープ情報を初期化
344 obj_LexScopes.Init(obp);
345
346
347 /////////////////////////////////////
348 // パラメータ用の変数データを考慮
349 /////////////////////////////////////
350
[75]351 //パラメータ用の変数データを考慮
352 for(i3=(int)pUserProc->RealParams().size()-1;i3>=0;i3--){
353 Parameter &param = *pUserProc->RealParams()[i3];
[3]354
[75]355 Variable *pVar = new Variable( param.GetVarName(), param, false, param.IsRef() );
[3]356
[75]357 if( param.IsArray() ){
358 pVar->SetArray( param.GetSubScriptsPtr() );
[3]359 }
360
[75]361 int varSize;
362 if( param.IsRef() == false && param.IsStruct() ){
[64]363 //構造体のByValパラメータ
[75]364 pVar->ThisIsParameter();
365 varSize=PTR_SIZE;
[3]366 }
367 else{
[75]368 if( param.IsArray() == false ){
369 varSize = pVar->GetMemorySize();
[3]370 }
371 else{
[75]372 varSize=PTR_SIZE;
[3]373 }
374 }
[75]375 AllLocalVarSize+=varSize;
376 pVar->offset=AllLocalVarSize;
[3]377
378 //レキシカルスコープ情報
[75]379 pVar->ScopeLevel=obj_LexScopes.GetNowLevel();
380 pVar->ScopeStartAddress=obj_LexScopes.GetStartAddress();
381 pVar->bLiving=TRUE;
[3]382
[75]383 pUserProc->localVars.push_back( pVar );
[3]384 }
385
386 //Thisポインタを示すローカルオフセット値をセット
387 extern int LocalVar_ThisPtrOffset;
388 LocalVar_ThisPtrOffset=AllLocalVarSize;
389
390 //スタックフレーム管理用クラスを初期化
391 extern CStackFrame *pobj_sf;
392 pobj_sf=new CStackFrame();
393
394
395 ///////////////////////
396 // ここからコード生成
397
[75]398 for(i3=(int)pUserProc->RealParams().size()-1;i3>=0;i3--){
399 Parameter &param = *pUserProc->RealParams()[i3];
[3]400 if(i3==3){
[75]401 if(param.IsReal()&&param.IsRef() == false){
[3]402 //movsd qword ptr[rsp+0x20],xmm3
403 op_movsd_MR(REG_XMM3,REG_RSP,0x20,MOD_BASE_DISP32);
404 }
405 else{
406 //mov qword ptr[rsp+0x20],r9
407 op_mov_MR(sizeof(_int64),REG_R9,REG_RSP,0x20,MOD_BASE_DISP32);
408 }
409 }
410 if(i3==2){
[75]411 if(param.IsReal()&&param.IsRef() == false){
[3]412 //movsd qword ptr[rsp+0x18],xmm2
413 op_movsd_MR(REG_XMM2,REG_RSP,0x18,MOD_BASE_DISP32);
414 }
415 else{
416 //mov qword ptr[rsp+0x18],r8
417 op_mov_MR(sizeof(_int64),REG_R8,REG_RSP,0x18,MOD_BASE_DISP32);
418 }
419 }
420 if(i3==1){
[75]421 if(param.IsReal()&&param.IsRef() == false){
[3]422 //movsd qword ptr[rsp+0x10],xmm1
423 op_movsd_MR(REG_XMM1,REG_RSP,0x10,MOD_BASE_DISP32);
424 }
425 else{
426 //mov qword ptr[rsp+0x10],rdx
427 op_mov_MR(sizeof(_int64),REG_RDX,REG_RSP,0x10,MOD_BASE_DISP32);
428 }
429 }
430 if(i3==0){
[75]431 if(param.IsReal()&&param.IsRef() == false){
[3]432 //movsd qword ptr[rsp+0x8],xmm0
433 op_movsd_MR(REG_XMM0,REG_RSP,0x8,MOD_BASE_DISP32);
434 }
435 else{
436 //mov qword ptr[rsp+0x8],rcx
437 op_mov_MR(sizeof(_int64),REG_RCX,REG_RSP,0x8,MOD_BASE_DISP32);
438 }
439 }
440 }
441
442 //ret用のアドレスを考慮
443 AllLocalVarSize+=sizeof(_int64);
444
445 //sub rsp,スタックフレームサイズ
446 int StackFrameSchedule;
447 op_sub_rsp(0xFFFFFFFF);
448 StackFrameSchedule=obp-sizeof(long);
449
450 //mov qword ptr[rsp+offset],reg ※スタックフレームを利用
451 pobj_sf->push(REG_RBX);
452 pobj_sf->push(REG_RSI);
453 pobj_sf->push(REG_RDI);
454 pobj_sf->push(REG_R12);
455 pobj_sf->push(REG_R13);
456 pobj_sf->push(REG_R14);
457 pobj_sf->push(REG_R15);
458
459 //ローカル変数のベース値
460 int BaseLocalVar;
461 BaseLocalVar=AllLocalVarSize;
462
[75]463 if( !pUserProc->ReturnType().IsNull() ){
[3]464 //戻り値が存在するとき
465
[75]466 const char *temp = pUserProc->GetName().c_str();
467 if( temp[0]==1&&temp[1]==ESC_OPERATOR ){
468 temp = "_System_ReturnValue";
469 }
[3]470
[75]471 if( pUserProc->ReturnType().IsStruct() ){
[64]472 //戻り値用の構造体(値型)はパラメータで引き渡される
[3]473 }
474 else{
[89]475 if( pUserProc->ReturnType().IsObject() ){
476 sprintf(temporary,"%s=Nothing%c%c%s",temp,1,ESC_AS, pUserProc->ReturnType().ToString().c_str() );
477 }
478 else{
479 //戻り値用の変数の定義
480 sprintf(temporary,"%s%c%c%s",temp,1,ESC_AS, pUserProc->ReturnType().ToString().c_str() );
481 }
[40]482
[3]483 OpcodeDim(temporary,0);
484 }
485 }
486
487 int RspOffsetSchedule=0;
488 int RspOffsetSchedule2;
489 if(bDebugCompile&&bDebugSupportProc==0){
490 //mov rdx, qword ptr[rsp+スタックフレームサイズ]
491 op_mov_RM(sizeof(_int64),REG_RDX,REG_RSP,0,MOD_BASE_DISP32);
492 RspOffsetSchedule=obp-sizeof(long);
493
494 //mov rcx,rsp
[64]495 op_mov_RR(REG_RCX,REG_RSP);
[3]496
497 //add rcx,スタックフレームサイズ+sizeof(_int64) ※ret用のサイズを考慮
[64]498 op_add_RV(REG_RCX,0);
[3]499 RspOffsetSchedule2=obp-sizeof(long);
500
501 //call _DebugSys_StartProc
[75]502 extern UserProc *pSub_DebugSys_StartProc;
[3]503 op_call(pSub_DebugSys_StartProc);
504 }
505
506 if(pobj_CompilingClass){
[131]507 if( pUserProc->GetName() == pobj_CompilingClass->GetName() ){
[3]508 ////////////////////////////////////
509 // コンストラクタをコンパイルするとき
510 ////////////////////////////////////
511
[17]512 //コンストラクタのコンパイル開始を通知
513 pobj_CompilingClass->NotifyStartConstructorCompile();
514
[27]515 //基底クラスかどうかの識別
516 //(継承元がインターフェイスの場合も基底クラスと見なす)
[3]517 BOOL bThisIsSuperClass;
518 if(pobj_CompilingClass->pobj_InheritsClass==0) bThisIsSuperClass=1;
[51]519 else if( pobj_CompilingClass->pobj_InheritsClass->GetConstructorMethod() == NULL ){
[3]520 //インターフェイスを継承したときはコンストラクタを持たない
521 bThisIsSuperClass=1;
522 }
523 else bThisIsSuperClass=0;
524
525 if(!bThisIsSuperClass){
526 /* サブクラスコンストラクタをコンパイルしているときは、
[27]527 基底クラスのコンストラクタを呼び出す */
[3]528
529 i3=cp+1;
530 while(IsCommandDelimitation(basbuf[i3])) i3++;
531 for(i4=0;;i3++,i4++){
532 if(!IsVariableChar(basbuf[i3])){
533 temporary[i4]=0;
534 break;
535 }
536 temporary[i4]=basbuf[i3];
537 }
[131]538 if( pobj_CompilingClass->pobj_InheritsClass->GetName() == temporary ){
[27]539 //基底クラスのコンストラクタを呼び出す
[3]540 cp=i3;
541 for(i4=0;;cp++,i4++){
542 if(IsCommandDelimitation(basbuf[cp])){
543 temporary[i4]=0;
544 break;
545 }
546 temporary[i4]=basbuf[cp];
547 }
548 if(!(temporary[0]=='('&&temporary[lstrlen(temporary)-1]==')')){
549 SetError(1,NULL,cp);
550 }
551 RemoveStringPare(temporary);
552
[89]553 Type dummyType;
554 CallProc( PROC_DEFAULT
555 , pobj_CompilingClass->pobj_InheritsClass->GetConstructorMethod()->pUserProc
556 , pobj_CompilingClass->pobj_InheritsClass->GetConstructorMethod()->pUserProc->GetName().c_str()
557 , temporary
558 , dummyType );
[3]559 }
560 else{
[27]561 //基底クラスのコンストラクタを暗黙的に呼び出す
[3]562 Opcode_CallProc("",
[75]563 pobj_CompilingClass->pobj_InheritsClass->GetConstructorMethod()->pUserProc,
[3]564 0,
565 "",
566 0);
567 }
568 }
569
570 //仮想関数テーブルを初期化
[141]571 if( pobj_CompilingClass->IsExistVirtualFunctions()
572 && !pobj_CompilingClass->IsAbstract() ){
[28]573 //関数テーブルに値をセット
[63]574 int offset = (int)pobj_CompilingClass->GetVtblGlobalOffset();
[3]575
[28]576 //mov rax,offset
577 op_mov_RV(sizeof(_int64),REG_RAX,offset);
578 obp-=sizeof(long);
579 pobj_DataTableSchedule->add();
580 obp+=sizeof(long);
[3]581
[28]582 //Thisポインタをrcxにコピー
583 SetThisPtrToReg(REG_RCX);
[3]584
[28]585 //mov qword ptr[rcx],rax
586 op_mov_MR(sizeof(_int64),REG_RAX,REG_RCX,0,MOD_BASE);
[3]587 }
588 }
[75]589 else if( pUserProc->IsDestructor() ){
[18]590 //デストラクタをコンパイルしたとき
591
592 //デストラクタのコンパイル開始を通知
593 pobj_CompilingClass->NotifyStartDestructorCompile();
594 }
[3]595 }
596
597 //////////////////////////////////////////
598 //////////////////////////////////////////
599 ////// プロシージャ内をコンパイル ////////
[90]600 if( pUserProc->IsAutoGeneration() ){
601 AutoGeneration( *pUserProc );
602 }
[75]603 else{
[90]604 if(pUserProc->IsMacro()){
605 CompileBuffer(ESC_ENDMACRO,0);
606 }
607 else{
608 if(pUserProc->IsSub()){
609 CompileBuffer(ESC_ENDSUB,0);
610 }
611 else if(pUserProc->IsFunction()){
612 CompileBuffer(ESC_ENDFUNCTION,0);
613 }
614 }
[75]615 }
[3]616 //////////////////////////////////////////
617 //////////////////////////////////////////
618
[17]619 if( pobj_CompilingClass ){
620
621 if( pobj_CompilingClass->IsCompilingConstructor() ){
622 // コンストラクタをコンパイルしていたとき
[18]623
[17]624 // コンストラクタのコンパイルが完了したことを通知
625 pobj_CompilingClass->NotifyFinishConstructorCompile();
626 }
[75]627 else if( pUserProc->IsDestructor() ){
[3]628 ////////////////////////////////////
629 //デストラクタをコンパイルしたとき
630 ////////////////////////////////////
631
[18]632 // デストラクタのコンパイルが完了したことを通知
633 pobj_CompilingClass->NotifyFinishDestructorCompile();
634
[3]635 if(pobj_CompilingClass->pobj_InheritsClass){
636 /* サブクラスのデストラクタをコンパイルしているときは、
[27]637 基底クラスのデストラクタを呼び出す */
[3]638
[135]639 const CMethod *method = pobj_CompilingClass->pobj_InheritsClass->GetDestructorMethod();
[51]640 if( method ){
[3]641 Opcode_CallProc("",
[75]642 method->pUserProc,
[3]643 0,
644 "",
645 0);
646 }
647 }
648 }
649 }
650
651 //ラベル用のメモリを解放
652 for(i3=0;i3<MaxLabelNum;i3++){
653 if(pLabelNames[i3].pName) HeapDefaultFree(pLabelNames[i3].pName);
654 }
655 HeapDefaultFree(pLabelNames);
656
657 //Goto未知ラベルスケジュールを解放
658 for(i3=0;i3<GotoLabelScheduleNum;i3++){
659 if(pGotoLabelSchedule[i3].pName){
660 SetError(6,pGotoLabelSchedule[i3].pName,pGotoLabelSchedule[i3].now_cp);
661 HeapDefaultFree(pGotoLabelSchedule[i3].pName);
662 }
663 else{
664 sprintf(temporary,"%d",pGotoLabelSchedule[i3].line);
665 SetError(6,temporary,pGotoLabelSchedule[i3].now_cp);
666 }
667 }
668 HeapDefaultFree(pGotoLabelSchedule);
669
670 //With情報のメモリを解放
671 for(i3=0;i3<WithInfo.num;i3++){
672 SetError(22,"With",WithInfo.pWithCp[i3]);
673 HeapDefaultFree(WithInfo.ppName[i3]);
674 }
675 HeapDefaultFree(WithInfo.ppName);
676 HeapDefaultFree(WithInfo.pWithCp);
677
678 //ローカルオブジェクト(レキシカルスコープレベル=0)の解放処理
[34]679 obj_LexScopes.CallDestructorsOfScopeEnd();
[3]680
[34]681 //プロシージャ抜け出しスケジュール(Exit Sub/Function)
682 for(i3=0;i3<ExitSubScheduleNum;i3++){
683 *((long *)(OpBuffer+pExitSubSchedule[i3]))=obp-(pExitSubSchedule[i3]+sizeof(long));
684 }
685 HeapDefaultFree(pExitSubSchedule);
686
687 if(bDebugCompile&&bDebugSupportProc==0){
688 //call _DebugSys_EndProc
[75]689 extern UserProc *pSub_DebugSys_EndProc;
[34]690 op_call(pSub_DebugSys_EndProc);
691 }
692
[75]693 if( !pUserProc->ReturnType().IsNull() ){
[3]694 //////////////////////////////////
695 // 戻り値をraxまたはxmm0に設定
696 //////////////////////////////////
697
698 RELATIVE_VAR RelativeVar;
699
[75]700 const char *temp = pUserProc->GetName().c_str();
701 if( temp[0]==1 && temp[1]==ESC_OPERATOR ){
[3]702 temp="_System_ReturnValue";
[75]703 }
704 GetVarOffsetReadWrite(temp,&RelativeVar,Type());
[3]705
[75]706 i3=pUserProc->ReturnType().GetBasicType();
[3]707
[64]708 if(i3==DEF_OBJECT || i3==DEF_STRUCT){
[3]709 SetVarPtrToReg(REG_RAX,&RelativeVar);
[64]710 if( i3==DEF_OBJECT ){
711 //mov rax,qword ptr[rax]
712 op_mov_RM( sizeof(_int64), REG_RAX, REG_RAX, 0, MOD_BASE );
713 }
[3]714 }
715 else if(i3==DEF_DOUBLE){
716 //64ビット実数型
717 SetXmmReg_DoubleVariable(&RelativeVar,REG_XMM0);
718 }
719 else if(i3==DEF_SINGLE){
720 //32ビット実数型
721 SetXmmReg_SingleVariable(&RelativeVar,REG_XMM0);
722 }
723 else if(IsWholeNumberType(i3)){
724 //整数型
725 SetReg_WholeVariable(i3,&RelativeVar,REG_RAX);
726 }
727 else SetError(300,NULL,cp);
728 }
729
730 //ローカル変数領域のサイズをスタックフレームに通知
731 int iLocalParmSize;
732 iLocalParmSize=AllLocalVarSize-BaseLocalVar;
733 pobj_sf->SetLocalParmSize(iLocalParmSize);
734
735 //ローカル変数アドレススケジュール
736 for(i3=0;i3<LocalVarAddrScheduleNum;i3++){
737 *((long *)(OpBuffer+pLocalVarAddrSchedule[i3]))+=AllLocalVarSize+pobj_sf->GetFrameSize();
738 }
739 HeapDefaultFree(pLocalVarAddrSchedule);
[75]740 foreach( Variable *pVar, pUserProc->localVars ){
[3]741 //後にデバッグで利用する
[75]742 pVar->offset = AllLocalVarSize + pobj_sf->GetFrameSize() - pVar->offset;
[3]743 }
744
745 //mov reg,qword ptr[rsp+offset] ※スタックフレームを利用
746 pobj_sf->pop(REG_R15);
747 pobj_sf->pop(REG_R14);
748 pobj_sf->pop(REG_R13);
749 pobj_sf->pop(REG_R12);
750 pobj_sf->pop(REG_RDI);
751 pobj_sf->pop(REG_RSI);
752 pobj_sf->pop(REG_RBX);
753
754 int iStackFrameSize;
755 iStackFrameSize=iLocalParmSize + pobj_sf->GetFrameSize();
756
757 //add rsp,スタックフレームサイズ
758 op_add_rsp(iStackFrameSize);
759
[142]760 //ret
761 op_ret();
[3]762
763
764 //デバッグ用
765 if(RspOffsetSchedule){
766 *((long *)(OpBuffer+RspOffsetSchedule))=iStackFrameSize;
767 *((long *)(OpBuffer+RspOffsetSchedule2))=iStackFrameSize+sizeof(_int64);
768 }
769
770
771 //スタックフレームスケジュール(subコマンド)
772 *((long *)(OpBuffer+StackFrameSchedule))=iStackFrameSize;
773
774 //スタックフレームスケジュールを実行
775 pobj_sf->RunningSchedule();
776 delete pobj_sf;
777 pobj_sf=0;
778
779
[75]780 pUserProc->endOpAddress=obp;
[3]781
[75]782
783 //重複エラー情報管理のメモリを解放
784 for(i3=0;i3<SynonymErrorNum;i3++) HeapDefaultFree(SynonymErrorWords[i3]);
785 HeapDefaultFree(SynonymErrorWords);
[3]786}
[91]787
788void CompileBufferInProcedure( UserProc &userProc ){
789 if( userProc.IsUsing() == false || userProc.IsCompiled() ) return;
790
791 _compile_proc( &userProc );
792
793 // ログを履く
794 char temporary[8192];
795 temporary[0]=0;
796 lstrcat( temporary, "------------------------------------------------------------------\n" );
[100]797 sprintf( temporary + lstrlen(temporary), "【 %s のコード情報】\n", userProc.GetName().c_str() );
[91]798 sprintf( temporary + lstrlen(temporary), "code size: %d bytes\n", userProc.GetCodeSize() );
799 lstrcat( temporary, "------------------------------------------------------------------\n" );
800 lstrcat( temporary, "\n" );
[169]801 trace_for_size( temporary );
[91]802}
[3]803void CompileLocal(){
[100]804 extern GlobalProc **ppSubHash;
[3]805 int i2;
806
807 extern BOOL bDll;
808 if(bDll){
809 //DLLの場合はグローバル変数を初期化するための関数を一番初めにコンパイルする
[100]810 UserProc *pUserProc=GetSubHash("_System_InitDllGlobalVariables");
[75]811 if(pUserProc){
[91]812 CompileBufferInProcedure( *pUserProc );
[3]813 }
814 else SetError(300,NULL,cp);
815 }
816
[94]817 //_System_TypeBase_InitializeUserTypesは一番最後にコンパイル
818 extern UserProc *pSubStaticMethod_System_TypeBase_InitializeUserTypes;
819 pSubStaticMethod_System_TypeBase_InitializeUserTypes->CompleteCompile();
820
[3]821 //_System_InitStaticLocalVariablesは一番最後にコンパイル
822 //※一般関数内の静的変数オブジェクトをすべて収集しなければならない
[75]823 extern UserProc *pSub_System_InitStaticLocalVariables;
824 pSub_System_InitStaticLocalVariables->CompleteCompile();
[3]825
826 //_System_Call_Destructor_of_GlobalObjectは一番最後にコンパイル
[75]827 extern UserProc *pSub_System_Call_Destructor_of_GlobalObject;
828 pSub_System_Call_Destructor_of_GlobalObject->CompleteCompile();
[3]829
[94]830repeat:
[100]831 GlobalProc *pGlobalProc;
[3]832 for(i2=0;i2<MAX_HASH;i2++){
[100]833 pGlobalProc=ppSubHash[i2];
834 while(pGlobalProc){
835 CompileBufferInProcedure( *pGlobalProc );
836 pGlobalProc=pGlobalProc->pNextData;
[3]837 }
838 }
839
[94]840 if( IsNeedProcCompile() ){
841 //プロシージャコンパイルによって、プロシージャコンパイルが必要になる場合
842 goto repeat;
843 }
844
845 //_System_TypeBase_InitializeUserTypesは最後のほうでコンパイル
846 pSubStaticMethod_System_TypeBase_InitializeUserTypes->KillCompileStatus();
847 CompileBufferInProcedure( *pSubStaticMethod_System_TypeBase_InitializeUserTypes );
848
849 if( IsNeedProcCompile() ){
850 //プロシージャコンパイルによって、プロシージャコンパイルが必要になる場合
851 for(i2=0;i2<MAX_HASH;i2++){
[100]852 pGlobalProc=ppSubHash[i2];
853 while(pGlobalProc){
854 CompileBufferInProcedure( *pGlobalProc );
855 pGlobalProc=pGlobalProc->pNextData;
[3]856 }
857 }
858 }
859
860 //_System_InitStaticLocalVariablesは一番最後にコンパイル
[75]861 pSub_System_InitStaticLocalVariables->KillCompileStatus();
[91]862 CompileBufferInProcedure( *pSub_System_InitStaticLocalVariables );
[3]863
864 //_System_Call_Destructor_of_GlobalObjectは一番最後にコンパイル
[75]865 pSub_System_Call_Destructor_of_GlobalObject->KillCompileStatus();
[91]866 CompileBufferInProcedure( *pSub_System_Call_Destructor_of_GlobalObject );
[3]867}
Note: See TracBrowser for help on using the repository browser.