1745 lines
69 KiB
NASM
Raw Normal View History

2001-01-01 00:00:00 +01:00
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