asm { //************************************ //See ::/Doc/Credits.DD. _MALLOC:: // Throws 'OutMem' PUSH RBP MOV RBP, RSP PUSH RSI PUSH RDI XOR RBX, RBX MOV RDX, U64 SF_ARG2[RBP] TEST RDX, RDX JNZ @@05 MOV RDX, U64 FS:CTask.addr[RBX] @@05: CMP U32 CTask.task_signature[RDX], TASK_SIGNATURE_VAL #assert CTask.task_signature == CHeapCtrl.hc_signature //location signature same JNE @@10 MOV RDX, U64 CTask.data_heap[RDX] @@10: CMP U32 CHeapCtrl.hc_signature[RDX], HEAP_CTRL_SIGNATURE_VAL JE @@15 PUSH RDX CALL &SysBadMAlloc JMP I32 _SYS_HLT @@15: MOV RAX, U64 SF_ARG1[RBP] PUSHFD ADD RAX, CMemUsed.start + 7 //round-up to I64 AND AL, 0xF8 #assert CMemUsed.start >= sizeof(CMemUnused) CMP RAX, CMemUsed.start JAE @@20 MOV RAX, CMemUsed.start @@20: CLI @@25: LOCK BTS U32 CHeapCtrl.locked_flags[RDX], HClf_LOCKED PAUSE //don't know if this instruction helps JC @@25 CMP RAX, MEM_HEAP_HASH_SIZE JAE @@30 MOV RSI, U64 CHeapCtrl.heap_hash[RAX + RDX] TEST RSI, RSI JZ @@35 MOV RCX, U64 CMemUnused.next[RSI] MOV U64 CHeapCtrl.heap_hash[RAX + RDX], RCX JMP I32 MALLOC_ALMOST_DONE //Big allocation @@30: ADD RAX, sizeof(CMemBlk) + MEM_PAG_SIZE - 1 SHR RAX, MEM_PAG_BITS PUSH RDX //preserve HeapCtrl PUSH RDX PUSH RAX CALL &MemPagTaskAlloc POP RDX TEST RAX, RAX JZ @@45 //Out of memory MOV RSI, RAX MOV EAX, U32 CMemBlk.pags[RSI] SHL RAX, MEM_PAG_BITS SUB RAX, sizeof(CMemBlk) ADD RSI, sizeof(CMemBlk) JMP I32 MALLOC_ALMOST_DONE //Little allocation, chunk-off piece from free list chunks @@35: LEA RSI, U64 CHeapCtrl.malloc_free_list - CMemUnused.next[RDX] @@40: MOV RBX, RSI MOV RSI, U64 CMemUnused.next[RBX] TEST RSI, RSI JNZ I32 @@60 PUSH RAX //-**** save byte size ADD RAX, 16 * MEM_PAG_SIZE - 1 SHR RAX, MEM_PAG_BITS PUSH RDX //preserve HeapCtrl PUSH RDX PUSH RAX CALL &MemPagTaskAlloc POP RDX TEST RAX, RAX JNZ @@50 //Out of memory @@45: LOCK BTR U32 CHeapCtrl.locked_flags[RDX], HClf_LOCKED POPFD PUSH TRUE MOV RAX, 'OutMem' PUSH RAX CALL I32 &throw JMP I32 MALLOC_FINAL_EXIT //Never gets here, hopefully. @@50: MOV RSI, RAX MOV EAX, U32 CMemBlk.pags[RSI] SHL RAX, MEM_PAG_BITS //Can it be combined with last chunk? (Never Free these chunks.) MOV RDI, U64 CHeapCtrl.last_mergable[RDX] LEA RBX, U64 [RSI + RAX] CMP RDI, RBX JNE @@55 PUSH RAX MOV EAX, U32 CMemBlk.pags[RDI] ADD U32 CMemBlk.pags[RSI],EAX //QueueRemove MOV RAX, U64 CMemBlk.next[RDI] MOV RBX, U64 CMemBlk.last[RDI] MOV U64 CMemBlk.last[RAX], RBX MOV U64 CMemBlk.next[RBX], RAX POP RAX @@55: MOV U64 CHeapCtrl.last_mergable[RDX], RSI LEA RSI, U64 sizeof(CMemBlk)[RSI] SUB RAX, sizeof(CMemBlk) LEA RBX, U64 CHeapCtrl.malloc_free_list - CMemUnused.next[RDX] MOV RDI, U64 CMemUnused.next[RBX] MOV U64 CMemUnused.next[RSI], RDI MOV U64 CMemUnused.size[RSI], RAX MOV U64 CMemUnused.next[RBX], RSI POP RAX //+**** JMP @@70 @@60: CMP U64 CMemUnused.size[RSI], RAX JB I32 @@40 JNE @@70 @@65: MOV RDI, U64 CMemUnused.next[RSI] MOV U64 CMemUnused.next[RBX], RDI JMP MALLOC_ALMOST_DONE @@70: SUB U64 CMemUnused.size[RSI], RAX //UPDATE FREE ENTRY CMP U64 CMemUnused.size[RSI], sizeof(CMemUnused) JAE @@75 //take from top of block ADD U64 CMemUnused.size[RSI], RAX //doesn't fit, undo JMP I32 @@40 @@75: ADD RSI, U64 CMemUnused.size[RSI] MALLOC_ALMOST_DONE: //RSI = res - CMemUsed.size //RAX = size + CMemUsed.size //RDX = HeapCtrl ADD U64 CHeapCtrl.used_u8s[RDX], RAX #if _CONFIG_HEAP_DEBUG //QueueInsert MOV RDI, U64 CHeapCtrl.last_um[RDX] MOV U64 CMemUsed.next[RDI], RSI MOV U64 CHeapCtrl.last_um[RDX], RSI MOV U64 CMemUsed.last[RSI], RDI LEA RDI, U64 CHeapCtrl.next_um - CMemUsed.next[RDX] MOV U64 CMemUsed.next[RSI], RDI //Caller1/Caller2 PUSH RDX MOV RDX, U64 [MEM_HEAP_LIMIT] MOV RDI, U64 SF_RIP[RBP] CMP RDI, RDX JB @@80 XOR RDI, RDI MOV U64 CMemUsed.caller1[RSI], RDI JMP @@90 @@80: MOV U64 CMemUsed.caller1[RSI], RDI MOV RDI, U64 SF_RBP[RBP] CMP RDI, RDX JB @@85 XOR RDI, RDI JMP @@90 @@85: MOV RDI, U64 SF_RIP[RDI] CMP RDI, RDX JB @@90 XOR RDI, RDI @@90: MOV U64 CMemUsed.caller2[RSI], RDI POP RDX #endif LOCK BTR U32 CHeapCtrl.locked_flags[RDX], HClf_LOCKED POPFD MOV U64 CMemUsed.size[RSI], RAX MOV U64 CMemUsed.hc[RSI], RDX LEA RAX, U64 CMemUsed.start[RSI] TEST U8 [SYS_SEMAS + SEMA_HEAPLOG_ACTIVE * DEFAULT_CACHE_LINE_WIDTH], 1 JZ @@105 PUSH RAX PUSH RAX MOV RAX, U64 [SYS_EXTERN_TABLE] MOV RAX, U64 EXT_HEAPLOG_MALLOC*8[RAX] TEST RAX, RAX JZ @@95 CALL RAX JMP @@100 @@95: ADD RSP, 8 @@100: POP RAX @@105: TEST U8 [SYS_HEAP_INIT_FLAG], 1 JZ MALLOC_FINAL_EXIT PUSH RAX MOV RCX, U64 CMemUsed.size - CMemUsed.start[RAX] SUB RCX, CMemUsed.start MOV RDI, RAX MOV AL, U8 [SYS_HEAP_INIT_VAL] REP_STOSB POP RAX MALLOC_FINAL_EXIT: POP RDI POP RSI POP RBP RET1 16 //************************************ _FREE:: //Be aware of heap_hash in MemPagTaskAlloc(). PUSH RBP MOV RBP, RSP PUSH RSI PUSH RDI TEST U8 [SYS_SEMAS + SEMA_HEAPLOG_ACTIVE * DEFAULT_CACHE_LINE_WIDTH], 1 JZ @@15 MOV RBX, U64 SF_ARG1[RBP] TEST RBX, RBX JZ @@05 MOV RAX, U64 CMemUsed.size - CMemUsed.start[RBX] TEST RAX, RAX JGE @@05 //Aligned alloced chunks have neg size ADD RBX, RAX @@05: PUSH RBX MOV RAX, U64 [SYS_EXTERN_TABLE] MOV RAX, U64 EXT_HEAPLOG_FREE*8[RAX] TEST RAX, RAX JZ @@10 CALL RAX JMP @@15 @@10: ADD RSP, 8 @@15: MOV RSI, U64 SF_ARG1[RBP] TEST RSI, RSI #if _CONFIG_HEAP_DEBUG JZ I32 FREE_DONE #else JZ FREE_DONE #endif MOV RAX, U64 CMemUsed.size - CMemUsed.start[RSI] TEST RAX, RAX JGE @@20 //Aligned alloced chunks have neg size. //The neg size is offset to start of CMemUsed struct. ADD RSI, RAX @@20: PUSHFD SUB RSI, CMemUsed.start MOV RDX, U64 CMemUsed.hc[RSI] CMP U32 CHeapCtrl.hc_signature[RDX], HEAP_CTRL_SIGNATURE_VAL JE @@25 ADD RSI, CMemUsed.start PUSH RSI CALL &SysBadFree JMP I32 _SYS_HLT @@25: MOV RAX, U64 CMemUsed.size[RSI] SUB U64 CHeapCtrl.used_u8s[RDX], RAX CLI @@30: LOCK BTS U32 CHeapCtrl.locked_flags[RDX], HClf_LOCKED PAUSE JC @@30 #if _CONFIG_HEAP_DEBUG //QueueRemove MOV RDX, U64 CMemUsed.next[RSI] MOV RDI, U64 CMemUsed.last[RSI] MOV U64 CMemUsed.last[RDX], RDI MOV U64 CMemUsed.next[RDI], RDX //Caller1/Caller2 MOV RDX, U64 [MEM_HEAP_LIMIT] MOV RDI, U64 SF_RIP[RBP] CMP RDI, RDX JB @@35 XOR RDI, RDI MOV U64 CMemUnused.caller1[RSI], RDI JMP @@45 @@35: MOV U64 CMemUnused.caller1[RSI], RDI MOV RDI, U64 SF_RBP[RBP] CMP RDI, RDX JB @@40 XOR RDI, RDI JMP @@45 @@40: MOV RDI, U64 SF_RIP[RDI] CMP RDI, RDX JB @@45 XOR RDI, RDI @@45: MOV U64 CMemUnused.caller2[RSI], RDI MOV RDX, U64 CMemUsed.hc[RSI] #endif CMP RAX, MEM_HEAP_HASH_SIZE JAE @@50 #assert CMemUnused.size == CMemUsed.size // MOV U64 CMemUnused.size[RSI], RAX MOV RBX, U64 CHeapCtrl.heap_hash[RAX + RDX] MOV U64 CMemUnused.next[RSI], RBX MOV U64 CHeapCtrl.heap_hash[RAX + RDX], RSI JMP @@55 @@50: SUB RSI, sizeof(CMemBlk) PUSH RDX PUSH RDX PUSH RSI CALL &MemPagTaskFree POP RDX @@55: LOCK BTR U32 CHeapCtrl.locked_flags[RDX], HClf_LOCKED POPFD FREE_DONE: POP RDI POP RSI POP RBP RET1 8 //************************************ _MSIZE:: PUSH RBP MOV RBP, RSP MOV RBX, U64 SF_ARG1[RBP] XOR RAX, RAX TEST RBX, RBX JZ @@10 MOV RAX, U64 CMemUsed.size - CMemUsed.start[RBX] TEST RAX, RAX JGE @@05 //Aligned alloced chunks have neg size ADD RBX, RAX MOV RAX, U64 CMemUsed.size - CMemUsed.start[RBX] @@05: SUB RAX, CMemUsed.start @@10: POP RBP RET1 8 //************************************ _MSIZE2:: PUSH RBP MOV RBP, RSP MOV RBX, U64 SF_ARG1[RBP] XOR RAX, RAX TEST RBX, RBX JZ @@10 MOV RAX, U64 CMemUsed.size-CMemUsed.start[RBX] TEST RAX, RAX JGE @@05 //Aligned alloced chunks have neg size ADD RBX, RAX @@05: MOV RAX, U64 CMemUsed.size - CMemUsed.start[RBX] @@10: POP RBP RET1 8 //************************************ _MHEAP_CTRL:: PUSH RBP MOV RBP, RSP MOV RBX, U64 SF_ARG1[RBP] XOR RAX, RAX TEST RBX, RBX JZ @@10 MOV RAX, U64 CMemUsed.size-CMemUsed.start[RBX] TEST RAX, RAX JGE @@05 //Aligned alloced chunks have neg size ADD RBX, RAX @@05: MOV RAX, U64 CMemUsed.hc - CMemUsed.start[RBX] @@10: POP RBP RET1 8 } _extern _FREE U0 Free(U8 *addr); //Free MAlloc()ed memory chunk. _extern _MSIZE I64 MSize( U8 *src); //Size of heap object. _extern _MSIZE2 I64 MSize2( U8 *src); //Internal size of heap object. _extern _MHEAP_CTRL CHeapCtrl *MHeapCtrl(U8 *src); //CHeapCtrl of object. _extern _MALLOC U8 *MAlloc(I64 size, CTask *mem_task=NULL); //Alloc memory chunk. //Accepts a CTask or CHeapCtrl. NULL allocs off current task's heap. U8 *ZMAlloc(I64 size) {//Alloc memory in System task's heap. return MAlloc(size, sys_task); } U8 *CAlloc(I64 size, CTask *mem_task=NULL) {//Accepts a CTask or CHeapCtrl. NULL allocs off current task's heap. U8 *res = MAlloc(size, mem_task); MemSet(res, 0, size); return res; } U8 *ZCAlloc(I64 size) {//Alloc and set to zero memory in System task's heap. return CAlloc(size, sys_task); } U8 *MAllocIdent(U8 *src, CTask *mem_task=NULL) {//Accepts a CTask or CHeapCtrl. NULL allocs off current task's heap. U8 *res; I64 size; if (!src) return NULL; size = MSize(src); res = MAlloc(size, mem_task); MemCopy(res, src, size); return res; } U8 *ZMAllocIdent(U8 *src) {//Alloc in System task's heap, ident copy of heap node. return MAllocIdent(src, sys_task); } U8 *MAllocAligned(I64 size, I64 alignment, CTask *mem_task=NULL, I64 misalignment=0) {//Only powers of two alignment. This is awful. I64 mask = alignment - 1; U8 *ptr = MAlloc(size + mask + sizeof(I64) + misalignment, mem_task), *res = (ptr + sizeof(I64) + mask) & ~mask + misalignment; res(I64 *)[-1] = ptr - res; #assert offset(CMemUsed.size) == offset(CMemUsed.start) - sizeof(I64) return res; } U8 *CAllocAligned(I64 size, I64 alignment, CTask *mem_task=NULL, I64 misalignment=0) {//Only powers of two alignment. This is awful. I64 mask = alignment-1; U8 *ptr = MAlloc(size + mask + sizeof(I64) + misalignment, mem_task), *res = (ptr + sizeof(I64) + mask) & ~mask + misalignment; res(I64 *)[-1] = ptr - res; #assert offset(CMemUsed.size) == offset(CMemUsed.start) - sizeof(I64) MemSet(res, 0, size); return res; } U8 *ReAlloc(U8 *ptr, U64 new_size, CTask *mem_task=NULL) {//Resize previously MAlloc'ed chunk. If new_size is zero then act as Free. //If ptr is NULL then act as MAlloc. if both are NULL/0 does nothing (Free(NULL)) //Useless for changing chunk sizes smaller than 8 bytes because MAlloc allocs 8 bytes at a time. U8 *res; if (!new_size) { Free(ptr); //we can free NULL return NULL; } res = MAlloc(new_size, mem_task); if (!ptr) return res; MemCopy(res, ptr, MinI64(MSize(ptr), new_size)); Free(ptr); return res; } U8 *ZReAlloc(U8 *ptr, I64 new_size) {//Realloc in System task's heap. return ReAlloc(ptr, new_size, sys_task); } U8 *StrNew(U8 *buf, CTask *mem_task=NULL) {//Accepts a CTask or CHeapCtrl. NULL allocs off current task's heap. U8 *res; I64 size; if (buf) { size = StrLen(buf) + 1; res = MAlloc(size, mem_task); MemCopy(res, buf, size); } else { res = MAlloc(1, mem_task); *res = 0; } return res; } U8 *ZStrNew(U8 *buf) {//Alloc copy of string in System task's heap. return StrNew(buf, sys_task); }