Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

1055 lines
32 KiB

TITLE LDAUX - Assembler side of LD.C
.xlist
include kernel.inc
include newexe.inc
include tdb.inc
include protect.inc
.list
DataBegin
externB fChksum
externW hExeHead
externW curTDB
externW PHTcount
externW kr1dsc
szFake32BitModuleName DB 128 DUP(0)
cFake32BitModuleName DW SIZE szFake32BitModuleName
ifdef WOW
szWinGDllModule DB 'WING', 0
szWinGDllFile DB '\SYSTEM\WING.DLL',0
externD lpWindowsDir
externW cBytesWinDir
endif ; WOW
ifdef DBCS_KEEP
ifdef WOW
public hModNotepad
hModNotepad DW 0
NotepadName DB 'notepad.exe', 0
endif ; WOW
endif ; DBCS_KEEP
DataEnd
externFP GlobalAlloc
externFP GlobalFree
externFP GlobalHandle
externFP GetExePtr
externFP FarMyUpper
externFP FarFindOrdinal
externFP FarEntProcAddress
externFP FarMyLock
externFP FarGetOwner
externFP AllocSelector
externFP IPrestoChangoSelector
externFP FreeSelector
externFP lstrcpyn
ifdef WOW
ifdef DBCS_KEEP
externFP WowGetModuleUsage
externFP MyGetAppWOWCompatFlags
endif ; DBCS_KEEP
externFP WowGetModuleFileName
externFP HandleAbortProc
externFP WowGetModuleHandle
endif
ifdef FE_SB
externFP FarMyIsDBCSLeadByte
endif
sBegin CODE
assumes CS,CODE
externNP LoadSegment
externNP MyLock
externNP GetOwner
if LDCHKSUM
externNP GetChksumAddr
endif
;-----------------------------------------------------------------------;
; GetSegPtr ;
; ;
; ;
; Arguments: ;
; DS:SI = FARPROC or DS = hModule and SI = segment# ;
; ;
; Returns: ;
; BX = 0 ;
; CX = segment# or zero if input invalid ;
; DS:SI -> segment table entry ;
; ;
; Error Returns: ;
; ;
; Registers Preserved: ;
; ;
; Registers Destroyed: ;
; ;
; Calls: ;
; ;
; History: ;
; ;
; Sun 24-Dec-1989 22:49:50 -by- David N. Weise [davidw] ;
; Added check for MPI thunk. ;
; ;
; Wed Oct 07, 1987 12:59:54p -by- David N. Weise [davidw] ;
; Added this nifty comment block. ;
;-----------------------------------------------------------------------;
assumes ds, nothing
assumes es, nothing
cProc GetSegPtr,<PUBLIC,NEAR>
cBegin nogen
xor bx,bx ; For cheap indexed addressing
cmp ds:[bx].ne_magic,NEMAGIC; Is this a thunk address?
jne gspFixed ; No, must be fixed procedure
dec si ; No, see if valid segment #
cmp si,ds:[bx].ne_cseg
jb @F
jmps gspfail ; No, fail
@@: mov cx,si ; Yes, load that segment
inc cx
jmps gspSegNo
; CX = segment number. Convert to segment info pointer.
gspSegNo:
push cx
dec cx
shl cx,1
mov si,cx
shl cx,1
shl cx,1
add si,cx
.errnz 10 - SIZE NEW_SEG1
add si,ds:[bx].ne_segtab
pop cx
jmps gspExit ; DS:SI -> segment info block
gspFixed: ; DS = code segment address
; check for MakeProcInstance thunk first because GetOwner in 386pmode
; ain't robust enough
cmp byte ptr ds:[si].0,0B8h ; Maybe, is this a mov ax,XXXX inst?
jnz gsp_not_mpit
cmp byte ptr ds:[si].3,0EAh ; Followed by far jump inst?
jnz gsp_not_mpit
cmp ds:[2],MPIT_SIGNATURE ; Is it in a mpi table?
jz gsp_yes_mpit
cmp ds:[TDB_MPI_THUNKS].2,MPIT_SIGNATURE
jnz gsp_not_mpit
gsp_yes_mpit:
lds si,ds:[si].4 ; get real procedure address
jmp GetSegPtr
gsp_not_mpit:
push ax
mov dx, ds ; Handle in dx
StoH dl ; Assume not fixed
cCall GetOwner,<ds>
or ax,ax
mov ds, ax
pop ax
jz gspfail
gspf1:
cmp ds:[bx].ne_magic,NEMAGIC; Is it a module DB?
jnz gspfail ; Nope, fail.
getting_closer:
mov si,ds:[bx].ne_segtab ; Yes, point to segment table
mov cx,ds:[bx].ne_cseg
gspLoop:
cmp dx,ds:[si].ns_handle ; Scan stmen
jz gspFound
HtoS dx ; May be fixed seg???
cmp dx,ds:[si].ns_handle ; Scan stmen
jz gspFound
StoH dx
add si,SIZE NEW_SEG1
loop gspLoop
gspfail:
xor cx,cx
jmps gspExit
gspFound:
sub cx,ds:[bx].ne_cseg ; Compute segment# from remainder
neg cx ; of loop count
inc cx ; 1-based segment#
gspExit:
ret
cEnd nogen
assumes ds, nothing
assumes es, nothing
cProc IGetCodeInfo,<PUBLIC,FAR>,<si,di>
parmD lpProc
parmD lpSegInfo
cBegin
lds si,lpProc
call getsegptr
mov ax,cx
jcxz gciExit
les di,lpSegInfo
mov cx,SIZE NEW_SEG1
cld
rep movsb
mov ax,ds:[bx].ne_align ; Return segment aligment
stosw
sub si,SIZE NEW_SEG1
sub di,SIZE NEW_SEG1+2
cmp si,ds:[bx].ne_pautodata
jne gciExit
mov ax,ds:[bx].ne_stack
add ax,ds:[bx].ne_heap
add es:[di].ns_minalloc,ax
gciExit:
smov es,ds ; put module handle (returned from getsegptr) into es
; user depends on this
mov cx,ax ; WHY IS THIS HERE?
;
; NOTE: USER assumes that AX == BOOL fSuccess and ES == hModule upon return
;
cEnd
assumes ds, nothing
assumes es, nothing
cProc GetCodeHandle,<PUBLIC,FAR>,<si,di>
parmD lpProc
cBegin
lds si,lpProc
call getsegptr
mov dx,cx
jcxz gchExit
mov ax,ds:[si].ns_handle
; test ds:[si].ns_flags,NSLOADED
; jnz gchExit
mov dx,-1
cCall LoadSegment,<ds,cx,dx,dx>
cCall MyLock,<ax>
xchg ax,dx
cmp ax,dx
je gchExit
; %OUT Need pmode lru stuff here
gchExit:
cEnd
assumes ds,nothing
assumes es,nothing
cProc CallProcInstance,<PUBLIC,FAR>
cBegin nogen
mov ax,ds ; AX = caller's DS
mov cx,es:[bx] ; CX = hInstance
jcxz cpx ; If zero, then use caller's DS
xchg ax,cx ; AX = hInstance, CX = caller's DS
test al,GA_FIXED ; Fixed?
jnz cpx ; Yes, all set
xchg bx,ax ; No, BX = hInstance, ES:AX ->
labelFP <PUBLIC,CallMoveableInstanceProc> ; procedure address.
HtoS bx ; Get selector
xchg bx,ax ; AX = segment address
mov ds,cx ; Restore DS
cpx: jmp dword ptr es:[bx][2] ; Jump to far procedure
cEnd nogen
sEnd CODE
sBegin NRESCODE
assumes CS,NRESCODE
externNP MapDStoDATA
externNP FindExeInfo
externNP FindExeFile
externNP NResGetPureName
assumes ds,nothing
assumes es,nothing
cProc CopyName,<PUBLIC,NEAR>,<si>
parmD pname
parmW pdst
parmW fUpper
cBegin
les si,pname
mov bx,pdst
mov cx,127
mov dx,fUpper
cn0:
lods byte ptr es:[si]
or al,al
jz cn1
or dx,dx
jz cn0a
ifdef FE_SB
call FarMyIsDBCSLeadByte ; test if a char is lead byte of DBC
jc @F ; jump if not a DBC
inc bx
mov ss:[bx],al ; store first byte of DBC
dec cx
jcxz cn1 ; pay attention for counter exhaust...
lods byte ptr es:[si] ; fetch a 2nd byte of DBC
jmps cn0a
@@:
endif
call FarMyUpper
cn0a:
inc bx
mov ss:[bx],al
loop cn0
cn1:
mov byte ptr ss:[bx+1],0
mov ax,bx
mov bx,pdst
sub ax,bx
mov ss:[bx],al
cEnd
assumes ds,nothing
assumes es,nothing
cProc IGetProcAddress,<PUBLIC,FAR>,<ds,si>
parmW hinstance
parmD pname
localV namebuf,130
cBegin
mov cx,hinstance
jcxz use_current_module
cCall GetExePtr,<cx>
xor dx,dx
jcxz gpdone1
mov si,ax
mov es,ax
test es:[ne_flags],NENOTP
jnz have_module_address
xor bx,bx
if KDEBUG
fkerror 00FFh,<Can not GetProcAddress a task.>,es,bx
endif
xor ax,ax
xor dx,dx
gpdone1:
jmps gpdone
use_current_module:
cCall MapDStoDATA
ReSetKernelDS
mov es,curTDB
UnSetKernelDS
mov si,es:[TDB_pModule]
have_module_address:
cmp seg_pname,0
jne gp0
mov ax,off_pname
jmps gpaddr
gp0:
lea bx,namebuf
xor dx,dx
cCall CopyName,<pname,bx,dx>
lea bx,namebuf
mov dx,-1
cCall FarFindOrdinal,<si,ssbx,dx>
cwd ; set DX to 0 if no ordinal
or ax,ax
jz gpdone ; if no ordinal, leave with 0:0
gpaddr:
cCall FarEntProcAddress,<si,ax>
gpdone:
mov cx,ax
or cx,dx
cEnd
;-----------------------------------------------------------------------;
; GetModuleHandle
;
; Returns the module handle, AKA segment address, of the named
; module. The pname should be a pointer to a name, however
; because we want FreeFontResource to take a file name we check
; for that one special case for file names.
;
; Entry:
; parmW pname
; or 0:instance handle
; Returns:
; AX = handle if loaded
; = 0 if not loaded
;
; Registers Destroyed:
; BX,CX,DX,ES
;
; History:
; Sat 18-Nov-1989 21:57:24 -by- David N. Weise [davidw]
; Adding the handling of exeheaders above the line by calling
; off to EMS_GetModuleHandle. A little while ago is when
; the checking for filenames was added.
;
; Tue 04-Apr-1989 01:42:12 -by- David N. Weise [davidw]
; This used to return DX = ExeHead, this is now returned in CX.
;
; Tue 28-Mar-1989 17:28:34 -by- David N. Weise [davidw]
; Put in the Excel hack and added this nifty comment block.
;-----------------------------------------------------------------------;
assumes ds,nothing
assumes es,nothing
cProc IGetModuleHandle,<PUBLIC,FAR>,<di,si>
parmD pname
localV nameBuf,130
cBegin
cCall MapDStoDATA
ReSetKernelDS
cmp pname.sel,0
ifndef DBCS_KEEP
jz gm1
else ; DBCS_KEEP
ifdef WOW
jnz gm_chk
jmp gm1
for Lotus Freelance Instration program
gm_chk:
call MyGetAppWOWCompatFlags
test ax, 4 ; WOWCF_AUDITNOTEPAD
jz gm_ne
lea si,NotepadName
les di,pname
mov cx,11
cmpsb
jnz gm_ne
mov ax,hModNotepad
or ax,ax
jz gm_ne
cmp ax,1
jnz gm_pe
mov bx,10
xor cx,cx
cCall GlobalAlloc,<cx,cx,bx>
or ax,ax
jz gm_hdl
mov hModNotepad,ax
gm_pe:
jmp short gmexit
gm_ne:
endif ; WOW
endif ; DBCS_KEEP
lea di,namebuf
xor dx,dx ; Try as is first
cCall CopyName,<pname,di,dx>
inc di ; point past count
push ax
cCall FindExeInfo,<ss,di,ax>
pop bx
or ax,ax
jnz gmexit
lea di,namebuf
mov dx,-1 ; Now force upper case
cCall CopyName,<pname,di,dx>
inc di ; point past count
push ax
cCall FindExeInfo,<ss,di,ax>
pop bx
or ax,ax
jnz gmexit
smov es,ss
call NResGetPureName
cCall FindExeFile,<ss,di>
jmps gmexit
gm1: cCall GetExePtr,<OFF_pname>
gmexit:
ifdef WOW
;
; If we didn't find a matching module, call WOW32 to see if
; the module name matches the module name of a child app
; spawned via WinOldAp. If it does, WOW32 will return
; that WinOldAp's hmodule. If not, WOW32 will return zero.
;
or ax,ax
jnz @F
lea di,namebuf[1]
cCall WowGetModuleHandle, <ss,di>
@@:
endif
mov dx,hExeHead ; return this as a freebie
cEnd
assumes ds,nothing
assumes es,nothing
cProc IGetModuleName,<PUBLIC,FAR>,<si,di>
parmW hinstance
parmD lpname
parmW nbytes
cBegin
mov ax, nbytes
or ax, ax
jnz @f
jmp gfx
@@: cCall GetExePtr,<hinstance>
or ax,ax
jz gmn
mov ds,ax
mov si,ds:[ne_restab] ; Module name is first entry in Res Name Table
xor cx,cx
mov cl,ds:[si] ; Get Len of module name
inc cl ; Space for NULL
inc si ; Goto start of module name
cmp cx,nbytes
jl gmn1
mov cx,nbytes
gmn1:
push lpname
push ds
push si
push cx
call lstrcpyn
mov ax,0DDh ; Return a true value
gmn:
cEnd
;-----------------------------------------------------------------------
; IsWingModule
; Checks if GetModuleFileHandle is called by Wing.dll
; on itself.
; and fills lpname if it is so
;
; Arguments:
; hinstance - GetModuleFileName's hinstance arg
; lpname - GetModuleFileName's lpname arg
; nbytes - GetModuleFileName's lpname arg
; exehdr - ExeHdr of hinstance
; rcsel - Return Code Selector that called GMFH
; Returns:
; AX = number of bytes copied if wing is calling GetModuleFileName on itself
; AX = 0 otherwise
;
;------------------------------------------------------------------------
cProc IsWingModule,<PUBLIC,FAR>,<si,di,ds,es>
parmW hinstance
parmD lpname
parmW nbytes
parmW exehdr
parmW rcsel
cBegin
mov es,exehdr
mov di,es:[ne_restab]
inc di ;got modulename
cCall MapDStoDATA
assumes DS, DATA
lea si, szWinGDllModule
mov cx,5
repe cmpsb
jnz short IWM_NoMatch
cCall getexeptr,<rcsel>
cmp ax,exehdr
jnz short IWM_NoMatch
mov cx, cBytesWinDir
add cx, 17 ;17=strlen("\system\wing.dll")+1;
cmp cx,nbytes
ja short IWM_NoMatch ;return righ away if doesn't fit
mov ax,cx
sub cx,17
les di,lpname
lds si,lpWindowsDir
cld
rep movsb
mov cx,17
lea si,szWinGDllFile
rep movsb
jmp short IWM_End
IWM_NoMatch:
mov ax,0
IWM_End:
cEnd
;
;NOTE: BX preserved because Actor 4.0 depends on this (bug 9078 - neilk)
;
cProc IGetModuleFileName,<PUBLIC,FAR>,<si,di,bx>
parmW hinstance
parmD lpname
parmW nbytes
cBegin
ifdef WOW
; take care of 32bit hInstance. 32bit hInstance has GDT/LDT bit clear
; the validation layer will pass such handles as valid so that we can
; thunk to 32bit GetModuleFileName if needed.
;
; in win32 hInstances are not global. therefore fake success by
; returning a valid module name for compatibility sake. Naturally,
; we will use our module name.
mov ax, hInstance
test al, 0100b ; Check for task aliases (see WOLE2.C) or BOGUSGDT
jnz GMFName_Normal
or ax, ax
jz GMFName_Normal
cCall MapDStoDATA
assumes DS, DATA
mov cx, cFake32BitModuleName
lea si, szFake32BitModuleName
cCall WowGetModuleFileName, <ax, ds, si, cx>
mov cx,ax
jmps GMF_CopyName32
GMFName_Normal:
endif
mov ax, nbytes
or ax, ax
jz gfx
cCall GetExePtr,<hinstance>
or ax,ax
jz gfx
mov ds,ax
cCall IsWinGModule,<hinstance,lpname,nbytes,ax,[bp+4]>
or ax, ax
jnz gfx
mov si,ds:[ne_pfileinfo]
xor cx,cx
mov cl,ds:[si].opLen
sub cx,opFile
lea si,[si].opFile
ifdef WOW
GMF_CopyName32:
endif
les di,lpname
cmp cx,nbytes
jl gf1
mov cx,nbytes
dec cx
gf1:
cld
mov ax,cx
rep movsb
mov es:[di],cl
gfx:
mov cx,ax
;** Nasty hack to support QuickWin libs (Fortran, QCWin)
;** The startup code assumes DX will be the segment of the
;** lpname parameter
mov dx,WORD PTR lpname[2]
cEnd
assumes ds,nothing
assumes es,nothing
cProc IGetModuleUsage,<PUBLIC,FAR>
parmW hinstance
cBegin
ifdef DBCS_KEEP
ifdef WOW
cCall MapDStoDATA
ReSetKernelDS
call MyGetAppWOWCompatFlags
test ax, 4 ; WOWCF_AUDITNOTEPAD
jz gu_ne
mov ax,hModNotepad
or ax,ax
jz gu_ne
cmp ax,hinstance
jnz gu_ne
push ax
cCall WowGetModuleUsage, <0>
or ax,ax
pop bx
jnz gux
cCall GlobalFree,<bx>
jmp short gux
gu_ne:
endif ; WOW
endif ; DBCS_KEEP
cCall GetExePtr,<hinstance>
or ax,ax
jz gux
got_one:
mov es,ax
mov ax,es:[ne_usage]
gux: mov cx,ax
cEnd
assumes ds,nothing
assumes es,nothing
cProc IGetInstanceData,<PUBLIC,FAR>,<si,di>
parmW hinstance
parmW psrcdst
parmW nbytes
cBegin
push ds
cCall GlobalHandle,<hinstance>
pop es ; Get caller's DS as destination
or dx,dx
jz gidone
mov ds,dx ; Source is passed instance
mov si,psrcdst ; Offsets are the same
mov di,si
mov ax,nbytes
mov cx,ax
jcxz gidone
cld
rep movsb
push es
pop ds
gidone: mov cx,ax
cEnd
sEnd NRESCODE
sBegin MISCCODE
assumes cs, misccode
assumes ds, nothing
assumes es, nothing
externNP MISCMapDStoDATA
;-----------------------------------------------------------------------;
; MakeProcInstance ;
; ;
; Cons together a far procedure address with a data segment address, ;
; in the form ;
; ;
; mov ax,DGROUP ;
; jmp far pproc ;
; ;
; This procedure allocates a fixed segment for a set of thunks and ;
; threads the thunks together on a free list within the segment. ;
; When a thunk segment is full, a new one is allocated and pushed ;
; onto the head of the list of thunk segments pointed to by ;
; TDB_MPI_Thunks. ;
; ;
; Arguments: ;
; parmD pproc ;
; parmW hinstance ;
; ;
; Returns: ;
; DX:AX = ProcInstance ;
; CX != 0 ;
; ;
; Error Returns: ;
; DX:AX = NULL ;
; CX = 0 ;
; ;
; Registers Preserved: ;
; DI,SI,DS ;
; ;
; Registers Destroyed: ;
; BX,ES ;
; ;
; Calls: ;
; MapDStoDATA ;
; GlobalAlloc ;
; FarMyLock ;
; ;
; History: ;
; ;
; Sat Sep 26, 1987 12:53:58p -by- David N. Weise [davidw] ;
; ReWrote it a while ago to work with EMS. ;
;-----------------------------------------------------------------------;
assumes ds,nothing
assumes es,nothing
cProc IMakeProcInstance,<PUBLIC,FAR>,<si,di>
parmD pproc
parmW hinstance
cBegin
SetKernelDSMisc
; We try to prohibit tasks from calling into each other.
; One way to do this is to not allow one app to MPI for
; another.
mov bx,[bp-2] ; Warning - assume DS pushed here!
mov ax,hinstance ; if hInstance == NULL,
or ax,ax ; use caller's DS
jz mpi1
push bx
cCall GlobalHandle,<ax>
pop bx
cmp dx,bx
jz mpi1
xor cx,cx
krDebugOut DEB_ERROR, "%dx2 MakeProcInstance only for current instance. "
mpi1: mov hinstance,bx
; If this is a library DS then it makes absolutely no sense to make a
; ProcInstance because library prologs setup DS already! In addition
; it is assumed in TestDSAX that only task DS's are in ProcInstances.
cCall FarGetOwner,<bx>
mov es, ax
NE_check_ES
mov dx,pproc.sel
mov ax,pproc.off
test es:[ne_flags],NENOTP
jz @F
jmp mpexit ; It's a library - return its address
@@:
; now try to get a thunklet
mov ax,curTDB
mov si,TDB_MPI_Thunks
mov es,ax
mov ax,es:[TDB_MPI_Sel]
mp2: mov es,ax
mov bx,es:[si].THUNKSIZE-2
or bx,bx
jz @F
jmp got_a_thunk
@@: mov ax,es:[si] ; Is there another block linked in?
xor si,si
or ax,ax
jnz mp2 ; Yes, look at it.
; No thunks available, make a block.
mov bx,THUNKSIZE * THUNKELEM
mov cx,GA_ZEROINIT
cCall GlobalAlloc,<cx,ax,bx>
or ax,ax
; jnz mpx
; cwd
jz mpfail
;mpx:
mov bx,ax
mov es,curTDB
xchg es:[TDB_MPI_Thunks],bx ; Link the new block in.
mov es,ax
; Initialize the new block.
mov es:[0],bx
mov es:[2],MPIT_SIGNATURE
mov bx,THUNKSIZE-2
mov cx,THUNKELEM-1
mp1: lea dx,[bx+THUNKSIZE]
.errnz THUNKELEM and 0FF00h
mov es:[bx],dx
mov bx,dx
loop mp1
mov es:[bx],cx
push es ; make the block into a code segment
cCall AllocSelector,<es>
pop es
or ax, ax
jnz @F
; AllocSelector Failed! Back out.
mov bx, es:[0] ; this is the old thunk val
mov ax, es ; this is the segment just allocated
mov es, curTDB
mov es:[TDB_MPI_Thunks], bx ; restore old value
cCall GlobalFree,<ax>
mpfail:
krDebugOut DEB_IERROR, "MakeProcInstance failed. Did you check return values?"
xor ax, ax
cwd
jmps mpexit
@@: mov di,ax
push es
cCall IPrestoChangoSelector,<di,es>
cCall FreeSelector,<di>
pop ax
xor bx,bx
jmp mp2
got_a_thunk:
push es
; we need a data alias so we can write into this thing
push bx
mov bx, kr1dsc
or bl, SEG_RING
cCall IPrestoChangoSelector,<es,bx>
mov es,ax
pop bx
mov ax,es:[bx]
mov es:[si].THUNKSIZE-2,ax
lea di,[bx-THUNKSIZE+2]
cld
mov dx,di ; save offset of thunk
mov al,0B8h ; mov ax,#
stosb
mov ax,hInstance
stosw
mov al,0EAh ; jmp far seg:off
stosb
mov ax,pproc.off
stosw
mov ax,pproc.sel
stosw
mov ax,dx ; recover offset of thunk
pop dx ; recover sel of thunk
mpexit:
mov cx,ax
or cx,dx
cEnd
;-----------------------------------------------------------------------;
; FreeProcInstance ;
; ;
; Frees the given ProcInstance. ;
; ;
; Arguments: ;
; parmD pproc ;
; ;
; Returns: ;
; AX != 0 Success ;
; CX != 0 Success ;
; ;
; Error Returns: ;
; AX == 0 ProcInstance not found. ;
; CX == 0 ProcInstance not found. ;
; ;
; Registers Preserved: ;
; DX,DI,SI,DS ;
; ;
; Registers Destroyed: ;
; BX ;
; ;
; Calls: ;
; nothing ;
; ;
; History: ;
; ;
; Sat Sep 26, 1987 12:25:42p -by- David N. Weise [davidw] ;
; ReWrote it a while ago to work with EMS. ;
;-----------------------------------------------------------------------;
assumes ds,nothing
assumes es,nothing
cProc IFreeProcInstance,<PUBLIC,FAR>,<di>
parmD pproc
cBegin
cCall MISCMapDStoDATA
ReSetKernelDS
mov ax,curTDB
mov es,ax
mov ax,es:[TDB_MPI_Sel]
mov bx,TDB_MPI_Thunks ; Point to start of first table.
fp0: or ax,ax ; Loop through linked tables.
jz fpdone
mov es,ax ; Point to table.
cmp ax,pproc.sel ; This the right ProcInstance table?
jz fp1
mov ax,es:[bx] ; No, get next table.
xor bx,bx
jmp fp0
fp1:
push bx
mov ax, kr1dsc
or al, SEG_RING
cCall IPrestoChangoSelector,<es,ax>
mov es,ax
pop bx
mov di,pproc.off ; Pick off the specific Proc.
xor ax,ax
cld
stosw ; Clear it out!
stosw
stosw
mov ax,di
xchg es:[bx].THUNKSIZE-2,ax ; Update free list.
stosw
mov ax,-1 ; Return success.
cCall HandleAbortProc, <pproc>
fpdone:
mov cx,ax
UnSetKernelDS
cEnd
assumes ds, nothing
assumes es, nothing
cProc DefineHandleTable,<PUBLIC,FAR>,<di>
parmW tblOffset
cBegin
cCall FarMyLock,<ds>
or ax,ax
jz dhtx
mov di,tblOffset
call MISCMapDStoDATA
ReSetKernelDS
push ds
mov ds,curTDB
UnSetKernelDS
or di,di ; resetting PHT?
jnz @F
mov dx,di
@@: mov word ptr ds:[TDB_PHT][2],dx
mov word ptr ds:[TDB_PHT][0],di
pop ds
ReSetKernelDS
or di,di ; Is the app removing the PHT?
jnz @F ; setting new
dec PHTcount ; resetting
jmps dhtx
@@: inc PHTcount ; bump the count of tasks with PHT's
assumes ds, nothing
mov es,ax
mov cx,es:[di] ; Get word count
add di,2 ; Point to first word
xor ax,ax
inc cx ; new handle table format (skip cwClear)
cld
rep stosw ; Zero words in private handle table
inc ax
dhtx:
cEnd
sEnd MISCCODE
end