| [206] | 1 | #include "stdafx.h" | 
|---|
|  | 2 |  | 
|---|
| [226] | 3 | #include <Compiler.h> | 
|---|
|  | 4 |  | 
|---|
| [3] | 5 | #include "../BasicCompiler_Common/common.h" | 
|---|
|  | 6 | #include "opcode.h" | 
|---|
|  | 7 |  | 
|---|
| [75] | 8 | void _call_constructor( const CClass *pobj_c,const char *CreateParameter,int ObjectSize,BOOL bSomeObjects){ | 
|---|
| [3] | 9 | //////////////////////////// | 
|---|
|  | 10 | // コンストラクタの呼び出し | 
|---|
|  | 11 | //////////////////////////// | 
|---|
|  | 12 |  | 
|---|
|  | 13 | //この関数を使用する場合は、 | 
|---|
|  | 14 | //・ebxにオブジェクトの個数(複数個の場合のみ) | 
|---|
|  | 15 | //・スタックフレームの先頭参照位置に先頭Thisポインタ | 
|---|
|  | 16 | //をセットしておかなければならない | 
|---|
|  | 17 |  | 
|---|
|  | 18 | int jnz_back; | 
|---|
|  | 19 |  | 
|---|
|  | 20 |  | 
|---|
|  | 21 | //jnzの番地 | 
|---|
|  | 22 | jnz_back=obp; | 
|---|
|  | 23 |  | 
|---|
|  | 24 | if(bSomeObjects){ | 
|---|
| [94] | 25 | SetError(); | 
|---|
| [3] | 26 | //mov qword ptr[rsp+offset],rbx     ※スタックフレームを利用 | 
|---|
|  | 27 | pobj_sf->push(REG_RBX); | 
|---|
|  | 28 |  | 
|---|
|  | 29 | // ※ここでプッシュされた値はコンストラクタのthisポインタとなる | 
|---|
|  | 30 | //mov qword ptr[rsp+offset],rax     ※スタックフレームを利用 | 
|---|
|  | 31 | pobj_sf->push(REG_RAX); | 
|---|
|  | 32 | } | 
|---|
|  | 33 |  | 
|---|
|  | 34 |  | 
|---|
|  | 35 | //////////////////////// | 
|---|
|  | 36 | // オーバーロードを解決 | 
|---|
|  | 37 | //////////////////////// | 
|---|
|  | 38 |  | 
|---|
| [206] | 39 | std::vector<const UserProc *> subs; | 
|---|
| [135] | 40 | pobj_c->GetMethods().Enum( pobj_c->GetName().c_str(), subs ); | 
|---|
| [50] | 41 |  | 
|---|
| [206] | 42 | const UserProc *pUserProc; | 
|---|
| [50] | 43 | if( subs.size() > 0 ){ | 
|---|
| [3] | 44 | //オーバーロードを解決 | 
|---|
| [131] | 45 | pUserProc=OverloadSolutionWithStrParam(pobj_c->GetName().c_str(), | 
|---|
| [75] | 46 | subs,CreateParameter,""); | 
|---|
| [3] | 47 |  | 
|---|
| [75] | 48 | if(!pUserProc) return; | 
|---|
| [3] | 49 | } | 
|---|
|  | 50 |  | 
|---|
|  | 51 | //コンストラクタを呼び出す | 
|---|
|  | 52 | Opcode_CallProc(CreateParameter, | 
|---|
| [75] | 53 | pUserProc, | 
|---|
| [3] | 54 | PROCFLAG_NEW,"",0); | 
|---|
|  | 55 |  | 
|---|
| [90] | 56 | { | 
|---|
|  | 57 | // 動的型情報をセットする | 
|---|
|  | 58 | // obj._System_SetType( _System_TypeBase.Search( strNamespace, name ) ) | 
|---|
|  | 59 | subs.clear(); | 
|---|
| [135] | 60 | pobj_c->GetMethods().Enum( "_System_SetType", subs ); | 
|---|
| [90] | 61 | if( subs.size() == 1 ){ | 
|---|
|  | 62 | char temporary[VN_SIZE]; | 
|---|
| [140] | 63 | sprintf( temporary, "ActiveBasic.Core._System_TypeBase.Search(\"\",\"%s\"))", pobj_c->GetName().c_str() ); | 
|---|
| [90] | 64 |  | 
|---|
|  | 65 | Opcode_CallProc(temporary, | 
|---|
|  | 66 | subs[0], | 
|---|
|  | 67 | PROCFLAG_NEW,"",0); | 
|---|
|  | 68 | } | 
|---|
|  | 69 | else{ | 
|---|
|  | 70 | SetError(); | 
|---|
|  | 71 | } | 
|---|
|  | 72 | } | 
|---|
|  | 73 |  | 
|---|
| [3] | 74 | if(bSomeObjects){ | 
|---|
|  | 75 | //mov rax,qword ptr[rsp+offset]     ※スタックフレームを利用 | 
|---|
|  | 76 | pobj_sf->pop(REG_RAX); | 
|---|
|  | 77 |  | 
|---|
|  | 78 | //mov rbx,qword ptr[rsp+offset]     ※スタックフレームを利用 | 
|---|
|  | 79 | pobj_sf->pop(REG_RBX); | 
|---|
|  | 80 |  | 
|---|
|  | 81 | //add eax,TypeSize | 
|---|
|  | 82 | OpBuffer[obp++]=(char)0x05; | 
|---|
|  | 83 | *((long *)(OpBuffer+obp))=ObjectSize; | 
|---|
|  | 84 | obp+=sizeof(long); | 
|---|
|  | 85 |  | 
|---|
|  | 86 | //sub ebx,1 | 
|---|
|  | 87 | OpBuffer[obp++]=(char)0x83; | 
|---|
|  | 88 | OpBuffer[obp++]=(char)0xEB; | 
|---|
|  | 89 | OpBuffer[obp++]=(char)0x01; | 
|---|
|  | 90 |  | 
|---|
|  | 91 | //jnz ↑ | 
|---|
|  | 92 | OpBuffer[obp++]=(char)0x0F; | 
|---|
|  | 93 | OpBuffer[obp++]=(char)0x85; | 
|---|
|  | 94 | *((long *)(OpBuffer+obp))=jnz_back-(obp+sizeof(long)); | 
|---|
|  | 95 | obp+=sizeof(long); | 
|---|
|  | 96 | } | 
|---|
|  | 97 | } | 
|---|
| [75] | 98 | void Operator_New( const CClass &classObj, const char *objectSizeStr, const char *parameter, const Type &baseType ){ | 
|---|
| [64] | 99 | int typeSize = classObj.GetSize(); | 
|---|
| [3] | 100 |  | 
|---|
| [64] | 101 | if(classObj.IsAbstract()){ | 
|---|
| [3] | 102 | //抽象クラスだったとき | 
|---|
| [131] | 103 | SetError(125,classObj.GetName().c_str(),cp); | 
|---|
| [3] | 104 | } | 
|---|
|  | 105 |  | 
|---|
|  | 106 | BOOL bSomeObjects=0; | 
|---|
| [64] | 107 | if(objectSizeStr[0]){ | 
|---|
| [3] | 108 | bSomeObjects=1; | 
|---|
|  | 109 |  | 
|---|
| [64] | 110 | int reg=REG_RAX; | 
|---|
| [75] | 111 | Type tempType; | 
|---|
|  | 112 | NumOpe(®,objectSizeStr,Type(),tempType); | 
|---|
|  | 113 | if( !tempType.IsWhole() ) SetError(49,NULL,cp); | 
|---|
| [3] | 114 |  | 
|---|
|  | 115 | //※添え字上限値であることを考慮 | 
|---|
|  | 116 | //add rax,1 | 
|---|
| [226] | 117 | compiler.codeGenerator.op_add_RV(REG_RAX,1); | 
|---|
| [3] | 118 |  | 
|---|
|  | 119 | //オブジェクトの個数をrbxに一時保持 | 
|---|
|  | 120 | //※rbxは関数が呼ばれても不変 | 
|---|
|  | 121 | //mov rbx,rax | 
|---|
| [226] | 122 | compiler.codeGenerator.op_mov_RR(REG_RBX,REG_RAX); | 
|---|
| [3] | 123 |  | 
|---|
|  | 124 | //imul rax,size | 
|---|
| [226] | 125 | compiler.codeGenerator.op_imul_RV(sizeof(_int64),REG_RAX,typeSize); | 
|---|
| [3] | 126 |  | 
|---|
| [123] | 127 | //add rax,OBJECT_HEAD_SIZE | 
|---|
| [226] | 128 | compiler.codeGenerator.op_add_RV(REG_RAX,OBJECT_HEAD_SIZE); | 
|---|
| [3] | 129 |  | 
|---|
|  | 130 | //mov rcx,rax | 
|---|
| [226] | 131 | compiler.codeGenerator.op_mov_RR(REG_RCX,REG_RAX); | 
|---|
| [3] | 132 | } | 
|---|
|  | 133 | else{ | 
|---|
|  | 134 | //オブジェクトの個数をrbxに一時保持 | 
|---|
|  | 135 | //※rbxは関数が呼ばれても不変 | 
|---|
|  | 136 | //mov rbx,1 | 
|---|
| [226] | 137 | compiler.codeGenerator.op_mov_RV(sizeof(_int64),REG_RBX,1); | 
|---|
| [3] | 138 |  | 
|---|
| [123] | 139 | //mov rcx,typeSize+OBJECT_HEAD_SIZE | 
|---|
| [226] | 140 | compiler.codeGenerator.op_mov_RV(sizeof(_int64),REG_RCX,typeSize+OBJECT_HEAD_SIZE); | 
|---|
| [3] | 141 | } | 
|---|
|  | 142 |  | 
|---|
| [75] | 143 | if( baseType.IsObject() ){ | 
|---|
|  | 144 | // オブジェクト インスタンス | 
|---|
|  | 145 | // ※DeleteはGCで処理 | 
|---|
| [3] | 146 |  | 
|---|
| [64] | 147 | //call _System_GC_malloc_ForObject | 
|---|
| [206] | 148 | extern const UserProc *pSub_System_GC_malloc_ForObject; | 
|---|
| [226] | 149 | compiler.codeGenerator.op_call(pSub_System_GC_malloc_ForObject); | 
|---|
| [64] | 150 | } | 
|---|
|  | 151 | else{ | 
|---|
| [75] | 152 | // オブジェクトポインタ | 
|---|
|  | 153 | // ※明示的なDeleteが必要 | 
|---|
| [3] | 154 |  | 
|---|
| [64] | 155 | //call _System_GC_malloc_ForObjectPtr | 
|---|
| [206] | 156 | extern const UserProc *pSub_System_GC_malloc_ForObjectPtr; | 
|---|
| [226] | 157 | compiler.codeGenerator.op_call(pSub_System_GC_malloc_ForObjectPtr); | 
|---|
| [64] | 158 | } | 
|---|
|  | 159 |  | 
|---|
|  | 160 |  | 
|---|
| [3] | 161 | /* | 
|---|
|  | 162 | 確保されたヒープ領域のポインタ(callocの戻り値eax)をpPtrとすると、 | 
|---|
| [123] | 163 | pPtr-=OBJECT_HEAD_SIZE ... ( sizeof(DWORD)*4 ) | 
|---|
| [3] | 164 | pPtr[0]=オブジェクトの個数 | 
|---|
|  | 165 | pPtr[1]=オブジェクトのサイズ | 
|---|
|  | 166 | pPtr[2]=デストラクタの関数ポインタ | 
|---|
| [123] | 167 | pPtr[3]=reserve | 
|---|
| [3] | 168 | */ | 
|---|
|  | 169 |  | 
|---|
|  | 170 |  | 
|---|
|  | 171 | //mov qword ptr[rax],rbx(オブジェクトの個数) | 
|---|
| [226] | 172 | compiler.codeGenerator.op_mov_MR(sizeof(_int64),REG_RBX,REG_RAX,0,MOD_BASE); | 
|---|
| [3] | 173 |  | 
|---|
|  | 174 | //add rax,PTR_SIZE | 
|---|
| [226] | 175 | compiler.codeGenerator.op_add_RV(REG_RAX,PTR_SIZE); | 
|---|
| [3] | 176 |  | 
|---|
|  | 177 |  | 
|---|
| [64] | 178 | //mov qword ptr[rax],typeSize(オブジェクトのサイズ) | 
|---|
| [226] | 179 | compiler.codeGenerator.op_mov_MV(sizeof(_int64),REG_RAX,0,NON_OFFSET,typeSize); | 
|---|
| [3] | 180 |  | 
|---|
|  | 181 | //add rax,PTR_SIZE | 
|---|
| [226] | 182 | compiler.codeGenerator.op_add_RV(REG_RAX,PTR_SIZE); | 
|---|
| [3] | 183 |  | 
|---|
|  | 184 |  | 
|---|
| [135] | 185 | const CMethod *method = classObj.GetDestructorMethod(); | 
|---|
| [64] | 186 | if( method == NULL ) return; | 
|---|
| [3] | 187 |  | 
|---|
| [123] | 188 | //mov rcx,DestructorProcAddr(デストラクタの関数ポインタ) | 
|---|
| [226] | 189 | compiler.codeGenerator.op_mov_RV(sizeof(_int64),REG_RCX,0); | 
|---|
| [3] | 190 | obp-=sizeof(long); | 
|---|
| [206] | 191 | pobj_SubAddrSchedule->add(&method->GetUserProc(),0); | 
|---|
|  | 192 | method->GetUserProc().Using(); | 
|---|
| [3] | 193 | obp+=sizeof(long); | 
|---|
|  | 194 |  | 
|---|
|  | 195 | //mov qword ptr[rax],rcx | 
|---|
| [226] | 196 | compiler.codeGenerator.op_mov_MR(sizeof(_int64),REG_RCX,REG_RAX,0,MOD_BASE); | 
|---|
| [3] | 197 |  | 
|---|
|  | 198 | //add rax,PTR_SIZE | 
|---|
| [226] | 199 | compiler.codeGenerator.op_add_RV(REG_RAX,PTR_SIZE); | 
|---|
| [3] | 200 |  | 
|---|
|  | 201 |  | 
|---|
| [123] | 202 | // リザーブ領域 | 
|---|
|  | 203 | //add rax,PTR_SIZE | 
|---|
| [226] | 204 | compiler.codeGenerator.op_add_RV(REG_RAX,PTR_SIZE); | 
|---|
| [123] | 205 |  | 
|---|
|  | 206 |  | 
|---|
| [3] | 207 | // ※ここでプッシュされた値はNew演算子の戻り値となる | 
|---|
|  | 208 | //mov qword ptr[rsp+offset],rax     ※スタックフレームを利用 | 
|---|
|  | 209 | pobj_sf->push(REG_RAX); | 
|---|
|  | 210 |  | 
|---|
|  | 211 |  | 
|---|
|  | 212 | ///////////////////////////////////////////////////////////////////// | 
|---|
|  | 213 |  | 
|---|
|  | 214 | //////////////////////////// | 
|---|
|  | 215 | // コンストラクタの呼び出し | 
|---|
|  | 216 | //////////////////////////// | 
|---|
|  | 217 |  | 
|---|
| [64] | 218 | _call_constructor(&classObj,parameter,typeSize,bSomeObjects); | 
|---|
| [3] | 219 |  | 
|---|
|  | 220 |  | 
|---|
|  | 221 | //mov rax,qword ptr[rsp+offset]     ※スタックフレームを利用 | 
|---|
|  | 222 | pobj_sf->pop(REG_RAX); | 
|---|
| [64] | 223 | } | 
|---|
|  | 224 | void OpcodeDelete(const char *Parameter, bool isSweeping){ | 
|---|
| [3] | 225 | int reg=REG_RAX; | 
|---|
| [75] | 226 | Type tempType; | 
|---|
|  | 227 | if( !NumOpe(®,Parameter,Type(),tempType) ){ | 
|---|
|  | 228 | return; | 
|---|
|  | 229 | } | 
|---|
|  | 230 | if(!( tempType.IsObjectPtr() || tempType.IsVoidPtr() )) SetError(122,NULL,cp); | 
|---|
| [3] | 231 |  | 
|---|
| [123] | 232 | //sub rax,OBJECT_HEAD_SIZE | 
|---|
| [226] | 233 | compiler.codeGenerator.op_sub_RV(sizeof(_int64),REG_RAX,OBJECT_HEAD_SIZE); | 
|---|
| [3] | 234 |  | 
|---|
|  | 235 | //mov qword ptr[rsp+offset],rax     ※スタックフレームを利用 | 
|---|
|  | 236 | pobj_sf->push(REG_RAX); | 
|---|
|  | 237 |  | 
|---|
|  | 238 |  | 
|---|
|  | 239 | //mov rbx,qword ptr[rax](オブジェクトの個数) | 
|---|
| [226] | 240 | compiler.codeGenerator.op_mov_RM(sizeof(_int64),REG_RBX,REG_RAX,0,MOD_BASE); | 
|---|
| [3] | 241 |  | 
|---|
|  | 242 | //add rax,PTR_SIZE | 
|---|
| [226] | 243 | compiler.codeGenerator.op_add_RV(REG_RAX,PTR_SIZE); | 
|---|
| [3] | 244 |  | 
|---|
|  | 245 |  | 
|---|
|  | 246 | //mov rsi,qword ptr[rax](オブジェクトのサイズ) | 
|---|
| [226] | 247 | compiler.codeGenerator.op_mov_RM(sizeof(_int64),REG_RSI,REG_RAX,0,MOD_BASE); | 
|---|
| [3] | 248 |  | 
|---|
|  | 249 | //add rax,PTR_SIZE | 
|---|
| [226] | 250 | compiler.codeGenerator.op_add_RV(REG_RAX,PTR_SIZE); | 
|---|
| [3] | 251 |  | 
|---|
|  | 252 |  | 
|---|
|  | 253 | //mov rdi,qword ptr[rax](デストラクタの関数ポインタ) | 
|---|
| [226] | 254 | compiler.codeGenerator.op_mov_RM(sizeof(_int64),REG_RDI,REG_RAX,0,MOD_BASE); | 
|---|
| [3] | 255 |  | 
|---|
|  | 256 | //add rax,PTR_SIZE | 
|---|
| [226] | 257 | compiler.codeGenerator.op_add_RV(REG_RAX,PTR_SIZE); | 
|---|
| [3] | 258 |  | 
|---|
|  | 259 |  | 
|---|
| [123] | 260 | // リザーブ領域 | 
|---|
|  | 261 | //add rax,PTR_SIZE | 
|---|
| [226] | 262 | compiler.codeGenerator.op_add_RV(REG_RAX,PTR_SIZE); | 
|---|
| [123] | 263 |  | 
|---|
|  | 264 |  | 
|---|
| [3] | 265 | //mov rcx,rax | 
|---|
| [226] | 266 | compiler.codeGenerator.op_mov_RR(REG_RCX,REG_RAX); | 
|---|
| [3] | 267 |  | 
|---|
|  | 268 |  | 
|---|
|  | 269 | //jnzの番地 | 
|---|
|  | 270 | int jnz_back; | 
|---|
|  | 271 | jnz_back=obp; | 
|---|
|  | 272 |  | 
|---|
| [64] | 273 | //mov qword ptr[rsp+offset],rcx     ※スタックフレームを利用 | 
|---|
|  | 274 | pobj_sf->push(REG_RCX); | 
|---|
| [3] | 275 |  | 
|---|
|  | 276 | //call rdi | 
|---|
|  | 277 | OpBuffer[obp++]=(char)0xFF; | 
|---|
|  | 278 | OpBuffer[obp++]=(char)0xD7; | 
|---|
|  | 279 |  | 
|---|
|  | 280 | //mov rcx,qword ptr[rsp+offset]     ※スタックフレームを利用 | 
|---|
|  | 281 | pobj_sf->pop(REG_RCX); | 
|---|
|  | 282 |  | 
|---|
|  | 283 | //add rcx,rsi | 
|---|
| [226] | 284 | compiler.codeGenerator.op_add64_reg(REG_RCX,REG_RSI); | 
|---|
| [3] | 285 |  | 
|---|
|  | 286 | //sub rbx,1 | 
|---|
| [226] | 287 | compiler.codeGenerator.op_sub_RV(sizeof(_int64),REG_RBX,1); | 
|---|
| [3] | 288 |  | 
|---|
|  | 289 | //jnz ↑ | 
|---|
|  | 290 | OpBuffer[obp++]=(char)0x0F; | 
|---|
|  | 291 | OpBuffer[obp++]=(char)0x85; | 
|---|
|  | 292 | *((long *)(OpBuffer+obp))=jnz_back-(obp+sizeof(long)); | 
|---|
|  | 293 | obp+=sizeof(long); | 
|---|
|  | 294 |  | 
|---|
|  | 295 |  | 
|---|
|  | 296 | ////////////////////////////////////////// | 
|---|
|  | 297 | // オブジェクトメンバ変数用のメモリを解放 | 
|---|
|  | 298 | ////////////////////////////////////////// | 
|---|
|  | 299 |  | 
|---|
|  | 300 | //mov rcx,qword ptr[rsp+offset]     ※スタックフレームを利用 | 
|---|
|  | 301 | pobj_sf->pop(REG_RCX); | 
|---|
|  | 302 |  | 
|---|
| [64] | 303 | if( isSweeping ){ | 
|---|
|  | 304 | //call _System_GC_free_for_SweepingDelete | 
|---|
| [206] | 305 | extern const UserProc *pSub_System_GC_free_for_SweepingDelete; | 
|---|
| [226] | 306 | compiler.codeGenerator.op_call(pSub_System_GC_free_for_SweepingDelete); | 
|---|
| [64] | 307 | } | 
|---|
|  | 308 | else{ | 
|---|
|  | 309 | //call free | 
|---|
| [206] | 310 | extern const UserProc *pSub_free; | 
|---|
| [226] | 311 | compiler.codeGenerator.op_call(pSub_free); | 
|---|
| [64] | 312 | } | 
|---|
| [3] | 313 | } | 
|---|