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.
3148 lines
73 KiB
3148 lines
73 KiB
PAGE ,132
|
|
TITLE LDBOOT - BootStrap procedure
|
|
?DFSTACK = 1
|
|
.xlist
|
|
include kernel.inc
|
|
include newexe.inc
|
|
include tdb.inc
|
|
include pdb.inc
|
|
include protect.inc
|
|
include gpcont.inc
|
|
ifdef WOW
|
|
include vint.inc
|
|
include doswow.inc
|
|
endif
|
|
.list
|
|
|
|
if 0; KDEBUG
|
|
ifdef WOW
|
|
BootTraceOn = 1
|
|
BootTrace macro char
|
|
push ax
|
|
mov ax,char
|
|
int 3
|
|
pop ax
|
|
endm
|
|
else
|
|
BootTraceOn = 1
|
|
BootTrace macro char
|
|
push char
|
|
call BootTraceChar
|
|
endm
|
|
endif
|
|
else
|
|
BootTrace macro char
|
|
endm
|
|
BootTraceOn = 0
|
|
endif
|
|
|
|
; Note that the following public constants require special handling
|
|
; for ROM Windows. If you add, delete, or change this list of constants,
|
|
; you must also check the ROM Image Builder.
|
|
|
|
public __ahshift
|
|
public __ahincr
|
|
public __0000h
|
|
public __0040h
|
|
public __A000h
|
|
public __B000h
|
|
public __B800h
|
|
public __C000h
|
|
public __D000h
|
|
public __E000h
|
|
public __F000h
|
|
public __ROMBIOS
|
|
public __WinFlags
|
|
public __flatcs
|
|
public __flatds
|
|
|
|
public __MOD_KERNEL
|
|
public __MOD_DKERNEL
|
|
public __MOD_USER
|
|
public __MOD_DUSER
|
|
public __MOD_GDI
|
|
public __MOD_DGDI
|
|
public __MOD_KEYBOARD
|
|
public __MOD_SOUND
|
|
public __MOD_SHELL
|
|
public __MOD_WINSOCK
|
|
public __MOD_TOOLHELP
|
|
public __MOD_MMEDIA
|
|
public __MOD_COMMDLG
|
|
ifdef FE_SB
|
|
public __MOD_WINNLS
|
|
public __MOD_WIFEMAN
|
|
endif ; FE_SB
|
|
|
|
__ahshift = 3
|
|
__ahincr = 1 shl __ahshift
|
|
__0000h = 00000h
|
|
__0040h = 00040h
|
|
__A000h = 0A000h
|
|
__B000h = 0B000h
|
|
__B800h = 0B800h
|
|
__C000h = 0C000h
|
|
__D000h = 0D000h
|
|
__E000h = 0E000h
|
|
__F000h = 0F000h
|
|
__ROMBIOS = 0F000h
|
|
__WinFlags = 1
|
|
__flatcs = 1Bh
|
|
__flatds = 23h
|
|
|
|
__MOD_KERNEL = 0h
|
|
__MOD_DKERNEL = 0h
|
|
__MOD_USER = 0h
|
|
__MOD_DUSER = 0h
|
|
__MOD_GDI = 0h
|
|
__MOD_DGDI = 0h
|
|
__MOD_KEYBOARD = 0h
|
|
__MOD_SOUND = 0h
|
|
__MOD_SHELL = 0h
|
|
__MOD_WINSOCK = 0h
|
|
__MOD_TOOLHELP = 0h
|
|
__MOD_MMEDIA = 0h
|
|
__MOD_COMMDLG = 0h
|
|
ifdef FE_SB
|
|
__MOD_WINNLS = 0h ; for IME
|
|
__MOD_WIFEMAN = 0h ; for WIFEMAN
|
|
endif ; FE_SB
|
|
|
|
BOOTSTACKSIZE = 512
|
|
EXTRASTACKSIZE = (4096-BOOTSTACKSIZE)
|
|
|
|
MultWIN386 EQU 16h ; Reserved to Win386
|
|
MW3_ReqInstall EQU 00h ; Installation check
|
|
MW3_ReqInstall_Ret_1 EQU 0FFh ; Return number 1
|
|
MW3_ReqInstall_Ret_2 EQU 01h ; Return number 2
|
|
MW3_ReqInstall_Ret_3 EQU 02h ; Return number 3
|
|
MW3_ReqInstall_Ret_4 EQU 03h ; Return number 4
|
|
DOS_FLAG_OFFSET EQU 086h
|
|
|
|
externFP lstrlen
|
|
externFP lstrcat
|
|
externFP lstrcpy
|
|
externFP IGlobalAlloc
|
|
externFP IGlobalFree
|
|
externFP IGlobalRealloc
|
|
externFP LoadModule
|
|
externFP IOpenFile
|
|
externFP lrusweep
|
|
externFP GetExePtr
|
|
externFP GetProfileInt
|
|
externFP GetPrivateProfileString
|
|
externFP GetPrivateProfileInt
|
|
externFP GetProcAddress
|
|
externFP GetTempDrive
|
|
externFP ExitKernel
|
|
externFP InternalEnableDOS
|
|
externFP FlushCachedFileHandle
|
|
externFP SetHandleCount
|
|
externFP IPrestoChangoSelector
|
|
externFP GPFault
|
|
externFP StackFault
|
|
externFP invalid_op_code_exception
|
|
externFP page_fault
|
|
ifdef WOW
|
|
externFP divide_overflow
|
|
externFP single_step
|
|
externFP breakpoint
|
|
endif
|
|
externFP DiagInit
|
|
|
|
ifdef WOW
|
|
externFP StartWOWTask
|
|
externFP AllocSelector_0x47
|
|
externW MOD_KERNEL
|
|
externW ModCount
|
|
externFP WOWGetTableOffsets
|
|
externFP WOWDosWowInit
|
|
externFP GetShortPathName
|
|
endif
|
|
|
|
if KDEBUG
|
|
externFP ValidateCodeSegments
|
|
endif
|
|
|
|
ifdef FE_SB
|
|
externFP GetSystemDefaultLangID
|
|
endif
|
|
|
|
externFP TermsrvGetWindowsDir
|
|
|
|
|
|
externW pStackBot
|
|
externW pStackMin
|
|
externW pStackTop
|
|
|
|
sBegin CODE
|
|
externFP Int21Handler
|
|
externFP Int10Handler
|
|
externD prevInt10proc
|
|
sEnd CODE
|
|
|
|
;------------------------------------------------------------------------
|
|
; Data Segment Variables
|
|
;------------------------------------------------------------------------
|
|
|
|
DataBegin
|
|
if SHERLOCK
|
|
externW gpEnable
|
|
endif
|
|
|
|
externB graphics
|
|
externB fBooting
|
|
externB Kernel_flags
|
|
externB fChkSum
|
|
externB fCheckFree
|
|
externB WOAName
|
|
externB grab_name
|
|
ifndef WOW
|
|
externB szUserPro
|
|
endif
|
|
externB szBootLoad
|
|
externB szCRLF
|
|
externB szMissingMod
|
|
externB szPleaseDoIt
|
|
externB fPadCode
|
|
;externW EMScurPID
|
|
;externW PID_for_fake
|
|
externW cBytesWinDir
|
|
externW cBytesSysDir
|
|
externW pGlobalHeap
|
|
externW hExeHead
|
|
externW MaxCodeSwapArea
|
|
externW curTDB
|
|
externW f8087
|
|
externB fastFP
|
|
externW topPDB
|
|
externW headTDB
|
|
externW winVer
|
|
externD Dos_Flag_Addr
|
|
|
|
ifndef WOW
|
|
externB WinIniInfo
|
|
externB PrivateProInfo
|
|
endif
|
|
externW gmove_stack
|
|
externW prev_gmove_SS
|
|
externW BaseDsc
|
|
externW WinFlags
|
|
externW hUser
|
|
externW hShell
|
|
externW MyCSSeg
|
|
externW MyDSSeg
|
|
externW MyCSAlias
|
|
externB fhCache
|
|
externW fhCacheEnd
|
|
externW fhCacheLen
|
|
externB fPokeAtSegments
|
|
externB fExitOnLastApp
|
|
externW segLoadBlock
|
|
externD pKeyProc
|
|
externD pKeyProc1
|
|
externW wDefRIP
|
|
|
|
externW cpShrunk
|
|
externW cpShrink
|
|
externW hLoadBlock
|
|
|
|
externD pTimerProc
|
|
externD pExitProc
|
|
externD pDisableProc
|
|
externD lpWindowsDir
|
|
externD lpSystemDir
|
|
|
|
;if KDEBUG and SWAPPRO
|
|
;externD prevIntF0proc
|
|
;endif
|
|
|
|
externD lpInt21 ; support for NOVELL stealing int 21h
|
|
|
|
ifdef WOW
|
|
externD pFileTable
|
|
externW cBytesSys16Dir
|
|
externD lpSystem16Dir
|
|
externB Sys16Suffix
|
|
externW cBytesSys16Suffix
|
|
externW cBytesSysWx86Dir
|
|
externD lpSystemWx86Dir
|
|
externB SysWx86Suffix
|
|
externW cBytesSysWx86Suffix
|
|
externD pPMDosCURDRV
|
|
externD pPMDosCDSCNT
|
|
externD pPMDosPDB
|
|
externD pPMDosExterr
|
|
externD pPMDosExterrLocus
|
|
externD pPMDosExterrActionClass
|
|
externD pDosWowData
|
|
|
|
globalW cbRealWindowsDir,0
|
|
|
|
WINDIR_BUFSIZE equ 121
|
|
|
|
achWindowsDir DB WINDIR_BUFSIZE DUP(?)
|
|
achRealWindowsDir DB WINDIR_BUFSIZE DUP(?)
|
|
achSystem16Dir DB 128 DUP(?)
|
|
achSystemWx86Dir DB 128 DUP(?)
|
|
|
|
public cbRealWindowsDir, achRealWindowsDir, achWindowsDir, achSystem16Dir
|
|
externB achTermSrvWindowsDir ; windows directory path (for win.ini)
|
|
|
|
|
|
endif
|
|
|
|
DataEnd
|
|
|
|
|
|
;------------------------------------------------------------------------
|
|
; INITDATA Variables
|
|
;------------------------------------------------------------------------
|
|
|
|
DataBegin INIT
|
|
|
|
ifndef WOW
|
|
; WOW doesn't muck with the WOAName buffer -- we just leave it
|
|
; as WINOLDAP.MOD
|
|
externB woa_286
|
|
externB woa_386
|
|
endif
|
|
externB bootExecBlock
|
|
externW oNRSeg
|
|
externW oMSeg
|
|
externW win_show
|
|
externD lpBootApp
|
|
|
|
staticW initTDBbias,0
|
|
staticW initSP,0
|
|
staticW initSSbias,0
|
|
staticW segNewCS,0
|
|
|
|
app_name db 68 dup(0)
|
|
|
|
DataEnd INIT
|
|
|
|
|
|
;------------------------------------------------------------------------
|
|
; EMSDATA Variables
|
|
;------------------------------------------------------------------------
|
|
|
|
|
|
;------------------------------------------------------------------------
|
|
|
|
externNP LoadSegment
|
|
externNP genter
|
|
externNP gleave
|
|
externNP GlobalInit
|
|
externNP DeleteTask
|
|
externNP BootSchedule
|
|
externNP InitFwdRef
|
|
|
|
externNP SaveState
|
|
externNP LKExeHeader
|
|
externNP GetPureName
|
|
externNP SegmentNotPresentFault
|
|
|
|
externNP LDT_Init
|
|
externNP alloc_data_sel
|
|
externNP get_physical_address
|
|
externNP set_physical_address
|
|
externNP set_sel_limit
|
|
externNP free_sel
|
|
externFP set_discarded_sel_owner
|
|
externNP SelToSeg
|
|
externNP DebugDefineSegment
|
|
externNP DebugFreeSegment
|
|
|
|
externNP SwitchToPMODE
|
|
externNP LKAllocSegs
|
|
|
|
ifdef WOW
|
|
externFP WOWFastBopInit
|
|
endif
|
|
|
|
|
|
if KDEBUG
|
|
if SWAPPRO
|
|
externB fSwapPro
|
|
externW hSwapPro
|
|
externW cur_dos_pdb
|
|
endif
|
|
endif
|
|
|
|
ifdef FE_SB
|
|
externFP FarMyIsDBCSTrailByte
|
|
endif
|
|
|
|
|
|
|
|
;------------------------------------------------------------------------
|
|
|
|
sBegin INITCODE
|
|
assumes cs,CODE
|
|
|
|
|
|
externD prevIntx6proc
|
|
externD prevInt0Cproc
|
|
externD prevInt0Dproc
|
|
externD prevInt0Eproc
|
|
externD prevInt21proc
|
|
externD prevInt3Fproc
|
|
ifdef WOW
|
|
externD prevInt01proc
|
|
externD prevInt03proc
|
|
externD oldInt00proc
|
|
externFP GlobalDosAlloc
|
|
externFP GlobalDosFree
|
|
endif
|
|
|
|
externNP TextMode
|
|
externNP InitDosVarP
|
|
externNP GrowSFTToMax
|
|
ifndef WOW
|
|
externNP SetUserPro
|
|
endif
|
|
externNP MyLock
|
|
externNP Shrink
|
|
externNP DebugDebug
|
|
externNP NoOpenFile
|
|
externNP NoLoadHeader
|
|
|
|
if SDEBUG
|
|
externNP DebugInit
|
|
endif
|
|
|
|
;if SWAPPRO
|
|
;externNP INTF0Handler
|
|
;endif
|
|
|
|
|
|
if SDEBUG
|
|
szModName db 'KERNEL',0
|
|
endif
|
|
|
|
ifdef WOW
|
|
externNP SetOwner
|
|
endif
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; Bootstrap ;
|
|
; ;
|
|
; Determines whether we should initialize in a smaller amount of ;
|
|
; memory, leaving room for an EEMS swap area. If so, we rep-move the ;
|
|
; code to a lower address, and tell the PSP to report less memory ;
|
|
; available. It then does lots more of stuff. ;
|
|
; ;
|
|
; Arguments: ;
|
|
; DS:0 = new EXE header ;
|
|
; ES:0 = Program Segment Prefix block (PSP) ;
|
|
; SS:SP = new EXE header ;
|
|
; CX = file offset of new EXE header (set by KernStub) ;
|
|
; DS = automatic data segment if there is one ;
|
|
; ;
|
|
; Returns: ;
|
|
; ;
|
|
; Error Returns: ;
|
|
; ;
|
|
; Registers Preserved: ;
|
|
; ;
|
|
; Registers Destroyed: ;
|
|
; ;
|
|
; Calls: ;
|
|
; ;
|
|
; History: ;
|
|
; ;
|
|
; Sat Jun 20, 1987 06:00:00p -by- David N. Weise [davidw] ;
|
|
; Made fast booting work with real EMS. ;
|
|
; ;
|
|
; Tue Apr 21, 1987 06:31:42p -by- David N. Weise [davidw] ;
|
|
; Added some more EMS support. ;
|
|
; ;
|
|
; Thu Apr 09, 1987 02:52:37p -by- David N. Weise [davidw] ;
|
|
; Put back in the movement down if EMS. ;
|
|
; ;
|
|
; Sat Mar 14, 1987 05:55:29p -by- David N. Weise [davidw] ;
|
|
; Added this nifty comment block. ;
|
|
;-----------------------------------------------------------------------;
|
|
|
|
assumes ds,DATA
|
|
assumes es,nothing
|
|
assumes ss,nothing
|
|
|
|
|
|
cProc BootStrap,<PUBLIC,NEAR>
|
|
cBegin nogen
|
|
if KDEBUG
|
|
jmp short Hello_WDEB_End
|
|
Hello_WDEB:
|
|
db 'Windows Kernel Entry',13,10,0
|
|
Hello_WDEB_End:
|
|
push ax
|
|
push es
|
|
push si
|
|
xor ax, ax
|
|
mov es, ax
|
|
mov ax, es:[68h*4]
|
|
or ax, es:[68h*4+2]
|
|
jz @F
|
|
mov ax,cs
|
|
mov es,ax
|
|
lea si,Hello_WDEB
|
|
mov ah,47h
|
|
int 68h
|
|
@@:
|
|
pop si
|
|
pop es
|
|
pop ax
|
|
endif
|
|
BootTrace 'a'
|
|
|
|
cmp ax, "KO" ; OK to boot; set by kernstub.asm
|
|
je @F
|
|
xor ax, ax
|
|
retf
|
|
@@:
|
|
|
|
; Set up addressibility to our code & data segments
|
|
|
|
mov MyCSSeg, cs
|
|
mov MyDSSeg, ds
|
|
|
|
ifdef WOW
|
|
; Get pointer to sft so that I can find do direct protect mode
|
|
; file IO operations
|
|
|
|
mov CS:MyCSDS, ds
|
|
|
|
push es
|
|
mov ah,52h ; get pointer to internal vars
|
|
int 21h
|
|
|
|
mov ax, es
|
|
shl ax, 4
|
|
add ax, DOS_FLAG_OFFSET
|
|
sub ax, 400h
|
|
mov Dos_Flag_Addr.off, ax
|
|
mov Dos_Flag_Addr.sel, 40h
|
|
|
|
;
|
|
; Retrieve some pointers from the DosWowData structure in DOS.
|
|
;
|
|
push di
|
|
push dx
|
|
mov di, es:[bx+6ah] ;kernel data pointer
|
|
|
|
mov pDosWowData.off, di
|
|
mov pDosWowData.sel, es
|
|
|
|
mov ax, word ptr es:[di].DWD_lpCurPDB
|
|
; mov dx, word ptr es:[di].DWD_lpCurPDB+2
|
|
mov pPMDosPDB.off,ax
|
|
mov pPMDosPDB.sel,0 ;to force gpfault if not ready
|
|
|
|
mov ax, word ptr es:[di].DWD_lpCurDrv
|
|
; mov dx, word ptr es:[di].DWD_lpCurDrv+2
|
|
mov pPMDosCURDRV.off,ax
|
|
mov pPMDosCURDRV.sel,0 ;to force gpfault if not ready
|
|
|
|
mov ax, word ptr es:[di].DWD_lpCDSCount
|
|
; mov dx, word ptr es:[di].DWD_lpCDSCount+2
|
|
mov pPMDosCDSCNT.off,ax
|
|
mov pPMDosCDSCNT.sel,0 ;to force gpfault if not ready
|
|
|
|
mov ax, word ptr es:[di].DWD_lpExterr
|
|
mov pPMDosExterr.off, ax
|
|
mov pPMDosExtErr.sel, 0
|
|
|
|
mov ax, word ptr es:[di].DWD_lpExterrLocus
|
|
mov pPMDosExterrLocus.off, ax
|
|
mov pPMDosExtErrLocus.sel, 0
|
|
|
|
mov ax, word ptr es:[di].DWD_lpExterrActionClass
|
|
mov pPMDosExterrActionClass.off, ax
|
|
mov pPMDosExtErrActionClass.sel, 0
|
|
|
|
pop dx
|
|
pop di
|
|
|
|
lea bx,[bx+sftHead]
|
|
mov pFileTable.off,bx
|
|
mov pFileTable.sel,es
|
|
pop es
|
|
|
|
endif ; WOW
|
|
|
|
BootTrace 'b'
|
|
call SwitchToPMODE ; BX -> LDT, DS -> CS(DS), SI start of memory
|
|
BootTrace 'c'
|
|
|
|
ReSetKernelDS
|
|
|
|
mov BaseDsc,si
|
|
call LDT_Init
|
|
BootTrace 'd'
|
|
|
|
ifdef WOW
|
|
call AllocSelector_0x47
|
|
call WOWFastBopInit
|
|
endif
|
|
|
|
if SDEBUG
|
|
|
|
; In protected mode, initialize the debugger interface NOW. Only real mode
|
|
; needs to wait until the global heap is initialized.
|
|
|
|
cCall DebugInit
|
|
|
|
; In protected mode, define where our temp Code & Data segments are
|
|
|
|
mov ax,codeOffset szModName
|
|
cCall DebugDefineSegment,<cs,ax,0,cs,0,0>
|
|
|
|
mov ax,codeOffset szModName
|
|
cCall DebugDefineSegment,<cs,ax,3,ds,0,1>
|
|
endif
|
|
BootTrace 'e'
|
|
|
|
BootTrace 'f'
|
|
|
|
ifdef WOW
|
|
push ax
|
|
push bx
|
|
mov bx, pDosWowData.sel ;use this segment
|
|
mov ax, 2
|
|
int 31h
|
|
mov pPMDosPDB.sel,ax ;make this a PM pointer
|
|
mov pPMDosCURDRV.sel,ax ;make this a PM pointer
|
|
mov pPMDosCDSCNT.sel,ax ;make this a PM pointer
|
|
mov pPMDosExterr.sel,ax
|
|
mov pPMDosExterrLocus.sel,ax
|
|
mov pPMDosExterrActionClass.sel,ax
|
|
pop bx
|
|
pop ax
|
|
|
|
push pDosWowData.sel
|
|
push pDosWowData.off
|
|
call WOWDosWowInit
|
|
endif
|
|
|
|
; InitDosVarP just records a few variables, it does no hooking.
|
|
; It does dos version checking and other fun stuff that
|
|
; must be done as soon as possible.
|
|
|
|
call InitDosVarP
|
|
or ax,ax
|
|
jnz inited_ok
|
|
Debug_Out "KERNEL: InitDosVarP failed"
|
|
mov ax,4CFFh ; Goodbye!
|
|
INT21
|
|
inited_ok:
|
|
BootTrace 'g'
|
|
|
|
push bx
|
|
|
|
mov bx, WinFlags
|
|
ifndef JAPAN ; should be removed because of IBM dependent code.
|
|
; InitFwdRef routine will set 80x87 status bit in WinFlags
|
|
; and exported location #178.
|
|
|
|
; Determine if there is a co-processor present.
|
|
|
|
int 11h ; get equipment word
|
|
test al,2 ; this is the IBM approved method
|
|
jz no_80x87 ; to check for an 8087
|
|
or bh,WF1_80x87
|
|
no_80x87:
|
|
endif ;NOT JAPAN
|
|
|
|
ifdef WOW
|
|
or bh,WF1_WINNT ; Set NT flag ON
|
|
endif
|
|
|
|
mov WinFlags,bx
|
|
|
|
pop bx
|
|
|
|
; Determine if running under Windows/386 or the 286 DOS Extender
|
|
|
|
push bx
|
|
mov ax,(MultWin386 SHL 8) OR MW3_ReqInstall ; Win/386 install check
|
|
int 2Fh
|
|
BootTrace 'h'
|
|
cmp al,MW3_ReqInstall_Ret_4 ; Under WIN386 in pmode?
|
|
or Kernel_flags[1],kf1_Win386
|
|
or byte ptr WinFlags,WF_PMODE or WF_ENHANCED
|
|
jmps InstallChkDone
|
|
|
|
NotUnderWin386:
|
|
or Kernel_flags[2],KF2_DOSX
|
|
or byte ptr WinFlags,WF_PMODE or WF_STANDARD
|
|
|
|
InstallChkDone:
|
|
BootTrace 'i'
|
|
pop bx
|
|
|
|
ifndef WOW
|
|
; WOW doesn't muck with the WOAName buffer -- we just leave it
|
|
; as WINOLDAP.MOD
|
|
|
|
push cx
|
|
push di
|
|
push si
|
|
cld
|
|
mov cx,8
|
|
smov es,ds
|
|
mov di,dataOffset WOAName
|
|
mov si,dataOffset woa_286
|
|
test Kernel_flags[2],KF2_DOSX
|
|
jnz @F
|
|
mov si,dataOffset woa_386
|
|
@@: rep movsb
|
|
pop si
|
|
pop di
|
|
pop cx
|
|
endif
|
|
BootTrace 'j'
|
|
|
|
mov ax,cx
|
|
mov cl,4
|
|
shr ax,cl
|
|
mov cpShrunk,ax
|
|
|
|
; Compute paragraph address of new EXE header from SS:SP
|
|
|
|
mov bx,sp ; SS:SP -> new EXE header
|
|
|
|
cCall get_physical_address,<ss>
|
|
add ax,bx
|
|
adc dx,0
|
|
BootTrace 'k'
|
|
cCall alloc_data_sel,<dx,ax,0,0FFFFh>
|
|
BootTrace 'l'
|
|
|
|
mov segLoadBlock,ax ; hinitexe:0 -> new EXE header
|
|
|
|
; calculate the TDB bias
|
|
|
|
mov bx,dataOffset boottdb
|
|
mov initTDBbias,bx
|
|
|
|
; calculate the SS bias
|
|
|
|
mov bx,dataOffset stackbottom
|
|
mov initSSbias,bx
|
|
|
|
; calculate the initial SP
|
|
|
|
mov si,dataOffset stacktop
|
|
sub si,dataOffset stackbottom
|
|
mov initSP,si
|
|
|
|
cCall get_physical_address,<ds>
|
|
add ax,bx
|
|
adc dx,0
|
|
BootTrace 'm'
|
|
FCLI
|
|
mov prev_gmove_SS, ss ; Switch stack while we set up new SS
|
|
smov ss, ds
|
|
mov sp, dataOFFSET gmove_stack
|
|
cCall set_physical_address,<prev_gmove_SS>
|
|
xor ax, ax
|
|
xchg ax, prev_gmove_SS
|
|
BootTrace 'n'
|
|
mov ss,ax ; switch to new stack
|
|
mov sp,si
|
|
FSTI
|
|
|
|
xor bp,bp ; zero terminate BP chain.
|
|
sub si,BOOTSTACKSIZE
|
|
mov ss:[pStackBot],sp
|
|
mov ss:[pStackMin],sp
|
|
mov ss:[pStackTop],si
|
|
|
|
cld
|
|
mov es,topPDB
|
|
|
|
externB szNoGlobalInit
|
|
|
|
BootTrace 'o'
|
|
mov ax, BaseDsc ; Free memory starts after this
|
|
mov bx,segLoadBlock ; Make boot image be first busy block
|
|
mov cx,es:[PDB_block_len] ; cx is end of memory
|
|
mov dx,MASTER_OBJECT_SIZE
|
|
cCall GlobalInit,<dx,bx,ax,cx>
|
|
|
|
jc @F ; passed through from ginit
|
|
or ax,ax
|
|
jnz mem_init_ok
|
|
@@:
|
|
mov dx, codeoffset szNoGlobalInit
|
|
smov ds, cs
|
|
mov ah, 9
|
|
int 21h ; Whine
|
|
mov ax, 4CFFh
|
|
int 21h ; And exit
|
|
;NoGlobalInit db "KERNEL: Unable to initialise heap",13,10,'$'
|
|
|
|
mem_init_ok:
|
|
mov hLoadBlock,ax ; Save handle to first busy block
|
|
|
|
BootTrace 'p'
|
|
|
|
mov pExitProc.sel,cs
|
|
mov pExitProc.off,codeOffset ExitKernel
|
|
|
|
; Find out where we live, and where win.com lives.
|
|
|
|
mov ds,topPDB
|
|
UnSetKernelDS
|
|
mov bx,ds
|
|
mov ds,ds:[PDB_environ]
|
|
xor si,si
|
|
cld
|
|
envloop1:
|
|
lodsb
|
|
or al,al ; end of item?
|
|
jnz envloop1
|
|
lodsb
|
|
or al,al ; end of environment?
|
|
jnz envloop1
|
|
lodsw ; ignore argc, DS:SI -> kernel path
|
|
|
|
ifdef WOW
|
|
smov es, ds ; now ES:SI -> kernel path
|
|
else
|
|
call get_windir ; on return, DS:DI -> windir= value
|
|
|
|
smov es,ds
|
|
|
|
; Record where to find the 'windows' directory.
|
|
BootTrace 'q'
|
|
|
|
SetKernelDS
|
|
or di,di
|
|
jz no_win_dir_yet
|
|
mov lpWindowsDir.sel,es
|
|
mov lpWindowsDir.off,di
|
|
mov cBytesWinDir,cx
|
|
|
|
; Now set pointer to WIN.INI to be
|
|
; in the Windows directory
|
|
push si
|
|
push es
|
|
smov es, ds
|
|
mov si, dataoffset szUserPro+6
|
|
mov di, si
|
|
inc cx
|
|
add di, cx ; Move up the string WIN.INI
|
|
std
|
|
mov cx, 4
|
|
rep movsw
|
|
; Now copy Windows directory
|
|
cld
|
|
mov cx, cBytesWinDir
|
|
mov di, dataoffset szUserPro
|
|
lds si, lpWindowsDir
|
|
UnSetKernelDS
|
|
rep movsb
|
|
mov byte ptr es:[di], '\'
|
|
pop es
|
|
pop si
|
|
endif
|
|
SetKernelDS
|
|
BootTrace 'r'
|
|
no_win_dir_yet:
|
|
|
|
sub sp,SIZE OPENSTRUC + 127
|
|
mov di,sp
|
|
regptr ssdi,ss,di
|
|
mov bx,OF_EXIST
|
|
cCall IOpenFile,<essi,ssdi,bx>
|
|
BootTrace 's'
|
|
inc ax ; Test for -1
|
|
jnz opn1 ; Continue if success
|
|
mov dx, codeoffset NoOpenFile
|
|
fail1:
|
|
push dx ; Save string pointer
|
|
call textmode ; Switch to text mode
|
|
pop dx
|
|
smov ds, cs
|
|
mov ah, 9
|
|
int 21h ; Tell user why we're bailing out
|
|
mov ax,4CFFh ; Goodbye!
|
|
INT21
|
|
opn1:
|
|
|
|
; Now simulate loading ourselves from memory
|
|
|
|
; Load new EXE header for KERNEL.EXE from memory
|
|
|
|
cCall LKExeHeader,<segLoadBlock,ssdi>
|
|
BootTrace 't'
|
|
add sp,SIZE OPENSTRUC + 127
|
|
or ax,ax
|
|
jnz @F
|
|
mov dx, codeoffset NoLoadHeader
|
|
jmp SHORT fail1
|
|
@@:
|
|
|
|
mov hExeHead,ax
|
|
mov es,ax
|
|
|
|
ifndef WOW
|
|
; Record where to find the 'system' directory.
|
|
|
|
mov di,es:[ne_pfileinfo]
|
|
lea di,es:[di].opFile
|
|
mov lpSystemDir.sel,es
|
|
mov lpSystemDir.off,di
|
|
mov dx,di
|
|
call GetPureName
|
|
sub di,dx
|
|
dec di
|
|
mov cBytesSysDir,di
|
|
|
|
BootTrace 'u'
|
|
endif
|
|
; ndef WOW
|
|
|
|
|
|
; Make room at end of kernel data segment for stack--NOTE: DGROUP is
|
|
; assumed to be segment 4!
|
|
|
|
mov bx,es:[ne_segtab]
|
|
add es:[bx+(3*size NEW_SEG1)].ns_minalloc,EXTRASTACKSIZE
|
|
|
|
|
|
; Determine size of kernel's largest discardable code segment
|
|
|
|
; don't bother with swap area's anymore. Its fixed 192k (3segs)
|
|
; picked this up from Win95
|
|
|
|
xor ax,ax ; No max
|
|
mov es:[ne_swaparea],ax
|
|
BootTrace 'v'
|
|
|
|
|
|
; Allocate memory for kernel segments
|
|
|
|
cCall LKAllocSegs,<es>
|
|
mov oNRSeg,ax
|
|
mov oMSeg, bx ; Misc segment
|
|
mov es,hExeHead
|
|
|
|
|
|
; If this is a pMode debug version, the code and data segments have already
|
|
; been defined to the debugger once. We're about to LoadSegment and define
|
|
; these segments again in their final location. Use a special form of
|
|
; DebugFreeSegment to force the debugger to pull out any breakpoints in these
|
|
; segments. If we don't do this, any existing breakpoints become INT 3's
|
|
; in the new copy of the segment and the code has to be patched by hand.
|
|
; If only the debugger was smart enough to 'move' the breakpoints when it
|
|
; saw a second define for an already loaded segment...
|
|
|
|
if SDEBUG
|
|
|
|
cCall DebugFreeSegment,<cs,-1>
|
|
cCall DebugFreeSegment,<MyCSDS,-1>
|
|
endif
|
|
|
|
; Load kernel code segment 1 (resident code)
|
|
|
|
mov si,1
|
|
mov ax,-1 ; Indicate loading from memory
|
|
cCall LoadSegment,<es,si,cs,ax>
|
|
or ax,ax
|
|
jnz ll1
|
|
fail2: jmp bootfail
|
|
ll1:
|
|
mov segNewCS,ax ; Save new CS value
|
|
|
|
|
|
; Load kernel data segment (segment 4)
|
|
|
|
mov si,4
|
|
mov ax,-1
|
|
cCall LoadSegment,<hExeHead,si,ds,ax>
|
|
or ax,ax
|
|
jz fail2
|
|
|
|
BootTrace 'w'
|
|
|
|
|
|
; locate the stack in the new segment
|
|
|
|
mov bx,ax
|
|
mov si,initSP
|
|
add si,EXTRASTACKSIZE
|
|
|
|
cCall get_physical_address,<ax>
|
|
add ax,initSSbias
|
|
adc dx,0
|
|
sub ax,10h
|
|
sbb dx,0
|
|
or ax,10h
|
|
FCLI
|
|
mov prev_gmove_SS, ss ; Switch stack while we set up new SS
|
|
smov ss, ds
|
|
mov sp, OFFSET gmove_stack
|
|
cCall set_physical_address,<prev_gmove_SS>
|
|
push bx
|
|
mov bx, si
|
|
xor cx, cx ; cx:bx=stack len (for set_sel_limit)
|
|
cCall set_sel_limit,<prev_gmove_SS>
|
|
pop bx
|
|
xor ax, ax
|
|
xchg ax, prev_gmove_SS
|
|
mov ss,ax ; Switch to new stack location
|
|
mov sp,si
|
|
FSTI
|
|
mov ax,bx
|
|
|
|
|
|
; zero the new TDB
|
|
|
|
push ax
|
|
|
|
cCall get_physical_address,<ax>
|
|
add ax,initTDBbias
|
|
adc dx,0
|
|
YAH_WELL = (SIZE TDB+15) and NOT 15
|
|
cCall alloc_data_sel,<dx,ax,0,YAH_WELL>
|
|
mov es,ax
|
|
xor ax,ax
|
|
mov di,ax
|
|
mov cx,SIZE TDB
|
|
cld
|
|
rep stosb
|
|
pop ax
|
|
|
|
; put the limits in the stack
|
|
|
|
xor bp,bp ; zero terminate BP chain.
|
|
mov ss:[pStackBot],sp
|
|
mov ss:[pStackMin],sp
|
|
mov ss:[pStackTop],10h
|
|
mov ss:[0],bp ; To mark this stack as NOT linked
|
|
mov ss:[2],bp ; to another, see PatchStack.
|
|
mov ss:[4],bp
|
|
|
|
; initialize the new TDB
|
|
|
|
sub sp,SIZE TASK_REGS
|
|
mov es:[TDB_taskSS],ss
|
|
mov es:[TDB_taskSP],sp
|
|
mov cx,topPDB
|
|
mov es:[TDB_PDB],cx ; save new PDB
|
|
mov es:[TDB_DTA].off,80h ; set initial DTA
|
|
mov es:[TDB_DTA].sel,cx
|
|
mov bx,1 ; BX = 1
|
|
mov es:[TDB_nEvents],bx ; Start this guy up!
|
|
mov es:[TDB_pModule],-1 ; EMS requires -1
|
|
mov bx,winVer
|
|
mov es:[TDB_ExpWinVer],bx ; Windows Version #
|
|
mov es:[TDB_sig],TDB_SIGNATURE ; Set signature word.
|
|
|
|
; initialize the task BP and DS
|
|
|
|
push es
|
|
les bx,dword ptr es:[TDB_taskSP]
|
|
mov es:[bx].TASK_BP,bp ; initial BP = 0
|
|
mov es:[bx].TASK_DS,bp ; initial DS = 0
|
|
pop es
|
|
|
|
mov ds,ax ; switch to new DS segment
|
|
mov ax,segNewCS ; recover new CS value
|
|
|
|
push cs ; to free the selector
|
|
|
|
; do a far return to the new code segment
|
|
|
|
push ax
|
|
mov ax,codeOffset new_code
|
|
push ax
|
|
ret_far ; FAR return to new code segment
|
|
|
|
public new_code
|
|
new_code:
|
|
|
|
pop ax
|
|
push es
|
|
cCall free_sel,<ax> ; Free old CS selector
|
|
cCall IPrestoChangoSelector,<cs,MyCSAlias> ; Change our CS Alias
|
|
|
|
mov es,MyCSAlias ; Change MyCSDS to new DS
|
|
assumes es,CODE
|
|
mov ax,ds
|
|
xchg ax,es:MyCSDS
|
|
assumes es,nothing
|
|
cCall free_sel,<ax> ; Free old DS selector
|
|
|
|
call DebugDebug ; do after MyCSDS changes
|
|
|
|
mov pExitProc.sel,cs ; reset this!!
|
|
|
|
;;; pusha
|
|
;;; cCall get_physical_address,<cs>
|
|
;;; add ax, codeOffset end_page_locked
|
|
;;; adc dx, 0
|
|
;;; mov bx, dx
|
|
;;; mov cx, ax
|
|
;;; mov di, cs
|
|
;;; lsl di, di
|
|
;;; sub di, codeOffset end_page_locked
|
|
;;; xor si, si
|
|
;;; mov ax, 0601h
|
|
;;; int 31h ; Unlock some of Kernel!
|
|
;;; popa
|
|
|
|
cCall SelToSeg,<ds> ; Save DS segment value
|
|
mov MyDSSeg,ax
|
|
|
|
cCall SelToSeg,<cs> ; Set segment equivalent
|
|
mov MyCSSeg, ax
|
|
|
|
; calculate the maximum amount that we will allow SetSwapAreaSize to set
|
|
|
|
mov ax,-1 ; They can only harm themselves.
|
|
|
|
mov MaxCodeSwapArea,ax
|
|
|
|
ifndef WOW ; WOW uses 32 bit profile api's
|
|
|
|
; Allocate a handle for WIN.INI
|
|
|
|
xor ax,ax
|
|
mov bx,GA_SHAREABLE shl 8 OR GA_MOVEABLE
|
|
cCall IGlobalAlloc,<bx,ax,ax>
|
|
mov [WinIniInfo.hBuffer], ax
|
|
or ax,ax
|
|
jz nowinini
|
|
mov bx,ax ; put handle into base register
|
|
mov ax,hExeHead
|
|
mov es,ax
|
|
call set_discarded_sel_owner
|
|
; Set up the filename
|
|
mov word ptr [WinIniInfo.lpProFile], dataoffset szUserPro
|
|
mov word ptr [WinIniInfo.lpProFile][2], ds
|
|
nowinini:
|
|
|
|
; Allocate a handle for Private Profiles
|
|
|
|
xor ax,ax
|
|
mov bx,GA_SHAREABLE shl 8 OR GA_MOVEABLE
|
|
cCall IGlobalAlloc,<bx,ax,ax>
|
|
mov [PrivateProInfo.hBuffer],ax
|
|
or ax,ax
|
|
jz noprivate
|
|
mov bx,ax ; put handle into base register
|
|
mov ax,hExeHead
|
|
mov es,ax
|
|
call set_discarded_sel_owner
|
|
noprivate:
|
|
endif; WOW
|
|
|
|
ifdef WOW
|
|
; Allocate a ~128K discardable code selector to hide the
|
|
; GrowHeap - heap not sorted bugs, also present in Win 3.1.
|
|
;
|
|
; Since our system DLLs (like krnl386, user, gdi etc) contain a few
|
|
; discardable codesgements (of approx 40K), we need not be precise
|
|
;
|
|
push es
|
|
mov ax, 0C000H ; hiword is 01h , totalsize 0x1c000 bytes
|
|
mov bx, GA_MOVEABLE OR GA_DISCCODE
|
|
cCall IGlobalAlloc,<bx,01h,ax>
|
|
or ax,ax
|
|
jz short nogrowheap
|
|
cCall SetOwner, <ax, hExeHead>
|
|
nogrowheap:
|
|
pop es
|
|
endif; WOW
|
|
|
|
; Now shrink off exe header and segment 1 of KERNEL.EXE
|
|
|
|
mov si,2 ; Segment number
|
|
xor ax,ax
|
|
xchg oNRSeg,ax
|
|
nofastboot:
|
|
sub ax,cpShrunk
|
|
mov cpShrink,ax ; Amount to shrink by
|
|
cCall MyLock,<hLoadBlock>
|
|
mov bx,ax
|
|
xchg segLoadBlock,ax
|
|
cCall get_physical_address,<ax>
|
|
mov cx,dx
|
|
xchg bx,ax
|
|
cCall get_physical_address,<ax>
|
|
sub bx,ax
|
|
sbb cx,dx
|
|
REPT 4
|
|
shr cx,1
|
|
rcr bx,1
|
|
ENDM
|
|
mov ax,bx
|
|
add cpShrink,ax
|
|
push ax
|
|
cCall Shrink
|
|
pop ax
|
|
sub cpShrunk,ax
|
|
|
|
; Load kernel code segment 2 (non-resident code)
|
|
|
|
cCall MyLock,<hLoadBlock>
|
|
mov bx,-1 ; Indicate loading from memory
|
|
cCall LoadSegment,<hExeHead,si,ax,bx>
|
|
or ax,ax
|
|
jnz ll2
|
|
pop es
|
|
jmp bootfail
|
|
ll2:
|
|
|
|
inc si ; On to segment 3
|
|
xor ax, ax
|
|
xchg ax, oMSeg ; Once back only!
|
|
or ax, ax
|
|
jnz nofastboot
|
|
|
|
pop es
|
|
|
|
|
|
call genter
|
|
smov ds, es
|
|
UnSetKernelDS
|
|
SetKernelDS es
|
|
;;;mov ax,EMScurPID ; In case an EMS fast boot is going down.
|
|
;;;mov ds:[TDB_EMSPID],ax
|
|
|
|
;;;mov ax,ds:[TDB_EMSPID]
|
|
;;;mov PID_for_fake,ax
|
|
;;;mov EMScurPID,ax
|
|
mov curTDB,ds
|
|
mov headTDB,ds
|
|
push es
|
|
|
|
; vectors
|
|
SaveVec MACRO vec
|
|
mov ax, 35&vec
|
|
DOSCALL
|
|
mov [di.off], bx
|
|
mov [di.sel], es
|
|
add di, 4
|
|
ENDM
|
|
|
|
push di
|
|
mov di, TDB_INTVECS
|
|
SaveVec 00h
|
|
SaveVec 02h
|
|
SaveVec 04h
|
|
SaveVec 06h
|
|
SaveVec 07h
|
|
SaveVec 3Eh
|
|
SaveVec 75h
|
|
|
|
ifdef WOW
|
|
;; Hook Int10 so we can filter calls in WOW (see intnn.asm)
|
|
push es
|
|
push ds
|
|
mov ax,3510h
|
|
INT21
|
|
mov ax,codeOffset prevInt10proc
|
|
SetKernelCSDword ax,es,bx
|
|
mov ax,2510h
|
|
smov ds,cs
|
|
mov dx,codeOFFSET Int10Handler
|
|
INT21
|
|
pop ds
|
|
pop es
|
|
endif
|
|
|
|
pop di
|
|
|
|
cCall SaveState,<ds>
|
|
pop es
|
|
mov ds,pGlobalHeap
|
|
call gleave
|
|
UnSetKernelDS es
|
|
|
|
mov ax, 32 ; Kernel wants a big handle table
|
|
cCall SetHandleCount,<ax>
|
|
|
|
SetKernelDS
|
|
|
|
;
|
|
; The following variable initialization is done here to avoid having
|
|
; relocations in Kernel's data segment.
|
|
;
|
|
mov lpInt21.off,codeOFFSET Int21Handler
|
|
mov lpInt21.sel,cs
|
|
|
|
;
|
|
; Before we hook exception handlers, make sure the DPMI exception
|
|
; handler stack is set up the way Windows likes it.
|
|
;
|
|
mov bl,6
|
|
mov ax,0202h ; DPMI get exception handler vector
|
|
int 31h
|
|
push cx
|
|
push dx
|
|
|
|
mov cx,cs
|
|
lea dx,fixing_stack
|
|
mov bl,6
|
|
mov ax,0203h ; DPMI set exception handler vector
|
|
int 31h
|
|
|
|
pop dx
|
|
pop cx
|
|
;
|
|
; Generate an invalid opcode exception fault. This causes DPMI to call
|
|
; our "exception handler."
|
|
;
|
|
db 0fh,0ffh
|
|
fixing_stack:
|
|
push bp
|
|
mov bp,sp ; BP -> BP RETIP RETCS EC IP CS FL SP SS
|
|
;
|
|
; Restore the previous invalid exception handler vector.
|
|
;
|
|
mov bl,6
|
|
mov ax,0203h
|
|
int 31h
|
|
;
|
|
; Replace the return address on the DPMI fault handler routine with
|
|
; our exit code.
|
|
;
|
|
lea ax,done_fixing_stack
|
|
mov [bp+8],ax
|
|
mov [bp+10],cs
|
|
|
|
lea ax,[bp+16]
|
|
mov ss:[pStackBot],ax
|
|
mov ss:[pStackMin],sp
|
|
mov ss:[pStackTop],offset pStackBot + 150
|
|
|
|
mov sp,bp
|
|
pop bp
|
|
retf
|
|
|
|
done_fixing_stack:
|
|
|
|
mov es, MyCSAlias
|
|
assumes es, CODE
|
|
|
|
|
|
; Hook not present fault for segment reloader.
|
|
|
|
mov ax,0202h ; Record old not present fault.
|
|
mov bl,0Bh
|
|
int 31h
|
|
mov prevInt3Fproc.off,dx
|
|
mov prevInt3Fproc.sel,cx
|
|
|
|
mov ax,0203h ; Hook not present fault.
|
|
mov cx,cs
|
|
mov dx,codeOffset SegmentNotPresentFault
|
|
int 31h
|
|
|
|
; Hook GP fault in order to terminate app.
|
|
|
|
mov bl, 0Dh ; GP Fault
|
|
mov ax, 0202h
|
|
int 31h
|
|
mov prevInt0Dproc.off, dx
|
|
mov prevInt0Dproc.sel, cx
|
|
|
|
mov ax, 0203h
|
|
mov cx, cs
|
|
mov dx, codeOffset GPFault
|
|
int 31h
|
|
|
|
; Hook invalid op-code in order to terminate app.
|
|
|
|
mov bl, 06h ; invalid op-code
|
|
mov ax, 0202h
|
|
int 31h
|
|
mov prevIntx6proc.off, dx
|
|
mov prevIntx6proc.sel, cx
|
|
|
|
mov ax, 0203h
|
|
mov cx, cs
|
|
mov dx, codeOffset invalid_op_code_exception
|
|
int 31h
|
|
|
|
; Hook stack fault in order to terminate app.
|
|
|
|
mov bl, 0Ch ; stack fault
|
|
mov ax, 0202h
|
|
int 31h
|
|
mov prevInt0Cproc.off, dx
|
|
mov prevInt0Cproc.sel, cx
|
|
|
|
mov ax, 0203h
|
|
mov cx, cs
|
|
mov dx, codeOffset StackFault
|
|
int 31h
|
|
|
|
; Hook bad page fault in order to terminate app.
|
|
|
|
mov bl, 0Eh ; page fault
|
|
mov ax, 0202h
|
|
int 31h
|
|
mov prevInt0Eproc.off, dx
|
|
mov prevInt0Eproc.sel, cx
|
|
|
|
mov ax, 0203h
|
|
mov cx, cs
|
|
mov dx, codeOffset page_fault
|
|
int 31h
|
|
|
|
ifdef WOW
|
|
|
|
; Hook divide overflow trap in order to get better WOW debugging.
|
|
|
|
mov bl, 00h ; divide overflow
|
|
mov ax, 0202h
|
|
int 31h
|
|
mov oldInt00proc.off, dx
|
|
mov oldInt00proc.sel, cx
|
|
|
|
mov ax, 0203h
|
|
mov cx, cs
|
|
mov dx, codeOffset divide_overflow
|
|
int 31h
|
|
|
|
; Hook single step trap in order to get better WOW debugging.
|
|
|
|
mov bl, 01h ; single step
|
|
mov ax, 0202h
|
|
int 31h
|
|
mov prevInt01proc.off, dx
|
|
mov prevInt01proc.sel, cx
|
|
|
|
mov ax, 0203h
|
|
mov cx, cs
|
|
mov dx, codeOffset single_step
|
|
int 31h
|
|
|
|
; Hook breakpoint trap in order to get better WOW debugging.
|
|
|
|
mov bl, 03h ; page fault
|
|
mov ax, 0202h
|
|
int 31h
|
|
mov prevInt03proc.off, dx
|
|
mov prevInt03proc.sel, cx
|
|
|
|
mov ax, 0203h
|
|
mov cx, cs
|
|
mov dx, codeOffset breakpoint
|
|
int 31h
|
|
|
|
endif
|
|
|
|
assumes es, nothing
|
|
|
|
; Do a slimy fix-up of __WinFlags containing processor and protect mode flags
|
|
|
|
xor ax,ax
|
|
mov dx,178
|
|
cCall GetProcAddress,<hExeHead,ax,dx>
|
|
mov ax,WinFlags
|
|
mov es:[bx],ax
|
|
|
|
ifdef WOW
|
|
; get WOW32 thunk table offsets and do fixups
|
|
|
|
; WARNING: WOW32 has a dependency on this being called after
|
|
; kernel is done booting and addresses are fixed
|
|
push ds
|
|
push dataoffset MOD_KERNEL
|
|
call far ptr WOWGetTableOffsets
|
|
|
|
mov si, dataoffset MOD_KERNEL
|
|
mov cx, ModCount ; # fixups to do
|
|
mov di, 570 ; first ordinal of the group (DANGER hardcoded from kernel.def)
|
|
|
|
Mexico:
|
|
|
|
push si
|
|
push di
|
|
push cx
|
|
|
|
push hExeHead
|
|
push 0
|
|
push di
|
|
call GetProcAddress
|
|
|
|
pop cx
|
|
pop di
|
|
pop si
|
|
mov ax,[si]
|
|
mov es:[bx],ax
|
|
|
|
inc si ; point to next word
|
|
inc si
|
|
inc di ; get next ordinal
|
|
|
|
loop Mexico
|
|
endif
|
|
|
|
; Do a very slimy fix-up of the runtime constant __0000h
|
|
|
|
cCall GetProcAddress,<hExeHead,0,183>
|
|
mov si,bx
|
|
mov bx,00000h
|
|
mov ax,2
|
|
int 31h
|
|
mov es:[si],ax
|
|
|
|
; Do a very slimy fix-up of the runtime constant __0040h
|
|
|
|
cCall GetProcAddress,<hExeHead,0,193>
|
|
mov si,bx
|
|
mov bx,00040h
|
|
mov ax,2
|
|
int 31h
|
|
mov es:[si],ax
|
|
|
|
; Do a very slimy fix-up of the runtime constant __ROMBIOS
|
|
|
|
cCall GetProcAddress,<hExeHead,0,173>
|
|
mov si,bx
|
|
mov bx,0F000h
|
|
mov ax,2
|
|
int 31h
|
|
mov es:[si],ax
|
|
|
|
; Do a very slimy fix-up of the runtime constant __F000h
|
|
|
|
cCall GetProcAddress,<hExeHead,0,194>
|
|
mov si,bx
|
|
mov bx,0F000h
|
|
mov ax,2
|
|
int 31h
|
|
mov es:[si],ax
|
|
|
|
; Do a very slimy fix-up of the runtime constant __A000h
|
|
|
|
cCall GetProcAddress,<hExeHead,0,174>
|
|
mov si,bx
|
|
mov bx,0A000h
|
|
mov ax,2
|
|
int 31h
|
|
mov es:[si],ax
|
|
|
|
; Do a very slimy fix-up of the runtime constant __B000h
|
|
|
|
cCall GetProcAddress,<hExeHead,0,181>
|
|
mov si,bx
|
|
mov bx,0B000h
|
|
mov ax,2
|
|
int 31h
|
|
mov es:[si],ax
|
|
|
|
; Do a very slimy fix-up of the runtime constant __B800h
|
|
|
|
cCall GetProcAddress,<hExeHead,0,182>
|
|
mov si,bx
|
|
mov bx,0B800h
|
|
mov ax,2
|
|
int 31h
|
|
mov es:[si],ax
|
|
|
|
; Do a very slimy fix-up of the runtime constant __C000h
|
|
|
|
cCall GetProcAddress,<hExeHead,0,195>
|
|
mov si,bx
|
|
mov bx,0C000h
|
|
mov ax,2
|
|
int 31h
|
|
mov es:[si],ax
|
|
|
|
; Do a very slimy fix-up of the runtime constant __D000h
|
|
|
|
cCall GetProcAddress,<hExeHead,0,179>
|
|
mov si,bx
|
|
mov bx,0D000h
|
|
mov ax,2
|
|
int 31h
|
|
mov es:[si],ax
|
|
|
|
; Do a very slimy fix-up of the runtime constant __E000h
|
|
|
|
cCall GetProcAddress,<hExeHead,0,190>
|
|
mov si,bx
|
|
mov bx,0E000h
|
|
mov ax,2
|
|
int 31h
|
|
mov es:[si],ax
|
|
|
|
ifndef WOW
|
|
cCall SetUserPro ; Get WIN.INI filename from environment
|
|
endif
|
|
|
|
CheckKernelDS
|
|
|
|
; Free high memory copy of KERNEL.EXE
|
|
|
|
cCall IGlobalFree,<hLoadBlock>
|
|
mov hLoadBlock,ax
|
|
|
|
.386
|
|
mov fs, ax
|
|
mov gs, ax
|
|
.286p
|
|
|
|
ifndef WOW
|
|
cmp lpWindowsDir.sel,ax
|
|
jnz got_win_dir
|
|
mov si,dataOffset szUserPro
|
|
mov di,dataOffset [WinIniInfo.ProBuf]
|
|
cCall IOpenFile,<dssi,dsdi,OF_PARSE>
|
|
lea di,[di].opfile
|
|
mov lpWindowsDir.sel,ds
|
|
mov lpWindowsDir.off,di
|
|
mov dx,di
|
|
call GetPureName
|
|
sub di,dx
|
|
dec di
|
|
mov cBytesWinDir,di
|
|
got_win_dir:
|
|
endif
|
|
|
|
ifdef WOW
|
|
; Regular Windows kernel sets up the Windows directory very early in
|
|
; boot, before WOW kernel loads WOW32.DLL.
|
|
;
|
|
; It turns out we can delay setting up the 16-bit copy of the
|
|
; windows directory (referred by lpWindowsDir) until here, where
|
|
; we're in protected mode and about to set up the system directories
|
|
; as well. This allows us to call a GetShortPathName thunk in
|
|
; WOW32.
|
|
;
|
|
|
|
mov ds,topPDB
|
|
UnSetKernelDS
|
|
mov ds,ds:[PDB_environ]
|
|
|
|
call get_windir ; on return, DS:DI -> windows dir
|
|
|
|
smov es,ds
|
|
SetKernelDS
|
|
|
|
; Record where to find the 'windows' directory.
|
|
BootTrace 'q'
|
|
|
|
mov lpWindowsDir.sel,es
|
|
mov lpWindowsDir.off,di
|
|
mov cBytesWinDir,cx
|
|
|
|
; Record where to find the 'system32' directory.
|
|
|
|
mov ax,hExeHead
|
|
mov es,ax
|
|
mov di,es:[ne_pfileinfo]
|
|
lea di,es:[di].opFile
|
|
mov lpSystemDir.sel,es
|
|
mov lpSystemDir.off,di
|
|
mov dx,di
|
|
call GetPureName
|
|
sub di,dx
|
|
dec di
|
|
mov cBytesSysDir,di
|
|
|
|
BootTrace 'u'
|
|
|
|
;
|
|
; Under WOW there are two system directories: \nt\system and \nt\system32.
|
|
; lpSystemDir points to \nt\system32, while lpSystem16Dir points to
|
|
; \nt\system. We return the latter from GetSystemDirectory, since we want
|
|
; applications to install their shared stuff in \nt\system, but we want
|
|
; to search \nt\system32 before \nt\system when loading modules.
|
|
; Set up lpSystem16Dir using the Windows dir + \system.
|
|
; Note that the string pointed to by lpSystem16Dir is not null terminated!
|
|
|
|
mov lpSystem16Dir.sel,ds
|
|
mov lpSystem16Dir.off,dataoffset achSystem16Dir
|
|
|
|
cld
|
|
mov si,dataoffset achRealWindowsDir
|
|
mov es,lpSystem16Dir.sel
|
|
mov di,lpSystem16Dir.off
|
|
mov cx,cbRealWindowsDir
|
|
rep movsb ; copy Windows dir
|
|
|
|
mov si,dataoffset Sys16Suffix
|
|
mov cx,cBytesSys16Suffix
|
|
rep movsb ; tack on "\system"
|
|
|
|
mov cx,cbRealWindowsDir
|
|
add cx,cBytesSys16Suffix
|
|
mov cBytesSys16Dir,cx
|
|
|
|
|
|
;
|
|
; build the Wx86 system directory "Windir\Sys32x86"
|
|
;
|
|
mov lpSystemWx86Dir.sel,ds
|
|
mov lpSystemWx86Dir.off,dataoffset achSystemWx86Dir
|
|
|
|
mov es,lpSystemWx86Dir.sel
|
|
mov di,lpSystemWx86Dir.off
|
|
mov cx,cbRealWindowsDir
|
|
mov si,dataoffset achRealWindowsDir
|
|
rep movsb ; copy System dir (Windows\System32)
|
|
|
|
mov si,dataoffset SysWx86Suffix
|
|
mov cx,cBytesSysWx86Suffix
|
|
rep movsb ; tack on "\Wx86"
|
|
|
|
mov cx,cbRealWindowsDir
|
|
add cx,cBytesSysWx86Suffix
|
|
mov cBytesSysWx86Dir,cx
|
|
|
|
|
|
@@:
|
|
; WOW DPMI Supports wants to call GlobalDosAlloc and GlobalDosFree so that
|
|
; We don't have to write the same code for DPMI support. So we call DPMI
|
|
; Here with the addresses of those routines so he can call us back.
|
|
|
|
mov bx,cs
|
|
mov si,bx
|
|
|
|
mov dx,codeOffset GlobalDosAlloc
|
|
mov di,codeOffset GlobalDosFree
|
|
|
|
mov ax,4f3h
|
|
int 31h
|
|
|
|
; Now that we've built up the system dir from the windows dir, set
|
|
; the windows dir to where it should be for this user.
|
|
push es
|
|
cld
|
|
mov di,offset achTermsrvWindowsDir
|
|
cCall TermsrvGetWindowsDir,<ds, di, MaxFileLen>
|
|
or ax, ax
|
|
jz @F ; ax = 0 -> error, just leave windows dir
|
|
|
|
smov es,ds
|
|
mov cx,-1
|
|
xor ax,ax
|
|
repnz scasb
|
|
not cx
|
|
dec cx ; length of windows directory
|
|
|
|
mov lpWindowsDir.sel,ds
|
|
mov lpWindowsDir.off,offset achTermsrvWindowsDir
|
|
mov cBytesWinDir,cx
|
|
@@:
|
|
pop es
|
|
|
|
endif
|
|
|
|
|
|
; Under win 2.11 we allowed the ":" syntax to replace the shell.
|
|
; We no longer allow this, however to avoid messing up people
|
|
; with batch files that have ":" in them we will strip the
|
|
; ":" out of the command line. But we retain the :: syntax
|
|
; for the OS/2 VM!!
|
|
|
|
; We also do the check for the /b switch here. This puts us in "diagnostic
|
|
; mode and we set a flag to say this. Later, we will call the DiagInit()
|
|
; function to open the log file, etc.
|
|
|
|
push ds
|
|
cld
|
|
mov ds,topPDB
|
|
UnSetKernelDS
|
|
push ds
|
|
pop es
|
|
mov si,80h
|
|
xor ax,ax
|
|
lodsb
|
|
or al,al
|
|
jz no_colon
|
|
mov cx,ax
|
|
@@: lodsb
|
|
cmp al,' '
|
|
loopz @B
|
|
cmp al,':'
|
|
jnz no_colon
|
|
cmp byte ptr [si],':'
|
|
jz no_colon
|
|
mov byte ptr [si][-1],' '
|
|
no_colon:
|
|
cmp al,'/' ;Switch character?
|
|
je CheckSwitch ;Yes
|
|
cmp al,'-' ;Other switch character?
|
|
jne NoSwitch ;Nope.
|
|
CheckSwitch:
|
|
lodsb ;Get next char
|
|
or al,32 ;Convert to lowercase if necessary
|
|
cmp al,'b' ;Diagnostic mode?
|
|
jnz NoSwitch ;Nope
|
|
cCall DiagInit ;Initialize diagnostic mode
|
|
mov WORD PTR [si-2],2020h ;Blank out the switch
|
|
NoSwitch:
|
|
pop ds
|
|
ReSetKernelDS
|
|
|
|
;** Reset secret flag for memory manager. Fixed segments will go
|
|
;** >1MB after this
|
|
and fBooting, NOT 2
|
|
|
|
;** We want to grow the SFT *before* loading the modules,
|
|
;** not after, like was the case previous to win3.1
|
|
call GrowSFTToMax ;add to SFT chain in p mode
|
|
|
|
if 1
|
|
; old ldinit.c
|
|
|
|
cld
|
|
push ds
|
|
smov es,ds
|
|
xor ax,ax
|
|
xor cx,cx
|
|
mov ds,topPDB
|
|
UnSetKernelDS
|
|
mov si,80h
|
|
lodsb
|
|
or al,al
|
|
jz gwaphics_done
|
|
mov cl,al
|
|
lodsb
|
|
cmp al,' '
|
|
mov al,ah
|
|
jnz gwaphics_done
|
|
lodsb
|
|
cmp al,':'
|
|
mov al,ah
|
|
jnz gwaphics_done
|
|
lodsb
|
|
cmp al,':'
|
|
mov al,ah
|
|
jnz gwaphics_done
|
|
|
|
mov di,dataOffset app_name
|
|
|
|
find_delineator:
|
|
lodsb
|
|
stosb
|
|
cmp al,' '
|
|
ja find_delineator
|
|
mov es:[di][-1],ah
|
|
mov ds:[80h],ah ; assume no arguments
|
|
jnz gwaphics_done
|
|
|
|
add cx,82h
|
|
sub cx,si
|
|
smov es,ds
|
|
mov di,80h
|
|
mov al,cl
|
|
stosb
|
|
rep movsb
|
|
|
|
gwaphics_done:
|
|
pop ds
|
|
ReSetKernelDS
|
|
or ax,ax
|
|
jz @F
|
|
mov graphics,0
|
|
@@:
|
|
else
|
|
cld
|
|
xor ax,ax
|
|
xor bx,bx
|
|
endif
|
|
jmps SlowBoot
|
|
|
|
|
|
bootfail:
|
|
mov al,1
|
|
cCall ExitKernel,<ax>
|
|
cEnd nogen
|
|
|
|
sEnd INITCODE
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; SlowBoot ;
|
|
; ;
|
|
; ;
|
|
; Arguments: ;
|
|
; ;
|
|
; Returns: ;
|
|
; ;
|
|
; Error Returns: ;
|
|
; ;
|
|
; Registers Preserved: ;
|
|
; ;
|
|
; Registers Destroyed: ;
|
|
; ;
|
|
; Calls: ;
|
|
; ;
|
|
; History: ;
|
|
; ;
|
|
; Sat Mar 14, 1987 05:52:22p -by- David N. Weise [davidw] ;
|
|
; Added this nifty comment block. ;
|
|
;-----------------------------------------------------------------------;
|
|
|
|
DataBegin INIT
|
|
|
|
BootSect db 'BOOT',0
|
|
BootFile db 'SYSTEM.INI',0
|
|
FilesCached db 'CACHEDFILEHANDLES',0
|
|
IdleSegLoad db 'LOADSEGMENTSATIDLE',0
|
|
ifdef WOW
|
|
ExitLastApp db 'CLOSEONLASTAPPEXIT',0
|
|
endif
|
|
|
|
if SHERLOCK
|
|
szGPC db 'GPCONTINUE',0
|
|
endif
|
|
|
|
szDebugSect db 'DEBUG',0
|
|
szOutputTo db 'OUTPUTTO', 0
|
|
szAux db 0 ;'AUX', 0 ; don't return a default
|
|
|
|
if KDEBUG
|
|
szKRInfo db 'KERNELINFO', 0
|
|
szKRBreak db 'KERNELBREAK', 0
|
|
szWin3Info db 'WIN3INFO', 0
|
|
szWin3Break db 'WIN3BREAK', 0
|
|
endif
|
|
|
|
BootBufLen equ 80
|
|
BootBuf db BootBufLen dup (?)
|
|
|
|
bootmods label byte
|
|
DB 'SYSTEM.DRV',0
|
|
winmods label byte
|
|
ifdef WOW
|
|
DB 'KEYBOARD.DRV',0
|
|
szAfterKeyboardDriver label byte ;Used so we can tell key driver loaded
|
|
DB 'MOUSE.DRV',0
|
|
|
|
ifdef FE_SB
|
|
szBeforeWifeMan label byte ;Used so we can tell key driver loaded
|
|
DB 'WIFEMAN.DLL', 0 ;WIFE manager has to be loaded before display driver
|
|
endif
|
|
DB 'VGA.DRV',0
|
|
DB 'SOUND.DRV',0
|
|
DB 'COMM.DRV',0
|
|
DB 'USER.EXE',0
|
|
DB 'GDI.EXE',0
|
|
ifdef FE_SB
|
|
szBeforeWinNls label byte ;Used so we can tell key driver loaded
|
|
DB 'WINNLS.DLL', 0 ;bug #112335
|
|
endif
|
|
|
|
else
|
|
DB 'KEYBOARD.DRV',0
|
|
szAfterKeyboardDriver label byte ;Used so we can tell key driver loaded
|
|
DB 'MOUSE.DRV',0
|
|
ifdef FE_SB
|
|
szBeforeWifeMan label byte ;Used so we can tell key driver loaded
|
|
DB 'WIFEMAN.DLL', 0 ;WIFE manager has to be loaded before
|
|
;display driver
|
|
endif
|
|
DB 'DISPLAY.DRV',0
|
|
DB 'SOUND.DRV',0
|
|
DB 'COMM.DRV',0
|
|
DB 'FONTS.FON',0
|
|
DB 'OEMFONTS.FON',0 ; For Internationals use.
|
|
DB 'GDI.EXE',0
|
|
DB 'USER.EXE',0
|
|
ifdef FE_SB
|
|
szBeforeWinNls label byte ;Used so we can tell key driver loaded
|
|
DB 'WINNLS.DLL', 0
|
|
endif
|
|
|
|
endif
|
|
|
|
ifdef WOW
|
|
defapp label byte
|
|
DB 'WOWEXEC.EXE',0
|
|
else
|
|
defapp label byte
|
|
DB 'PROGMAN.EXE',0
|
|
endif
|
|
|
|
Shell label byte
|
|
DB 'WOWSHELL',0
|
|
|
|
;** Ordinal strings for two of the Keyboard driver exports
|
|
keymodstr DB 'KEYBOARD',0
|
|
keyprocstr DB '#5',0 ; keyprocstr = AnsiToOem
|
|
keyprocstr1 DB '#6',0 ; keyprocstr = OemToAnsi
|
|
|
|
DataEnd INIT
|
|
|
|
sBegin INITCODE
|
|
assumes cs,CODE
|
|
assumes ds,nothing
|
|
assumes es,nothing
|
|
|
|
if BootTraceOn
|
|
cProc BootTraceChar,<PUBLIC,NEAR,NODATA>, <ax, dx>
|
|
parmW char
|
|
cBegin
|
|
mov dx, 3fdh
|
|
@@: in al, dx
|
|
test al, 20h
|
|
jz @B
|
|
mov ax, char
|
|
mov dx, 3f8h
|
|
out dx, al
|
|
cEnd
|
|
|
|
endif ; BootTraceOn
|
|
|
|
|
|
|
|
cProc ttywrite,<PUBLIC,NEAR,NODATA>, <ds, si>
|
|
parmD s
|
|
cBegin
|
|
; cCall lstrlen,<s>
|
|
; mov cx, ax
|
|
; lds bx, s
|
|
; mov bx,1
|
|
; mov ah,40h
|
|
; int 21h
|
|
lds si, s
|
|
cld
|
|
mov ah, 2
|
|
tty1:
|
|
lodsb
|
|
mov dl, al
|
|
or dl, dl
|
|
jz tty2
|
|
int 21h
|
|
jmps tty1
|
|
tty2:
|
|
cEnd
|
|
|
|
ifdef WOW
|
|
cProc LoadFail,<PUBLIC,NEAR,NODATA>, <ds,si,di,cx>
|
|
parmD s
|
|
cBegin
|
|
;** Put Up a Dialog Box If we can't load a module
|
|
|
|
; since szPleaseDoIt resides in protected cs we can't concat the module
|
|
; name to it -- so we need to copy it to the stack
|
|
|
|
;szPleaseDoIt db "Please re-install the following module to your system32
|
|
; directory: ",13,10,9,9,0
|
|
mov di,sp ; save current stack pointer
|
|
sub sp,100h ; allocate for new concat'd string on the stack
|
|
mov si,sp ; save the start of the stack string
|
|
|
|
push ss ; copy szPleaseDoIt to the stack buffer
|
|
push si
|
|
push cs
|
|
push codeOFFSET szPleaseDoIt
|
|
call lstrcpy
|
|
|
|
push ss ; concat module name to string on stack
|
|
push si
|
|
lds cx, s ; get the module name
|
|
push ds
|
|
push cx
|
|
call lstrcat
|
|
|
|
push ss ;push finished stack string now
|
|
push si
|
|
push cs ;szMissingMod db "KERNEL: Missing 16-bit module:",0
|
|
push codeOFFSET szMissingMod ; Set Caption
|
|
push 0 ;No left button
|
|
push SEB_CLOSE + SEB_DEFBUTTON ;Button 1 style
|
|
push 0 ;No right button
|
|
externFP kSYSERRORBOX
|
|
call kSYSERRORBOX ;Put up the system message
|
|
|
|
mov sp,di ; restore sp
|
|
cEnd
|
|
else // non-WOW
|
|
cProc LoadFail,<PUBLIC,NEAR,NODATA>, <ds>
|
|
parmD s
|
|
cBegin
|
|
SetKernelDS
|
|
cCall TextMode
|
|
mov bx, dataoffset szCRLF
|
|
cCall ttywrite, dsbx
|
|
mov bx, dataoffset szCRLF
|
|
cCall ttywrite, dsbx
|
|
mov bx, dataoffset szBootLoad
|
|
cCall ttywrite, dsbx
|
|
cCall ttywrite, s
|
|
mov bx, dataoffset szCRLF
|
|
cCall ttywrite, dsbx
|
|
cEnd
|
|
endif
|
|
|
|
cProc SlowBoot,<PUBLIC,NEAR>
|
|
cBegin nogen
|
|
|
|
CheckKernelDS
|
|
ReSetKernelDS
|
|
|
|
GPPI macro sect, key, defval, file, storeit
|
|
push ds
|
|
push dataoffset sect
|
|
|
|
push ds
|
|
push dataoffset key
|
|
|
|
push defval
|
|
|
|
push ds
|
|
push dataoffset file
|
|
|
|
call GetPrivateProfileInt
|
|
ifb <storeit>
|
|
mov defval, ax
|
|
endif
|
|
endm
|
|
|
|
|
|
GPPS macro sect, key, defval, here, hereLen, file
|
|
push ds
|
|
push dataoffset sect
|
|
|
|
push ds
|
|
push dataoffset key
|
|
|
|
push ds
|
|
push dataoffset defval
|
|
|
|
push ds
|
|
push dataoffset here
|
|
|
|
push hereLen
|
|
|
|
push ds
|
|
push dataoffset file
|
|
|
|
call GetPrivateProfileString
|
|
endm
|
|
|
|
GPPS1 macro sect, key, defval, here, hereLen, file
|
|
push ds
|
|
push dataoffset sect
|
|
|
|
push ds
|
|
push key
|
|
|
|
push ds
|
|
push defval
|
|
|
|
push ds
|
|
push dataoffset here
|
|
|
|
push hereLen
|
|
|
|
push ds
|
|
push dataoffset file
|
|
|
|
call GetPrivateProfileString
|
|
endm
|
|
|
|
GPI macro sect, key, defval, storeit
|
|
push ds
|
|
push dataoffset sect
|
|
|
|
push ds
|
|
push dataoffset key
|
|
|
|
push defval
|
|
|
|
call GetProfileInt
|
|
ifb <storeit>
|
|
mov defval, ax
|
|
endif
|
|
endm
|
|
|
|
if SHERLOCK
|
|
GPI szKernel, szGPC, gpEnable
|
|
endif
|
|
|
|
if KDEBUG
|
|
ifdef DISABLE
|
|
GPPI szDebugSect, szKRInfo, _krInfoLevel, BootFile
|
|
|
|
GPPI szDebugSect, szKRBreak, _krBreakLevel, BootFile
|
|
|
|
GPPI szDebugSect, szWin3Info, _Win3InfoLevel, Bootfile
|
|
|
|
GPPI szDebugSect, szWin3Break, _Win3BreakLevel, Bootfile
|
|
endif
|
|
endif ;KDEBUG
|
|
|
|
GPPS szDebugSect, szOutputTo, szAux, BootBuf, BootBufLen, BootFile
|
|
or ax, ax
|
|
jz @F
|
|
cmp ax, BootBufLen-2
|
|
jz @F
|
|
|
|
mov ah, 3ch ; creat file (zero length)
|
|
xor cx, cx
|
|
mov dx, dataOffset BootBuf
|
|
DOSCALL
|
|
jc @F
|
|
|
|
mov bx, ax
|
|
mov ah, 3eh ; now close it
|
|
DOSCALL
|
|
jc @F
|
|
|
|
mov ax, 3d42h ; open inherit, deny none, read/write
|
|
DOSCALL
|
|
jc @F
|
|
|
|
mov bx, ax ; dup handle
|
|
mov cx, 3 ; force AUX to be this file
|
|
mov ah, 46h
|
|
DOSCALL
|
|
|
|
mov ah, 3eh ; close temp file
|
|
DOSCALL
|
|
mov wDefRIP, 'i' ; since AUX is redirected, assume Ignore RIP
|
|
@@:
|
|
|
|
GPPI BootSect, FilesCached, MAXFHCACHELEN, BootFile, nostore
|
|
cmp ax, MINFHCACHELEN ; Validate length
|
|
jae @F
|
|
mov ax, MINFHCACHELEN
|
|
@@: cmp ax, MAXFHCACHELEN
|
|
jbe @F
|
|
mov ax, MAXFHCACHELEN
|
|
@@: mov fhCacheLen, ax ; Adjust # of cached file handles
|
|
shl ax, 1
|
|
shl ax, 1
|
|
add ax, dataoffset fhCache
|
|
mov fhCacheEnd, ax
|
|
|
|
GPPI BootSect, IdleSegLoad, 1, BootFile, nostore
|
|
mov fPokeAtSegments, al
|
|
|
|
ifdef WOW
|
|
GPPI BootSect, ExitLastApp, 0, BootFile, nostore
|
|
mov fExitOnLastApp, al
|
|
endif
|
|
|
|
mov bootExecBlock.lpfcb1.off,dataOffset win_show
|
|
mov bootExecBlock.lpfcb1.sel,ds
|
|
mov es,topPDB
|
|
|
|
mov bootExecBlock.lpcmdline.off,80h
|
|
mov bootExecBlock.lpcmdline.sel,es
|
|
|
|
mov lpBootApp.off,dataOffset app_name
|
|
mov lpBootApp.sel,ds
|
|
cmp graphics,1
|
|
jnz sb1
|
|
|
|
mov lpBootApp.off,dataOffset defapp
|
|
mov lpBootApp.sel,ds
|
|
|
|
sb1: mov di,dataOffset bootMods
|
|
|
|
sbloop1:
|
|
cCall LoadNewExe,<di>
|
|
SetKernelDS es
|
|
mov cx,-1
|
|
xor ax,ax
|
|
cld
|
|
repnz scasb
|
|
cmp di,dataOffset winmods
|
|
jb sbloop1
|
|
|
|
cmp graphics,1
|
|
jz sbloop2
|
|
; cCall InitFwdRef
|
|
jmps sb4
|
|
UnSetKernelDS es
|
|
|
|
sbloop2: ; load USER.EXE, GDI.EXE
|
|
ifdef FE_SB
|
|
;** If we just load the fae east modules, we want to
|
|
;** checks the system locale vale which is far east
|
|
;** locale.
|
|
;** If system locale is not far east, then we skip
|
|
;** to load far east module.
|
|
cmp di,dataOffset szBeforeWifeMan
|
|
je SB_DoFarEastModule
|
|
cmp di,dataOffset szBeforeWinNls
|
|
jne SB_DoLoadModule
|
|
|
|
SB_DoFarEastModule:
|
|
cCall GetSystemDefaultLangID
|
|
; return: ax is system locale value
|
|
cmp ax,411h
|
|
je SB_DoLoadModule
|
|
cmp ax,412h
|
|
je SB_DoLoadModule
|
|
cmp ax,404h
|
|
je SB_DoLoadModule
|
|
cmp ax,804h
|
|
je SB_DoLoadModule
|
|
cmp ax,0c04h
|
|
je SB_DoLoadModule
|
|
|
|
; skip to next module
|
|
mov cx,-1
|
|
xor ax,ax
|
|
cld
|
|
repnz scasb
|
|
jmp SB_NotKeyboardDriver
|
|
SB_DoLoadModule:
|
|
endif
|
|
cCall LoadNewExe,<di>
|
|
push ax ; Save hInst return value
|
|
SetKernelDS es
|
|
mov cx,-1
|
|
xor ax,ax
|
|
cld
|
|
repnz scasb
|
|
pop si ; Get hInst of latest module in SI
|
|
|
|
;** If we just loaded the keyboard driver, we want to
|
|
;** point our explicit link variables to the AnsiToOem and
|
|
;** OemToAnsi functions so OpenFile can use them for the
|
|
;** remaining boot modules
|
|
cmp di,dataOffset szAfterKeyboardDriver
|
|
jne SB_NotKeyboardDriver
|
|
|
|
push ds ; Save regs we care about
|
|
push di
|
|
regptr pStr,ds,bx
|
|
|
|
mov bx,dataOffset keyprocstr
|
|
cCall GetProcAddress,<si,pStr>
|
|
mov pKeyProc.off,ax
|
|
mov pKeyProc.sel,dx
|
|
|
|
mov bx,dataOffset keyprocstr1
|
|
cCall GetProcAddress,<si,pStr>
|
|
mov pKeyProc1.off,ax
|
|
mov pKeyProc1.sel,dx
|
|
|
|
pop di
|
|
pop ds
|
|
|
|
SB_NotKeyboardDriver:
|
|
cmp di,dataOffset defapp
|
|
jb sbloop2
|
|
|
|
; OPTIMIZE BEGIN
|
|
|
|
; OPTIMIZE END
|
|
|
|
sb4:
|
|
cCall InitFwdRef ; Gets stuff we can't dynalink to
|
|
cCall InternalEnableDOS ; Enable int21 hooks
|
|
;sb4:
|
|
call check_TEMP
|
|
|
|
; Get the shell name from SYSTEM.INI
|
|
|
|
mov ax,dataoffset Shell
|
|
GPPS1 BootSect, ax, lpBootApp.off, BootBuf, BootBufLen, BootFile
|
|
|
|
;** Here we need to convert the command line to ANSI
|
|
cmp WORD PTR pKeyProc1[2],0 ; Paranoia...
|
|
jz @F
|
|
|
|
;** Zero terminate the string before passing to OemToAnsi
|
|
mov es,bootExecBlock.lpcmdline.sel
|
|
mov bl,es:[80h] ; Get length byte
|
|
xor bh,bh
|
|
xchg BYTE PTR es:[bx+81h],bh ; Replace terminator char with zero
|
|
|
|
;** Call the keyboard driver
|
|
push es ; Save the seg reg
|
|
push bx ; Save offset and char saved
|
|
push es
|
|
push 81h ; Always starts here (fixed offset)
|
|
push es
|
|
push 81h
|
|
cCall [pKeyProc1] ; Call OemToAnsi in keyboard driver
|
|
pop bx ; Get saved info
|
|
pop es
|
|
mov al,bh ; Saved char from string
|
|
xor bh,bh
|
|
mov BYTE PTR es:[bx+81h],al ; Replace the character
|
|
@@:
|
|
|
|
mov ax,dataOffset bootExecBlock
|
|
regptr dsax,ds,ax
|
|
cmp graphics,1
|
|
jz @F
|
|
|
|
cCall LoadModule,<lpBootApp,dsax>
|
|
jmps asdf
|
|
|
|
@@: cCall FlushCachedFileHandle,<hUser> ; in case 100 fonts are loaded
|
|
farptr lpBootBuf,ds,di
|
|
mov di, dataoffset BootBuf
|
|
cCall LoadModule,<lpBootBuf,dsax>
|
|
|
|
asdf:
|
|
cmp ax,32
|
|
jbe sb6
|
|
|
|
cCall GetExePtr,<ax>
|
|
mov hShell, ax
|
|
jmp bootdone
|
|
|
|
sb6:
|
|
ReSetKernelDS
|
|
les bx, lpBootApp
|
|
krDebugOut DEB_ERROR, "BOOT: unable to load @ES:BX"
|
|
cCall LoadFail,<lpBootApp>
|
|
|
|
; cmp pDisableProc.sel,0 ; Is there a USER around yet?
|
|
; jz @F
|
|
; cCall pDisableProc
|
|
;@@:
|
|
jmp bootfail
|
|
UnSetKernelDS
|
|
cEnd nogen
|
|
|
|
|
|
;------------------------------------------------------------------------
|
|
|
|
assumes ds,nothing
|
|
assumes es,nothing
|
|
|
|
cProc LoadNewExe,<PUBLIC,NEAR>,<si,di>
|
|
parmW pname
|
|
cBegin
|
|
farptr lpparm,ax,ax
|
|
farptr lpBootBuf,ds,di
|
|
mov di, dataoffset BootBuf
|
|
|
|
CheckKernelDS
|
|
ReSetKernelDS
|
|
ifdef WOW
|
|
; ATM Alters system.ini registry boot section to load its own SYSTEM.DRV
|
|
; However on WOW this causes us to fail to load correctly. Also we
|
|
; would be hard pushed to support the current 16 bit ATM since it relies
|
|
; on internals of 16 bit GDI which are not present in WOW.
|
|
; For this Beta I'm going to ignore the bootsection of the registry
|
|
; when loading drivers and thus ATM will not get installed.
|
|
; At least the user will be protected from not being able to boot WOW
|
|
; BUGBUG - Consider
|
|
; MattFe Oct9 92
|
|
|
|
mov di,pname
|
|
else
|
|
GPPS1 BootSect, pname, pname, BootBuf, BootBufLen, BootFile
|
|
endif
|
|
xor ax,ax
|
|
cCall LoadModule,<lpBootBuf,lpparm>
|
|
cmp ax,2
|
|
jne lne1
|
|
krDebugOut DEB_ERROR, "Can't find @DS:DI"
|
|
; kerror ERR_LDBOOT,<BOOT: Unable to find file - >,ds,di
|
|
jmps lne4
|
|
lne1:
|
|
cmp ax,11
|
|
jne lne2
|
|
krDebugOut DEB_ERROR, "Invalid EXE file @DS:DI"
|
|
; kerror ERR_LDBOOT,<BOOT: Invalid .EXE file - >,ds,di
|
|
jmps lne4
|
|
lne2:
|
|
cmp ax,15
|
|
jnz lne3
|
|
krDebugOut DEB_ERROR, "Invalid protect mode EXE file @DS:DI"
|
|
; kerror ERR_LDBOOT,<BOOT: Invalid protect mode .EXE file - >,ds,di
|
|
jmps lne4
|
|
|
|
lne3:
|
|
cmp ax, 4
|
|
jne lne3a
|
|
krDebugOut DEB_ERROR, "Out of files (set FILES=30 in CONFIG.SYS) @DS:DI"
|
|
; kerror ERR_LDFILES,<BOOT: Out of files, (set FILES=30 in CONFIG.SYS) loading - >,ds,di
|
|
jmps lne4
|
|
|
|
lne3a:
|
|
cmp ax, 32
|
|
jae lnex
|
|
|
|
NoLoadIt:
|
|
; kerror ERR_LDBOOT,<BOOT: Unable to load - >,ds,pname
|
|
krDebugOut DEB_ERROR, "Unable to load @DS:DI (#ax)"
|
|
lne4:
|
|
cCall LoadFail,dsdi
|
|
mov ax,1
|
|
cCall ExitKernel,<ax>
|
|
lnex:
|
|
UnSetKernelDS
|
|
cEnd
|
|
|
|
sEnd INITCODE
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; BootDone ;
|
|
; ;
|
|
; Boot is done when all of modules are loaded. Here we do a bit of ;
|
|
; clean up, such as validating code segments, initing free memory to ;
|
|
; CCCC, freeing up the fake TDB, and finally reallocating the init ;
|
|
; code away. ;
|
|
; ;
|
|
; Arguments: ;
|
|
; none ;
|
|
; ;
|
|
; Returns: ;
|
|
; nothing ;
|
|
; ;
|
|
; Error Returns: ;
|
|
; ;
|
|
; Registers Preserved: ;
|
|
; ;
|
|
; Registers Destroyed: ;
|
|
; all ;
|
|
; ;
|
|
; Calls: ;
|
|
; ;
|
|
; History: ;
|
|
; ;
|
|
; Wed Apr 15, 1987 08:53:23p -by- David N. Weise [davidw] ;
|
|
; Added this nifty comment block. ;
|
|
;-----------------------------------------------------------------------;
|
|
|
|
|
|
DataBegin INIT
|
|
|
|
externB beg_initdata
|
|
|
|
szKernel DB 'KERNEL',0
|
|
szWindows DB 'WINDOWS',0
|
|
|
|
if KDEBUG
|
|
szDebugOptions DB 'DebugOptions',0
|
|
szDebugFilter DB 'DebugFilter',0
|
|
|
|
szChecksum DB 'EnableSegmentChecksum',0
|
|
szSweepFreak DB 'LRUSweepFrequency',0
|
|
sz80x87 DB 'NoUse80x87',0
|
|
szFastFP DB 'FastFP', 0
|
|
|
|
externW DebugOptions
|
|
externW DebugFilter
|
|
|
|
ifdef DISABLE
|
|
externB fLoadTrace
|
|
endif
|
|
|
|
endif ; KDEBUG
|
|
|
|
ifdef SDEBUG
|
|
szEMSDebug DB 'EnableEMSDebug',0
|
|
endif
|
|
szgrab_386 DB '386GRABBER',0
|
|
|
|
if SWAPPRO
|
|
szSwapPro DB 'SwapProfile',0
|
|
szSwapFile DB 'SWAPPRO.DAT',0
|
|
endif
|
|
|
|
DataEnd INIT
|
|
|
|
|
|
sBegin INITCODE
|
|
assumes cs,CODE
|
|
assumes ds,nothing
|
|
assumes es,nothing
|
|
|
|
externB beg_initcode
|
|
|
|
|
|
cProc BootDone,<PUBLIC,NEAR>
|
|
cBegin nogen
|
|
|
|
SetKernelDS
|
|
|
|
if KDEBUG
|
|
|
|
; Get win.ini [Windows] DebugOptions
|
|
GPI szWindows, szDebugOptions, DebugOptions
|
|
; Get win.ini [Windows] DebugFilter
|
|
GPI szWindows, szDebugFilter, DebugFilter
|
|
|
|
; Now set various internal flags based on DebugOptions
|
|
|
|
xor ax,ax
|
|
test DebugOptions,DBO_CHECKHEAP
|
|
jz @F
|
|
inc ax
|
|
@@:
|
|
mov es,pGlobalHeap
|
|
mov es:[hi_check],ax
|
|
|
|
test DebugOptions,DBO_CHECKFREE
|
|
jz @F
|
|
or Kernel_flags,kf_check_free
|
|
@@:
|
|
ifdef DISABLE
|
|
xor ax,ax
|
|
test DebugOptions,DBO_LOADTRACE
|
|
jz @F
|
|
mov fLoadTrace, al
|
|
@@:
|
|
endif ; DISABLE
|
|
test DebugOptions,DBO_DISABLEGPTRAPPING
|
|
jz wants_trapping
|
|
|
|
mov ax,0203h ; Reset GP fault.
|
|
mov bl,0Dh
|
|
mov cx,prevInt0Dproc.sel
|
|
mov dx,prevInt0Dproc.off
|
|
int 31h
|
|
|
|
mov ax,0203h ; Reset invalid op-code exception.
|
|
mov bl,06h
|
|
mov cx,prevIntx6proc.sel
|
|
mov dx,prevIntx6proc.off
|
|
int 31h
|
|
|
|
mov ax,0203h ; Reset page fault.
|
|
mov bl,0Eh
|
|
mov cx,prevInt0Eproc.sel
|
|
mov dx,prevInt0Eproc.off
|
|
int 31h
|
|
wants_trapping:
|
|
|
|
if SWAPPRO
|
|
GPI szKernel, szSwapPro, 1, nostore
|
|
mov fSwapPro, al
|
|
|
|
mov bx,TopPDB
|
|
mov ah,50h
|
|
pushf
|
|
FCLI
|
|
call prevInt21Proc
|
|
|
|
lea dx,szSwapFile
|
|
xor cx,cx
|
|
mov ah,3Ch
|
|
pushf
|
|
FCLI
|
|
call prevInt21Proc
|
|
mov hSwapPro,ax
|
|
|
|
mov bx,cur_dos_pdb
|
|
mov ah,50h
|
|
pushf
|
|
FCLI
|
|
call prevInt21Proc
|
|
endif
|
|
GPI szKernel, szChecksum, 1, nostore
|
|
mov fChkSum,al
|
|
|
|
GPI szKernel, sz80x87, 0, nostore
|
|
or ax,ax
|
|
jz wants_8087
|
|
mov f8087,0
|
|
and WinFlags,NOT WF1_80x87 ;Turn off corresponding WinFlags bit
|
|
wants_8087:
|
|
GPI szKernel, szFastFP, 1, nostore
|
|
mov fastFP, al
|
|
|
|
GPI szKernel, szSweepFreak, 500, nostore
|
|
else
|
|
mov ax, 500
|
|
endif ; KDEBUG
|
|
|
|
ifdef WOW
|
|
xor ax,ax
|
|
endif
|
|
|
|
or ax,ax
|
|
jz nolrusweep
|
|
|
|
test WinFlags[1], WF1_PAGING
|
|
jnz short nolrusweep
|
|
|
|
mov bx,codeOffset lrusweep
|
|
regptr csbx,cs,bx
|
|
xor dx,dx
|
|
cCall pTimerProc,<dx,ax,csbx>
|
|
nolrusweep:
|
|
|
|
if SDEBUG
|
|
GPI szKernel, szEMSDebug, 0, nostore
|
|
or ax,ax
|
|
jz no_EMS_debug_wanted
|
|
or Kernel_flags,kf_EMS_debug
|
|
no_EMS_debug_wanted:
|
|
endif
|
|
|
|
if LDCHKSUM
|
|
cCall ValidateCodeSegments
|
|
endif
|
|
|
|
if KDEBUG
|
|
mov fCheckFree,0
|
|
endif
|
|
|
|
; Get the shell name from SYSTEM.INI
|
|
|
|
GPPS BootSect, szgrab_386, szgrab_386, grab_name, 128, BootFile
|
|
|
|
mov es,curTDB ; ES = TDB of fake task
|
|
push es
|
|
cCall DeleteTask,<es> ; Flush bogus task
|
|
pop es
|
|
xor dx,dx
|
|
mov es:[TDB_sig],dx ; Clear signature word.
|
|
mov curTDB,dx ; Let someone else be current task
|
|
|
|
; switch to the temp stack since we're about to Realloc the present one away
|
|
|
|
mov ax, ss
|
|
|
|
FCLI
|
|
SetKernelDS ss
|
|
mov sp,dataOffset gmove_stack
|
|
FSTI
|
|
|
|
cCall free_sel,<ax>
|
|
|
|
|
|
; Shrink DGROUP down to its post initialization size
|
|
|
|
mov cx,dataOFFSET beg_initdata ; don't need init data
|
|
|
|
; dx doubles as high word and specifies fixed
|
|
; reallocation
|
|
xor dx,dx
|
|
cCall IGlobalReAlloc,<ds,dx,cx,dx> ; Realloc DGROUP
|
|
|
|
xor dx,dx
|
|
|
|
|
|
; Now shrink the resident CODE segment
|
|
|
|
mov cx,codeOFFSET beg_initcode ; dont need init code
|
|
|
|
; cCall IGlobalReAlloc,<cs,dxcx,dx>
|
|
; jmps BootSchedule ; Jump to schedule first app
|
|
|
|
push cs ; Arguments to GlobalReAlloc
|
|
push dx
|
|
push cx
|
|
push dx
|
|
push cs ; Where GlobalReAlloc will eventually return
|
|
mov ax,codeOFFSET BootSchedule
|
|
push ax
|
|
jmp near ptr IGlobalReAlloc ; Jump to GlobalReAlloc
|
|
|
|
UnSetKernelDS ss
|
|
UnSetKernelDS
|
|
cEnd nogen
|
|
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; check_TEMP
|
|
;
|
|
; If the environment variable TEMP points to garbage then GetTempFile
|
|
; screws up. We fix it by wiping out the TEMP string if it points
|
|
; to garbage.
|
|
;
|
|
; Entry:
|
|
; none
|
|
;
|
|
; Returns:
|
|
;
|
|
; Registers Preserved:
|
|
; all
|
|
;
|
|
; History:
|
|
; Thu 11-May-1989 16:39:34 -by- David N. Weise [davidw]
|
|
; Wrote it.
|
|
;-----------------------------------------------------------------------;
|
|
|
|
assumes ds,nothing
|
|
assumes es,nothing
|
|
|
|
cProc check_TEMP,<PUBLIC,NEAR>,<ds,es>
|
|
ifdef FE_SB
|
|
localW pMyBuf
|
|
endif
|
|
cBegin
|
|
pusha
|
|
SetKernelDS
|
|
sub sp,130
|
|
mov di,sp
|
|
CheckKernelDS
|
|
mov ds,TopPDB
|
|
UnSetKernelDS
|
|
mov ds,ds:[PDB_environ]
|
|
xor si, si ; assume DS:SI points to environment
|
|
cCall GetTempDrive,<si>
|
|
smov es,ss
|
|
cld
|
|
stosw
|
|
ifdef FE_SB
|
|
mov pMyBuf,di ; save string offset(exclude D:)
|
|
endif
|
|
stmp2:
|
|
lodsw
|
|
or al,al ; no more enviroment
|
|
ifdef FE_SB
|
|
jnz @F ; I hate this
|
|
jmp stmpNo1
|
|
@@:
|
|
else
|
|
jz stmpNo1
|
|
endif
|
|
cmp ax,'ET' ; Look for TEMP=
|
|
jne stmp3
|
|
lodsw
|
|
cmp ax,'PM'
|
|
jne stmp3
|
|
lodsb
|
|
cmp al,'='
|
|
je stmpYes
|
|
stmp3: lodsb
|
|
or al,al
|
|
jnz stmp3
|
|
jmp stmp2
|
|
stmpYes:
|
|
push si ; save pointer to TEMP
|
|
push ds
|
|
|
|
push si ; spaces are legal, but they
|
|
lookForSpace: ; confuse too many apps, so
|
|
lodsb ; we treat them as illegal
|
|
cmp al, ' '
|
|
jz stmpFoundSpace
|
|
or al, al
|
|
jnz lookForSpace
|
|
pop si
|
|
cmp byte ptr [si+1],':'
|
|
jne stmpnodrive
|
|
and byte ptr [si],NOT 20h ; springboard needs this!
|
|
dec di
|
|
dec di
|
|
stmpnodrive:
|
|
lodsb
|
|
or al,al
|
|
jz stmpNo
|
|
stosb
|
|
jmp stmpnodrive
|
|
|
|
stmpNo:
|
|
mov ax,'~\'
|
|
cmp es:[di-1],al ; does it already end in \
|
|
jnz stmpNoF ; no, just store it
|
|
dec di ; override it
|
|
ifdef FE_SB
|
|
push si
|
|
mov si,pMyBuf
|
|
call FarMyIsDBCSTrailByte ;is that '\' a DBCS trailing byte?
|
|
cmc
|
|
adc di,0 ;skip it if yes.
|
|
pop si
|
|
endif
|
|
stmpNoF:
|
|
stosw
|
|
xor ax,ax
|
|
stosb ; zero terminate it
|
|
pop es ; recover pointer to TEMP
|
|
pop di
|
|
smov ds,ss
|
|
mov dx,sp
|
|
mov ax,5B00h
|
|
xor cx,cx
|
|
DOSCALL
|
|
jnc stmpClose
|
|
cmp al,80 ; Did we fail because the file
|
|
jz stmpNo1 ; already exists?
|
|
stmpNukeIt:
|
|
sub di,5 ; Get the TEMP= part.
|
|
@@: mov al,'x'
|
|
xchg al,es:[di]
|
|
inc di
|
|
or al,al
|
|
jnz @B
|
|
mov byte ptr es:[di-1],0
|
|
jmps stmpNo1
|
|
|
|
stmpClose:
|
|
mov bx,ax
|
|
mov ah,3Eh
|
|
DOSCALL
|
|
mov ah,41h
|
|
DOSCALL
|
|
|
|
stmpNo1:
|
|
add sp,130
|
|
popa
|
|
cEnd
|
|
stmpFoundSpace:
|
|
pop si
|
|
pop es
|
|
pop di
|
|
jmps stmpNukeIt
|
|
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; get_windir
|
|
;
|
|
; Get a pointer to the 'windows' directory.
|
|
;
|
|
; Entry:
|
|
; DS => environment string
|
|
;
|
|
; Returns:
|
|
; CX = length of string
|
|
; DI => WFP of 'windows' directory
|
|
;
|
|
; Registers Preserved:
|
|
; all
|
|
;
|
|
; History:
|
|
; Mon 16-Oct-1989 23:17:23 -by- David N. Weise [davidw]
|
|
; Wrote it!
|
|
;-----------------------------------------------------------------------;
|
|
|
|
assumes ds,nothing
|
|
assumes es,nothing
|
|
|
|
ifdef WOW
|
|
;-----------------------------------------------------------------------;
|
|
; original get_windir looks for the environment variable 'windir' (all
|
|
; lowercase) to set the 'windows' directory.
|
|
;
|
|
; On NT the equivalent environment variable is 'SystemRoot' (all
|
|
; uppercase). Hence the code is different.
|
|
;
|
|
; - nanduri
|
|
;
|
|
; There are some customers that used the undocumented trick on win31
|
|
; of moving windir to some network location by putting win.com there.
|
|
; The result would be that the "Windows Directory" would point
|
|
; to the network location, and the system directory would be local.
|
|
; Currently, the way WINDIR is supported involves hacks in a couple
|
|
; of different places. What would be best is if you could do a
|
|
; "set windir=xxxx" in your autoexec.nt and we would look for windir
|
|
; here and we would code it to emulate win31 behavior. However, that's
|
|
; broken right now, and windir is only regenerated after krnlx86.exe
|
|
; has finished booting. So the approach taken here is to look for
|
|
; a new environment variable called win16dir, and if it exists, make
|
|
; the windows directory point to it. Systemroot is still used to
|
|
; generate the system directory.
|
|
;
|
|
; We want to allow NT to be installed into a directory with a long
|
|
; name, so we use GetShortPathName on the directory we get from
|
|
; either SystemRoot or Win16Dir variables.
|
|
; -- DaveHart 9-Feb-96
|
|
;-----------------------------------------------------------------------;
|
|
|
|
szSystemRoot DB 'SYSTEMROOT=',0
|
|
szWin16Dir DB 'WIN16DIR=',0
|
|
|
|
cProc get_windir,<PUBLIC,NEAR>
|
|
cBegin nogen
|
|
|
|
push es
|
|
mov ax, cs ; the string is in 'cs', see above
|
|
mov es, ax
|
|
mov di, codeoffset szSystemRoot
|
|
call get_env_var_ptr
|
|
|
|
push dx
|
|
mov dx, ds
|
|
mov es, dx
|
|
SetKernelDS
|
|
mov si, dataoffset achRealWindowsDir
|
|
regptr esdi,es,di
|
|
regptr dssi,ds,si
|
|
mov cx, WINDIR_BUFSIZE
|
|
push dx
|
|
cCall GetShortPathName, <esdi, dssi, cx>
|
|
pop dx
|
|
mov cbRealWindowsDir,ax
|
|
mov ds, dx ;restore ds
|
|
pop dx
|
|
assumes ds,nothing
|
|
|
|
push cx
|
|
push di
|
|
|
|
mov ax, cs ; the string is in 'cs', see above
|
|
mov es, ax
|
|
mov di, codeoffset szWin16Dir
|
|
call get_env_var_ptr
|
|
or di, di ;does win16dir exist?
|
|
jz gw_not
|
|
add sp, 4 ;throw away systemroot
|
|
jmp short gw_cont
|
|
|
|
gw_not:
|
|
pop di
|
|
pop cx
|
|
|
|
gw_cont:
|
|
; Now ds:di points to the Windows directory string in
|
|
; the environment block. It may be a long pathname,
|
|
; so fix it up.
|
|
smov es, ds
|
|
SetKernelDS
|
|
mov si, dataoffset achWindowsDir
|
|
regptr esdi,es,di
|
|
regptr dssi,ds,si
|
|
mov cx, WINDIR_BUFSIZE
|
|
cCall GetShortPathName, <esdi, dssi, cx>
|
|
mov cx, ax
|
|
smov ds, es
|
|
assumes ds,nothing
|
|
|
|
pop es
|
|
ret
|
|
|
|
cEnd nogen
|
|
|
|
cProc get_env_var_ptr,<PUBLIC,NEAR>
|
|
cBegin nogen
|
|
cld
|
|
push si
|
|
xor si,si
|
|
|
|
push di
|
|
mov cx,-1
|
|
xor ax,ax
|
|
repnz scasb
|
|
not cx
|
|
dec cx ; length of szSystemRoot
|
|
pop di
|
|
|
|
gw_cmp:
|
|
mov al, [si]
|
|
or al, al
|
|
jz gw_exit
|
|
|
|
push di
|
|
push cx
|
|
repz cmpsb ; compare the inputstring with szSystemRoot
|
|
pop cx
|
|
pop di
|
|
|
|
jnz gw_next ; not szSystemRoot
|
|
xor cx,cx ; yes szSystemRoot, cx=0 indicates so
|
|
mov di,si
|
|
|
|
gw_next:
|
|
lodsb
|
|
or al,al
|
|
jnz gw_next ; skip to the terminating NULL.
|
|
or cx,cx ; cx==0 implies... found szSystemRoot
|
|
jnz gw_cmp ; compare with the next environment string
|
|
mov cx,si ; here if found szSystemRoot.
|
|
sub cx,di
|
|
mov ax,di
|
|
dec cx
|
|
|
|
|
|
gw_exit:
|
|
mov di,ax
|
|
pop si
|
|
ret
|
|
cEnd nogen
|
|
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; original get_windir is within the 'else' 'endif' block
|
|
;
|
|
;-----------------------------------------------------------------------;
|
|
|
|
else
|
|
|
|
cProc get_windir,<PUBLIC,NEAR>
|
|
cBegin nogen
|
|
cld
|
|
push si
|
|
xor di,di
|
|
xor si,si
|
|
gw: lodsw
|
|
or al,al ; no more enviroment
|
|
jz gw_exit
|
|
cmp ax,'iw' ; Look for windir=
|
|
jne @F
|
|
lodsw
|
|
cmp ax,'dn'
|
|
jne @F
|
|
lodsw
|
|
cmp ax,'ri'
|
|
jne @F
|
|
lodsb
|
|
cmp al,'='
|
|
je gw_got_it
|
|
@@: lodsb
|
|
or al,al
|
|
jnz @B
|
|
jmp gw
|
|
gw_got_it:
|
|
mov di,si
|
|
@@: lodsb
|
|
or al,al
|
|
jnz @B
|
|
mov cx,si
|
|
sub cx,di
|
|
dec cx
|
|
gw_exit:
|
|
pop si
|
|
ret
|
|
cEnd nogen
|
|
|
|
endif
|
|
|
|
|
|
sEnd INITCODE
|
|
|
|
;------------------------------------------------------------------------
|
|
|
|
sBegin STACK
|
|
|
|
; Boot TDB
|
|
|
|
boottdb equ this byte
|
|
DB SIZE TDB dup (0)
|
|
|
|
; do a clumsy paragraph alignment
|
|
|
|
rept 16
|
|
if ($ - boottdb) and 0Fh
|
|
db 0
|
|
endif
|
|
endm
|
|
|
|
; Dummy arena entry so boot SS looks like a valid object
|
|
|
|
DB 'M'
|
|
DW -1
|
|
DW (BOOTSTACKSIZE + 31)/16
|
|
DB 0
|
|
DW 5 DUP (0)
|
|
|
|
; Another in case we have to tweek the low order bit of SS
|
|
|
|
DB 'M'
|
|
DW -1
|
|
DW (BOOTSTACKSIZE + 15)/16
|
|
DB 0
|
|
DW 5 DUP (0)
|
|
|
|
; Boot stack
|
|
|
|
stackbottom equ this word
|
|
DB BOOTSTACKSIZE DUP (0)
|
|
stacktop equ this word
|
|
|
|
DW -1
|
|
|
|
sEnd STACK
|
|
|
|
end BootStrap
|