|
|
PAGE ,132 TITLE GALLOC - Global memory allocator
.sall .xlist include kernel.inc include protect.inc include wowcmpat.inc .list
.386
DataBegin
externB Kernel_flags externB fBooting externB fCheckFree externW pGlobalHeap externW Win386_Blocks externW FreeArenaCount gsearch_state_machine dw 0 gsearch_compact_first dw 0
public ffixedlow ffixedlow db 0
DataEnd
sBegin CODE assumes CS,CODE
externNP gcompact externNP gmovebusy externNP gslide externNP galign externNP genter externNP gleave ifdef WOW externFP MyGetAppWOWCompatFlagsEx endif externNP gwin386discard externNP GetDPMIFreeSpace externNP InnerShrinkHeap externNP get_physical_address externNP set_physical_address externNP alloc_arena_header externNP free_arena_header ifndef WOW_x86 externNP get_blotto endif externNP PreallocArena externNP DPMIProc
;-----------------------------------------------------------------------; ; gsearch ; ; ; ; Searches from start to finish for a free global object to allocate ; ; space from. For a fixed request it tries to get the space as low as ; ; possible, moving movable out of the way if neccessary. For movable ; ; requests it also starts at the bottom. For discardable code it ; ; starts from the top, only using the first free block it finds. ; ; If at first blush it can't find a block it compacts memory and tries ; ; again. ; ; When it finally finds a block it gsplices it in, makes sure the ; ; arena headers are fine, and returns the allocated block. ; ; Called from within the global memory manager's critical section. ; ; ; ; Arguments: ; ; AX = allocations flags ; ; EBX = #bytes ; ; CX = owner field value ; ; DS:DI = address of global arena information structure ; ; ; ; Returns: ; ; AX = data address of block allocated or NULL ; ; BX = ga_prev or ga_next ; ; DX = allocation flags ; ; ; ; Error Returns: ; ; ZF = 1 ; ; AX = 0 ; ; BX = ga_prev or ga_next ; ; DX = size of largest free block ; ; ; ; Registers Preserved: ; ; DI,DS ; ; ; ; Registers Destroyed: ; ; CX,SI,ES ; ; ; ; Calls: ; ; galign ; ; gfindfree ; ; fmovebusy ; ; gcheckfree ; ; gcompact ; ; gsplice ; ; gzero ; ; gmarkfree ; ; ; ; History: ; ; ; ; ; ; 19-Aug-95 davehart: Win 3.1 tries to grow the heap before ; ; compacting it, for WOW we want to compact first to be a good ; ; multitasking neighbor. ; ; ; ; Wed Jul 22, 1987 11:15:19p -by- David N. Weise [davidw] ; ; Fixed BOGUS BLOCK freeing yet again. ; ; ; ; Sun May 10, 1987 11:29:38p -by- David N. Weise [davidw] ; ; Added the state machine to handle the case of wanting to search ; ; both global arenas. ; ; ; ; Sat Feb 28, 1987 06:31:11p -by- David N. Weise [davidw] ; ; Putting in support for allocating discardable code from EEMS land. ; ; ; ; Tue Dec 30, 1986 01:54:50p -by- David N. Weise [davidw] ; ; Made sure it freed any bogus blocks created. ; ; ; ; Thu Nov 20, 1986 04:00:06p -by- David N. Weise [davidw] ; ; Rewrote it use the global free list. Also made it put fixed requests ; ; as low as possible and to search again after a compact. ; ; ; ; Mon Nov 17, 1986 11:41:49a -by- David N. Weise [davidw] ; ; Added support for the free list of global partitions. ; ; ; ; Tue Sep 23, 1986 04:35:39p -by- David N. Weise [davidw] ; ; Added this nifty comment block. ; ;-----------------------------------------------------------------------;
assumes ds,nothing assumes es,nothing
cProc gsearch,<PUBLIC,NEAR> cBegin nogen
CheckKernelDS fs ReSetKernelDS fs
cmp word ptr [di].gi_free_count, 96 jae short compact_first mov gsearch_compact_first, 0 mov gsearch_state_machine,codeOFFSET grow_heap jmp short look_again
compact_first: mov gsearch_compact_first, 1 mov gsearch_state_machine,codeOFFSET do_compact_nodiscard
look_again: push ebx ; Save requested size push cx ; Save owner push ax ; Save flags clc ; Instead of add call galign ; Get requested size in EDX push edx ; Save adjusted requested size
; see if there is any one free space large enough for the request first
mov cx,[di].gi_free_count jcxz were_hosed_from_start mov esi,[di].phi_first is_there_one: mov esi,ds:[esi].pga_freenext cmp edx,ds:[esi].pga_size jbe short got_one loop is_there_one were_hosed_from_start: jmp space_not_found
got_one: mov ebx,pga_prev ; search backwards test al,GA_ALLOCHIGH jz short alloc_low
;------ allocate disc code -------
public alloc_high alloc_high: mov cx,[di].gi_free_count mov esi,[di].phi_last ; Start with last entry. alloc_high_loop: mov esi,ds:[esi].pga_freeprev cmp edx,ds:[esi].pga_size jbe afound ; Yes, exit search loop loop alloc_high_loop jmp space_not_found
;------ allocate moveable ---------
public alloc_low alloc_low: mov ebx,pga_next ; Search forwards. test al,GA_MOVEABLE jz short alloc_fixed call gcheckfree ; Room for requested size? jb space_not_found jmp afound
;------ allocate fixed ------------
public alloc_fixed alloc_fixed: mov esi,[di].phi_first ; Start with first entry. mov cx,[di].hi_count mov esi,ds:[esi+ebx] ; Skip first sentinel
; maybe ALLOC_DOS or regular fixed. test ah, GA_ALLOC_DOS jnz alloc_fixed_loop test fBooting, 2 ; are we past KERNEL loading jnz alloc_fixed_loop ; N: give him under 1MB line ; check to see if we want old behaviour ; this is to fix some mm drivers that mmsystem.dll will load test ffixedlow, 1 ; want FIXED under 1MB? jnz alloc_fixed_loop ; Y: give him under 1MB line ; start fixed blocks > 1MB. This will leave room for GlobalDOSAllocs ; Memory < 1MB will also get used for moveable blocks as we start ; from phi_first for these. These can be moved out of the way when ; we need to alloc GA_DOS_ALLOC requests.
fast_forward_to_1MB_line: ; need to store this ptr ; for speed? cmp ds:[esi].pga_address, 100000h ; > 1Mb? jae skip_not_there ; OK for fixed allocs now mov esi,ds:[esi+ebx] loop fast_forward_to_1MB_Line
; when here it means we have no free blocks > 1MB!!! ; but have a block < 1MB. space_not_found will discard ; enough to move blocks up to generate room for this fixed blk. jmp space_not_found
skip_not_there: if KDEBUG cmp [esi].pga_owner, GA_NOT_THERE je @f krDebugOut <DEB_ERROR OR DEB_KrMemMan>, "NOT_THERE block not there!" @@: endif mov esi,ds:[esi+ebx]
alloc_fixed_loop: push cx push esi call is_there_theoretically_enough_space cmp eax,edx jb short nope pop esi pop cx call can_we_clear_this_space jz short anext1 call gcheckfree ; Yes, room for requested size? jb short anext1 ; Now we have enough free space, ; slide it back as far as possible. push eax ; This is to prevent us blocking push edx ; the growth of moveable blocks. push ebx ; AND to keep pagelocked code together call PreallocArena jz short no_sliding
mov ebx, pga_prev ; Sliding backwards keep_sliding: call gslide jnz keep_sliding
no_sliding: pop ebx pop edx pop eax
pop edx pop cx test ch, GA_ALLOC_DOS push cx push edx jz afound cmp ds:[esi].pga_address, 100000h ; > 1Mb? jb afound jmp gsearch_fail
nope: or eax,eax jz short hosed_again anext: add sp, 6 ; get rid of CX, ESI on the stack anext1: mov esi,ds:[esi+ebx] loop alloc_fixed_loop jmps and_again
hosed_again: pop esi pop cx and_again:
; no one space big enough, try compacting
public space_not_found space_not_found: pop edx ; get adjusted size pop ax ; get flags push ax push edx ; test al,GA_ALLOCHIGH ; jnz short ask_for_what_we_need ; add edx,0400h ; ask for 1k more ; jnc short ask_for_what_we_need ; no overflow ; mov edx,-1 ask_for_what_we_need:
jmp gsearch_state_machine ;------------------------------
public do_compact_nodiscard ; for debugging
do_compact_nodiscard: mov gsearch_state_machine,codeOFFSET grow_heap
; ; Before growing the heap try compacting without ; discarding. This step isn't executed unless there ; were 96 or more free blocks at entry to gsearch. ;
test ds:[di].gi_cmpflags, GA_NODISCARD jnz short dcn_nodiscard
or ds:[di].gi_cmpflags, GA_NODISCARD call gcompact and ds:[di].gi_cmpflags, NOT GA_NODISCARD jmp short over_compact
dcn_nodiscard: call gcompact jmp short over_compact
public do_compact ; for debugging
do_compact: mov gsearch_state_machine,codeOFFSET gsearch_fail
; ; If we tried compacting before and GA_NODISCARD was set, ; there is no need to compact again, since we already ; compacted with GA_NODISCARD before attempting to ; grow the heap. If GA_NODISCARD was not set, our earlier ; compact forced it on, so it's worth trying again since ; we may be able to discard enough to satisfy the request. ;
cmp gsearch_compact_first, 0 je short @f test ds:[di].gi_cmpflags, GA_NODISCARD jnz short gsearch_fail
@@: call gcompact
over_compact: pop edx pop ax pop cx pop ebx jmp look_again
public grow_heap ; for debugging
grow_heap: mov gsearch_state_machine,codeOFFSET do_compact
push edx ; can we get the space from DPMI? call GrowHeap pop edx jnc short over_compact ; heap grew, go look again
call InnerShrinkHeap ; try to give back DPMI blocks so DPMI ; mmgr can defragment its memory jz short do_compact ; heap did not shrink
push edx ; gave some back, try to get it again call GrowHeap pop edx jnc short over_compact ; heap grew, go look again doomed:
;------------------------------
public gsearch_fail ; for debugging
gsearch_fail: ; get size of largest free block .errnz doomed-gsearch_fail xor edx,edx mov cx,[di].gi_free_count jcxz gs_failure mov esi,[di].phi_first largest_loop: mov esi,ds:[esi].pga_freenext mov eax,ds:[esi].pga_size cmp edx,eax jae short new_smaller mov edx,eax new_smaller: loop largest_loop gs_failure: pop eax ; adjusted requested size pop ax ; AX = flags
pop cx ; CX = owner field pop eax ; waste requested size xor eax,eax ; Return zero, with ZF = 1 ret
; Here when we have a block big enough. ; ES:DI = address of block ; AX = size of block, including header ; DX = requested size, including header ; BX = ga_prev if backwards search and ga_next if forwards search
afound: mov ecx,ds:[esi].pga_size ; Use actual size of free block sub ecx,edx ; (found size - requested size) jecxz no_arena_needed call PreallocArena jnz short no_arena_needed
; ; Detect infinite loop here. If we are out of arenas, and ; the compact failed once, then another one isn't going to do much ; good. ; cmp gsearch_state_machine,codeOFFSET gsearch_fail je gsearch_fail
pop edx ; get adjusted size pop ax ; get flags push ax push edx and ds:[di].gi_cmpflags, NOT (GA_NODISCARD+GA_NOCOMPACT) or ds:[di].gi_cmpflags, COMPACT_ALLOC mov edx, -1 ; Discard the world! jmp do_compact
no_arena_needed: mov eax,ds:[esi].pga_freeprev call gdel_free ; remove the alloc block from freelist jecxz aexitx cmp bl,pga_prev ; Yes, scanning forwards or backwards? je short abackward ; Backwards. call gsplice ; FS:ESI = block we are allocating jmps aexit ; EDX = block to mark as free abackward: neg edx add edx,ds:[esi].pga_size ; Scanning backwards. Put extra space call gsplice xchg edx, esi jmps aexit
; Here with allocated block ; AX = data address or zero if nothing allocated ; ES:DI = address of block to mark as busy and zero init if requested ; EDX = address of block to mark as free
aexitx: xor edx,edx ; Assume nothing extra to free aexit: pop ecx ; waste adjusted requested size pop cx ; Restore flags pop ds:[esi].pga_owner ; Mark block as busy with owner field value add sp, 4 ; waste requested size mov ds:[esi].pga_lruprev,edi mov ds:[esi].pga_lrunext,edi
push esi mov esi, edx ; Free any extra space mov edx, eax ; Previous free block call gmarkfree pop esi
mov dx, cx mov al,GA_SEGTYPE and al,dl test dh,GAH_NOTIFY jz short no_notify or al,GAH_NOTIFY no_notify: mov ds:[esi].pga_flags,al ; Store segment type bits mov eax,esi ; AX = address of client data
test cl,GA_ZEROINIT ; Want it zeroed? jz short aexit1 ; No, all done
push eax ifdef WOW_x86 ;; On NT we try never to set selectors when we don't need to since it is a ;; slow operation - a system call. In this case we can use selector 23h ;; which points to all flat vdm memory as data push es push bx push edi
mov bx,FLAT_SEL mov es,bx
mov ecx,ds:[esi].pga_size ; Yes, zero paragraphs push ecx
shr ecx, 2 ; # dwords to clear mov edi, ds:[esi].pga_address xor eax, eax
cld rep stos dword ptr es:[edi]
pop ecx pop edi pop bx pop es else cCall get_blotto mov ecx,ds:[esi].pga_size ; Yes, zero paragraphs push bx mov bx,ax ; from beginning of client data call gzero ; zero them pop bx endif; WOW_x86 pop eax aexit1:
or eax,eax ret ; Return AX points to client portion UnSetKernelDS FS ; of block allocated. cEnd nogen
;------------------------------------------------------------------ ; ; ChangeAllocFixedBehaviour ; GlobalAlloc(FIXED) used to return address < 1MB if possible in 3.1 ; It doesn't anymore in Chicago. You need to call this API if you ; want the old behaviour. Bad things will happen if you switch to old ; behaviour and forget to switch it back. ; MMSYSTEM.DLL loads some drivers that may expect this behaviour and ; they are the only callers of this fun. at the time of its writing. ; ENTRY = flags ; 0 = chicago behaviour ; 1 = win31 behaviour ; EXIT ; old bahaviour before this change (can be used to restore) ; ;------------------------------------------------------------------
cProc ChangeAllocFixedBehaviour,<PUBLIC,FAR> parmW flags cBegin GENTER32 CheckKernelDS FS ReSetKernelDS fs
mov ax, flags xchg al, ffixedlow
GLEAVE32 cEnd
cProc GrowHeap,<PUBLIC,NEAR> cBegin nogen CheckKernelDS FS ReSetKernelDS fs cmp FreeArenaCount, 4 ; 3 for below, 1 for a gsplice later jb short gh_fail
pushad push edx ; Save requested size ; If they want more than 64k, assume they ; want a big block and just allocate that amount. cmp edx, 64*1024 ; Want more than 64k? jae short ask_for_it ; yes, just round up mov edx, 128*1024 ; no, try for 128k ask_for_it: mov ebx, edx add ebx, 4096-1 ; Round up to 4k multiple and bx, NOT (4096-1)
cCall MyGetAppWOWCompatFlagsEx ; check if we need to pad it test ax, WOWCFEX_BROKENFLATPOINTER jz short @f add ebx, 4096*4 ; make it a bit bigger @@:
push ebx ; Length we will ask for mov cx, bx shr ebx, 16 DPMICALL 0501h ; Allocate Memory Block ; Get our memory jnc short got_more_memory pop ebx ; Toss length we asked for
; Couldn't get our 1st choice, how call GetDPMIFreeSpace ; much is available mov ebx, eax ; ebx = largest available
pop edx ; Requested size cmp ebx, edx ; Enough for request? jbe SHORT gh_fail_pop
push edx ; Expected on stack below push ebx ; Length we will ask for mov cx, bx shr ebx, 16 DPMICALL 0501h ; Allocate Memory Block ; Get our memory jnc short got_more_memory add sp, 8 ; Toss requested size & len asked for
gh_fail_pop: popad gh_fail: stc ; No chance mate! ret ; Now we have a new block ; Sort it in to the heap ; Create NOT THERE blocks to got_more_memory: ; bracket the new block inc Win386_Blocks pop edx ; Length of block allocated if KDEBUG mov eax, edx shr eax, 16 krDebugOut <DEB_TRACE OR DEB_KrMemMan>, "GrowHeap: #ax#DX allocated" endif shl ebx, 16 mov bx, cx ; EBX has linear address pop ecx ; Toss original length shl esi, 16 mov si, di ; ESI has WIN386 handle xor edi, edi cCall alloc_arena_header,<ebx> mov ecx, eax ; First not there arena mov [ecx].pga_size, edi mov [ecx].pga_sig, GA_SIGNATURE mov [ecx].pga_owner, GA_NOT_THERE mov [ecx].pga_handle, di mov [ecx].pga_flags, di mov [ecx].pga_lrunext, esi ; Save WIN386 handle here mov [ecx].pga_lruprev, edi cCall alloc_arena_header,<ebx> mov [eax].pga_size, edx ; Free block push ebx ; Save address add edx, ebx ; Address of end of block mov ebx, eax mov [ecx].pga_next, ebx mov [ebx].pga_prev, ecx mov [ebx].pga_owner, di cCall alloc_arena_header,<edx> mov edx, eax mov [ebx].pga_next, edx mov [edx].pga_prev, ebx mov [edx].pga_size, edi mov [edx].pga_owner, GA_NOT_THERE mov [edx].pga_handle, di mov [edx].pga_flags, di mov [edx].pga_sig, GA_SIGNATURE mov [edx].pga_lrunext, edi mov [edx].pga_lruprev, edi
pop eax ; Address of block sort_it: mov esi, [edi].phi_first cmp eax, [esi].pga_address ; Below start of heap? ja short sort_loop ; no, sort it in ; int 3 ; [this code never reached] mov [esi].pga_address, eax ; yes, adjust sentinel jmps link_it_in ; Sentinel now points to new block
sort_loop: mov esi, [esi].pga_next cmp [esi].pga_next, esi ; At end? je short sort_found ; yes, put here cmp [esi].pga_owner, GA_NOT_THERE jne short sort_loop mov esi, [esi].pga_prev ; Will go after previous block.
sort_found: ; Block will go after ESI cmp [esi].pga_next, esi ; This the sentinel? jne short link_it_in ; no, link it in mov eax, [edx].pga_address ; yes, adjust sentinel mov [esi].pga_address, eax sub eax, [di].gi_reserve ; Adjust fence mov [di].gi_disfence_lo, ax shr eax, 16 mov [di].gi_disfence_hi, ax mov esi, [esi].pga_prev ; New block goes before sentinel
link_it_in: ; Link it in after ESI mov [ecx].pga_prev, esi xchg [esi].pga_next, ecx mov [edx].pga_next, ecx mov [ecx].pga_prev, edx
add [di].hi_count, 3 ; Three more entries in heap mov esi, ebx xor edx, edx call gmarkfree ; To be picked up next time around popad clc ret UnSetKernelDS FS cEnd nogen
; Input - DWORD - Old Arena offset from burgermaster ; ; Output - None
cProc FreeHeapDib,<PUBLIC,FAR> parmD OldArena cBegin CheckKernelDS FS ReSetKernelDS fs
; Make sure arena before and after are GA_NOT_THERE mov ebx,OldArena mov edx,ds:[ebx].pga_prev cmp ds:[edx].pga_owner,GA_NOT_THERE jne short fhd5 mov ecx,ds:[ebx].pga_next cmp ds:[ecx].pga_owner,GA_NOT_THERE je short fhd7 fhd5: if KDEBUG krDebugOut <DEB_TRACE OR DEB_KrMemMan>, "FreeHeapDIB: Corrupt DIB Block" endif fhd7: ;Free all the three arenas. First fixup the arena list. ; edx - First GA_NOT_THERE arena ; ebx - Actual DIB arean ; ecx - Last GA_NOT_THERE arena
mov eax,ds:[edx].pga_prev mov ebx,ds:[ecx].pga_next mov ds:[eax].pga_next,ebx mov ds:[ebx].pga_prev,eax
mov ds:[edx].pga_handle,0 cCall free_arena_header,<edx>
mov ds:[ecx].pga_handle,0 cCall free_arena_header,<ecx>
mov edx,OldArena mov ds:[edx].pga_handle,0 cCall free_arena_header,<edx>
xor di,di sub [di].hi_count, 3 ; Three less entries in heap dec Win386_Blocks UnSetKernelDS FS cEnd
; Input - DWORD - Dib Address ; DWORD - Old Arena offset from burgermaster ; ; Output - eax = new arena if operation successful ; eax = NULL if operation failed
cProc GrowHeapDib,<PUBLIC,FAR> parmD OldArena parmD NewAddress cBegin CheckKernelDS FS ReSetKernelDS fs cmp FreeArenaCount, 4 ; 3 for below, 1 for a gsplice later jae short ghd_start xor eax,eax ret
; Now we have a new block. Sort it in to the heap. Create ; NOT THERE blocks as well.
ghd_start: inc Win386_Blocks mov ebx,OldArena mov edx,ds:[ebx].pga_size ; Length of block if KDEBUG mov eax, edx shr eax, 16 krDebugOut <DEB_TRACE OR DEB_KrMemMan>, "GrowHeapDIB: #ax#DX allocated" endif mov ebx,NewAddress ; Ebx is the address of new block mov esi, ebx ; ESI has WIN386 handle xor edi, edi cCall alloc_arena_header,<ebx> mov ecx, eax ; First not there arena mov [ecx].pga_size, edi mov [ecx].pga_sig, GA_SIGNATURE mov [ecx].pga_owner, GA_NOT_THERE mov [ecx].pga_handle, di mov [ecx].pga_flags, di mov [ecx].pga_lrunext, esi ; Save WIN386 handle here mov [ecx].pga_lruprev, edi cCall alloc_arena_header,<ebx> mov [eax].pga_size, edx ; DIB block push ebx ; Save address add edx, ebx ; Address of end of block mov ebx, eax mov [ecx].pga_next, ebx mov [ebx].pga_prev, ecx push ecx mov ecx,OldArena mov ax, [ecx].pga_handle mov [ebx].pga_handle,ax mov ax, [ecx].pga_owner mov [ebx].pga_owner,ax mov al, [ecx].pga_count mov [ebx].pga_count,al inc [ebx].pga_count ; make sure it doesn't move mov al, [ecx].pga_pglock mov [ebx].pga_pglock,al mov al, [ecx].pga_flags mov [ebx].pga_flags,al mov al, [ecx].pga_selcount mov [ebx].pga_selcount,al mov [ebx].pga_lrunext, edi mov [ebx].pga_lruprev, edi pop ecx cCall alloc_arena_header,<edx> mov edx, eax mov [ebx].pga_next, edx mov [edx].pga_prev, ebx mov [edx].pga_size, edi mov [edx].pga_owner, GA_NOT_THERE mov [edx].pga_handle, di mov [edx].pga_flags, di mov [edx].pga_sig, GA_SIGNATURE mov [edx].pga_lrunext, edi mov [edx].pga_lruprev, edi
pop eax ; Address of block mov esi, [edi].phi_first cmp eax, [esi].pga_address ; Below start of heap? ja short ghd_sort_loop ; no, sort it in ; int 3 ; [this code never reached] mov [esi].pga_address, eax ; yes, adjust sentinel jmps ghd_link_it_in ; Sentinel now points to new block
ghd_sort_loop: mov esi, [esi].pga_next cmp [esi].pga_next, esi ; At end? je short ghd_sort_found ; yes, put here cmp [esi].pga_owner, GA_NOT_THERE jne short ghd_sort_loop mov esi, [esi].pga_prev ; Will go after previous block.
ghd_sort_found: ; Block will go after ESI cmp [esi].pga_next, esi ; This the sentinel? jne short ghd_link_it_in ; no, link it in mov eax, [edx].pga_address ; yes, adjust sentinel mov [esi].pga_address, eax sub eax, [di].gi_reserve ; Adjust fence mov [di].gi_disfence_lo, ax shr eax, 16 mov [di].gi_disfence_hi, ax mov esi, [esi].pga_prev ; New block goes before sentinel
ghd_link_it_in: ; Link it in after ESI mov [ecx].pga_prev, esi xchg [esi].pga_next, ecx mov [edx].pga_next, ecx mov [ecx].pga_prev, edx
add [di].hi_count, 3 ; Three more entries in heap mov eax,ebx UnSetKernelDS FS cEnd
;-----------------------------------------------------------------------; ; is_there_theoretically_enough_space ; ; Starting at the given arena checks to see if there are enough ; continuous free and unlocked moveable blocks. ; ; Entry: ; CX = arenas to search ; EDX = size requested ; DS = BurgerMaster ; FS:ESI = arena to start from ; ; Returns: ; EAX = 0 => not enough space and no more memory left to search ; EAX = 1 => not enough space in this block, but maybe.... ; otherwise ; EAX = size of block ; ; Registers Destroyed: ; CX,ESI ; ; Registers Preserved: ; BX,DX,DI,ES ; ; History: ; Mon 05-Sep-1988 15:21:14 -by- David N. Weise [davidw] ; Moved it here from gsearch so that grealloc could use it as well. ;-----------------------------------------------------------------------;
assumes ds,nothing assumes es,nothing
cProc is_there_theoretically_enough_space,<PUBLIC,NEAR> cBegin nogen
xor eax,eax ittes: cmp ds:[esi].pga_owner,di jne short is_it_moveable add eax,ds:[esi].pga_size push ebx push eax mov bx,word ptr [di].gi_disfence_hi ; See if begin of reserve area shl ebx, 16 mov bx,word ptr [di].gi_disfence_lo ; is above end of free block mov eax, ds:[esi].pga_address add eax, ds:[esi].pga_size ; Address of end of free block sub eax,ebx ja short ittes_above_fence ; All below fence? ittes_below_fence: pop eax ; yes, we can use it pop ebx jmps this_ones_free
ittes_above_fence: cmp eax, ds:[di].gi_reserve jae ittes_below_fence mov ebx, eax ; portion above the fence pop eax ; Total size so far sub eax,ebx ; No, Reduce apparent size of free block pop ebx cmp eax,edx jae short theoretically_enough jmps absolutely_not
is_it_moveable: cmp ds:[esi].pga_owner,GA_NOT_THERE ; Against end of heap partition? je short theoretically_not ; no room here. test ds:[esi].pga_handle,GA_FIXED ; See if movable. jnz short theoretically_not cmp ds:[esi].pga_count,0 jne short theoretically_not ; See if locked. add eax,ds:[esi].pga_size this_ones_free: cmp eax,edx jae short theoretically_enough mov esi,ds:[esi].pga_next loop ittes
; For the case of gsearch we should never get here for two reasons. ; 1) It should be impossible to have no discardable code loaded, in ; this case we would have failed in the above loop. 2) We checked ; at the very start for a free block somewhere that could have ; satisfied the request. In our mucking around to load as low as ; possible we destroyed this free block and we did not produce a free ; block we could use. However we know from debugging code in 2.03 ; that this happens extremely rarely. Because of the rareness of ; this event we will not try to recover, instead we simply fail the call.
absolutely_not: mov eax,-1 ; return EAX = 0 theoretically_not: inc eax ; DX is even, => cmp the same. theoretically_enough: ret cEnd nogen
;-----------------------------------------------------------------------; ; can_we_clear_this_space ; ; Attempts to make a free space starting at the address moved in. ; To do this it moves moveable out of the area. The only subtlety ; involves a free block that stradles the end of wanted area. This ; may get broken into two pieces, the lower piece gets temporary marked ; as allocated and BOGUS, the upper piece will remain free. ; ; Entry: ; CX = max number of blocks to look at ; EDX = size wanted in bytes ; DS = BurgerMaster ; FS:ESI = beginning arena ; ; Returns: ; ZF = 0 ; FS:ESI points to free space ; ZF = 1 ; couldn't free the space up ; ; Registers Destroyed: ; AX,SI ; ; History: ; Mon 05-Sep-1988 16:48:31 -by- David N. Weise [davidw] ; Moved it out of gsearch so that grealloc could use it. ;-----------------------------------------------------------------------;
assumes ds,nothing assumes es,nothing
cProc can_we_clear_this_space,<PUBLIC,NEAR> cBegin nogen CheckKernelDS FS ReSetKernelDS FS push ecx push esi cmp di,[di].gi_free_count ; any free blocks? jz short cwcts_fail mov eax, esi ; Beginning of space we want. cmp di,ds:[esi].pga_owner ; Is it free? jnz short can_we_move_it mov ecx,edx cmp ecx,ds:[esi].pga_size ja short asdf or eax,eax ; return ZF = 0 cwcts_fail: pop esi pop ecx ret
asdf: mov esi,ds:[esi].pga_next
public can_we_move_it can_we_move_it: push ebx push ecx push edx push esi cmp ds:[esi].pga_owner,GA_BOGUS_BLOCK jnz short not_bogus xor edx,edx call gmarkfree jmp restart not_bogus: mov ebx, ds:[eax].pga_address add ebx, edx ; EBX is the end of the space we want mov cx,[di].gi_free_count mov edx,ds:[esi].pga_size ; Yes, try to find a place for the mov esi,[di].phi_first ; moveable block look_loop: call PreAllocArena ; Needed for gmovebusy or gsplice jz couldnt_clear_it mov esi,ds:[esi].pga_freenext push esi mov esi, ds:[esi].pga_address cmp esi, ds:[eax].pga_address ; It defeats our purpose to move the jb short check_this_out ; block to a free space we want. cmp esi, ebx jb short is_there_hope check_this_out: pop esi push eax call gcheckfree push ecx jb short inopportune_free_space pop ecx pop eax
pop edx ; EDX = moveable block for gmovebusy mov ebx,pga_next call gmovebusy ; Move moveable block out of the way mov esi,edx ; Replace the ESI on the stack, ; the free block may have grown ; downward with the gmovebusy. pop edx pop ecx pop ebx pop ecx ; WAS pop esi but esi set above now pop ecx jmp can_we_clear_this_space
inopportune_free_space: pop ecx pop eax loop look_loop jmps couldnt_clear_it
public is_there_hope is_there_hope: pop esi push eax push ecx
mov ecx, ds:[esi].pga_address add ecx, ds:[esi].pga_size ; ECX end of block mov ax, [di].gi_disfence_hi shl eax, 16 mov ax, [di].gi_disfence_lo ; EAX == fence
sub eax, ecx ; Fence - End jae short below_reserved ; Block is below fence neg eax ; End - Fence cmp eax, ds:[di].gi_reserve jae short below_reserved ; Block is above reserved sub ecx, eax ; End - (End - Fence) ; Gives Fence in ECX below_reserved: sub ecx, ebx ; Adjust size of free block jbe inopportune_free_space ; No room here
overlap: cmp ecx,edx ; Is it big enough? jbe short inopportune_free_space
mov edx, ebx sub edx, ds:[esi].pga_address ; Calculate overlap
pop ecx pop eax
; cut off the first piece for the original alloc
push ds:[esi].pga_freeprev call gdel_free call gsplice
; DS:ESI = addr of block to mark as busy, FS:EDX = addr of block to mark as free
mov ds:[esi].pga_owner,GA_BOGUS_BLOCK mov ds:[esi].pga_lruprev,edi mov ds:[esi].pga_lrunext,edi mov esi, edx pop edx ; previous free block call gmarkfree ; Free any extra space restart: pop edx ; WAS pop es pop edx pop ecx pop ebx pop esi pop ecx jmp can_we_clear_this_space
; If here then failure! see if we made a bogus block!
couldnt_clear_it: pop esi ; recover block we wanted moved check_again: mov edx,ds:[esi].pga_next cmp edx, ds:[edx].pga_next ; At end of arenas? je short no_bogus_block ; Yes, let's go cmp ds:[edx].pga_address, ebx ; EBX points to where bogus block ja short no_bogus_block ; would be. je short is_it_bogus mov esi,edx jmps check_again is_it_bogus: cmp ds:[esi].pga_owner,GA_BOGUS_BLOCK jnz short no_bogus_block xor edx,edx call gmarkfree no_bogus_block:
if KDEBUG mov cx,[di].hi_count mov esi,[di].phi_first bogus_all: cmp ds:[esi].pga_owner,GA_BOGUS_BLOCK jnz short not_me int 3 not_me: mov esi,ds:[esi].pga_next loop bogus_all endif pop edx pop ecx pop ebx pop esi pop ecx xor eax,eax ; return ZF = 1 ret UnSetKernelDS FS cEnd nogen
;-----------------------------------------------------------------------; ; gcheckfree ; ; ; ; Checks the size of the passed free block against the passed desired ; ; size, making sure that the limitations of the code reserve area are ; ; not violated for objects other than discardable code. It also checks ; ; for Phantoms. ; ; ; ; Arguments: ; ; FS:ESI = address of free block ; ; DX = #bytes needed ; ; DS:DI = address of global arena information structure ; ; ; ; Returns: ; ; CF = 0 block big enough ; ; EAX = apparent size of free block ; ; ; ; Error Returns: ; ; none ; ; ; ; Registers Preserved: ; ; All ; ; ; ; Registers Destroyed: ; ; ; ; Calls: ; ; nothing ; ; ; ; History: ; ; ; ; Thu 27-Apr-1989 10:38:05 -by- David N. Weise [davidw] ; ; Fixed this to work in pmode. ; ; ; ; Thu Apr 02, 1987 10:45:22p -by- David N. Weise [davidw] ; ; Added Phantom support. ; ; ; ; Tue Sep 23, 1986 05:54:51p -by- David N. Weise [davidw] ; ; Added this nifty comment block. ; ;-----------------------------------------------------------------------;
cProc gcheckfree,<PUBLIC,NEAR> cBegin nogen mov eax,ds:[esi].pga_size ; Compute size of free block test byte ptr [di].gi_cmpflags,GA_DISCCODE jnz short gcftest ; Discardable code not restricted
push ebx mov ebx, [di].phi_last ; Last sentinel mov ebx, ds:[ebx].pga_address cmp ebx, ds:[esi].pga_address ; Above sentinel? jbe short gcftest1 ; yes, not in reserved area! push eax mov bx,word ptr [di].gi_disfence_hi ; See if begin of reserve area shl ebx, 16 mov bx,word ptr [di].gi_disfence_lo ; is above end of free block add eax, ds:[esi].pga_address ; EAX address of end of block sub ebx, eax pop eax jae short gcftest1 ; Yes, return actual size of free block neg ebx sub eax,ebx ; No, Reduce apparent size of free block ja short gcftest1 ; Is it more than what is free?
xor eax,eax ; Yes, then apparent size is zero ; Nothing left, set apparent size to 0 gcftest1: pop ebx jmps gcftest gcfrsrv1: ; Yes, then apparent size is zero xor eax,eax ; Nothing left, set apparent size to 0 gcftest: cmp eax,edx ; Return results of the comparison ret cEnd nogen
;-----------------------------------------------------------------------; ; gsplice [should be named gslice, ; ; since splice means combine two into one - donc] ; ; ; ; Splits one block into two. ; ; ; ; Arguments: ; ; EDX = size in bytes of new block to make ; ; DS:ESI = address of existing block ; ; DS:DI = address of global arena information structure ; ; ; ; Returns: ; ; EDX = address of new block ; ; ; Error Returns: ; ; nothing ; ; ; ; Registers Preserved: ; ; AX,BX,DX,DI,DS,ES ; ; ; ; Registers Destroyed: ; ; CX ; ; ; ; Calls: ; ; nothing ; ; History: ; ; ; ; Tue Sep 23, 1986 03:50:30p -by- David N. Weise [davidw] ; ; Added this nifty comment block. ; ;-----------------------------------------------------------------------;
cProc gsplice,<PUBLIC,NEAR>,<EAX,EBX> cBegin
mov ebx,ds:[esi].pga_size sub ebx,edx ; get size of 2nd new block made mov eax, ds:[esi].pga_address add eax, edx cCall alloc_arena_header,<eax>
inc [di].hi_count ; Adding new arena entry mov ecx, eax xchg ds:[esi].pga_next,ecx ; and old.next = new mov ds:[ecx].pga_prev,eax ; [old old.next].prev = new
mov ds:[eax].pga_next,ecx ; new.next = old old.next mov ds:[eax].pga_prev,esi
mov ds:[eax].pga_size,ebx mov ds:[eax].pga_sig,GA_SIGNATURE mov ds:[eax].pga_owner,di ; Zero owner & handle fields mov ds:[eax].pga_flags,0 ; For good measure. mov ds:[eax].pga_handle,di
mov ds:[esi].pga_size,edx
mov edx, eax gsplice_ret: cEnd
;-----------------------------------------------------------------------; ; gjoin ; ; ; ; Merges a block into his previous neighbor. ; ; ; ; Arguments: ; ; FS:ESI = address of block to remove ; ; DS:DI = address of global arena information structure ; ; ; ; Returns: ; ; nothing ; ; ; ; Error Returns: ; ; nothing ; ; ; ; Registers Preserved: ; ; AX,BX,CX,DX,DI,SI,DS,ES ; ; ; ; Registers Destroyed: ; ; SI ; ; ; ; Calls: ; ; gdel_free ; ; ; ; History: ; ; ; ; Mon Nov 17, 1986 11:41:49a -by- David N. Weise [davidw] ; ; Added support for the free list of global partitions. ; ; ; ; Tue Sep 23, 1986 03:58:00p -by- David N. Weise [davidw] ; ; Added this nifty comment block. ; ;-----------------------------------------------------------------------;
cProc gjoin,<PUBLIC,NEAR> cBegin nogen if KDEBUG cmp esi, ds:[esi].pga_prev jne short ok int 3 int 3 ok: endif push eax ; assumes one is on freelist push edx dec [di].hi_count call gdel_free mov eax,ds:[esi].pga_size mov edx,ds:[esi].pga_next ; Get address of block after mov esi,ds:[esi].pga_prev ; Get address of block before cCall free_arena_header,<ds:[edx].pga_prev> ; Free arena being removed mov ds:[edx].pga_prev,esi ; Fix up block after mov ds:[esi].pga_next, edx ; and the one before add ds:[esi].pga_size,eax ; Recompute size of block pop edx pop eax ret cEnd nogen
;-----------------------------------------------------------------------; ; gzero ; ; ; ; Fills the given area with zeros. ; ; ; ; Arguments: ; ; BX = address of first paragraph ; ; ECX = Bytes to clear ; ; ; ; Returns: ; ; BX = 0 ; ; ; ; Error Returns: ; ; ; ; Registers Preserved: ; ; AX,DX,DI,SI,DS,ES ; ; ; ; Registers Destroyed: ; ; CX ; ; Calls: ; ; nothing ; ; ; ; History: ; ; ; ; Tue Sep 23, 1986 04:08:55p -by- David N. Weise [davidw] ; ; Added this nifty comment block. ; ;-----------------------------------------------------------------------;
cProc gzero,<PUBLIC,NEAR> cBegin nogen
; Assumptions: on entry, BX contains selector to start of block, and is ; for a scratch descriptor that can be modified. ECX contains the # of ; bytes to be zeroed.
push es push eax push edi push ecx
shr ecx, 2 ; # dwords to clear mov es, bx xor eax, eax xor edi, edi
cld rep stos dword ptr es:[edi]
pop ecx pop edi pop eax pop es ret
cEnd nogen
;-----------------------------------------------------------------------; ; gmarkfree ; ; ; ; Marks a block as free, coalesceing it with any free blocks before ; ; or after it. This does not free any handles. ; ; ; ; Arguments: ; ; EDX = the first free object before this one ; ; 0 if unknown ; ; Ring 1 if no free list update wanted ; ; Ring 0 if free list update required ; ; FS:ESI = block to mark as free. ; ; DS:DI = address of global arena information structure ; ; ; ; Returns: ; ; ZF = 1 if freed a fixed block ; ; DX = 0 ; ; ZF = 0 if freed a moveable block ; ; DX = handle table entry ; ; FS:ESI = block freed (may have been coalesced) ; ; ; ; Error Returns: ; ; ; ; Registers Preserved: ; ; AX,BX,CX,DX,DS ; ; ; ; Registers Destroyed: ; ; none ; ; ; ; Calls: ; ; gjoin ; ; gadd_free ; ; ; ; History: ; ; ; ; Mon Nov 17, 1986 11:41:49a -by- David N. Weise [davidw] ; ; Added support for the free list of global partitions. ; ; ; ; Sun Nov 09, 1986 01:35:08p -by- David N. Weise [davidw] ; ; Made the debugging version fill all free space with CCCC. ; ; ; ; Wed Sep 24, 1986 10:27:06p -by- David N. Weise [davidw] ; ; Added this nifty comment block. ; ;-----------------------------------------------------------------------;
cProc gmarkfree,<PUBLIC,NEAR> cBegin nogen
call gadd_free or esi,esi jz short gmf_exit
; Mark this block as free by clearing the owner field.
mov ds:[esi].pga_sig,GA_SIGNATURE mov ds:[esi].pga_owner,di ; Mark as free mov ds:[esi].pga_flags,0 ; For good measure.
; Remember the handle value in DX, before setting to zero.
xor dx,dx xchg ds:[esi].pga_handle,dx
; Try to coalesce with next block, if it is free
push ds:[esi].pga_prev ; save previous block mov esi,ds:[esi].pga_next ; ESI = next block cmp ds:[esi].pga_owner,di ; Is it free? jne short free2 ; No, continue call gjoin ; Yes, coalesce with block we are freeing free2: pop esi ; ESI = previous block cmp ds:[esi].pga_owner,di ; Is it free? jne short free3 ; No, continue mov esi,ds:[esi].pga_next ; Yes, coalesce with block we are freeing; call gjoin free3: cmp ds:[esi].pga_owner,di ; Point to free block? je short free4 ; Yes, done mov esi,ds:[esi].pga_next ; No, leave ES pointing at free block free4: call gwin386discard gmf_exit: or dx,dx ret cEnd nogen
;-----------------------------------------------------------------------; ; gadd_free ; ; ; ; Links in the given partition into the global free list. ; ; ; ; Arguments: ; ; EDX = the first free object before this one ; ; 0 if unknown ; ; odd if no free list update wanted ; ; DS:DI = BurgerMaster ; ; FS:ESI = free global object to add ; ; ; ; Returns: ; ; ; ; Error Returns: ; ; ; ; Registers Preserved: ; ; all ; ; ; ; Registers Destroyed: ; ; ; ; Calls: ; ; nothing ; ; ; ; History: ; ; ; ; Sun Nov 09, 1986 02:42:53p -by- David N. Weise [davidw] ; ; Wrote it. ; ;-----------------------------------------------------------------------;
cProc gadd_free,<PUBLIC,NEAR> cBegin nogen push eax push edx
test dl,1 jnz short no_update_wanted
or esi,esi ; this happens with gmovebusy jz short no_update_wanted inc [di].gi_free_count mov edx, esi need_a_home_loop: mov edx, ds:[edx].pga_prev cmp ds:[edx].pga_owner, di ; Found a free block? je short found_a_home cmp ds:[edx].pga_prev, edx ; Sentinel? jne short need_a_home_loop found_a_home: mov eax, ds:[edx].pga_freenext mov ds:[esi].pga_freenext, eax mov ds:[esi].pga_freeprev, edx mov ds:[edx].pga_freenext, esi mov ds:[eax].pga_freeprev, esi
if KDEBUG call check_free_list endif gaf_exit: no_update_wanted: pop edx pop eax ret cEnd nogen
;-----------------------------------------------------------------------; ; gdel_free ; ; ; ; Removes a partition from the global free list. ; ; ; ; Arguments: ; ; DS:DI = BurgerMaster ; ; FS:ESI = arena header of partition ; ; ; ; Returns: ; ; ; ; Error Returns: ; ; ; ; Registers Preserved: ; ; All ; ; ; ; Registers Destroyed: ; ; ; ; Calls: ; ; nothing ; ; ; ; History: ; ; ; ; Sun Nov 09, 1986 02:43:26p -by- David N. Weise [davidw] ; ; Wrote it. ; ;-----------------------------------------------------------------------;
cProc gdel_free,<PUBLIC,NEAR> cBegin nogen
push edi push esi mov edi,ds:[esi].pga_freeprev mov esi,ds:[esi].pga_freenext mov ds:[edi].pga_freenext,esi mov ds:[esi].pga_freeprev,edi pop esi pop edi
dec [di].gi_free_count if KDEBUG call check_free_list endif ret cEnd nogen
if KDEBUG ;-----------------------------------------------------------------------; ; check_free_list ; ; ; ; Checks the global free list for consistency. ; ; ; ; Arguments: ; ; none ; ; ; ; Returns: ; ; ; ; Error Returns: ; ; ; ; Registers Preserved: ; ; All ; ; ; ; Registers Destroyed: ; ; ; ; Calls: ; ; nothing ; ; ; ; History: ; ; ; ; Wed Oct 29, 1986 10:13:42a -by- David N. Weise [davidw] ; ; Wrote it. ; ;-----------------------------------------------------------------------;
cProc check_free_list,<PUBLIC,NEAR> cBegin nogen push eax push ebx push cx push esi push ds SetKernelDS cmp fCheckFree,0 jnz short cfl_outta_here test Kernel_flags,kf_check_free jnz short cfl_check_it_out cfl_outta_here: jmp all_done cfl_check_it_out: mov ds,pGlobalHeap UnSetKernelDS mov esi,[di].phi_first mov cx,[di].gi_free_count or cx,cx jz all_done mov eax, ds:[esi].pga_freenext mov esi, eax check_chain_loop: mov ebx,ds:[esi].pga_freeprev mov esi,ds:[esi].pga_freenext cmp ds:[ebx].pga_freenext,eax jz short prev_okay prev_bad: push esi mov esi, eax kerror 0FFh,<free_list: prev bad>,bx,si mov eax, esi pop esi prev_okay: cmp ds:[esi].pga_freeprev,eax jnz short next_bad mov ebx,esi jmps next_okay next_bad: mov bx, ax kerror 0FFh,<free_list: next bad>,bx,es next_okay: mov eax,esi loop check_chain_loop SetKernelDS mov ds,pGlobalHeap UnSetKernelDS cmp [di].phi_last,eax jz short all_done mov bx, ax kerror 0FFh,<free_list: count bad>,[di].phi_last,bx all_done: pop ds pop esi pop cx pop ebx pop eax ret cEnd nogen
endif
cProc ValidateFreeSpaces,<PUBLIC,FAR> cBegin nogen ret cEnd nogen
sEnd CODE
end
|