852 lines
25 KiB
NASM
Raw Normal View History

2001-01-01 00:00:00 +01:00
TITLE LCOMPACT - Local memory allocator, compaction procedures
include kernel.inc
errnz la_prev ; This code assumes la_prev = 0
sBegin CODE
assumes CS,CODE
; These are all the external subroutines needed by this source file.
;
externNP <henum> ; HANDLE.ASM
externNP <ljoin,lfree,lfreeadd,lfreedelete> ; LALLOC.ASM
externNP <lnotify,lalign> ; LINTERF.ASM
externFP <GlobalHandle, GlobalRealloc>
;externFP <LocalHeapSize>
; These are all of the internal subroutines defined in this source file.
;
PUBLIC lmove, lbestfit, lcompact
;-----------------------------------------------------------------------;
; lmove ;
; ;
; Moves a moveable block into the top part of a free block. ;
; ;
; Arguments: ;
; BX = address of free block ;
; SI = address of busy block to move ;
; ;
; Returns: ;
; SI = old busy address (new free block) ;
; DI = new busy address ;
; BX = old free block ;
; AX = block after old free block ;
; ;
; Error Returns: ;
; ;
; Registers Preserved: ;
; ;
; Registers Destroyed: ;
; CX,DX,ES ;
; ;
; Calls: ;
; ;
; History: ;
; ;
; Tue Oct 14, 1986 06:22:28p -by- David N. Weise [davidw] ;
; Added this nifty comment block. ;
;-----------------------------------------------------------------------;
cProc lmove,<PUBLIC,NEAR>
cBegin nogen
mov cx,[si].la_next ; Calculate #bytes in busy block
sub cx,si
add si,cx ; SI = bottom of busy block
mov di,[bx].la_next ; DI = bottom of free block
cmp si,bx ; Are the busy and free blocks adjacent?
je move1 ; Yes, always room for free header
mov ax,di ; Calculate the new busy block location
sub ax,cx
sub ax,LA_MINBLOCKSIZE ; See if there is room for two blocks
cmp ax,bx ; in this free block
jae move1 ; Yes, continue
mov ax,di ; No, AX = block after free block
mov di,bx ; New busy block will replace free block
add di,cx ; with some extra slop at end
jmp short move2
move1:
mov ax,di ; AX = block after free block
move2:
dec si ; Predecrement for moving backwards
dec si ; on bogus Intel hardware
dec di
dec di
shr cx,1 ; Move words
push ds ; Initialize for rep movsw
pop es
std ; Move down as may overlap
rep movsw
cld ; Don't hose careless ones
inc si ; More bogosity.
inc si ; SI = old busy address (new free block)
inc di ; DI = new busy address
inc di ; BX = old free block
movex: ; AX = block after old free block
ret
cEnd nogen
;-----------------------------------------------------------------------;
; lbestfit ;
; ;
; Searches backwards for the largest moveable block that will fit ;
; in the passed free block. ;
; ;
; Arguments: ;
; BX = free block ;
; CX = #arena entries left to examine ;
; ;
; Returns: ;
; SI = address of moveable block or zero ;
; ;
; Error Returns: ;
; ;
; Registers Preserved: ;
; ;
; Registers Destroyed: ;
; ;
; Calls: ;
; ;
; History: ;
; ;
; Tue Oct 14, 1986 06:25:46p -by- David N. Weise [davidw] ;
; Wrote it. ;
;-----------------------------------------------------------------------;
cProc lbestfit,<PUBLIC,NEAR>
cBegin nogen
push bx
push cx
push dx
xor si,si ; Have not found anything yet
push si
mov dx,[bx].la_next ; Compute max size to look for
sub dx,bx
bfloop:
mov ax,[bx].la_prev ; Get previous block pointer
test al,LA_BUSY ; Is this block busy?
jz bfnext ; No, continue
test al,LA_MOVEABLE ; Is this block moveable?
jz bfnext ; No, continue
mov si,[bx].la_handle ; Yes, is this block locked?
cmp [si].lhe_count,0
jne bfnext ; No, continue
mov ax,[bx].la_next ; Yes, compute size of this moveable block
sub ax,bx ; Compare to size of free block
cmp ax,dx ; Is it bigger?
ja bf2 ; Yes, continue
pop si ; No, Recover largest block so far
or si,si ; First time?
jz bf0 ; Yes, special case
add ax,si ; No, is this block better than
cmp ax,[si].la_next ; ...the best so far?
jbe bf1 ; No, continue
bf0: mov si,bx ; Yes, remember biggest block
bf1: push si ; Save largest block so far
bf2: mov ax,[bx].la_prev ; Skip past this block
bfnext:
and al,LA_MASK
mov bx,ax
loop bfloop
bfexit:
pop si
pop dx
pop cx
pop bx
ret
cEnd nogen
;-----------------------------------------------------------------------;
; lcompact ;
; ;
; Compacts the local heap. ;
; ;
; Arguments: ;
; DX = minimum #contiguous bytes needed ;
; DI = address of local heap information ;
; ;
; Returns: ;
; AX = size of largest contiguous free block ;
; BX = arena header of largest contiguous free block ;
; DX = minimum #contiguous bytes needed ;
; ;
; Error Returns: ;
; ;
; Registers Preserved: ;
; ;
; Registers Destroyed: ;
; CX,SI,ES ;
; ;
; Calls: ;
; ;
; History: ;
; ;
; Wed March 11, 1987 -by- Bob Gunderson [bobgu] ;
; Added code to maintain free list while compacting. ;
; ;
; Tue Oct 14, 1986 06:27:37p -by- David N. Weise [davidw] ;
; Added this nifty comment block. ;
;-----------------------------------------------------------------------;
cProc lcompact,<PUBLIC,NEAR>
cBegin nogen
x = LHE_DISCARDABLE+1
x = x shl 8
x = x or 1
IncLocalStat ls_lcompact
push si
mov word ptr [di].hi_ncompact,x
errnz <hi_ncompact-hi_dislevel+1>
cmp [di].hi_freeze,0 ; Is the heap frozen?
je compact1 ; No, continue
dec [di].hi_ncompact ; Yes, prevent discarding
compact1:
IncLocalStat ls_cloop
push dx ; Save what we are looking for
xor ax,ax ; Haven't found a free block yet
push ax ; Remember what we have found here
mov bx,[di].hi_last ; Start at end of arena
mov cx,[di].hi_count
; Loop to find the next free block
;
cfreeloop:
IncLocalStat ls_cexamine
mov ax,[bx].la_prev
test al,LA_BUSY
jz cfreefound
and al,LA_MASK
mov bx,ax
cflnext:
loop cfreeloop
compact2:
pop bx ; Recover largest free block so far
pop dx ; Recover size needed
mov ax,bx ; Compute size in AX
or ax,ax ; Did we find a free block?
jz cexit1 ; No, done
sub ax,[bx].la_next ; Yes, compute size of largest free block
neg ax
dec [di].hi_ncompact ; Any other possibilities?
jl cexit ; No, get out now
cmp ax,dx ; Yes, Did we get size we needed?
jb compact3 ; No, try next possibility
cexit:
cexit1:
pop si
ret ; Yes, return to caller
compact3:
push dx
push bx
dec [di].hi_dislevel ; Down to next discard level
jz compact2 ; Ooops, no more, try next step
inc [di].hi_ncompact ; Still discarding
xor si,si ; Start enumerating handle table entries
cdloop:
call henum ; Get next discardable handle
jz cdexit ; No, more see if we discarded anything
push cx ; Got one, see if okay to discard
mov cl,LN_DISCARD ; AX = LN_DISCARD
xchg ax,cx ; CX = discardable flags
mov bx,si ; BX = handle
call lnotify
pop cx
or ax,ax ; Is it still discardable?
jz cdloop ; No, skip this handle
mov bx,[si].lhe_address ; Get true address of a block
sub bx,SIZE LocalArena ; BX = address of block to free
call lfree ; Free the block associated with this handle
xor ax,ax ; Zero the true address field in the
mov [si].lhe_address,ax
or [si].lhe_flags,LHE_DISCARDED ; and mark as discarded
or [di].hi_ncompact,80h ; Remember we discarded something
jmp cdloop
cdexit:
test [di].hi_ncompact,80h ; No, did we discarded something?
jz compact2
xor [di].hi_ncompact,80h ; Yes, clear flag
pop bx
pop dx
jmp compact1 ; and try compacting again
; Here when we have a free block. While the preceeding block is
; moveable, then move it down and put the free space in it place.
; When the preceeding block is not moveable, then search backwards
; for one or more moveable blocks that are small enough to fit
; in the remaining free space. Advance to previous block when
; no more blocks to examine.
cfreefound:
IncLocalStat ls_cfree
cmp [di].hi_freeze,0 ; Is the heap frozen?
jne cffexit ; No, continue
; and al,LA_MASK ; Already clear for free blocks
mov si,ax ; SI = previous block
test byte ptr [si].la_prev,LA_MOVEABLE
jz cfreefill ; Skip if not moveable
mov si,[si].la_handle ; Point to handle table entry
cmp [si].lhe_count,0 ; Is it locked?
jne cfreefill ; Yes, skip this block
; Here if previous block is moveable. Slide it up to the top of the
; free block and make the old busy block free. This is easy as
; we are not really adding or taking anything away from the arena
;
push cx ; Save loop state regs
push di
mov si,ax ; SI = busy block before free block
call lfreedelete ; remove [BX] from free list
call lmove ; Move busy block down
mov [di].la_prev,si ; Link in new busy block
or byte ptr [di].la_prev,LA_MOVEABLE+LA_BUSY
mov bx,ax
mov [di].la_next,bx ; la_next field from old free block
and [bx].la_prev,LA_ALIGN
or [bx].la_prev,di
mov [si].la_next,di ; Link in old busy block (new free block)
and byte ptr [si].la_prev,LA_MASK ; mark it free
push si
push bx
mov bx,si
xor si,si
call lfreeadd ; add new free block to free list
pop bx
pop si
mov bx,[di].la_handle ; Modify handle table entry to point
lea ax,[di].SIZE LocalArena
mov [bx].lhe_address,ax ; to new location of client data
pop di ; Restore arena info pointer
IncLocalStat ls_cmove
mov al,LN_MOVE ; Tell client we moved it
lea cx,[si].SIZE LocalArena ; CX = old client data
call lnotify ; BX = handle
pop cx ; restore #arena entries left
mov bx,si ; BX = address of free block
mov si,[bx].la_prev ; SI = previous block
test byte ptr [si].la_prev,LA_BUSY ; Is it free?
jnz cffnext ; No, continue
call ljoin ; Yes, coelesce with block in BX
;;; DO NOT change cx, ljoin leaves BX pointing after the free block
;;; dec cx ; ...keep CX in sync
cffnext:
jmp cflnext ; Go process next free block
; Here when done with a free block. Keep track of the largest free block
; seen so far.
;
cffexit:
pop si ; Recover largest free block so far
cmp si,bx ; Same as current?
je cff1 ; Yes, no change then
test [bx].la_prev,LA_BUSY ; No, is current free?
jnz cff1 ; No, ignore it then
or si,si ; Yes, First time?
jz cff0 ; Yes, special case
mov ax,[si].la_next ; No, compute size of largest free block
sub ax,si
add ax,bx ; Compare to size of this free block
cmp [bx].la_next,ax ; Is it bigger?
jbe cff1 ; No, do nothing
cff0: mov si,bx ; Yes, remember biggest free block
cff1: push si ; Save largest free block so far
cff2: mov bx,[bx].la_prev ; Skip past this free block
and bl,LA_MASK ; (it might not be a free block)
jmp cffnext
; Here if previous block is NOT moveable. Search backwards for the
; largest moveable block that will fit in this free block. As long
; as a block is found, move it to the top of the free block, free it
; from it's old location and shrink the current free block to accomodate
; the change.
;
cfreefill:
call lbestfit ; Search for best fit
or si,si ; Find one?
jz cffexit ; No, all done with this free block
push cx ; Save loop state regs
push di
push [bx].la_prev ; Save busy block before free block
call lfreedelete ; remove [BX] from free list
call lmove ; Move it down
; SI = old busy address (new free block)
; DI = new busy address
; BX = old free block
; AX = block after old free block
pop cx ; Recover la_prev field of old free block
cmp bx,di ; Did we consume the free block?
je cff ; Yes, then CX has what we want
mov cx,bx ; No, then busy block will point to what
mov [bx].la_next,di ; is left of the free block and vs.
cff:
mov [di].la_prev,cx ; Link in new busy block
or byte ptr [di].la_prev,LA_MOVEABLE+LA_BUSY
mov [di].la_next,ax ; la_next field from old free block
xchg di,ax
and [di].la_prev,LA_ALIGN
or [di].la_prev,ax
xchg di,ax
lea cx,[di].SIZE LocalArena
cmp bx,di ; Did we create a free block?
je cffff
push si
xor si,si
call lfreeadd ; Add [BX] to free list
pop si
cffff:
cmp bx,di ; Did we create a free block?
mov bx,di ; Current block is busy block
mov di,[di].la_handle ; Modify handle table entry to point
xchg [di].lhe_address,cx ; to new location of client data
pop di ; Restore arena info pointer
pop ax ; restore #arena entries left
je cfff ; No, arena count okay
inc ax ; Keep arena count on target
inc [di].hi_count
cfff:
push bx ; Save current block pointer
push ax ; Save #arena entries left
mov al,LN_MOVE ; Tell client we moved it (CX = old addr)
mov bx,[bx].la_handle ; BX = handle
call lnotify
pop cx ; restore #arena entries left
and byte ptr [si].la_prev,not LA_MOVEABLE ; Clear moveable bit
mov bx,si ; BX = address of old busy block to free
push [di].hi_count
call lfree ; Mark it as free
pop si
sub si,[di].hi_count
sub cx,si ; Keep arena count on target
pop bx ; BX = current block
jmp cff2 ; Move to previous block.
cEnd nogen
;-----------------------------------------------------------------------;
; lshrink ;
; ;
; Shrinks the local heap. ;
; ;
; Arguments: ;
; DS:DI = address of local heap information block ;
; BX = Minimum size to shrink heap ;
; ;
; Returns: ;
; AX = New size of local heap ;
; ;
; Error Returns: ;
; None. ;
; ;
; Registers Preserved: ;
; None ;
; Registers Destroyed: ;
; All ;
; Calls: ;
; ;
; History: ;
; ;
; Fri July 17, 1987 -by- Bob Gunderson [bobgu] ;
; Changed logic so we don't call lcompact every time we move ;
; a block, just call it whenever we can't find room. ;
; ;
; Fri June 12, 1987 -by- Bob Gunderson [bobgu] ;
; Wrote it. ;
;-----------------------------------------------------------------------;
cProc lshrink,<PUBLIC,NEAR>
cBegin nogen
; Bound the minimum size to the size specified at LocalInit() time.
mov ax,[di].li_minsize
cmp ax,bx ; specified smaller than min ?
jbe lshr_around ; no - use what's specified
mov bx,ax ; yes - use real min.
lshr_around:
push bx ; Save minimum heap size
mov ax,[di].hi_last
add ax,SIZE LocalArenaFree
sub ax,[di].hi_first ; ax = current heap size
cmp ax,bx ; already small enough ?
ja lshr_around1
jmp lshr_exit ; yes - that was quick...
lshr_around1:
; local compact to get as much contiguous free space as possible.
stc
call lalign
call lcompact ; compact everything
xor dx,dx ; start with null flag
; Start with last block in heap.
mov si,[di].hi_last
loop1:
; SI = address of block just moved.
; This section moves moveable blocks (one at a time) up
; into free blocks below (higher addresses) all fixed blocks.
; The blocks they vacate are freed. The sentenal block is then
; moved, the heap is re-compacted. Then a test is made to see if
; the heap has shrunk enough, if not
; we start this loop again. We continue this until we can't
; move anything or we have moved enough to satisfy the minimum
; size specified by BX when called.
mov si,[si].la_prev ; get next block to move
and si,LA_MASK ; zap the flag bits
mov ax,[si].la_prev ; get this blocks flags
test ax,LA_BUSY ; is this block free?
jz lshr_gobblefree ; yes - gobble this free block
test ax,LA_MOVEABLE ; is it moveable ?
jz lshr_cantmove ; no - exit
mov bx,[si].la_handle ; get its handle
cmp [bx].lhe_count,0 ; is it locked ?
jnz lshr_cantmove ; yes - exit
mov cx,[si].la_next
sub cx,si ; get block size in cx
; Find the first free block (after fixed blocks) that is big
; enough for this block
loop2:
call lfirstfit
jnc lshr_gotblock ; got one - continue
; If we have already tried compacting, nothing more to do
or dx,dx
jnz lshr_cantmove ; nothing more to do...
; Not enough room found, recompact the local heap and test again
push si
push cx
stc
call lalign
call lcompact ; compact everything
pop cx
pop si
mov dx,1 ; flag to say we have done this
jmp loop2 ; try again
; DI = local arena info block address
; SI = block we didn't move (next is last block moved)
lshr_cantmove:
mov si,[si].la_next ; get last block moved
jmp lshr_alldone ; cleanup and exit
lshr_gobblefree:
; SI = address of free block
; Block to move is already free, remove it from the free list and mark
; it a busy.
mov bx,si
call lfreedelete
or [bx].la_prev,LA_BUSY
jmps lshr_shiftup
lshr_gotblock:
; DI - Arena info block address
; SI - Address of block to move
; BX - Address of free block to use
; CX - Size of block to move
call lfreedelete ; remove it from the free list
xchg di,bx
mov ax,[di].la_next
sub ax,cx
sub ax,LA_MINBLOCKSIZE ; Room to split free block into
cmp ax,di ; two blocks ?
jb lshr_nosplit ; no - don't split the block
push bx
push si
mov bx,cx
lea bx,[bx+di] ; bx = address of new free block to add
; link this new block in
mov [bx].la_prev,di
mov ax,[di].la_next
mov [bx].la_next,ax
mov [di].la_next,bx
xchg ax,bx
and [bx].la_prev,LA_ALIGN ; Zap only high bits
or [bx].la_prev,ax ; and set new previous pointer
mov bx,ax
mov si,[di].la_free_prev ; previous free block
call lfreeadd ; add the new smaller free block
pop si
pop bx
inc [bx].hi_count ; bump the block count (we added
; a block)
lshr_nosplit:
; BX - Arena info block address
; SI - Address of block to move
; DI - Address of free block to use
; CX - Size of block to move
; copy the flags from the old block
mov ax,[si].la_prev
and ax,LA_ALIGN
or [di].la_prev,ax
push si
push di
; don't copy the block headers (but do copy the handle)
add si,la_fixedsize
add di,la_fixedsize
sub cx,la_fixedsize
; We can take # bytes/2 as # words to move because all blocks
; start on even 4 byte boundary.
shr cx,1 ; bytes / 2 = words
push ds
pop es ; same segment
cld ; auto-increment
rep movsw ; Head 'em up! Move 'em out!
pop di
pop si
; BX - Arena info block address
; SI - Address of block to move
; DI - Address of free block to use
; Fixup the handle table pointer
push bx
mov bx,[si].la_handle
lea ax,[di].SIZE LocalArena
mov [bx].lhe_address,ax
pop di ; DI = info block address
; Mark the old block as busy fixed
and [si].la_prev,LA_MASK ; clear old bits
or [si].la_prev,LA_BUSY ; put in busy fixed bit
jmps lshr_shiftup
; Time to shift the setenal block up, recompact and look for more
; space...
; DI = local arena info block address
; SI = block just freed
public foo
foo:
lshr_shiftup:
mov ax,[di].hi_last
mov bx,ax
sub bx,si ; less what is already removed
sub ax,bx
add ax,SIZE LocalArenaFree ; size of sentenal block
sub ax,[di].hi_first ; ax = current heap size
pop bx ; get min size
push bx ; and put it back on stack
cmp ax,bx ; already small enough ?
jbe lshr_alldone ; yes - exit
xor dx,dx ; localcompact flag
jmp loop1 ; and back for more
lshr_alldone:
; DI = local arena info block address
; SI = last block moved
; Time to shift the sentenal block up and realloc our heap
pop bx ; Minimum size of the heap
push bx
call slide_sentenal ; slide the sentenal block up
; get our data segment handle
push ds
call GlobalHandle ; ax = handle of DS
or ax,ax
jz lshr_exit ; this can't happen.....
push ax ; hMem
; determine how big we should be
mov ax,[di].hi_last
add ax,SIZE LocalArenaFree
xor cx,cx
push cx
push ax ; # bytes in DS
push cx ; no wFlags
call GlobalRealloc
; local compact to get as much contiguous free space as possible.
lshr_exit:
stc
call lalign
call lcompact ; compact everything
pop ax ; clean the stack
mov ax,[di].hi_last
add ax,SIZE LocalArenaFree
sub ax,[di].hi_first ; ax = current heap size
ret
cEnd nogen
;-------------------------------------------------------------------------
; Find first free block after last fixed block. We do this by scanning
; through the free list looking for a free block with the next physical
; block being moveable (remember we just did a compact...). Then, starting
; with this free block, find the first free block that is at least
; CX bytes long.
;
; Entry:
; DI = local arena header info block address
; CX = # bytes needed
;
; Exit:
; BX = Address of free block to use
;
; Error Return:
; Carry set if no free block big enough
;
; History:
;
; Fri June 12, 1987 -by- Bob Gunderson [bobgu]
; Wrote it.
;-------------------------------------------------------------------------
cProc lfirstfit,<PUBLIC,NEAR>
cBegin nogen
push ax
mov bx,[di].hi_last
mov bx,[bx].la_free_prev
; mov bx,[di].hi_first
; mov bx,[bx].la_free_next
lffloop:
cmp bx,[bx].la_free_prev
; cmp bx,[bx].la_free_next
jz lff_err_exit
cmp cx,[bx].la_size
jbe lff_exit
mov bx,[bx].la_free_prev
; mov bx,[bx].la_free_next
jmp lffloop
lff_err_exit:
stc
jmps lff_done
lff_exit:
clc ; good return
lff_done:
pop ax
ret
cEnd nogen
;-------------------------------------------------------------------------
; slide_sentenal
; This routine is called during the LocalShrink() operation.
; Make all the blocks between the one at SI and the sentenal go
; away. Then check to see if we have shrunk the heap too much. If
; so, then add a free block at SI big enough to bump the size up (this
; will be moved later by LocalCompact). Now move the sentenal block
; up after the last block.
;
; Entry:
; BX = requested minimum size of the heap
; SI = last block actually moved (put sentenal here)
; DI = local arena info block address
;
; Exit:
;
; Error Return:
;
; History:
;
; Fri Aug 14, 1987 -by- Bob Gunderson [bobgu]
; Changed things again to insure we don't shrink to heap too much.
;
; Fri July 17, 1987 -by- Bob Gunderson [bobgu]
; Was too slow, so changed the logic.
;
; Fri June 12, 1987 -by- Bob Gunderson [bobgu]
; Wrote it.
;-------------------------------------------------------------------------
cProc slide_sentenal,<PUBLIC,NEAR>
cBegin nogen
push bx
; Get sentenal block
mov bx,[di].hi_last
; Insure that we aren't trying to move to ourselves
cmp si,bx
jz slide_exit ; nothing to do....
; count the number of block we are removing so that we can update
; the block count in the arena header.
push si
xor cx,cx
slide_loop:
cmp si,bx
jz slide_loop_exit
inc cx
mov si,[si].la_next
jmp slide_loop
slide_loop_exit:
pop si
sub [di].hi_count,cx ; decrement the count of blocks
; Would the heap be too small if the sentenal were moved to the block
; pointed to by si? If so, add a free block at SI to make up the
; difference.
mov ax,si
sub ax,[di].hi_first
add ax,la_freefixedsize ; size of heap in ax
pop bx ; minimum in bx
cmp ax,bx
ja slide_ok ; everything is ok...
sub bx,ax ; bx = size to grow
cmp bx,LA_MINBLOCKSIZE
jbe slide_ok ; not enough for a block
clc
call lalign ; make it even 4 byte boundary
mov bx,dx
; add a free block of bx bytes
lea ax,[si+bx] ; address of new "next" block
mov [si].la_next,ax ; make block point at "next"
and [si].la_prev,LA_MASK ; make it a free block
xor bx,bx
xchg si,bx ; bx = block to add, si = 0
call lfreeadd ; preserves ax
inc [di].hi_count ; bump block count
mov si,ax ; and bump si to "next" block
mov [si].la_prev,bx ; set new block's previous pointer
slide_ok:
; move the sentenal block up by copying the words.
mov bx,[di].hi_last ; old sentenal block
mov [si].la_next,si ; new sentenal points to self
and [si].la_prev,LA_MASK ; remove any old flags
or [si].la_prev,LA_BUSY ; make it busy
mov ax,[bx].la_size
mov [si].la_size,ax ; move in the size
mov bx,[bx].la_free_prev
mov [si].la_free_prev,bx ; move in previous free block
mov [bx].la_free_next,si ; point prev free block to this one
mov [si].la_free_next,si ; point to itself
; Now fixup the arena header block to point to the new setenal
; position.
mov [di].hi_last,si
; And we are done...
jmps slide_exit1
slide_exit:
pop bx
slide_exit1:
ret
cEnd nogen
sEnd CODE
end