#include "../BasicCompiler_Common/common.h" #include "Opcode.h" void SystemProc(char *name){ if(lstrcmp(name,"_System_GetEip")==0){ //mov eax,dword ptr[esp] OpBuffer[obp++]=(char)0x8B; OpBuffer[obp++]=(char)0x04; OpBuffer[obp++]=(char)0x24; //ret OpBuffer[obp++]=(char)0xC3; } else if(lstrcmp(name,"_System_InitDllGlobalVariables")==0){ //////////////////////////////////////// // DLLのグローバル領域をコンパイル //////////////////////////////////////// extern BOOL bDll; if(!bDll){ //ret OpBuffer[obp++]=(char)0xC3; return; } extern BOOL bCompilingGlobal; BOOL bBackCompilingGlobal; bBackCompilingGlobal=bCompilingGlobal; bCompilingGlobal=1; int BackCp; BackCp=cp; cp=-1; extern BOOL bDebugCompile; if(bDebugCompile){ //デバッグ用の変数を定義 DebugVariable(); } //GC用の変数を定義 InitGCVariables(); GetGlobalDataForDll(); bCompilingGlobal=bBackCompilingGlobal; cp=BackCp; //ret OpBuffer[obp++]=(char)0xC3; } else if(lstrcmp(name,"_System_InitStaticLocalVariables")==0){ //静的ローカルオブジェクトのコンストラクタ呼び出し extern int MaxGlobalVarNum; extern VARIABLE *GlobalVar; int i; for(i=0;ibUse==0||psi->bCompile) return; psi->bCompile=1; extern BOOL bSystemProc; if(memcmp(psi->name,"_System_",8)==0) bSystemProc=1; else bSystemProc=0; extern BOOL bDebugSupportProc; if(memcmp(psi->name,"_DebugSys_",10)==0){ if(!bDebugCompile){ psi->pVar=(VARIABLE *)HeapAlloc(hHeap,0,1); psi->VarNum=0; return; } bDebugSupportProc=1; } else bDebugSupportProc=0; psi->CompileAddress=obp; if(psi->bSystem){ extern int MaxLocalVarNum; extern int AllLocalVarSize; MaxLocalVarNum=0; AllLocalVarSize=0; SystemProc(psi->name); psi->EndOpAddr=obp; psi->pVar=(VARIABLE *)HeapAlloc(hHeap,0,1); psi->VarNum=0; return; } cp=psi->address; for(;;cp++){ if(IsCommandDelimitation(basbuf[cp])) break; } cp--; //ローカル変数に関する情報 extern VARIABLE *LocalVar; extern int MaxLocalVarNum; extern int AllLocalVarSize; LocalVar=(VARIABLE *)HeapAlloc(hHeap,0,1); MaxLocalVarNum=0; AllLocalVarSize=0; //ローカル変数アドレススケジュール extern DWORD *pLocalVarAddrSchedule; extern int LocalVarAddrScheduleNum; pLocalVarAddrSchedule=(DWORD *)HeapAlloc(hHeap,0,1); LocalVarAddrScheduleNum=0; //パラメータ用の変数データを考慮 for(i3=psi->RealParmNum-1;i3>=0;i3--){ //変数データを追加 LocalVar=(VARIABLE *)HeapReAlloc(hHeap,0,LocalVar,(MaxLocalVarNum+1)*sizeof(VARIABLE)); lstrcpy(LocalVar[MaxLocalVarNum].name,psi->pRealParmInfo[i3].name); // TODO: パラメータのConst定義の指定が未完成 LocalVar[MaxLocalVarNum].bConst=false; LocalVar[MaxLocalVarNum].bArray=psi->pRealParmInfo[i3].bArray; memcpy(LocalVar[MaxLocalVarNum].SubScripts,psi->pRealParmInfo[i3].SubScripts,MAX_ARRAYDIM*sizeof(int)); LocalVar[MaxLocalVarNum].type=psi->pRealParmInfo[i3].type; if(LocalVar[MaxLocalVarNum].type==DEF_OBJECT){ if(psi->bExport) SetError(24,NULL,cp); } LocalVar[MaxLocalVarNum].u.index=psi->pRealParmInfo[i3].u.index; if(psi->pRealParmInfo[i3].bByVal&&psi->pRealParmInfo[i3].type==DEF_OBJECT){ //実態オブジェクトのByValパラメータ LocalVar[MaxLocalVarNum].fRef=REF_PARAMETER | OBJECT_PARAMETER; VarSize=PTR_SIZE; } else{ if(psi->pRealParmInfo[i3].bByVal&&psi->pRealParmInfo[i3].bArray==0){ LocalVar[MaxLocalVarNum].fRef=0; VarSize=GetTypeSize(LocalVar[MaxLocalVarNum].type,LocalVar[MaxLocalVarNum].u.index); if(VarSize%PTR_SIZE) VarSize+=PTR_SIZE-(VarSize%PTR_SIZE); } else{ LocalVar[MaxLocalVarNum].fRef=REF_PARAMETER; VarSize=PTR_SIZE; } } AllLocalVarSize+=VarSize; LocalVar[MaxLocalVarNum].offset=AllLocalVarSize; //レキシカルスコープ情報 LocalVar[MaxLocalVarNum].ScopeLevel=obj_LexScopes.GetNowLevel(); LocalVar[MaxLocalVarNum].ScopeStartAddress=obj_LexScopes.GetStartAddress(); LocalVar[MaxLocalVarNum].bLiving=TRUE; MaxLocalVarNum++; } //Thisポインタを示すローカルオフセット値をセット extern int LocalVar_ThisPtrOffset; LocalVar_ThisPtrOffset=AllLocalVarSize; BaseOffset=AllLocalVarSize; //ret用のアドレスを考慮 AllLocalVarSize+=sizeof(long); //コンパイル中の関数が属するクラス pobj_CompilingClass=psi->pobj_ParentClass; //コンパイルスタートをクラス管理クラスに追加 pobj_DBClass->StartCompile( psi ); //コンパイル中の関数 extern SUBINFO *pCompilingSubInfo; pCompilingSubInfo=psi; /////////////////////// // ここからコード生成 //sub esp,AllLocalVarSize(スケジュール) op_sub_esp(0xFFFFFFFF); LocalVarSchedule=obp-sizeof(long); //push ebp op_push(REG_EBP); //mov ebp,esp OpBuffer[obp++]=(char)0x8B; OpBuffer[obp++]=(char)0xEC; //push ebx op_push(REG_EBX); //push esi OpBuffer[obp++]=(char)0x56; //push edi OpBuffer[obp++]=(char)0x57; if(psi->ReturnType!=-1){ //戻り値が存在するとき char *temp; if(psi->name[0]==1&&psi->name[1]==ESC_OPERATOR) temp="_System_ReturnValue"; else temp=psi->name; if(psi->ReturnType==DEF_OBJECT && psi->isReturnRef == false){ //戻り値用オブジェクトのコンストラクタを呼び出す if(psi->u.Return_pobj_c->ConstructorMemberSubIndex!=-1){ sprintf(temporary,"%s.%s()",temp,psi->u.Return_pobj_c->name); OpcodeOthers(temporary); } } else{ //戻り値用の変数の定義 if( psi->isReturnRef ){ sprintf(temporary,"%c%c",1,ESC_BYREF); } else temporary[0]=0; sprintf(temporary+lstrlen(temporary),"%s%c%c",temp,1,ESC_AS); GetTypeName(psi->ReturnType,psi->u.ReturnIndex,temporary+lstrlen(temporary)); OpcodeDim(temporary,0); } } //プロシージャ抜け出しスケジュール(Exit Sub/Function) extern DWORD *pExitSubSchedule; extern int ExitSubScheduleNum; pExitSubSchedule=(DWORD *)HeapAlloc(hHeap,0,1); ExitSubScheduleNum=0; //ラベル用のメモリを確保 extern LABEL *pLabelNames; extern int MaxLabelNum; pLabelNames=(LABEL *)HeapAlloc(hHeap,0,1); MaxLabelNum=0; //Gotoラベルスケジュール extern GOTOLABELSCHEDULE *pGotoLabelSchedule; extern int GotoLabelScheduleNum; pGotoLabelSchedule=(GOTOLABELSCHEDULE *)HeapAlloc(hHeap,0,1); GotoLabelScheduleNum=0; //With情報のメモリを確保 extern WITHINFO WithInfo; WithInfo.ppName=(char **)HeapAlloc(hHeap,0,1); WithInfo.pWithCp=(int *)HeapAlloc(hHeap,0,1); WithInfo.num=0; //重複エラー情報管理のメモリを確保 extern char **SynonymErrorWords; extern int SynonymErrorNum; SynonymErrorNum=0; SynonymErrorWords=(char **)HeapAlloc(hHeap,0,1); //Continueアドレスを初期化 extern DWORD dwContinueAddress; dwContinueAddress=-1; if(bDebugCompile&&bDebugSupportProc==0){ //push dword ptr[ebp+(AllLocalVarSize-BaseOffset)](スケジュール) OpBuffer[obp++]=(char)0xFF; OpBuffer[obp++]=(char)0xB5; EspOffsetSchedule=obp; obp+=sizeof(long); //push dword ptr[ebp](以前のebp) OpBuffer[obp++]=(char)0xFF; OpBuffer[obp++]=(char)0x75; OpBuffer[obp++]=(char)0x00; //call _DebugSys_StartProc extern SUBINFO *pSub_DebugSys_StartProc; op_call(pSub_DebugSys_StartProc); } if(pobj_CompilingClass){ if(lstrcmp(psi->name,pobj_CompilingClass->name)==0){ //////////////////////////////////// // コンストラクタをコンパイルするとき //////////////////////////////////// //コンストラクタのコンパイル開始を通知 pobj_CompilingClass->NotifyStartConstructorCompile(); //基底クラスかどうかの識別 //(継承元がインターフェイスの場合も基底クラスと見なす) BOOL bThisIsSuperClass; if(pobj_CompilingClass->pobj_InheritsClass==0) bThisIsSuperClass=1; else if(pobj_CompilingClass->pobj_InheritsClass->ConstructorMemberSubIndex==-1){ //インターフェイスを継承したときはコンストラクタを持たない bThisIsSuperClass=1; } else bThisIsSuperClass=0; if(!bThisIsSuperClass){ /* サブクラスコンストラクタをコンパイルしているときは、 基底クラスのコンストラクタを呼び出す */ i3=cp+1; while(IsCommandDelimitation(basbuf[i3])) i3++; for(i4=0;;i3++,i4++){ if(!IsVariableChar(basbuf[i3])){ temporary[i4]=0; break; } temporary[i4]=basbuf[i3]; } if(lstrcmp(temporary, pobj_CompilingClass->pobj_InheritsClass->name)==0){ //基底クラスのコンストラクタを呼び出す cp=i3; for(i4=0;;cp++,i4++){ if(IsCommandDelimitation(basbuf[cp])){ temporary[i4]=0; break; } temporary[i4]=basbuf[cp]; } if(!(temporary[0]=='('&&temporary[lstrlen(temporary)-1]==')')){ SetError(1,NULL,cp); } RemoveStringPare(temporary); i3=pobj_CompilingClass->pobj_InheritsClass->ConstructorMemberSubIndex; Opcode_CallProc(temporary, pobj_CompilingClass->ppobj_Method[i3]->psi, 0, "", 0); } else{ //基底クラスのコンストラクタを暗黙的に呼び出す i3=pobj_CompilingClass->pobj_InheritsClass->ConstructorMemberSubIndex; Opcode_CallProc("", pobj_CompilingClass->ppobj_Method[i3]->psi, 0, "", 0); } } //実体クラスを持つメンバのコンストラクタ(引数有りを除く)を呼び出す int offset; int MemberTypeSize; int MemberObjectNum; if(pobj_CompilingClass->vtbl_num) offset=PTR_SIZE; else offset=0; for(i3=0;i3iMemberNum;i3++){ CMember *pMember = pobj_CompilingClass->ppobj_Member[i3]; offset=GetSizeOfClassMember(pobj_CompilingClass,pMember->name,NULL); if(pMember->TypeInfo.type==DEF_OBJECT){ if(pMember->TypeInfo.u.pobj_Class->ConstructorMemberSubIndex!=-1){ i5=pMember->TypeInfo.u.pobj_Class->ConstructorMemberSubIndex; if(pMember->TypeInfo.u.pobj_Class->ppobj_Method[i5]->psi->RealParmNum==1){ //Thisポインタを第一引数に持ち合わせるため、引数なしの場合はこの値が1になる MemberTypeSize= GetTypeSize(pMember->TypeInfo.type, pMember->TypeInfo.u.lpIndex); MemberObjectNum= JumpSubScripts(pMember->SubScripts); for(i4=0;i4TypeInfo.u.pobj_Class->ppobj_Method[i5]->psi); } } } } } //仮想関数テーブルを初期化 if(pobj_CompilingClass->vtbl_num&& pobj_CompilingClass->IsAbstract()==false){ //関数テーブルに値をセット offset=pobj_CompilingClass->GetVtblGlobalOffset(); //mov eax,offset OpBuffer[obp++]=(char)0xB8; *((long *)(OpBuffer+obp))=offset; pobj_DataTableSchedule->add(); obp+=sizeof(long); //Thisポインタをecxにコピー SetThisPtrToReg(REG_ECX); //mov dword ptr[ecx],eax OpBuffer[obp++]=(char)0x89; OpBuffer[obp++]=(char)0x01; } } else if(psi->name[0]=='~'){ //デストラクタをコンパイルしたとき //デストラクタのコンパイル開始を通知 pobj_CompilingClass->NotifyStartDestructorCompile(); } } ////////////////////////////////////////// ////////////////////////////////////////// ////// プロシージャ内をコンパイル //////// if(psi->dwType==SUBTYPE_SUB) CompileBuffer(ESC_ENDSUB,0); else if(psi->dwType==SUBTYPE_FUNCTION) CompileBuffer(ESC_ENDFUNCTION,0); else if(psi->dwType==SUBTYPE_MACRO) CompileBuffer(ESC_ENDMACRO,0); ////////////////////////////////////////// ////////////////////////////////////////// if( pobj_CompilingClass ){ if( pobj_CompilingClass->IsCompilingConstructor() ){ // コンストラクタをコンパイルしていたとき // コンストラクタのコンパイルが完了したことを通知 pobj_CompilingClass->NotifyFinishConstructorCompile(); } else if(psi->name[0]=='~'){ //////////////////////////////////// //デストラクタをコンパイルしたとき //////////////////////////////////// // デストラクタのコンパイルが完了したことを通知 pobj_CompilingClass->NotifyFinishDestructorCompile(); if(pobj_CompilingClass->pobj_InheritsClass){ /* サブクラスのデストラクタをコンパイルしているときは、 基底クラスのデストラクタを呼び出す */ i3=pobj_CompilingClass->pobj_InheritsClass->DestructorMemberSubIndex; if(i3!=-1){ Opcode_CallProc("", pobj_CompilingClass->ppobj_Method[i3]->psi, 0, "", 0); } } //実体クラスを持つメンバのデストラクタを呼び出す //※コンストラクタと逆順序で呼び出す int offset; int MemberTypeSize; int MemberObjectNum; offset=GetTypeSize(DEF_OBJECT,(LONG_PTR)pobj_CompilingClass); for(i3=pobj_CompilingClass->iMemberNum-1;i3>=0;i3--){ CMember *pMember = pobj_CompilingClass->ppobj_Member[i3]; MemberTypeSize= GetTypeSize(pMember->TypeInfo.type, pMember->TypeInfo.u.lpIndex); MemberObjectNum= JumpSubScripts(pMember->SubScripts); offset-=MemberTypeSize*MemberObjectNum; if(pMember->TypeInfo.type==DEF_OBJECT){ if(pMember->TypeInfo.u.pobj_Class->DestructorMemberSubIndex!=-1){ for(i4=MemberObjectNum-1;i4>=0;i4--){ //Thisポインタをecxにコピー SetThisPtrToReg(REG_ECX); //add ecx,offset OpBuffer[obp++]=(char)0x81; OpBuffer[obp++]=(char)0xC1; *((long *)(OpBuffer+obp))=offset+i4*MemberTypeSize; obp+=sizeof(long); //push ecx op_push(REG_ECX); //call destructor i5=pMember->TypeInfo.u.pobj_Class->DestructorMemberSubIndex; op_call(pMember->TypeInfo.u.pobj_Class->ppobj_Method[i5]->psi); } } } } } } //ラベル用のメモリを解放 for(i3=0;i3ReturnType!=-1){ //戻り値をeax、edxに設定 RELATIVE_VAR RelativeVar; LONG_PTR lpIndex; char *temp; if(psi->name[0]==1&&psi->name[1]==ESC_OPERATOR) temp="_System_ReturnValue"; else temp=psi->name; GetVarOffsetReadWrite(temp,&i3,&RelativeVar,&lpIndex); i3=psi->ReturnType; if(i3==DEF_OBJECT){ SetVarPtrToEax(&RelativeVar); } else if(i3==DEF_DOUBLE){ //fld qword ptr[ebp+offset] OpBuffer[obp++]=(char)0xDD; OpBuffer[obp++]=(char)0x85; *((long *)(OpBuffer+obp))=RelativeVar.offset; AddLocalVarAddrSchedule(); obp+=sizeof(long); } else if(i3==DEF_SINGLE){ //fld dword ptr[ebp+offset] OpBuffer[obp++]=(char)0xD9; OpBuffer[obp++]=(char)0x85; *((long *)(OpBuffer+obp))=RelativeVar.offset; AddLocalVarAddrSchedule(); obp+=sizeof(long); } else if(i3==DEF_INT64||i3==DEF_QWORD){ //mov eax,dword ptr[ebp+offset] OpBuffer[obp++]=(char)0x8B; OpBuffer[obp++]=(char)0x85; *((long *)(OpBuffer+obp))=RelativeVar.offset; AddLocalVarAddrSchedule(); obp+=sizeof(long); //mov edx,dword ptr[ebp+offset+sizeof(long)] OpBuffer[obp++]=(char)0x8B; OpBuffer[obp++]=(char)0x95; *((long *)(OpBuffer+obp))=RelativeVar.offset+sizeof(long); AddLocalVarAddrSchedule(); obp+=sizeof(long); } else if(i3==DEF_LONG||i3==DEF_DWORD|| IsPtrType(i3)){ //mov eax,dword ptr[ebp+offset] OpBuffer[obp++]=(char)0x8B; OpBuffer[obp++]=(char)0x85; *((long *)(OpBuffer+obp))=RelativeVar.offset; AddLocalVarAddrSchedule(); obp+=sizeof(long); } else if(i3==DEF_INTEGER||i3==DEF_WORD){ //xor eax,eax(eaxを0に初期化する) op_zero_reg(REG_EAX); //mov ax,word ptr[ebp+offset] OpBuffer[obp++]=(char)0x66; OpBuffer[obp++]=(char)0x8B; OpBuffer[obp++]=(char)0x85; *((long *)(OpBuffer+obp))=RelativeVar.offset; AddLocalVarAddrSchedule(); obp+=sizeof(long); } else if(i3==DEF_CHAR||i3==DEF_BYTE||i3==DEF_BOOLEAN){ //xor eax,eax(eaxを0に初期化する) op_zero_reg(REG_EAX); //mov al,byte ptr[ebp+offset] OpBuffer[obp++]=(char)0x8A; OpBuffer[obp++]=(char)0x85; *((long *)(OpBuffer+obp))=RelativeVar.offset; AddLocalVarAddrSchedule(); obp+=sizeof(long); } } //ローカル変数アドレススケジュール for(i3=0;i3bCdecl){ //ret 0 OpBuffer[obp++]=(char)0xC3; } else{ //ret BaseOffset(パラメータ分のスタック領域を解放) OpBuffer[obp++]=(char)0xC2; *((_int16 *)(OpBuffer+obp))=(_int16)BaseOffset; obp+=sizeof(_int16); } psi->EndOpAddr=obp; psi->pVar=LocalVar; psi->VarNum=MaxLocalVarNum; //ローカル変数のネーム情報は後に解放する } void CompileLocal(){ extern SUBINFO **ppSubHash; int i2; SUBINFO *psi; extern BOOL bDll; if(bDll){ //DLLの場合はグローバル変数を初期化するための関数を一番初めにコンパイルする psi=GetSubHash("_System_InitDllGlobalVariables"); if(psi){ CompileBufferInProcedure(psi); } else SetError(300,NULL,cp); } //_System_InitStaticLocalVariablesは一番最後にコンパイル //※一般関数内の静的変数オブジェクトをすべて収集しなければならない extern SUBINFO *pSub_System_InitStaticLocalVariables; pSub_System_InitStaticLocalVariables->bCompile=1; //_System_Call_Destructor_of_GlobalObjectは一番最後にコンパイル extern SUBINFO *pSub_System_Call_Destructor_of_GlobalObject; pSub_System_Call_Destructor_of_GlobalObject->bCompile=1; Repeat: for(i2=0;i2pNextData; } } //プロシージャコンパイルによって、プロシージャコンパイルが必要になる場合 for(i2=0;i2bUse&&psi->bCompile==0){ goto Repeat; } psi=psi->pNextData; } } //_System_InitStaticLocalVariablesは一番最後にコンパイル pSub_System_InitStaticLocalVariables->bCompile=0; CompileBufferInProcedure(pSub_System_InitStaticLocalVariables); //_System_Call_Destructor_of_GlobalObjectは一番最後にコンパイル pSub_System_Call_Destructor_of_GlobalObject->bCompile=0; CompileBufferInProcedure(pSub_System_Call_Destructor_of_GlobalObject); }