TITLE GCOMPACT - Global memory compactor .sall .xlist include kernel.inc .list WM_COMPACTING = 041h .386p include protect.inc DataBegin externB Kernel_Flags externW WinFlags externW Win386_Blocks externW PagingFlags externD gcompact_start externD gcompact_timer externD pPostMessage externD NextCandidate fSwitchStacks DB 0 fUpDown DB 0 DataEnd externFP GlobalCompact ifdef WOW externFP VirtualFree endif sBegin CODE assumes CS,CODE assumes fs, nothing externNP glrudel externNP gmarkfree externNP gcheckfree externNP gdel_free externNP gsplice externNP gnotify externNP genter externNP gleave externNP Enter_gmove_stack externNP Leave_gmove_stack if KDEBUG externFP ValidateFreeSpaces endif externNP set_selector_address32 ifndef WOW_x86 externNP get_rover_232 endif externNP AssociateSelector32 externNP alloc_arena_header externNP free_arena_header externNP mark_sel_NP externNP PreAllocArena externNP DPMIProc externNP get_physical_address if KDEBUG externNP CheckGlobalHeap endif ;-----------------------------------------------------------------------; ; gcompact ; ; ; ; Compacts the global heap. ; ; ; ; Arguments: ; ; DX = minimum #contiguous bytes needed ; ; DS:DI = address of global heap information ; ; ; ; Returns: ; ; AX = size of largest contiguous free block ; ; ES:DI = arena header of largest contiguous free block ; ; DX = minimum #contiguous bytes needed ; ; ; ; Error Returns: ; ; ; ; Registers Preserved: ; ; SI ; ; ; ; Registers Destroyed: ; ; CX ; ; ; ; Calls: ; ; gcmpheap ; ; gcheckfree ; ; gdiscard ; ; ; ; History: ; ; ; ; Thu Sep 25, 1986 05:34:32p -by- David N. Weise [davidw] ; ; Added this nifty comment block. ; ;-----------------------------------------------------------------------; assumes ds, nothing assumes es, nothing cProc gcompact, cBegin nogen CheckKernelDS fs ReSetKernelDS fs push esi smov es,40h mov ax,es:[6Ch] ; get the BIOS ticker count sub gcompact_timer.lo,ax or Kernel_Flags[1],kf1_MEMORYMOVED mov si, [di].gi_cmpflags push si push si test WinFlags[1], WF1_PAGING jz short @F ; First time: or [di].gi_cmpflags, GA_NOCOMPACT ; No movement @@: gcompactl: if KDEBUG call ValidateFreeSpaces endif push edx ; Save requested size cmp [di].gi_reserve,edi ; Is there a reserve swap area? je short gcompact1 ; No, then dont compact lower heap mov esi,[di].phi_first ; Yes, compact lower heap mov ebx,pga_next call gcmpheap gcompact1: mov esi,[di].phi_last ; Compact upper heap mov ebx,pga_prev call gcmpheap pop edx ; Get requested size mov esi,eax ; ES points to largest free block or eax,eax ; Did we find a free block? jz short gcompact2 ; No, try discarding call gcheckfree ; Yes, see if block big enough jae short gcompactxx ; Yes, all done gcompact2: ; Discarding allowed? cmp [di].hi_freeze,di ; Heap frozen? jne short gcompactxx ; Yes, return test [di].gi_cmpflags,GA_NODISCARD jnz short gcompactx ; No, return test WinFlags[1], WF1_PAGING jnz short @F test [di].gi_cmpflags, GA_NOCOMPACT ; Ignore flag if paging jnz short gcompactx @@: push esi call gdiscard ; No, try discarding pop ecx ; Saved ESI may be bogus if gdiscard ; discarded anything... jnz short gcompactl ; Compact again if anything discarded mov esi, ecx ; Nothing discarded so ES OK. gcompactx: test WinFlags[1], WF1_PAGING jz short gcompactxx pop si ; Original flags mov [di].gi_cmpflags, si or si, GA_NOCOMPACT+GA_NODISCARD push si ; Abort next time test [di].gi_cmpflags, GA_NOCOMPACT jz short gcompactl gcompactxx: add sp,2 ; Toss working flags push ax push dx push es mov ax,40h mov es,ax mov ax,es:[6Ch] mov si,ax cmp pPostMessage.sel,0 ; is there a USER around yet? jz short tock add gcompact_timer.lo,ax sub ax,gcompact_start.lo cmp ax,546 ; 30 secs X 18.2 tics/second jb short tock cmp ax,1092 ; 60 secs ja short tick mov cx,gcompact_timer.lo ; poor resolution of timer! jcxz short tick xchg ax,cx xor dx,dx xchg ah,al ; shl 8 DX:AX xchg dl,al div cx cmp ax,32 ; < 12.5% ? jb short tick mov ah,al mov bx,-1 ; broadcast mov cx,WM_COMPACTING xor dx,dx cCall pPostMessage, tick: mov gcompact_start.lo,si mov gcompact_timer.lo,0 tock: pop es pop dx pop ax pop [di].gi_cmpflags pop esi ; Restore SI ret UnSetKernelDS fs cEnd nogen ;-----------------------------------------------------------------------; ; gcmpheap ; ; ; ; ; ; Arguments: ; ; EBX = pga_prev or pga_next ; ; EDX = minimum #contiguous bytes needed ; ; DS:DI = address of global heap information ; ; FS = Kernel DATA ; ; ; ; Returns: ; ; EAX = largest free block ; ; ; ; Error Returns: ; ; ; ; Registers Preserved: ; ; ; ; Registers Destroyed: ; ; AX,CX ; ; ; ; Calls: ; ; gslide ; ; gbestfit ; ; ; ; History: ; ; ; ; Thu Sep 25, 1986 05:38:16p -by- David N. Weise [davidw] ; ; Added this nifty comment block. ; ;-----------------------------------------------------------------------; assumes ds,nothing assumes es,nothing cProc gcmpheap, cBegin nogen CheckKernelDS fs ReSetKernelDS fs mov NextCandidate, -1 xor eax,eax ; Nothing found yet push eax ; Save largest free block so far gchloop: cmp ds:[esi].pga_owner,di je short gchfreefnd cmp ds:[esi].pga_owner, GA_NOT_THERE jne short gchnext mov NextCandidate, -1 gchnext: mov esi, ds:[esi+ebx] cmp esi, ds:[esi+ebx] ; Sentinel? jne short gchloop ; no, continue gchexit: pop eax ; Return largest free block in AX ret gchfreefnd: test [di].gi_cmpflags,GA_NOCOMPACT jnz short gchmaxfree ; No, just compute max free. cmp [di].hi_freeze,di ; Heap frozen? jne short gchmaxfree ; Yes, just compute max free. push esi test [di].gi_cmpflags,COMPACT_ALLOC jz short no_hack call gcheckfree ; Allocating, this big enough? jb short no_hack ; yes, STOP NOW! cmp bl,pga_prev ; Compacting upper heap? jnz short no_hack test [di].gi_cmpflags,GA_DISCCODE jz short no_hack cmp edx,ds:[esi].pga_size ja short no_hack mov esi,ds:[esi].pga_next test ds:[esi].pga_flags,GA_DISCCODE jnz short hack cmp ds:[esi].pga_owner,GA_SENTINAL jz short hack cmp ds:[esi].pga_owner,GA_NOT_THERE jnz short no_hack hack: pop esi pop eax mov eax,esi ret no_hack: pop esi test byte ptr WinFlags[1], WF1_PAGING ; Paging? jnz short best_it ; yes, don't slide call PreAllocArena jz short gchmaxfree push edx call gslide pop edx jnz short gchfreefnd best_it: push edx call gfirstfit pop edx gchmaxfree: cmp bl,pga_prev ; Compacting upper heap? jne short gchnext ; No, dont compute largest free block cmp ds:[esi].pga_owner,di ; Is current free? jne gchnext ; No, ignore it then pop eax ; Recover largest free block so far push edx cmp esi,eax ; Same as current? je short gchmf2 ; Yes, no change then push eax cmp ds:[di].gi_reserve,edi ; Is there a code reserve area? je short gchmf0 ; No, continue test ds:[di].gi_cmpflags,GA_DISCCODE ; If allocating disc jnz short gchmf0 ; code ignore the fence mov ax, [di].gi_disfence_hi shl eax, 16 mov ax, [di].gi_disfence_lo mov edx, ds:[esi].pga_size ; Size of this block add edx, ds:[esi].pga_address ; End of this block sub eax, edx ; Fence above it? jae short gchmf0 ; yes, use the whole block neg eax ; Amount above the fence cmp eax, ds:[di].gi_reserve ; Above reserve? jae short gchmf0 ; yes, use all of it sub edx, eax ; Subtract off that above fence ja short gchmf00 ; Use as size of block pop eax jmps gchmf2 gchmf0: mov edx, ds:[esi].pga_size ; Size of this block gchmf00: pop eax or eax,eax ; First time? jz short gchmf1 ; Yes, special case cmp edx, ds:[eax].pga_size ; Is it bigger? jb short gchmf2 ; No, do nothing gchmf1: mov eax,esi ; Yes, remember biggest free block gchmf2: pop edx push eax ; Save largest free block so far test PagingFlags, 2 ; Idle time compaction! jnz gchexit jmp gchnext ; Go process next block UnSetKernelDS fs cEnd nogen ;-----------------------------------------------------------------------; ; gslide ; ; ; ; Sees if next/previous block can slide into the passed free block. ; ; ; ; Arguments: ; ; FS:ESI = free block ; ; DS:DI = address of global heap information ; ; CX = #arena entries left to examine ; ; EBX = pga_next or pga_prev ; ; ; ; Returns: ; ; ZF = 0 if block found and moved into passed free block ; ; FS:ESI points to new free block ; ; FS:EDX points to new free block ; ; ; ; ZF = 1 if no block found ; ; FS:ESI points to original free block ; ; ; ; Error Returns: ; ; ; ; Registers Preserved: ; ; ; ; Registers Destroyed: ; ; EAX,EDX ; ; ; ; Calls: ; ; ; ; History: ; ; ; ; Thu Sep 25, 1986 05:58:25p -by- David N. Weise [davidw] ; ; Added this nifty comment block. ; ;-----------------------------------------------------------------------; assumes ds, nothing assumes es, nothing cProc gslide, cBegin nogen push esi mov esi,ds:[ebx+esi] mov edx,esi ; Source of move in EDX call gmoveable pop esi jz short gslide_no_move call gmovebusy ; Handle exact fits etc!! call gpagingcandidate if KDEBUG cmp edx, ds:[esi+ebx] ; Are we adjacent? je short gslide_adjacent int 3 int 3 gslide_adjacent: endif mov esi, edx or esi, esi ; ZF = 0 gslide_no_move: ret cEnd nogen ;-----------------------------------------------------------------------; ; gmove ; ; ; ; Moves a moveable block into the top part of a free block. The low ; ; order bit of the source and destination may be either set or reset. ; ; If set, then this routine does NOT move the arena header paragraph. ; ; If the bit is reset, then the arena header is moved. Only the low ; ; order bit of EDX is examined, and the low order bit of ESI is assumed ; ; to the same. ; ; ; ; Arguments: ; ; DS:DI = master object ; ; ES:0 = address of destination block ; ; FS:ESI = arena of destination block ; ; FS:EDX = arena of source block ; ; ECX = # bytes to move ; ; ; ; Returns: ; ; DS:DI = master object (it may have moved) ; ; ; ; Error Returns: ; ; ; ; Registers Preserved: ; ; AX,BX,CX,DX,DI,SI,ES ; ; ; ; Registers Destroyed: ; ; none ; ; ; ; Calls: ; ; gnotify ; ; ; ; History: ; ; ; ; Thu Sep 25, 1986 03:31:51p -by- David N. Weise [davidw] ; ; Added this nifty comment block. ; ;-----------------------------------------------------------------------; assumes ds, nothing assumes es, nothing cProc gmove, cBegin nogen CheckKernelDS fs ReSetKernelDS fs push es pushad mov eax, edx push esi ; Save destination mov edx, ecx ; # bytes passed in in ECX mov bx, ds:[eax].pga_handle ; BX = handle of source Handle_To_Sel bl mov cx, bx ; CX = client data address of dest. push eax push edx ; Save #paragraphs mov ax, GN_MOVE push cx ; Save client data address call gnotify ; Call global notify procedure pop cx pop edx ; EDX = #paragraphs to move pop esi ; ESI = source arena pop edi ; EDI = destination arena ; Save DS value AFTER call to gnotify, as the master object might be the ; block we are moving and thus changed by the global heap notify proc. push gs smov gs, ds mov ax,ss ; Are we about to move the stack? cmp ax,cx mov cx,0 ; ...assume no jne short stack_no_move mov cx, ax ; Selector does not change! call Enter_gmove_stack ; switch to temporary stack stack_no_move: mov fSwitchStacks,cl ; Remember if we switched xor cx, cx ; Ready for result of compare mov eax, gs:[edi].pga_address cmp gs:[esi].pga_address, eax adc ch, 0 mov fUpDown,ch mov ecx, edx ; # bytes to move shr ecx, 2 ; # dwords to move jecxz all_done cmp fUpDown, 0 jnz short move_it_up ; MOVE IT DOWN cld xor esi, esi jmps move_it move_it_up: std mov esi, ecx dec esi shl esi,2 move_it: mov ds, bx ; DS:SI = first word in source block mov edi, esi ifdef WOW_x86 smov es,FLAT_SEL ; 23:EAX => pga_address or taget add edi,eax endif; WOW_x86 rep movs dword ptr [edi], dword ptr [esi] ; 386 BUG, ECX, ESI, EDI ARE NOW TRASHED all_done: smov ds, gs cCall set_selector_address32, ; Update source selector cmp fSwitchStacks,cl ; Switch to new stack if any je short move_exit call Leave_gmove_stack move_exit: pop gs popad pop es cld ; Protect people like MarkCl from themselves ret UnSetKernelDS fs cEnd nogen ;-----------------------------------------------------------------------; ; gbestfit ; ; ; ; Searches for the largest moveable block that will fit in the passed ; ; free block. ; ; ; ; Arguments: ; ; DS:ESI = free block ; ; DS:DI = address of global heap information ; ; CX = #arena entries left to examine ; ; EBX = pga_next or pga_prev ; ; ; ; Returns: ; ; ZF = 1 if block found & moved into free block w/ no extra room. ; ; DS:ESI = busy block before/after new busy block. ; ; ; ; ZF = 0 if DS:ESI points to a free block, either the ; ; original one or what is left over after moving a block ; ; into it. ; ; ; ; Error Returns: ; ; ; ; Registers Preserved: ; ; ; ; Registers Destroyed: ; ; DX,SI ; ; ; ; Calls: ; ; gmoveable ; ; gmovebusy ; ; ; ; History: ; ; ; ; Thu Sep 25, 1986 05:52:12p -by- David N. Weise [davidw] ; ; Added this nifty comment block. ; ;-----------------------------------------------------------------------; assumes ds,nothing assumes es,nothing cProc gfirstfit, cBegin nogen CheckKernelDS fs ReSetKernelDS fs push ecx mov edx, esi ; Save free block gbfrestart: call PreAllocArena ; Abort if no free arenas jz short gbfabort mov eax,ds:[edx].pga_size ; Compute max size to look for jmps gbfnext gbfloop: cmp ds:[esi].pga_owner,di ; Is this block busy? je short gbfnext ; No, continue cmp ds:[esi].pga_size,eax ; Yes, is block bigger than max size? ja short gbfnext ; Yes, continue cmp ds:[esi].pga_owner,GA_NOT_THERE ; Is this even here? je short gbfnext ; No, continue call gmoveable ; Yes, is it moveable jnz short gbffound gbfnext: ; No, continue mov esi, ds:[esi+ebx] ; Skip past this block cmp esi, ds:[esi+ebx] ; Sentinel? jne gbfloop gbfabort: mov esi, edx ; Return original free block in ESI jmps gfirstfit1 ; Nothing found! gbffound: xchg edx, esi ; Source of move call gmovebusy ; Yes, move it into free block mov NextCandidate, -1 call gpagingcandidate mov NextCandidate, -1 xchg esi, edx ; Put free block in esi call gwin386discard xchg esi, edx cmp edx, ds:[esi+ebx] ; Blocks adjacent? je short gfirstfit1 ; Yes, may have coalesced. ; Return busy block in ESI mov esi, ds:[esi+ebx] ; Get block after busy block cmp ds:[esi].pga_owner,di ; Is this block busy? jne short gfirstfit1 ; Yes, fit was exact xchg edx, esi ; EDX is new free block jmps gbfrestart ; Start search from old busy block gfirstfit1: pop ecx ret UnSetKernelDS fs cEnd nogen ;-----------------------------------------------------------------------; ; gmovebusy ; ; ; ; Subroutine to move a busy block to a free block of the same size, ; ; preserving the appropriate arena header fields, freeing the old ; ; busy block and updating the handle table entry to point to the ; ; new location of the block. ; ; ; ; [tonyg] ; ; The above has been inaccurate for a while - the destination is NOT ; ; necessarily free, NOR is it always the same size! ; ; ; ; It will now handle everything gslidecommon used to do! ; ; ; ; Arguments: ; ; BX = ga_prev or ga_next ; ; DS:EDX = old busy block location ; ; DS:ESI = new busy block location ; ; DS:DI = address of global heap information ; ; ; ; Returns: ; ; DS:ESI = points to new busy block arena header ; ; DS:EDX = points to free block where block used to be ; ; (may be coalesced) ; ; ; ; Error Returns: ; ; ; ; Registers Preserved: ; ; EBX,ECX ; ; ; ; Registers Destroyed: ; ; EAX ; ; ; ; Calls: ; ; ; ; History: ; ; ; ; Mon Jun 22, 1987 11:39:56p -by- David N. Weise [davidw] ; ; Made it jump off to gslidecommon when appropriate. ; ; ; ; Mon Oct 27, 1986 10:17:16p -by- David N. Weise [davidw] ; ; Made the lru list be linked arenas, so we must keep the list correct ; ; here. ; ; ; ; Thu Sep 25, 1986 05:49:25p -by- David N. Weise [davidw] ; ; Added this nifty comment block. ; ;-----------------------------------------------------------------------; assumes ds, nothing assumes es, nothing cProc gmovebusy, cBegin nogen if KDEBUG cmp ds:[esi].pga_owner, GA_NOT_THERE jne short @F AAARRRGGGHHH: int 3 int 3 @@: cmp ds:[edx].pga_owner, GA_NOT_THERE je AAARRRGGGHHH endif push ecx mov ecx,ds:[edx].pga_size ; ECX = size of source cmp ds:[esi].pga_owner,di ; Is destination busy? jne gmbexactfit ; Yes, then don't create extra block mov eax, ds:[esi].pga_freeprev call gdel_free ; Take off free list now! cmp ecx,ds:[esi].pga_size ; No, source and destination same size? je short gmbexactfit ; Yes, then don't create extra block jb short gmbsplice ; Destination is larger, split it if KDEBUG ; MUST BE ADJACENT IF DESTINATION cmp ds:[esi+ebx], edx ; SMALLER THAN SOURCE!!! je short gmb_adjust int 3 int 3 gmb_adjust: endif push ecx ; source length push ds:[esi].pga_size ; destination length cmp bl, pga_next je short gmb_down ; Moving busy block up mov eax, ds:[edx].pga_address ; Correct destination address add eax, ds:[esi].pga_size mov ds:[esi].pga_address, eax call gmb_gmove jmps gmb_adjusted gmb_down: ; Moving busy block down call gmb_gmove mov ecx, ds:[esi].pga_address ; Correct new free block address add ecx, ds:[edx].pga_size mov ds:[edx].pga_address, ecx gmb_adjusted: pop ds:[edx].pga_size ; Swap sizes pop ds:[esi].pga_size jmps gmb_moved gmbsplice: push ecx push edx mov edx, ecx ; # bytes in block to make cmp bl, pga_prev je short gmb_backward call gsplice ; Split the block jmps gmb_spliced gmb_backward: neg edx add edx, ds:[esi].pga_size ; Second block will be used call gsplice xchg esi, edx gmb_spliced: mov ds:[esi].pga_owner,1 ; Mark new block busy push esi ; New free block mov esi, edx ; Block to free mov edx, eax call gmarkfree pop esi ; New free block pop edx ; Source for copy pop ecx gmbexactfit: call gmb_gmove gmb_moved: mov eax, esi mov esi, edx xor edx, edx call gmarkfree ; Free old block mov ecx,esi ; New free block mov esi,eax ; New block or dx,dx jz short gmb1 mov ds:[esi].pga_handle,dx ; Set back link to handle in new block cCall AssociateSelector32, ; Associate with new arena xor dx,dx ; Set Z flag gmb1: mov edx,ecx gmbexit: pop ecx ret cEnd nogen ; ; Common code for gmovebusy ; cProc gmb_gmove, cBegin nogen push ecx ; # bytes to move push dword ptr ds:[edx].pga_count push ds:[edx].pga_owner push ds:[edx].pga_lruprev push ds:[edx].pga_lrunext pop ds:[esi].pga_lrunext ; Copy client words to new header pop ds:[esi].pga_lruprev pop ds:[esi].pga_owner pop dword ptr ds:[esi].pga_count cmp ds:[esi].pga_lruprev,edi jz short no_link cmp [di].gi_lruchain,edx jnz short didnt_move_head mov [di].gi_lruchain,esi didnt_move_head: mov ecx,ds:[edx].pga_lruprev mov ds:[ecx].pga_lrunext,esi ; Update the lru list mov ecx,ds:[edx].pga_lrunext mov ds:[ecx].pga_lruprev,esi ; Update the lru list no_link: pop ecx ifndef WOW_x86 call get_rover_232 endif jmp gmove ; Move the client data cEnd nogen ;-----------------------------------------------------------------------; ; gmoveable ; ; ; ; Tests if an ojbect is moveable. Non moveable blocks are: ; ; Fixed blocks, moveable blocks that are locked, moveable blocks ; ; going up, discardable code going down. ; ; ; ; Arguments: ; ; FS:ESI = arena header of object ; ; DS:DI = address of global heap information ; ; BX = ga_next or ga_prev ; ; ; ; Returns: ; ; ZF = 0 if object moveable ; ; ZF = 1 if object not moveable ; ; ; ; Error Returns: ; ; ; ; Registers Preserved: ; ; All ; ; ; ; Registers Destroyed: ; ; ; ; Calls: ; ; nothing ; ; ; ; History: ; ; ; ; Wed Oct 15, 1986 05:04:39p -by- David N. Weise [davidw] ; ; Moved he_count to ga_count. ; ; ; ; Thu Sep 25, 1986 05:42:17p -by- David N. Weise [davidw] ; ; Added this nifty comment block. ; ;-----------------------------------------------------------------------; assumes ds, nothing assumes es, nothing cProc gmoveable, cBegin nogen test ds:[esi].pga_handle,GA_FIXED ; If no handle then fixed jnz short gmfixed cmp ds:[esi].pga_count,bh ; If locked then fixed jne short gmfixed test ds:[esi].pga_flags,GA_DISCCODE ; If discardable code jz short gmnotcode cmp bl,pga_next ; Discardable code can only ret ; move up in memory gmnotcode: cmp [di].gi_reserve,edi ; If no reserved code area? je short gmokay ; Then anything can move up cmp bl,pga_prev ; Otherwise can only move down ret ; in memory gmfixed: or bh,bh ; Return with ZF = 1 if ret ; not moveable gmokay: or esi,esi ret cEnd nogen ;-----------------------------------------------------------------------; ; gdiscard ; ; ; ; Subroutine to walk LRU chain, discarding objects until the #paras ; ; discarded, plus the biggest free block is greater than the #paras ; ; we are looking for. ; ; ; ; Arguments: ; ; EAX = size of largest free block so far ; ; EDX = minimum #bytes needed ; ; DS:DI = address of global heap information ; ; ; ; Returns: ; ; ZF = 0 if one or more objects discarded. ; ; ZF = 1 if no objects discarded. ; ; ; ; Error Returns: ; ; ; ; Registers Preserved: ; ; AX,DX,DI ; ; ; ; Registers Destroyed: ; ; BX,CX,ES ; ; ; ; Calls: ; ; ; ; History: ; ; Mon Oct 27, 1986 09:34:45p -by- David N. Weise [davidw] ; ; The glru list was reworked to link the arenas, not using the handle ; ; table as a middle man. Because of this change glruprev was moved ; ; inline and the code shortened up again. ; ; ; ; Wed Oct 15, 1986 05:04:39p -by- David N. Weise [davidw] ; ; Moved he_count to ga_count. ; ; ; ; Thu Sep 25, 1986 05:45:31p -by- David N. Weise [davidw] ; ; Shortened it up a bit and added this nifty comment block. ; ;-----------------------------------------------------------------------; assumes ds, nothing assumes es, nothing cProc gdiscard, cBegin nogen push eax push edx mov [di].hi_ncompact,0 ; Clear compaction flag sub edx,eax ; How much to discard before mov [di].hi_distotal,edx ; compacting again. xor ebx,ebx ; EBX = amount of DISCCODE below fence test [di].gi_cmpflags,GA_DISCCODE jnz short fence_not_in_effect0 mov cx,[di].gi_lrucount jcxz fence_not_in_effect0 ; All done if LRU chain empty mov ax, [di].gi_disfence_hi shl eax, 16 mov ax, [di].gi_disfence_lo push edx mov edx, eax add edx, ds:[edi].gi_reserve mov esi,[di].gi_lruchain ; ESI -> most recently used (ga_lruprev gdloop0: ; is the least recently used) mov esi,ds:[esi].pga_lruprev ; Move to next block in LRU chain test ds:[esi].pga_flags,GA_DISCCODE ; Discardable code? jz short gdloop0a ; No, ignore cmp edx, ds:[esi].pga_address jbe short gdinclude cmp eax, ds:[esi].pga_address ; Yes, is this code fenced off? jbe short gdloop0a ; No, ignore gdinclude: add ebx,ds:[esi].pga_size ; Yes, accumulate size of discardable gdloop0a: ; code below the fence loop gdloop0 pop edx fence_not_in_effect0: mov esi,[di].gi_lruchain cmp [di].gi_lrucount, 0 je short gdexit push ds:[esi].pga_lruprev push [di].gi_lrucount gdloop: pop cx pop eax jcxz gdexit ; No more see if we discarded anything mov esi, eax ; ES on stack may be invalid if count 0 dec cx push ds:[esi].pga_lruprev ; Save next handle from LRU chain push cx cmp ds:[esi].pga_count,0 ; Is this handle locked? jne short gdloop ; Yes, ignore it then test [di].gi_cmpflags,GA_DISCCODE jnz short fence_not_in_effect test ds:[esi].pga_flags,GA_DISCCODE jz short fence_not_in_effect or ebx,ebx ; Discardable code below fence? jz short gdloop ; No, cant discard then cmp ebx,ds:[esi].pga_size ; Yes, more than size of this block? jb short gdloop ; No, cant discard then sub ebx,ds:[esi].pga_size ; Yes, reduce size of code below fence fence_not_in_effect: push ebx call DiscardCodeSegment pop ebx jnz short discarded_something test [di].hi_ncompact,10h ; did a GlobalNotify proc free enough? jz short gdloop jmps enough_discarded discarded_something: test [di].hi_ncompact,10h ; did a GlobalNotify proc free enough? jnz short enough_discarded or [di].hi_ncompact,1 ; Remember we discarded something sub [di].hi_distotal,eax ; Have we discarded enough yet? ja short gdloop ; No, look at next handle enough_discarded: pop cx ; Flush enumeration counter pop ecx ; and saved ESI gdexit: cmp [di].hi_ncompact,0 ; Return with Z flag set or clear pop edx pop eax ret cEnd nogen ;-----------------------------------------------------------------------; ; DiscardCodeSegment ; ; ; ; Discards the given segment. Calls gnotify to fix stacks, entry ; ; points, thunks, and prologs. Then glrudel removes it from the lru ; ; list and gmarkfree finally gets rid of it. ; ; ; ; Arguments: ; ; DS:DI => BurgerMaster ; ; ES = Address of segment to discard ; ; ; ; Returns: ; ; AX = size discarded ; ; ZF = 0 ok ; ; ; ; Error Returns: ; ; ; ; Registers Preserved: ; ; DI,SI,DS,ES ; ; ; ; Registers Destroyed: ; ; BX,CX,DX ; ; ; ; Calls: ; ; gnotify ; ; glrudel ; ; gmarkfree ; ; ; ; History: ; ; ; ; Fri Jun 12, 1987 -by- Bob Matthews [bobm] ; ; Made FAR. ; ; ; ; Sun Apr 19, 1987 12:05:40p -by- David N. Weise [davidw] ; ; Moved it here from InitTask, so that FirstTime could use it. ; ;-----------------------------------------------------------------------; assumes ds, nothing assumes es, nothing cProc DiscardCodeSegment, cBegin nogen push esi mov bx,ds:[esi].pga_handle ; BX = handle mov al,GN_DISCARD ; AX = GN_DISCARD call gnotify jz short cant_discard ; Skip this handle if not discardable call glrudel ; Delete handle from LRU chain push ds:[esi].pga_owner ; Save owner field mov eax,ds:[esi].pga_size ; Save size xor edx,edx call gmarkfree ; Free the block associated with this handle mov bx,dx pop cx ; Owner cCall mark_sel_NP, cant_discard: pop esi ret cEnd nogen ;-----------------------------------------------------------------------; ; ShrinkHeap ; ; ; ; Tries to return DPMI memory blocks to DPMI memory manager. ; ; ; ; Arguments: ; ; ; ; Returns: ; ; ; ; Error Returns: ; ; ; ; Registers Preserved: ; ; ; ; Registers Destroyed: ; ; ; ; Calls: ; ; InnerShrinkHeap ; ; ; ;-----------------------------------------------------------------------; assumes ds, nothing assumes es, nothing cProc ShrinkHeap, cBegin nogen push ds GENTER32 ReSetKernelDS FS cCall InnerShrinkHeap jnz short sh_maybe_more and PagingFlags, NOT 8 ; Don't call back if # win386 ; didn't change sh_maybe_more: GLEAVE32 UnSetKernelDS FS pop ds ret cEnd nogen ;-----------------------------------------------------------------------; ; InnerShrinkHeap ; ; ; ; Checks heap to see if there are any blocks to return to Win386 ; ; Compacts if there are Win386 blocks around AND there is more ; ; than 512k free. ; ; Returns any completely free Win386 block to Win386. ; ; ; ; Arguments: ; ; FS = Kernel's DS ; ; EDI = 0 ; ; ; ; Returns: ; ; Z flag set if no blocks returned, Z clear if 1 or more returned.; ; ; ; Error Returns: ; ; ; ; Registers Preserved: ; ; All ; ; ; ; Registers Destroyed: ; ; ; ; Calls: ; ; UnlinkWin386Block ; ; ; ;-----------------------------------------------------------------------; assumes ds, nothing assumes es, nothing cProc InnerShrinkHeap, cBegin nogen CheckKernelDS FS ReSetKernelDS FS pushad push Win386_Blocks cmp Win386_Blocks, 0 je short sh_done ; First count up free blocks mov esi, [edi].phi_first scan_loop: mov esi, [esi].pga_freenext scan_next: cmp esi, [esi].pga_next ; Sentinel? je short sh_done ; yes, all done mov ebx, [esi].pga_prev cmp [ebx].pga_owner, GA_NOT_THERE jne short scan_loop mov ecx, [esi].pga_next cmp [ecx].pga_owner, GA_NOT_THERE jne short scan_loop mov eax, [ecx].pga_next ; Block after NOT_THERE block cmp eax, [eax].pga_next ; Sentinel? je short sh_done ; yes, don't try to unlink ; Have block to return push [esi].pga_freeprev ; Current block will be freed cCall UnlinkWin386Block pop esi ; Continue before block freed jmps scan_loop sh_done: pop ax ; Starting value of Win386_Blocks cmp ax, Win386_Blocks ; Set Z flag if heap didn't shrink popad UnSetKernelDS FS ret cEnd ;-----------------------------------------------------------------------; ; UnlinkWin386Block ; ; ; ; Returns a block to Win386 and unlinks it from the heap. ; ; ; ; Arguments: ; ; EBX Block previous to block to be unlinked ; ; ESI Block to be unlinked ; ; ECX Block after to block to be unlinked ; ; ; ; Returns: ; ; ESI Block previous to EBX ; ; ; ; Error Returns: ; ; ; ; Registers Preserved: ; ; ECX, DX, EDI, DS, ES ; ; ; ; Registers Destroyed: ; ; EAX, EBX, EDX ; ; ; ; Calls: ; ; GlobalCompact ; ; UnlinkWin386Block ; ; ; ; History: ; ; ; ;-----------------------------------------------------------------------; assumes ds, nothing assumes es, nothing cProc UnlinkWin386Block, cBegin nogen CheckKernelDS FS ReSetKernelDS FS push dx mov edx, [ecx].pga_next ; Block after all this cmp edx, [edx].pga_next ; Last sentinel? je RSHORT dont_do_it ; Never free last block if KDEBUG push edx mov eax, [esi].pga_size mov edx, eax shr edx, 16 krDebugOut , "UnlinkWin386Block: releasing #dx#AX bytes" pop edx endif push esi call gdel_free ; Remove from free list ifdef WOW push edx push ebx push ecx mov eax,MEM_RELEASE mov edi,[esi].pga_size cCall VirtualFree,<[ebx].pga_lrunext,edi,eax> pop ecx pop ebx pop edx else mov esi, [ebx].pga_lrunext ; Saved WIN386 handle mov di, si shr esi, 16 ; Put in SI:DI DPMICALL 0502h ; Free Memory Block endif; WOW xor edi, edi pop esi dec Win386_Blocks mov eax, [ebx].pga_prev ; Block before all this mov [eax].pga_next, edx ; Unlink them. mov [edx].pga_prev, eax cCall free_arena_header, ; Free the arena headers cCall free_arena_header, cCall free_arena_header, mov esi, eax ; For gfreeall sub [edi].hi_count, 3 if KDEBUG call CheckGlobalHeap endif dont_do_it: pop dx ret UnSetKernelDS FS cEnd nogen cProc gpagingcandidate, cBegin nogen CheckKernelDS FS ReSetKernelDS FS test byte ptr WinFlags[1], WF1_PAGING ; Paging? jz short gpc_not_paging pushad mov ebx, ds:[esi].pga_address mov esi, ds:[esi].pga_size add esi, ebx ; End of region shr ebx, 12 shr esi, 12 cmp ebx, NextCandidate jb short gpc_use_this_page mov ebx, NextCandidate ; Start the region here gpc_use_this_page: cmp esi, ebx jne short call_win386 mov NextCandidate, ebx jmps gpc_done call_win386: mov NextCandidate, -1 sub esi, ebx ; number of pages mov di, si shr esi, 16 mov cx, bx shr ebx, 16 DPMICALL 0700h ; Page Candidate gpc_done: popad gpc_not_paging: ret UnSetKernelDS FS cEnd nogen assumes ds, nothing assumes es, nothing cProc gwin386discard, cBegin nogen CheckKernelDS FS ReSetKernelDS FS cmp ds:[esi].pga_size, 4096 jb short not_a_chance ; Quick exit cmp ds:[esi].pga_size, 16*1024 jb short inform_later pushad mov ebx, ds:[esi].pga_address mov esi, ds:[esi].pga_size mov di, si shr esi, 16 ; SI:DI is # bytes to discard mov cx, bx shr ebx, 16 ; BX:CX is first bytes to discard DPMICALL 0703h popad jmps not_a_chance inform_later: or PagingFlags, 1 not_a_chance: ret UnSetKernelDS FS cEnd nogen cProc DiscardFreeBlocks, cBegin nogen push es push ds GENTER32 ReSetKernelDS FS and PagingFlags, NOT 1 mov esi, ds:[di].phi_first mov esi, ds:[esi].pga_freenext dfb_next: cmp esi, ds:[esi].pga_next ; Sentinel? je short dfb_done ; yes, all done push esi push edi cmp ds:[esi].pga_size, 4096 jb short no_win386discard ; Quick exit mov ebx, ds:[esi].pga_address mov esi, ds:[esi].pga_size add esi, ebx ; First byte past block shr esi, 12 ; Page of this byte add ebx, 0fffh shr ebx, 12 ; First page we can discard sub esi, ebx ; # pages we can discard jbe short no_win386discard ; none to discard mov di, si shr esi, 16 ; SI:DI is # pages to discard mov cx, bx shr ebx, 16 ; BX:CX is first page to discard DPMICALL 0701h ; Say goodbye, pages no_win386discard: pop edi pop esi mov esi, ds:[esi].pga_freenext jmps dfb_next dfb_done: GLEAVE32 UnSetKernelDS FS pop ds pop es ret cEnd nogen ;--------------------------------------------------------------------------- ; ; guc_findfree ; ; helper function for guncompact ; ; finds the next free block below the address ; in ECX into which the block pointed to by ESI will fit. ; ; Entry: ; ECX = maximum address (desired swap area) ; DS:DI = global heap info ; ESI = arena of block to move ; ; Exit: ; Carry clear if block found, set if not ; EDX = free block arena ; ; Uses: ; EAX, EDX ; ; Preserves: ; EBX, ECX, ESI, EDI ; ; History: ; Fri Jun 7, 1991 9:38 -by- Craig Critchley [craigc] ; Wrote it... ; ;-------------------------------------------------------------------------- cProc guc_findfree, cBegin nogen mov edx, ds:[di].phi_last mov edx, ds:[edx].pga_freeprev gucff_check: cmp edx, ds:[edx].pga_prev ; if at start, not found jz short gucff_notfound mov eax, ds:[edx].pga_address ; is it out of swap area add eax, ds:[edx].pga_size cmp eax, ecx jae short gucff_nextblock mov eax, ds:[esi].pga_size ; does it fit cmp ds:[edx].pga_size, eax jb short gucff_nextblock clc ; return it in EDX ret gucff_nextblock: mov edx, ds:[edx].pga_freeprev ; previous free block jmp short gucff_check gucff_notfound: stc ; return error ret cEnd nogen ;----------------------------------------------------------------------------- ; ; guncompact - ; ; this function moves segments that are not free or discardable code ; out of the intended swap area. don't take the name too seriously. ; ; Entry: ; ECX = size of intended swap area ; DS:DI = global heap info ; FS = global heap ; ; Exit: ; Carry clear if space could be cleared, set if not ; ; Registers used: ; EAX, EBX, ECX, EDX, ESI ; ; Called by: ; greserve ; ; Calls: ; gmoveable ; gmovebusy ; ; History: ; Fri Jun 7, 1991 9:38 -by- Craig Critchley [craigc] ; Wrote it... ; ;----------------------------------------------------------------------------- cProc guncompact, cBegin mov edx, [di].phi_last ; point to last block mov ebx, pga_prev sub ecx, ds:[edx].pga_address ; find desired code fence neg ecx guc_trymovingblock: mov esi, ds:[edx+ebx] ; block under current one... cmp ds:[esi].pga_owner, 0 ; don't move free blocks jz short guc_skipblock test ds:[esi].pga_flags, GA_DISCCODE ; don't move discardable code jnz short guc_skipblock cmp ds:[esi].pga_owner, GA_NOT_THERE ; ignore not-there's jz short guc_skipblock call gmoveable ; can this block be moved? jz short guc_error ; if not, swap area toast push edx call guc_findfree ; find a block to move it into pop eax jc short guc_error ; error if didn't find one push eax xchg edx, esi call gmovebusy ; move it xchg edx, esi pop edx ;;; can we just fall thru the block ought to be free now... jmp short guc_trymovingblock ; move what's there now guc_skipblock: mov edx, esi ; this block now first cleared cmp ds:[edx].pga_address, ecx ; big enough? ja short guc_trymovingblock guc_done: clc ; success-o-mundo!! ret guc_error: stc ; if not, we could not ret ; clear swap area cEnd sEnd CODE end