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

569 lines
16 KiB
NASM

TITLE TASKING.ASM - WOW Tasking Support
PAGE ,132
;
; WOW v1.0
;
; Copyright (c) 1991, Microsoft Corporation
;
; TASKING.ASM
; WOW Tasking Support on 16 bit side - also see wkman.c
;
; History:
; 23-May-91 Matt Felton (mattfe) Created
; 12-FEB-92 Cleanup
;
.xlist
include kernel.inc
include tdb.inc
include newexe.inc
include wow.inc
include vint.inc
.list
.386
if KDEBUG
externFP OutputDebugString
endif
DataBegin
externB Kernel_InDOS
externB Kernel_flags
externB InScheduler
ifndef WOW
externB fProfileDirty
endif
externW WinFlags
externW cur_drive_owner
externW curTDB
externW Win_PDB
externW LockTDB
externW headTDB
externW hShell
externW pGlobalHeap
externW hExeHead
externW f8087
externD pIsUserIdle
externW wCurTaskSS
externW wCurTaskBP
globalw gwHackpTDB,0
globalw gwHackTaskSS,0
globalw gwHackTaskSP,0
externW PagingFlags
DataEnd
ifdef WOW
externFP ExitKernelThunk
externFP WowInitTask
externFP Yield
externFP WOW16CallNewTaskRet
externFP WowKillTask
externFP WOW16DoneBoot
externFP WOWGetCurrentDirectory
externFP ExitCall
externFP WowSyncTask
endif
sBegin CODE
assumes cs,CODE
assumes ds,NOTHING
assumes es,NOTHING
externNP LoadSegment
externNP DeleteTask
externNP InsertTask
externNP ShrinkHeap
ifdef WOW
externNP Int21Handler
endif
if SDEBUG
externNP DebugSwitchOut
externNP DebugSwitchIn
endif
;-----------------------------------------------------------------------;
; StartWOWTask ;
; ;
; Start a WOW 32 Task ;
; ;
; Arguments: ;
; Parmw1 -> TDB Of New Task ;
; Parmw2 New Task SS ;
; Parmw3 New Task SP ;
; Returns: ;
; AX - TRUE/FALSE if we we able to create a new WOW task ;
; Error Returns: ;
; none ;
; Registers Preserved: ;
; ;
; Registers Destroyed: ;
; BX,CX,DX ;
; Calls: ;
; See Notes ;
; History: ;
; ;
; Fre 24-May-1991 14:30 -by- Matthew A. Felton [mattfe] ;
; Created ;
;-----------------------------------------------------------------------;
; The 16-Bit Kernel has created the new task and its ready to go
; 1. Temporarily Switch to New Task Stack S2
; 2. Fake Far Return from Thunk to point to StartW16Task
; 3. Thunk to WOW32 - This will create a new Thread and call Win32 InitTask()
; Note InitTask() will not return from the non-preemptive scheduler until
; This thread does a Yield.
; 4. When the thunk returns is jmps to WOW16TaskStarted on S2 leaving
; S2 ready for the new task when it returns
; 5. Restore S1 - Original Task Stack
; 6. Return Back to LoadModule who will Yield and start T2 going.
assumes ds, nothing
assumes es, nothing
cProc StartWOWTask,<PUBLIC,FAR>
parmW pTDB
parmW wTaskSS
parmW wTaskSP
cBegin
SetKernelDS ds
mov es,pTDB ; Get New Task TDB, will use later
push curTDB ; Save our TDB
push bp ; save BP (we don't exit wow16cal directly)
mov di,ss ; Save Current Task Stack
mov cx,sp ; di=ss cx=sp
; switch to new task stack temporarily
FCLI ; Disable Ints Till We get Back
; To presever new fram
mov si,wTaskSP ; Grab this before SS goes away
mov ss,wTaskSS ; Switch to new task stack.
mov sp,si ;
mov curTDB,es ; Set curTDB to new task
;
pushf
pop ax
or ax,0200h ; Re-enable interrupts
push ax
push es ; hTask
mov es, es:[TDB_pModule] ; exehdr
mov ax, es:[ne_expver]
mov dx, es:[ne_flags]
and dx, NEWINPROT ; prop. font bit
push es ; hModule
push es ; Pointer to module name
push es:ne_restab
push es ; Pointer to module path
push word ptr es:ne_crc+2
push dx ; expwinver. argument for WOWINITTASK
push ax
mov es, curTDB ; resotre es, just being safe
; thunk to wow32 to create a thread and task
push cs ; Fake Out FAR Return to StartW16Task
push offset StartW16Task
jmp WowInitTask ; Start Up the W32 Thread
public WOW16TaskStarted
WOW16TaskStarted:
mov ss,di ; Restore Calling Task Stack
mov sp,cx
pop bp
pop curTDB ; Restore real CurTDB
FSTI ; OK we look like old task
cEnd
;
; First code executed By New Task - Setup Registers and Go
;
StartW16Task:
add sp,+12
pop ax ; Flags
mov bp,sp
xchg ax,[bp+20] ; Put flags down & pick up cs
xchg ax,[bp+18] ; Put cs down & pick up ip
xchg ax,[bp+16] ; Put ip down & pick up bp
mov bp,ax
dec bp
pop dx
pop bx
pop es
pop cx
pop ax
pop di
pop si
pop ds
call SyncTask
iret ; App Code Starts From Here
; ExitKernel
;
; Enter When the 16 bit Kernel is going away -- never returns.
; In WOW this is a register-args wrapper for the stack-args
; thunk ExitKernelThunk.
public ExitKernel
ExitKernel:
cCall ExitKernelThunk, <ax>
INT3_NEVER ; ExitKernel never returns.
; BootSchedule
;
; Entered When Bogus Boot Task goes Away - treated same as task exit.
public BootSchedule
BootSchedule:
CheckKernelDS DS ; Make Sure we Address Kernel DS
mov [curTDB],0 ; Make No Task the Current Task
jmp WOW16DONEBOOT
; ExitSchedule
;
; We get here when a 16 bit task has exited - go kill this task
; Win32 non-preemptive scheduler will wake someone else to run.
public ExitSchedule
ExitSchedule:
CheckKernelDS DS ; Make Sure we Address Kernel DS
mov ax,[LockTDB] ; If I am the locked TDB then clear flag
cmp ax,[curTDB]
jnz @f
mov [LockTDB],0
@@:
mov [curTDB],0 ; Make No Task the Current Task
call ShrinkHeap
jmp WowKillTask ; Go Kill Myself (NEVER RETURNS)
;-----------------------------------------------------------------------;
; SwitchTask - This is NOT a subroutine DO NOT CALL IT
;
; This routine does a Win 3.1 compatible task switch.
;
; Arguments:
; AX == Next Tasks TDB pointer
; Returns:
; nothing
; Error Returns:
; nothing
; Registers Preserved:
;
; Registers Destroyed:
;
; Calls:
; SaveState
; RestoreState
;
; History:
; 22-May-91 Matt Felton (MattFe) Created
; Using idea's from Win 3.1 Schedule.Asm
;
;-----------------------------------------------------------------------;
assumes ds, nothing
assumes es, nothing
public SwitchTask
SwitchTask:
CheckKernelDS ds
ReSetKernelDS ds
inc InScheduler ; set flag for INT 24...
cmp curTDB,0 ; Previous Task Gone Away ?
jnz @f ; No ->
; Yes
mov di,ax ; DI = New TDB
mov ds,ax ; DS = New TDB
jmps dont_save_state ; Get Set for this new guy
@@:
push ax ; Save Next Tasks TDB pointer
; COMPAT 22-May-91 Mattfe, Idle callout for funky screen drivers is done by
; the Windows scheduler - that will not happen either from WOW. INT 28
; and Win386 1689 int2f call
; There was PokeAtSegments code which during idle time brought back segments !
; Do Debuggers Care that the stack frame of registers looks like a Windows stack frame
; when we do the debugger callout ? - check with someone in CVW land.
mov es,curTDB ; ES = Previous Task TDB
mov ax,es ; Don't SS,SP For DEAD Task
or ax,ax
jz @F
mov ax,wCurTaskSS ; FAKE Out TDB_taskSS,SP so that
mov es:[TDB_taskSS],ax ; The Old Task Task_BP looks right
mov ax,wCurTaskBP
sub ax,(Task_BP-Task_DX)
mov es:[TDB_taskSP],ax
@@:
pop ds ; DS = Next Task TDB
UnSetKernelDS ds
if KDEBUG
; Assertion Check TDB_taskSS == SS for current Task
mov ax,ds:[TDB_taskSS]
mov di,ss
cmp di,ax
jz @F
; int 3
@@:
endif; KDEBUG
mov di,ds ; DI = destination task
xor si,si ; SI is an argument to RestoreState
mov ax,es ; NOTE TDB_SS,SP Are note Correct
or ax,ax ; might affect debugger compatability.
jz short dont_save_state
cmp es:[TDB_sig],TDB_SIGNATURE
jnz short dont_save_state
mov si,es ; SI = Old Task
cCall SaveState,<si>
if SDEBUG
push ds
mov ds,ax
call DebugSwitchOut ; Note Stack Frame is not Compatible
pop ds ; Do we care ?
endif
dont_save_state:
SetKernelDS es
mov curTDB,di
mov ax, ds:[TDB_PDB] ; Set our idea of the PDB
mov Win_PDB, ax
SetKernelDS es
cmp di,0 ; the first task, will never get 0
jz dont_restore_state
or Kernel_flags,kf_restore_CtrlC OR kf_restore_disk
if SDEBUG
call DebugSwitchIn
endif
dont_restore_state:
; Switch to new task stack.
mov curTDB,di
dec InScheduler ; reset flag for INT 24
SetKernelDS ds ; Set the Kernel DS again
;the word at [vf_wES] is a selector that's about to be popped into
;the ES in WOW16CallNewTaskRet. this selector could be the TDB of
;a task that just died, in which case it's invalid. so let's
;shove something in there that won't cause a GP when we do the POP ES.
;
; In some cases we are switching tasks while returning to a callback.
; When this happens our stack is a CBVDMFRAME instead of a VDMFRAME.
; We only want to shove a safe ES,FS,GS value when it's a VDMFRAME and
; we're returning from an API call rather than calling back to
; 16-bit code.
;
; Regardless of which frame we're using, ss:sp points to wRetID.
mov bx, sp
cmp WORD PTR ss:[bx], RET_DEBUGRETURN ; if (wRetID > RET_DEBUGRETURN)
ja dont_stuff_regs ; goto dont_stuff_regs
sub bx,vf_wTDB + 2 ;bx is now start of VDMFRAME struct
mov ss:[bx+vf_wES],es ;put something safe in there
;Win31 does not save fs, gs over task switches, so we zero them out here
mov word ptr ss:[bx+vf_wFS],0 ;put something safe in there
mov word ptr ss:[bx+vf_wGS],0 ;put something safe in there
dont_stuff_regs:
if KDEBUG
mov bx, sp
cmp WORD PTR ss:[bx], RET_TASKSTARTED
jne @f
INT3_NEVER ; We need to stuff ES for RET_TASKSTARTED
; if we hit this breakpoint.
@@:
endif
; Hung App Support
; If the new task is the one we want to kill then force it to exit
mov bx,curTDB ; if ( curTDB == LockTDB )
cmp bx,LockTDB
jnz SW_DontKillIt
mov ax,4CFFH ; YES -> Exit
DOSCALL
INT3_NEVER
SW_DontKillIt:
jmp WOW16CallNewTaskRet ; Continue with the new task.
;-----------------------------------------------------------------------;
; SaveState ;
; ;
; Saves the state of the current MS-DOS process. This means the per ;
; task interrupt vectors, the drive and directory, EEMS land if any, ;
; and old app stuff if any. ;
; ;
; Arguments: ;
; parmW destination ;
; ;
; Returns: ;
; DS returned in AX. ;
; ;
; Error Returns: ;
; ;
; Registers Preserved: ;
; ;
; Registers Destroyed: ;
; ;
; Calls: ;
; ;
; History: ;
; ;
; Mon 07-Aug-1989 21:53:42 -by- David N. Weise [davidw] ;
; Removed the WinOldApp support. ;
; ;
; Tue Feb 03, 1987 08:21:53p -by- David N. Weise [davidw] ;
; Got rid of the rest of the DOS version dependencies. ;
; ;
; Thu Jan 22, 1987 03:15:15a -by- David N. Weise [davidw] ;
; Took out the saving of the ^C state, DTA address, and ErrorMode. ;
; ;
; Sun Jan 04, 1987 04:40:44p -by- David N. Weise [davidw] ;
; Added this nifty comment block. ;
;-----------------------------------------------------------------------;
assumes ds, nothing
assumes es, nothing
cProc SaveState,<PUBLIC,NEAR>,<si,di,ds>
parmW destination
cBegin
cld
SetKernelDS
mov ax,f8087
UnSetKernelDS
mov ds,destination
or ax,ax
if 0
jz short no_fstcw
.8087
fstcw ds:[TDB_FCW]
endif
no_fstcw:
test ds:[TDB_Drive],10000000b; if hi bit set....
jnz short ss_ret ; ...no need to get dir
mov ah,19h
DOSCALL
mov dl,al
inc dl
or al,10000000b
mov ds:[TDB_Drive],al ; save it (A=0, B=1, etc.)
mov si,TDB_LFNDirectory
mov byte ptr [si],'\' ; set "\"
inc si
; get Long path
cCall WowGetCurrentDirectory,<80h, ds, si>
or dx, dx
jz short ss_ret
mov byte ptr [si-1],0 ; indicate error with null byte
ss_ret: mov ax,ds
cEnd
;-----------------------------------------------------------------------;
; RestoreState ;
; ;
; Restores the MS-DOS interrupt vectors in real mode. ;
; ;
; Arguments: ;
; none ;
; Returns: ;
; none ;
; Error Returns: ;
; none ;
; Registers Preserved: ;
; BX,CX,DX,DI,SI,DS,ES ;
; Registers Destroyed: ;
; AX ;
; Calls: ;
; nothing ;
; History: ;
; ;
; Mon 07-Aug-1989 21:53:42 -by- David N. Weise [davidw] ;
; Removed the WinOldApp support. ;
; ;
; Tue Feb 03, 1987 08:21:53p -by- David N. Weise [davidw] ;
; Got rid of the rest of the DOS version dependencies. ;
; ;
; Thu Jan 22, 1987 03:15:15a -by- David N. Weise [davidw] ;
; Took out the restoring of the ^C state, DTA address, and ErrorMode. ;
; ;
; Sun Jan 04, 1987 04:45:31p -by- David N. Weise [davidw] ;
; Added this nifty comment block. ;
;-----------------------------------------------------------------------;
; SyncTask
;
; Enter: when new task starts
; check if the new task is blocked by appshelp
cProc SyncTask,<PUBLIC,NEAR> <ax,bx,cx,dx,di,si,es,ds>
cBegin
STL:
cCall WowSyncTask
cmp ax, 0
je @F
jg STL
call ExitCall
@@:
cEnd
sEnd CODE
end