2020-09-30 16:53:55 +02:00

473 lines
10 KiB
NASM

include kernel.inc
include gpfix.inc
ifdef WOW
include protect.inc
endif
sBegin DATA
globalW sels, <1> ; end of list is 1 to fool DOS386 DPMI
globalW selscount, <0> ; length of selman16 list
IF KDEBUG
externW ArenaSel
externW SelTableLen
externD SelTableStart
ENDIF
sEnd DATA
sBegin CODE
assumes CS,CODE
assumes DS,NOTHING
assumes ES,NOTHING
assumes FS,NOTHING
assumes GS,NOTHING
externFP AllocSelectorArray
externFP IFreeSelector
externW gdtdsc
ifdef WOW
externNP DPMIProc
endif
.386p
IF 0 ;KDEBUG - WOW doesn't have FastAndDirty... routines from win95 krn32.asm
extern FastAndDirtyIsMovableHandle:far16
ENDIF
WOW_DPMIFUNC_0C equ 04f2h
;*********************************************************************
; MapSL - Maps a selector:offset pair to a linear address
;
; Arguments: dAddrFar16 - selector:offset to map
;
; Returns: eax = linear address
; dx:ax = linear address
;
; Register Usage: eax,ebx,edx
;
; *Must preserve es*. Some win32s thunking code depends on this.
;*********************************************************************
cProc MapSL, <PUBLIC, FAR>
cBegin nogen
mov bx,sp
mov bx, ss:[bx+4+2] ;Get selector
test bl, 4
jz MapSL_mightbegdt
IF 0 ;KDEBUG - WOW doesn't have FastAndDirty... routines from win95 krn32.asm
push bx
push es
xor cx,cx
mov es,cx
push bx
call FastAndDirtyIsMovableHandle
or ax,ax
jz MapSL_memok
mov bx,sp
add bx, 4
mov ax, ss:[bx+4+2] ;Get offending selector
mov dx, ss:[bx+2] ;Get return address seg
mov bx, ss:[bx] ;Get return address off
krDebugOut DEB_ERROR, "*** MapSL(16) called on unfixed selector #ax. Ret addr: #dx:#bx"
MapSL_memok:
pop cx
beg_fault_trap MapSL_bad_es
MapSL_restore_es:
mov es,cx
end_fault_trap
pop bx
jmp MapSL_memchkdone
MapSL_bad_es:
fault_fix_stack
xor cx,cx
jmp MapSL_restore_es
MapSL_memchkdone:
ENDIF ;KDEBUG
mov gs, [gdtdsc]
and bl, not 7 ;Mask ring bits
beg_fault_trap MapSL_fault_Trap
test byte ptr gs:[bx+5], 80h
end_fault_trap
jz MapSL_invalid
mov ah, gs:[bx+7] ; get upper 8 bits base
mov al, gs:[bx+4] ; get next byte of base
shl eax, 16 ; move to proper place
mov ax, gs:[bx+2] ; get lower 16 bits of base
xor edx, edx
mov bx, sp
mov dx, ss:[bx+4] ; get offset of current pointer
add eax, edx ; find complete 32 bit base
MapSL_exit2:
xor cx,cx ;Don't give apps access to the LDT
mov gs,cx
MapSL_exit:
;; dx:ax <- eax
shld edx, eax, 16 ;dx <- hiword of eax
retf 4
;;--------------------------------------------------------------
;; Error case: Selector out of range.
;;--------------------------------------------------------------
MapSL_fault_trap:
fault_fix_stack
IF KDEBUG
mov bx, sp
mov bx, ss:[bx+4+2]
krDebugOut DEB_ERROR, "MapSL(#bx): out of range selector"
ENDIF
xor eax,eax
jmp MapSL_exit2
;;--------------------------------------------------------------
;; Error case: Selector not present
;;--------------------------------------------------------------
MapSL_invalid:
IF KDEBUG
mov bx, sp
mov bx, ss:[bx+4+2]
krDebugOut DEB_ERROR, "MapSL(#bx): selector not present."
ENDIF
xor eax,eax
jmp MapSL_exit2
;;--------------------------------------------------------------
;; Potential error case: GDT selector or NULL?
;;--------------------------------------------------------------
MapSL_mightbegdt:
or bx,bx
jnz MapSL_gdt
;; Special case: Return original argument.
mov bx,sp
mov eax, ss:[bx+4]
shld edx, eax, 16 ;dx <- hiword of eax
retf 4
;;--------------------------------------------------------------
;; Error case: GDT selector
;;--------------------------------------------------------------
MapSL_gdt:
IF KDEBUG
krDebugOut DEB_ERROR, "***MapSL(16) called on GDT selector #bx"
ENDIF
xor eax,eax
jmp MapSL_exit
cEnd nogen
;****************************************************************
; MapLS - Allocate a new 64k limit selector that maps to linear address.
;
; Returns: Selector:offset in eax & dx:ax (offset is always zero).
; *Must preserve es because some win32s thunking code depends on it.*
;****************************************************************
cProc MapLS, <PUBLIC, FAR>
cBegin nogen
mov bx, sp
mov dx, ss:[bx+4+2]
or dx,dx
jz MapLS_special
mov gs, [gdtdsc]
SetKernelDS fs
mov eax, ss:[bx+2] ;put lo-word in hi-half of eax
mov bx, fs:[sels]
cmp bx, 1
jz MapLS_moresel ;cache is empty: go to slow case
mov cx, gs:[bx] ;remove from linked list
mov fs:[sels], cx
dec fs:[selscount]
MapLS_gotit:
and bl, not 7
or ax, -1 ;Set limit to 64k
mov gs:[bx], eax ;Init low half of descriptor
xchg dh, dl
shl edx, 16 ;Put high 16 bits of base up hi
mov dl, 1+2+16+32+64+128 ;accessed, write, app, ring3, present
rol edx, 8
mov gs:[bx+4], edx ;init upper half of descriptor
mov cx, 1
DPMICALL WOW_DPMIFUNC_0C ; Write shadow LDT entry thru to system LDT.
mov dx, bx
or dl, 7
mov ax, dx
shl eax, 16 ;EAX = DX:AX = NEWSEL:0 = alias
MapLS_exit2:
xor cx,cx
mov gs,cx ;Don't give apps access to LDT
MapLS_exit:
retf 4
UnsetKernelDS fs
MapLS_special:
mov eax, ss:[bx+4]
;; dx already has correct value
retf 4
UnsetKernelDS fs
;------------------------------------------------------------------------
; K16 selman cache is empty. Time to get more from krnl386.
;------------------------------------------------------------------------
MapLS_moresel: ; we need some more selectors here
push es
;Don't use more than one here. get_sel always takes
; multi-selector requests from DPMI, and these
; usually won't be given back because we'll free
; them one by one.
cCall AllocSelectorArray, <1>
or ax,ax
jnz MapLS_moreselendloop
;Uh oh.
krDebugOut DEB_ERROR, "MapLS(16): Couldn't get any more selectors!"
xor ax,ax
; Fall thru.
MapLS_moreselendloop:
krDebugOut DEB_TRACE, "Selman(16) allocated selector #AX from KRNL386."
pop cx
beg_fault_trap MapLS_bades
MapLS_restore_es:
mov es,cx
end_fault_trap
cwd ; put possible failure code in DX
cwde ; and in EAX
or ax, ax
jz MapLS_exit2
mov gs, [gdtdsc]
SetKernelDS fs
xchg bx,ax
and bl, not 7
push bx
mov bx,sp
add bx, 2
mov dx, ss:[bx+4+2]
mov eax, ss:[bx+2]
pop bx
jmp MapLS_gotit
MapLS_bades:
fault_fix_stack
xor cx,cx
jmp MapLS_restore_es
UnsetKernelDS fs
cEnd nogen
;***********************************************************************
; UnMapLS
;***********************************************************************
cProc UnMapLS, <PUBLIC, FAR>
cBegin nogen
if KDEBUG
mov bx, sp
push ds
mov bx, ss:[bx+4+2]
or bx,bx
jz UnMapLS_ok
test bl, 4
jz UnMapLS_gdt
or bl, 7
SetKernelDS
mov ax, [gdtdsc]
lsl dx,ax
cmp bx,dx
ja UnMapLS_pastldt
mov ds,ax
UnsetKernelDS
test byte ptr ds:[bx-2], 080h
jz UnMapLS_notpresent
SetKernelDS
and bl, not 7
shr bx, 1
cmp bx, [SelTableLen]
jae UnMapLS_notgh
movzx ebx,bx
add ebx, [SelTableStart]
mov ds, [ArenaSel]
UnsetKernelDS
mov eax, ds:[ebx]
or eax,eax
jnz UnMapLS_gh
UnMapLS_notgh:
;Fallthru
UnMapLS_ok:
pop ds
endif
mov bx, sp
mov bx, ss:[bx+4+2]
and bl, not 7 ; point to LDT entry
jz UnMapLS_null
xor eax,eax
mov es,eax ;In case es contains selector we're trashing
; fs and gs are loaded below.
if KDEBUG
mov ax, ss
and al, not 7
cmp ax, bx
je UnMapLS_inuse
xor ax, ax
endif
SetKernelDS fs
mov gs, [gdtdsc]
if KDEBUG
test WORD PTR gs:[bx+5], 80h
jz UnMapLS_invalid
endif
mov ax, fs:[sels]
mov gs:[bx], eax ; linked list of avail sels
xor eax,eax
mov gs:[bx+4], eax ; zero out entry
mov fs:[sels], bx ; new list head sel
inc fs:[selscount]
UnMapLS_checkcount:
if KDEBUG
cmp fs:[selscount], 4
else
cmp fs:[selscount], 42
endif
jb UnMapLS_Null
dec fs:[selscount]
mov bx, fs:[sels]
mov ax, gs:[bx]
mov fs:[sels],ax
; Make it look like a valid limit-1 selector so FreeSelector doesn't
; free the ones after it.
mov dword ptr gs:[bx], 1
mov dword ptr gs:[bx+4], 0000f300h
or bx, 7
push ds
push es
push fs
push gs
push bx
xor cx,cx ;Just in case of one these registers contained
mov ds,cx ;the segment being freed.
mov es,cx
cCall IFreeSelector,<bx>
pop bx
pop gs
pop fs
pop cx
beg_fault_trap UnMapLS_bades
UnMapLS_restore_es:
mov es,cx
end_fault_trap
pop cx
beg_fault_trap UnMapLS_badds
UnMapLS_restore_ds:
mov ds,cx
end_fault_trap
krDebugOut DEB_TRACE, "Selman(16) returning selector #BX to KRNL386"
jmp UnMapLS_checkcount
UnMapLS_null:
xor eax, eax ; EAX = DX:AX = 0 (OK)
cwd
mov gs,ax ; Cut off access to LDT
UnMapLS_exit:
retf 4
UnMapLS_bades:
fault_fix_stack
xor cx,cx
jmp UnMapLS_restore_es
UnMapLS_badds:
fault_fix_stack
xor cx,cx
jmp UnMapLS_restore_ds
if KDEBUG
UnMapLS_invalid:
krDebugOut DEB_ERROR, "UnMapLS(#bx) invalid selector"
or al, 1 ; AX != 0
jmps UnMapLS_exit
UnMapLS_inuse:
krDebugOut DEB_ERROR, "UnMapLS(#bx) attempting to free ss"
or al, 1 ; AX != 0
jmps UnMapLS_exit
UnMapLS_gdt:
mov bx,sp
add bx,2
mov bx, ss:[bx+4+2]
krDebugOut DEB_ERROR, "UnMapLS(#bx) GDT selector. Type 'k'"
jmp UnMapLS_ok ;in trouble if ignores this.
UnMapLS_pastldt:
mov bx,sp
add bx,2
mov bx, ss:[bx+4+2]
krDebugOut DEB_ERROR, "UnMapLS(#bx) Selector out of range. Type 'k'"
jmp UnMapLS_ok ;in trouble if ignores this.
UnMapLS_notpresent:
mov bx,sp
add bx,2
mov bx, ss:[bx+4+2]
krDebugOut DEB_ERROR, "UnMapLS(#bx) Selector not present. Probably part of a freelist! Type 'k' and '.wl'"
jmp UnMapLS_ok ;in trouble if ignores this.
UnMapLS_gh:
mov bx,sp
add bx,2
mov bx, ss:[bx+4+2]
krDebugOut DEB_ERROR, "UnMapLS(#bx) Selector is a global handle! Type 'k' and '.dg'"
jmp UnMapLS_ok ;in trouble if ignores this.
endif
cEnd nogen
sEnd CODE
end