#include "../BasicCompiler_Common/common.h" #include "Opcode.h" void SystemProc( const 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(); //クラスに属する静的メンバを定義 CMember::InitStaticMember(); 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;iIsUsing() == false || pUserProc->IsCompiled() ) return; pUserProc->CompleteCompile(); extern BOOL bSystemProc; if(memcmp(pUserProc->GetName().c_str(),"_System_",8)==0) bSystemProc=1; else bSystemProc=0; extern BOOL bDebugSupportProc; if(memcmp(pUserProc->GetName().c_str(),"_DebugSys_",10)==0){ if(!bDebugCompile){ pUserProc->pVar=(VARIABLE *)HeapAlloc(hHeap,0,1); pUserProc->VarNum=0; return; } bDebugSupportProc=1; } else bDebugSupportProc=0; pUserProc->beginOpAddress=obp; if(pUserProc->IsSystem()){ extern int MaxLocalVarNum; extern int AllLocalVarSize; MaxLocalVarNum=0; AllLocalVarSize=0; SystemProc(pUserProc->GetName().c_str()); pUserProc->endOpAddress=obp; pUserProc->pVar=(VARIABLE *)HeapAlloc(hHeap,0,1); pUserProc->VarNum=0; return; } cp=pUserProc->GetCodePos(); 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=(int)pUserProc->RealParams().size()-1;i3>=0;i3--){ Parameter ¶m = *pUserProc->RealParams()[i3]; //変数データを追加 LocalVar=(VARIABLE *)HeapReAlloc(hHeap,0,LocalVar,(MaxLocalVarNum+1)*sizeof(VARIABLE)); lstrcpy( LocalVar[MaxLocalVarNum].name,param.GetVarName().c_str() ); // TODO: パラメータのConst定義の指定が未完成 LocalVar[MaxLocalVarNum].bConst=false; LocalVar[MaxLocalVarNum].bArray=param.IsArray()?TRUE:FALSE; memcpy(LocalVar[MaxLocalVarNum].SubScripts,param.GetSubScriptsPtr(),MAX_ARRAYDIM*sizeof(int)); LocalVar[MaxLocalVarNum].type=param.GetBasicType(); if(LocalVar[MaxLocalVarNum].type==DEF_OBJECT ||LocalVar[MaxLocalVarNum].type==DEF_STRUCT){ if(pUserProc->IsExport()) SetError(24,NULL,cp); } LocalVar[MaxLocalVarNum].u.index=param.GetIndex(); if( param.IsRef() == false && param.IsStruct() ){ //構造体のByValパラメータ LocalVar[MaxLocalVarNum].fRef=REF_PARAMETER | OBJECT_PARAMETER; VarSize=PTR_SIZE; } else{ if( param.IsRef() == false && param.IsArray() == false ){ 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=pUserProc->GetParentClassPtr(); //コンパイルスタートをクラス管理クラスに追加 pobj_DBClass->StartCompile( pUserProc ); //コンパイル中の関数 extern UserProc *pCompilingUserProc; pCompilingUserProc=pUserProc; /////////////////////// // ここからコード生成 //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( !pUserProc->ReturnType().IsNull() ){ //戻り値が存在するとき const char *temp = pUserProc->GetName().c_str(); if( temp[0]==1&&temp[1]==ESC_OPERATOR ){ temp = "_System_ReturnValue"; } if( pUserProc->ReturnType().IsStruct() ){ //戻り値用の構造体(値型)はパラメータで引き渡される } else{ //戻り値用の変数の定義 sprintf(temporary,"%s%c%c%s",temp,1,ESC_AS, pUserProc->ReturnType().ToString().c_str() ); 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 UserProc *pSub_DebugSys_StartProc; op_call(pSub_DebugSys_StartProc); } if(pobj_CompilingClass){ if( pUserProc->GetName() == pobj_CompilingClass->name ){ //////////////////////////////////// // コンストラクタをコンパイルするとき //////////////////////////////////// //コンストラクタのコンパイル開始を通知 pobj_CompilingClass->NotifyStartConstructorCompile(); //基底クラスかどうかの識別 //(継承元がインターフェイスの場合も基底クラスと見なす) BOOL bThisIsSuperClass; if(pobj_CompilingClass->pobj_InheritsClass==0) bThisIsSuperClass=1; else if( pobj_CompilingClass->pobj_InheritsClass->GetConstructorMethod() == NULL ){ //インターフェイスを継承したときはコンストラクタを持たない 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); Opcode_CallProc(temporary, pobj_CompilingClass->pobj_InheritsClass->GetConstructorMethod()->pUserProc, 0, "", 0); } else{ //基底クラスのコンストラクタを暗黙的に呼び出す Opcode_CallProc("", pobj_CompilingClass->pobj_InheritsClass->GetConstructorMethod()->pUserProc, 0, "", 0); } } //実体クラスを持つメンバのコンストラクタ(引数有りを除く)を呼び出す for(i3=0;i3iMemberNum;i3++){ CMember *pMember = pobj_CompilingClass->ppobj_Member[i3]; if(pMember->TypeInfo.type==DEF_OBJECT){ // オブジェクトメンバを発見したとき sprintf(temporary, "This.%s=%c%c%s()", pMember->name, 1, ESC_NEW, pMember->TypeInfo.u.pobj_Class->name ); OpcodeCalc( temporary ); } } //仮想関数テーブルを初期化 if(pobj_CompilingClass->vtbl_num&& pobj_CompilingClass->IsAbstract()==false){ //関数テーブルに値をセット int offset = (int)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( pUserProc->IsDestructor() ){ //デストラクタをコンパイルしたとき //デストラクタのコンパイル開始を通知 pobj_CompilingClass->NotifyStartDestructorCompile(); } } ////////////////////////////////////////// ////////////////////////////////////////// ////// プロシージャ内をコンパイル //////// if(pUserProc->IsMacro()) CompileBuffer(ESC_ENDMACRO,0); else{ if(pUserProc->IsSub()) CompileBuffer(ESC_ENDSUB,0); else if(pUserProc->IsFunction()) CompileBuffer(ESC_ENDFUNCTION,0); } ////////////////////////////////////////// ////////////////////////////////////////// if( pobj_CompilingClass ){ if( pobj_CompilingClass->IsCompilingConstructor() ){ // コンストラクタをコンパイルしていたとき // コンストラクタのコンパイルが完了したことを通知 pobj_CompilingClass->NotifyFinishConstructorCompile(); } else if( pUserProc->IsDestructor() ){ //////////////////////////////////// //デストラクタをコンパイルしたとき //////////////////////////////////// // デストラクタのコンパイルが完了したことを通知 pobj_CompilingClass->NotifyFinishDestructorCompile(); if(pobj_CompilingClass->pobj_InheritsClass){ /* サブクラスのデストラクタをコンパイルしているときは、 基底クラスのデストラクタを呼び出す */ CMethod *method = pobj_CompilingClass->pobj_InheritsClass->GetDestructorMethod(); if( method ){ Opcode_CallProc("", method->pUserProc, 0, "", 0); } } //実体クラスを持つメンバのデストラクタ呼び出しはGCに任せる /* //※コンストラクタと逆順序で呼び出す 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){ CMethod *method = pMember->TypeInfo.u.pobj_Class->GetDestructorMethod(); if( method ){ 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 op_call( method->pUserProc ); } } } }*/ } } //ラベル用のメモリを解放 for(i3=0;i3ReturnType().IsNull() ){ //戻り値をeax、edxに設定 RELATIVE_VAR RelativeVar; LONG_PTR lpIndex; const char *temp = pUserProc->GetName().c_str(); if( temp[0]==1 && temp[1]==ESC_OPERATOR ){ temp="_System_ReturnValue"; } GetVarOffsetReadWrite(temp,&i3,&RelativeVar,&lpIndex); i3=pUserProc->ReturnType().GetBasicType(); if(i3==DEF_OBJECT || i3==DEF_STRUCT){ SetVarPtrToEax(&RelativeVar); if( i3==DEF_OBJECT ){ //mov eax,dword ptr[eax] op_mov_RM( sizeof(long), REG_EAX, REG_EAX, 0, MOD_BASE ); } } 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 || (isUnicode&&i3==DEF_CHAR)){ //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_SBYTE||i3==DEF_BYTE||i3==DEF_BOOLEAN || (isUnicode==false&&i3==DEF_CHAR)){ //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;i3IsCdecl() ){ //ret 0 OpBuffer[obp++]=(char)0xC3; } else{ //ret BaseOffset(パラメータ分のスタック領域を解放) OpBuffer[obp++]=(char)0xC2; *((_int16 *)(OpBuffer+obp))=(_int16)BaseOffset; obp+=sizeof(_int16); } pUserProc->endOpAddress=obp; pUserProc->pVar=LocalVar; pUserProc->VarNum=MaxLocalVarNum; //重複エラー情報管理のメモリを解放 for(i3=0;i3CompleteCompile(); //_System_Call_Destructor_of_GlobalObjectは一番最後にコンパイル extern UserProc *pSub_System_Call_Destructor_of_GlobalObject; pSub_System_Call_Destructor_of_GlobalObject->CompleteCompile(); Repeat: for(i2=0;i2pNextData; } } //プロシージャコンパイルによって、プロシージャコンパイルが必要になる場合 for(i2=0;i2IsUsing() && pUserProc->IsCompiled() == false ){ goto Repeat; } pUserProc=pUserProc->pNextData; } } //_System_InitStaticLocalVariablesは一番最後にコンパイル pSub_System_InitStaticLocalVariables->KillCompileStatus(); CompileBufferInProcedure(pSub_System_InitStaticLocalVariables); //_System_Call_Destructor_of_GlobalObjectは一番最後にコンパイル pSub_System_Call_Destructor_of_GlobalObject->KillCompileStatus(); CompileBufferInProcedure(pSub_System_Call_Destructor_of_GlobalObject); }