948 lines
25 KiB
NASM
948 lines
25 KiB
NASM
|
;/*
|
|||
|
; * Microsoft Confidential
|
|||
|
; * Copyright (C) Microsoft Corporation 1991
|
|||
|
; * All Rights Reserved.
|
|||
|
; */
|
|||
|
;===========================================================================
|
|||
|
;
|
|||
|
; FILE: GLBLINIT.ASM
|
|||
|
;
|
|||
|
;===========================================================================
|
|||
|
|
|||
|
;===========================================================================
|
|||
|
;Declaration of include files
|
|||
|
;===========================================================================
|
|||
|
|
|||
|
;
|
|||
|
;---------------------------------------------------------------------------
|
|||
|
;
|
|||
|
; M024 : B#5495. Added "Insufficient memory" message when FORMAT cannot
|
|||
|
; allocate memory for FAT, Directory... etc
|
|||
|
;
|
|||
|
;---------------------------------------------------------------------------
|
|||
|
;
|
|||
|
debug equ 0
|
|||
|
|
|||
|
.xlist
|
|||
|
|
|||
|
INCLUDE BPB.INC
|
|||
|
INCLUDE DOSEQUS.INC
|
|||
|
INCLUDE DOSMAC.INC
|
|||
|
INCLUDE SYSCALL.INC
|
|||
|
INCLUDE FOREQU.INC
|
|||
|
INCLUDE FORMACRO.INC
|
|||
|
INCLUDE IOCTL.INC
|
|||
|
INCLUDE FORSWTCH.INC
|
|||
|
INCLUDE SAFEDEF.INC
|
|||
|
INCLUDE SYSVAR.INC
|
|||
|
.list
|
|||
|
|
|||
|
|
|||
|
;===========================================================================
|
|||
|
; Data segment
|
|||
|
;===========================================================================
|
|||
|
|
|||
|
DATA SEGMENT PUBLIC PARA 'DATA'
|
|||
|
|
|||
|
SECTORS_FOR_MIRROR EQU 7 ; # extra buffer sectors
|
|||
|
; required by Mirror utility,
|
|||
|
; apart from FAT & Root Dir
|
|||
|
|
|||
|
;===========================================================================
|
|||
|
; Declarations for all publics in other modules used by this module
|
|||
|
;===========================================================================
|
|||
|
|
|||
|
;Bytes
|
|||
|
EXTRN msgCrLF :BYTE
|
|||
|
EXTRN DriveToFormat :BYTE
|
|||
|
EXTRN ClustBound_Flag :BYTE
|
|||
|
EXTRN FileStat :BYTE
|
|||
|
EXTRN SystemDriveLetter :BYTE
|
|||
|
EXTRN SecPerClus :BYTE
|
|||
|
EXTRN CMCDDFlag :BYTE
|
|||
|
|
|||
|
;Words
|
|||
|
EXTRN SwitchMap :WORD
|
|||
|
EXTRN mSize :WORD
|
|||
|
EXTRN mStart :WORD
|
|||
|
EXTRN ClustBound_Buffer_Seg :WORD
|
|||
|
EXTRN Paras_per_fat :WORD
|
|||
|
|
|||
|
;Pointers and DWORDs
|
|||
|
EXTRN DirectorySector :DWORD
|
|||
|
EXTRN FatSpace :DWORD
|
|||
|
EXTRN FatSector :DWORD
|
|||
|
EXTRN DirBuf :DWORD
|
|||
|
EXTRN TotalClusters :DWORD
|
|||
|
|
|||
|
;Messages
|
|||
|
EXTRN msgFormatNotSupported :BYTE
|
|||
|
EXTRN msgCantZThisDrive :BYTE
|
|||
|
EXTRN msgCantZWithQ :BYTE
|
|||
|
EXTRN msgCantZFAT16 :BYTE
|
|||
|
EXTRN msgCantZFAT32 :BYTE
|
|||
|
EXTRN msgZFAT32Huge :BYTE
|
|||
|
EXTRN msgZFAT32TooHuge :BYTE
|
|||
|
EXTRN msgOutOfMemory :BYTE
|
|||
|
EXTRN msgInsufficientMemory :BYTE
|
|||
|
IFNDEF OPKBLD
|
|||
|
EXTRN msgNoSysSwitch :BYTE
|
|||
|
ENDIF ;OPKBLD
|
|||
|
|
|||
|
;Structures
|
|||
|
EXTRN SavedParams :BYTE
|
|||
|
EXTRN DeviceParameters :BYTE
|
|||
|
EXTRN IsExtRAWIODrv :BYTE
|
|||
|
EXTRN Bios :BYTE
|
|||
|
EXTRN dos :BYTE
|
|||
|
EXTRN Command :BYTE
|
|||
|
IFDEF DBLSPACE_HOOKS
|
|||
|
EXTRN DblSpaceBin :BYTE
|
|||
|
ENDIF
|
|||
|
|
|||
|
PUBLIC FATNotAllInMem
|
|||
|
PUBLIC FATSecCntInMem
|
|||
|
|
|||
|
FATNotAllInMem db 0
|
|||
|
FATSecCntInMem dd 0
|
|||
|
|
|||
|
ifdef NEC_98
|
|||
|
AllocSectorSize DW ? ; fixed #16585
|
|||
|
endif
|
|||
|
DATA ENDS
|
|||
|
|
|||
|
;===========================================================================
|
|||
|
; Executable code segment
|
|||
|
;===========================================================================
|
|||
|
|
|||
|
CODE SEGMENT PUBLIC PARA 'CODE'
|
|||
|
ASSUME CS:CODE, DS:DATA, ES:DATA
|
|||
|
|
|||
|
|
|||
|
;===========================================================================
|
|||
|
; Declarations for all publics in other modules used by this module
|
|||
|
;===========================================================================
|
|||
|
|
|||
|
;Functions
|
|||
|
|
|||
|
;Labels
|
|||
|
EXTRN FatalExit :NEAR
|
|||
|
EXTRN ReadDos :NEAR
|
|||
|
EXTRN SysPrm :NEAR
|
|||
|
EXTRN exclusive_access_failed :NEAR
|
|||
|
EXTRN IsLockErr? :NEAR
|
|||
|
EXTRN Calc_Big32_Fat :NEAR
|
|||
|
EXTRN Calc_Big16_Fat :NEAR
|
|||
|
EXTRN GetTotalClusters :NEAR
|
|||
|
EXTRN Yes? :NEAR
|
|||
|
|
|||
|
;===========================================================================
|
|||
|
; Declarations for all publics in this module
|
|||
|
;===========================================================================
|
|||
|
|
|||
|
PUBLIC Global_Init
|
|||
|
PUBLIC GetDeviceParameters
|
|||
|
PUBLIC ModifyDevPrmsForZSwich
|
|||
|
|
|||
|
; for debug
|
|||
|
|
|||
|
PUBLIC Copy_Device_Parameters
|
|||
|
PUBLIC Alloc_Dir_Buf
|
|||
|
PUBLIC Alloc_Fat_Buf
|
|||
|
PUBLIC Alloc_Fat_Sec_Buf
|
|||
|
PUBLIC Alloc_DirBuf2
|
|||
|
PUBLIC Alloc_Cluster_Buf
|
|||
|
IFDEF OPKBLD
|
|||
|
PUBLIC Do_Switch_S
|
|||
|
ENDIF ;OPKBLD
|
|||
|
|
|||
|
|
|||
|
;===========================================================================
|
|||
|
;
|
|||
|
; Global_Init : This procedure first gets the default drive parameters.
|
|||
|
; It then allocates buffer space for the root directory
|
|||
|
; sector, FAT,a fat sector, a file header and first
|
|||
|
; root DIR sector based on these parameters. It
|
|||
|
; then checks for the /s switch and if this is present,
|
|||
|
; a buffer is allocated for the system files and these
|
|||
|
; are read into memory. A prompt to insert the system
|
|||
|
; disk will be given in the case of removable media.
|
|||
|
;
|
|||
|
;===========================================================================
|
|||
|
|
|||
|
Global_Init proc near
|
|||
|
|
|||
|
lea DX, DeviceParameters ; Get the default drive parameters
|
|||
|
mov DeviceParameters.DP_SpecialFunctions, 0
|
|||
|
.errnz EDP_SPECIALFUNCTIONS NE DP_SPECIALFUNCTIONS
|
|||
|
call GetDeviceParameters
|
|||
|
|
|||
|
jnc GotDeviceParameters
|
|||
|
call IsLockErr?
|
|||
|
jnc @f
|
|||
|
jmp exclusive_access_failed
|
|||
|
@@:
|
|||
|
Message msgFormatNotSupported
|
|||
|
stc ; Let the jump to FatalExit be made
|
|||
|
ret ; in the main routine, upon returning
|
|||
|
|
|||
|
GotDeviceParameters:
|
|||
|
ifdef NEC_98 ; fixed #16585
|
|||
|
cmp DeviceParameters.DP_DeviceType,DEV_HARDDISK
|
|||
|
je @f
|
|||
|
cmp DeviceParameters.DP_DeviceType,DEV_OPTICAL
|
|||
|
je @f
|
|||
|
mov AllocSectorSize,0400h
|
|||
|
jmp short Set_ok
|
|||
|
@@:
|
|||
|
push bx
|
|||
|
mov bx, DeviceParameters.DP_BPB.oldBPB.BPB_BytesPerSector
|
|||
|
mov AllocSectorSize,bx
|
|||
|
pop bx
|
|||
|
Set_ok:
|
|||
|
endif ; fixed #16585
|
|||
|
call Copy_Device_Parameters ; Save the device parameters
|
|||
|
; for when we exit
|
|||
|
cmp SecPerClus,0
|
|||
|
je NoZSwtch
|
|||
|
mov ax,1 ; Noisy version
|
|||
|
call ModifyDevPrmsForZSwich
|
|||
|
jc gi_err
|
|||
|
NoZSwtch:
|
|||
|
ifdef NEC_98
|
|||
|
; If 3.5"MO, allcate memory after updating default BPB
|
|||
|
cmp DeviceParameters.DP_DeviceType, DEV_HARDDISK ; Hard disk?
|
|||
|
.errnz EDP_DEVICETYPE NE DP_DEVICETYPE
|
|||
|
jne @F ; No
|
|||
|
|
|||
|
test DeviceParameters.DP_DeviceAttributes, 1 ; Removable?
|
|||
|
.errnz EDP_DEVICEATTRIBUTES NE DP_DEVICEATTRIBUTES
|
|||
|
jz $$IF100 ; Yes
|
|||
|
@@:
|
|||
|
endif
|
|||
|
call Alloc_Dir_Buf ; Allocate root directory buffer
|
|||
|
jc gi_memerr
|
|||
|
|
|||
|
call Alloc_Fat_Buf ; Allocate FAT buffer
|
|||
|
jc gi_memerr
|
|||
|
|
|||
|
call Alloc_Fat_Sec_Buf ; Allocate fat sector buffer
|
|||
|
jc gi_memerr
|
|||
|
|
|||
|
call Alloc_DirBuf2 ; Allocate 1-sector buffer DirBuf (general-
|
|||
|
; purpose use)
|
|||
|
jc gi_memerr
|
|||
|
|
|||
|
call Alloc_Cluster_Buf ; get room for retry buffer
|
|||
|
|
|||
|
IFDEF OPKBLD
|
|||
|
call Do_Switch_S ; Load system files if needed
|
|||
|
ELSE
|
|||
|
test SwitchMap,SWITCH_S
|
|||
|
jz NoS1 ; Carry clear if jump
|
|||
|
Message msgNoSysSwitch
|
|||
|
jmp short gi_err
|
|||
|
|
|||
|
NoS1:
|
|||
|
ENDIF ;OPKBLD
|
|||
|
; carry flag determined by Do_Switch_S
|
|||
|
ifdef NEC_98
|
|||
|
$$IF100:
|
|||
|
endif
|
|||
|
ret
|
|||
|
|
|||
|
gi_memerr:
|
|||
|
Message msgInsufficientMemory
|
|||
|
gi_err:
|
|||
|
stc
|
|||
|
ret
|
|||
|
|
|||
|
Global_Init endp
|
|||
|
|
|||
|
; =========================================================================
|
|||
|
;
|
|||
|
; ModifyDevPrmsForZSwich:
|
|||
|
; Modify the device parameters for a different sec/clus value
|
|||
|
;
|
|||
|
; Input:
|
|||
|
; DeviceParameters set
|
|||
|
; AX != 0 for noisy version (do the message thing)
|
|||
|
; Output:
|
|||
|
; carry set if problem.
|
|||
|
;
|
|||
|
; =========================================================================
|
|||
|
|
|||
|
ModifyDevPrmsForZSwich proc near
|
|||
|
|
|||
|
push ax ; Save noisy switch on stack
|
|||
|
.386
|
|||
|
movzx cx,SecPerClus
|
|||
|
.8086
|
|||
|
or cx,cx
|
|||
|
jz MFZDoneJ ; Carry clear if jump
|
|||
|
cmp CMCDDFlag, Yes
|
|||
|
jne DrvOk1
|
|||
|
DispErrMsg:
|
|||
|
mov dx,offset data:msgCantZThisDrive
|
|||
|
DispErrMsgSet:
|
|||
|
pop ax
|
|||
|
push ax
|
|||
|
or ax,ax
|
|||
|
jz MFZDoneErrJ
|
|||
|
call Display_Interface
|
|||
|
MFZDoneErrJ:
|
|||
|
stc
|
|||
|
MFZDoneJ:
|
|||
|
jmp MFZDone
|
|||
|
|
|||
|
DrvOk1:
|
|||
|
;;
|
|||
|
;; This used to ignore things if you weren't actually making a change.
|
|||
|
;; This is now out as it is important to poke Mr. User if he is making
|
|||
|
;; a huge FAT drive.
|
|||
|
;;
|
|||
|
;; cmp cl,DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerCluster
|
|||
|
;; je MFZDoneJ ; Carry clear if jump
|
|||
|
;;;;
|
|||
|
mov dx,offset data:msgCantZWithQ
|
|||
|
test SwitchMap,SWITCH_Q ; Check for quick format
|
|||
|
jnz DispErrMsgSet
|
|||
|
mov DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerCluster,cl
|
|||
|
cmp DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerFAT,0
|
|||
|
je IsFAT32
|
|||
|
call Calc_Big16_Fat
|
|||
|
mov dx,0002h
|
|||
|
xor ax,ax ; DX:AX = 128k bytes
|
|||
|
mov cx,DeviceParameters.DP_BPB.oldBPB.BPB_BytesPerSector
|
|||
|
div cx ; AX = Sectors in 128k
|
|||
|
cmp ax,DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerFAT
|
|||
|
jb Bad16
|
|||
|
.386
|
|||
|
ja MFZDone ; Carry clear if jump
|
|||
|
.8086
|
|||
|
call GetTotalClusters
|
|||
|
.386
|
|||
|
cmp TotalClusters,0000FFF0h
|
|||
|
.8086
|
|||
|
cmc ; Change "JB Is-OK" to "JNC Is-OK"
|
|||
|
jnc MFZDone
|
|||
|
Bad16:
|
|||
|
mov dx,offset data:msgCantZFAT16
|
|||
|
jmp short DispErrMsgSet
|
|||
|
|
|||
|
IsFAT32:
|
|||
|
call Calc_Big32_Fat
|
|||
|
call GetTotalClusters
|
|||
|
.386
|
|||
|
cmp TotalClusters,65536-10
|
|||
|
.8086
|
|||
|
mov dx,offset data:msgCantZFAT32
|
|||
|
jb DispErrMsgSet
|
|||
|
.386
|
|||
|
mov eax,TotalClusters
|
|||
|
inc eax ; EAX now count including clus 0 and 1
|
|||
|
;
|
|||
|
; GUI SCANDISK is a 16-bit windows app. GlobalAlloc in 16-bit windows is
|
|||
|
; limited to 16Meg-64k per block. Check if we have made a drive with a huge FAT
|
|||
|
; that may result in slow disk util perf.
|
|||
|
;
|
|||
|
cmp eax, ((16 * 1024 * 1024) - (64 * 1024)) / 4
|
|||
|
je short MFZDone ; carry clear if jump
|
|||
|
cmc ; Turn carry around so clear if below limit
|
|||
|
jnc short MFZDone
|
|||
|
.8086
|
|||
|
pop ax
|
|||
|
push ax
|
|||
|
or ax,ax ; Noisy?
|
|||
|
jz MFZDone ; No, Carry clear if jump
|
|||
|
Message msgZFAT32Huge
|
|||
|
call Yes? ; Carry clear if YES, set if NO
|
|||
|
pushf
|
|||
|
Message msgCrlf
|
|||
|
popf
|
|||
|
MFZDone:
|
|||
|
pop ax
|
|||
|
ret
|
|||
|
|
|||
|
ModifyDevPrmsForZSwich endp
|
|||
|
|
|||
|
; =========================================================================
|
|||
|
;
|
|||
|
; GetDeviceParameters:
|
|||
|
; Get the device parameters
|
|||
|
;
|
|||
|
; Input:
|
|||
|
; DriveToFormat
|
|||
|
; DX - pointer to device parameters
|
|||
|
; =========================================================================
|
|||
|
|
|||
|
GetDeviceParameters proc near
|
|||
|
|
|||
|
mov AX, (IOCTL shl 8) or GENERIC_IOCTL
|
|||
|
mov bl, DriveToFormat
|
|||
|
inc bl
|
|||
|
mov CX, (EXTRAWIO shl 8) or GET_DEVICE_PARAMETERS
|
|||
|
int 21H
|
|||
|
jc TryOldForm
|
|||
|
mov IsExtRAWIODrv,1
|
|||
|
DoRet:
|
|||
|
jc realdoret
|
|||
|
push bx
|
|||
|
mov bx,dx
|
|||
|
cmp [bx.DP_BPB.oldBPB.BPB_TotalSectors],0
|
|||
|
je realdoretP
|
|||
|
;
|
|||
|
; the WORD total sectors field is non-zero, make sure the DWORD
|
|||
|
; total sectors field is 0. Having BigTotalSectors be a DWORD
|
|||
|
; version of TotalSectors in this case is SUPPOSED to be perfectly
|
|||
|
; ok but it turns out that several apps (mostly SETUP apps) get
|
|||
|
; upset about this (on floppies in particular).
|
|||
|
;
|
|||
|
mov [bx.DP_BPB.oldBPB.BPB_BigTotalSectors],0
|
|||
|
mov [bx.DP_BPB.oldBPB.BPB_BigTotalSectors+2],0
|
|||
|
realdoretP:
|
|||
|
pop bx
|
|||
|
clc
|
|||
|
realdoret:
|
|||
|
return
|
|||
|
|
|||
|
TryOldForm:
|
|||
|
mov IsExtRAWIODrv,0
|
|||
|
mov AX, (IOCTL shl 8) or GENERIC_IOCTL
|
|||
|
mov bl, DriveToFormat
|
|||
|
inc bl
|
|||
|
mov CX, (RAWIO shl 8) or GET_DEVICE_PARAMETERS
|
|||
|
int 21H
|
|||
|
jmp short DoRet
|
|||
|
|
|||
|
GetDeviceParameters endp
|
|||
|
|
|||
|
;==========================================================================
|
|||
|
;
|
|||
|
; Copy_Device_Parameters : This procedure saves a copy of the original
|
|||
|
; device parameters in the structure
|
|||
|
; SavedParams.
|
|||
|
;
|
|||
|
;==========================================================================
|
|||
|
|
|||
|
Copy_Device_Parameters proc near
|
|||
|
|
|||
|
lea SI, DeviceParameters
|
|||
|
lea DI, SavedParams
|
|||
|
mov CX, size EA_DeviceParameters
|
|||
|
push DS
|
|||
|
pop ES
|
|||
|
rep movsb
|
|||
|
ret
|
|||
|
|
|||
|
Copy_Device_Parameters endp
|
|||
|
|
|||
|
|
|||
|
;==========================================================================
|
|||
|
;
|
|||
|
; Alloc_Dir_Buf : This procedure allocates a memory block for the root
|
|||
|
; directory buffer, based on the device parameters only.
|
|||
|
;
|
|||
|
; Inputs : DeviceParameters.DP_BPB.oldBPB.BPB_BytesPerSector
|
|||
|
; Outputs : CY CLEAR - DirectorySector pointer to buffer
|
|||
|
; CY SET - failure
|
|||
|
; Modifies : AX, BX, DirectorySector
|
|||
|
;
|
|||
|
;==========================================================================
|
|||
|
|
|||
|
Alloc_Dir_Buf proc near
|
|||
|
; DirectorySector =
|
|||
|
; malloc( Bytes Per Sector )
|
|||
|
ifdef NEC_98 ; fixed #16585
|
|||
|
mov BX, AllocSectorSize
|
|||
|
else
|
|||
|
mov BX, DeviceParameters.DP_BPB.oldBPB.BPB_BytesPerSector
|
|||
|
endif
|
|||
|
.errnz EDP_BPB NE DP_BPB
|
|||
|
add BX, 0fH
|
|||
|
.386
|
|||
|
shr bx, 4 ; Divide by 16 to get #paragraphs
|
|||
|
.8086
|
|||
|
mov AH, Alloc
|
|||
|
int 21h
|
|||
|
jc Exit_Alloc_Dir_Buf
|
|||
|
; Base address of newly allocated
|
|||
|
; block is AX:0000
|
|||
|
mov WORD PTR DirectorySector+2,AX
|
|||
|
xor AX,AX
|
|||
|
mov WORD PTR DirectorySector,AX
|
|||
|
|
|||
|
Exit_Alloc_Dir_Buf:
|
|||
|
ret
|
|||
|
|
|||
|
Alloc_Dir_Buf endp
|
|||
|
|
|||
|
;==========================================================================
|
|||
|
;
|
|||
|
; Alloc_Fat_Buf : This procedure allocates a memory block for the FAT
|
|||
|
; buffer, based on the device parameters only. In order
|
|||
|
; to ensure there is enough buffer space for the Mirror
|
|||
|
; utility, the FatSpace buffer is initially allocated
|
|||
|
; with size:
|
|||
|
; FAT + RootDir + 6 sectors + 1 surplus sector
|
|||
|
; which is all the buffer space required by Mirror.
|
|||
|
;
|
|||
|
; Inputs : DeviceParameters.DP_BPB.oldBPB.BPB_BytesPerSector
|
|||
|
; DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerFat
|
|||
|
; DeviceParameters.DP_BPB.BGBPB_BigSectorsPerFat
|
|||
|
; DeviceParameters.DP_BPB.oldBPB.BPB_RootEntries
|
|||
|
;
|
|||
|
; Outputs : CY CLEAR - FatSpace pointer to buffer
|
|||
|
; CY SET - failure
|
|||
|
;
|
|||
|
; Modifies : AX, BX, DX, FatSpace
|
|||
|
;
|
|||
|
;==========================================================================
|
|||
|
|
|||
|
Alloc_Fat_Buf proc near
|
|||
|
|
|||
|
xor ax,ax
|
|||
|
mov FATNotAllInMem,al ; Assume FAT will fit in mem
|
|||
|
.386
|
|||
|
ifdef NEC_98 ; fixed #16585
|
|||
|
cmp DeviceParameters.DP_DeviceType, DEV_3INCH1440KB
|
|||
|
jne short $$IF101
|
|||
|
cmp DeviceParameters.DP_BPB.oldBPB.BPB_TotalSectors, 0
|
|||
|
je short $$IF101
|
|||
|
movzx EAX, AllocSectorSize
|
|||
|
jmp short $$EN101
|
|||
|
$$IF101:
|
|||
|
movzx EAX, DeviceParameters.DP_BPB.oldBPB.BPB_BytesPerSector
|
|||
|
$$EN101:
|
|||
|
else
|
|||
|
movzx EAX, DeviceParameters.DP_BPB.oldBPB.BPB_BytesPerSector
|
|||
|
endif
|
|||
|
.errnz EDP_BPB NE DP_BPB
|
|||
|
add EAX, 0fH ; round up for next para
|
|||
|
shr eax, 4 ; convert to paras
|
|||
|
movzx ebx,DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerFat
|
|||
|
or bx,bx
|
|||
|
jnz short GotFSz
|
|||
|
mov ebx,dword ptr DeviceParameters.DP_BPB.BGBPB_BigSectorsPerFat
|
|||
|
GotFSz:
|
|||
|
mul ebx
|
|||
|
or edx,edx
|
|||
|
jz short NotHi
|
|||
|
GotBigFat:
|
|||
|
mov FATNotAllInMem,1
|
|||
|
mov eax,128*1024
|
|||
|
movzx ebx,DeviceParameters.DP_BPB.oldBPB.BPB_BytesPerSector
|
|||
|
xor edx,edx
|
|||
|
div ebx
|
|||
|
mov FATSecCntInMem,eax
|
|||
|
mov ax,(128*1024)/16
|
|||
|
jmp short SetRestFatBuf
|
|||
|
|
|||
|
NotHi:
|
|||
|
cmp eax,(128*1024)/16 ; FAT bigger than 128k?
|
|||
|
ja short GotBigFat ; Yes
|
|||
|
SetRestFatBuf:
|
|||
|
.8086
|
|||
|
mov BX,AX ; Save FAT size in paras in BX
|
|||
|
mov Paras_per_fat,BX ; Set paras_per_fat here, to
|
|||
|
; avoid having to calculate it later
|
|||
|
; Now add on root dir + extra sectors
|
|||
|
mov AX,DeviceParameters.DP_BPB.oldBPB.BPB_RootEntries
|
|||
|
;; shl ax,5 ; * 32 bytes per dor entry
|
|||
|
;; shr ax,4 ; / 16 bytes per para
|
|||
|
;; Combine above two shifts....
|
|||
|
shl AX,1 ; AX = para size of root dir
|
|||
|
|
|||
|
add BX,AX ; BX = FAT + root dir
|
|||
|
|
|||
|
ifdef NEC_98 ; fixed #16585
|
|||
|
cmp DeviceParameters.DP_DeviceType, DEV_3INCH1440KB
|
|||
|
jne $$IF102
|
|||
|
cmp DeviceParameters.DP_BPB.oldBPB.BPB_TotalSectors, 0
|
|||
|
je $$IF102
|
|||
|
mov AX, AllocSectorSize
|
|||
|
jmp short $$EN102
|
|||
|
$$IF102:
|
|||
|
mov AX, DeviceParameters.DP_BPB.oldBPB.BPB_BytesPerSector
|
|||
|
$$EN102:
|
|||
|
else
|
|||
|
mov AX, DeviceParameters.DP_BPB.oldBPB.BPB_BytesPerSector
|
|||
|
endif
|
|||
|
add AX, 0fH ; round up for next para
|
|||
|
.386
|
|||
|
shr AX, 4 ; convert to paras
|
|||
|
.8086
|
|||
|
mov CX,SECTORS_FOR_MIRROR ; CX = # additional sectors needed by Mirror
|
|||
|
mul CX ; AX = total extra sector size in paras
|
|||
|
|
|||
|
add BX,AX ; BX = FAT + root dir + extra sectors
|
|||
|
; in paras
|
|||
|
mov AH,Alloc
|
|||
|
int 21h
|
|||
|
jc Exit_Alloc_Fat_Buf
|
|||
|
|
|||
|
mov WORD PTR FatSpace+2,AX
|
|||
|
xor AX,AX
|
|||
|
mov WORD PTR FatSpace,AX
|
|||
|
|
|||
|
Exit_Alloc_Fat_Buf:
|
|||
|
ret
|
|||
|
|
|||
|
Alloc_Fat_Buf endp
|
|||
|
|
|||
|
;==========================================================================
|
|||
|
;
|
|||
|
; Alloc_Fat_Sec_Buf : This procedure allocates a memory block for the fat
|
|||
|
; sector buffer which is used when copying chains from
|
|||
|
; the old FAT to the new FAT.
|
|||
|
;
|
|||
|
; Inputs : DeviceParameters.DP_BPB.oldBPB.BPB_BytesPerSector
|
|||
|
; Outputs : CY CLEAR - FatSector pointer to buffer
|
|||
|
; CY SET - failure
|
|||
|
; Modifies : AX, BX, FatSector
|
|||
|
;
|
|||
|
;==========================================================================
|
|||
|
|
|||
|
Alloc_Fat_Sec_Buf proc near
|
|||
|
; FatSector =
|
|||
|
; malloc( Bytes Per Sector )
|
|||
|
ifdef NEC_98 ; fixed #16585
|
|||
|
mov BX, AllocSectorSize
|
|||
|
else
|
|||
|
mov BX, DeviceParameters.DP_BPB.oldBPB.BPB_BytesPerSector
|
|||
|
endif
|
|||
|
.errnz EDP_BPB NE DP_BPB
|
|||
|
add BX, 0fH
|
|||
|
.386
|
|||
|
shr BX, 4 ; Divide by 16 to get #paragraphs
|
|||
|
.8086
|
|||
|
mov AH, Alloc
|
|||
|
int 21h
|
|||
|
jc Exit_Alloc_Fat_Sec_Buf
|
|||
|
; Base address of newly allocated
|
|||
|
; block is AX:0000
|
|||
|
mov WORD PTR FatSector+2,AX
|
|||
|
xor AX,AX
|
|||
|
mov WORD PTR FatSector,AX
|
|||
|
|
|||
|
Exit_Alloc_Fat_Sec_Buf:
|
|||
|
ret
|
|||
|
|
|||
|
Alloc_Fat_Sec_Buf endp
|
|||
|
|
|||
|
;==========================================================================
|
|||
|
;
|
|||
|
; Alloc_DirBuf2 : This procedure allocates a memory block for a 1-sector
|
|||
|
; buffer. This buffer is used when reading in the boot
|
|||
|
; sector in Phase1.
|
|||
|
;
|
|||
|
; Inputs : DeviceParameters.DP_BPB.oldBPB.BPB_BytesPerSector
|
|||
|
; Outputs : CY CLEAR - DirBuf pointer to buffer
|
|||
|
; CY SET - failure
|
|||
|
; Modifies : AX, BX, DirBuf
|
|||
|
;
|
|||
|
;==========================================================================
|
|||
|
|
|||
|
Alloc_DirBuf2 proc near
|
|||
|
; DirBuf =
|
|||
|
; malloc( Bytes Per Sector )
|
|||
|
ifdef NEC_98 ; fixed #16585
|
|||
|
mov BX, AllocSectorSize
|
|||
|
else
|
|||
|
mov BX, DeviceParameters.DP_BPB.oldBPB.BPB_BytesPerSector
|
|||
|
endif
|
|||
|
.errnz EDP_BPB NE DP_BPB
|
|||
|
add BX, 0fH
|
|||
|
.386
|
|||
|
shr BX, 4 ; Divide by 16 to get #paragraphs
|
|||
|
.8086
|
|||
|
mov AH, Alloc
|
|||
|
int 21h
|
|||
|
jc Exit_Alloc_DirBuf2
|
|||
|
; Base address of newly allocated
|
|||
|
; block is AX:0000
|
|||
|
mov WORD PTR DirBuf+2,AX
|
|||
|
xor AX,AX
|
|||
|
mov WORD PTR DirBuf,AX
|
|||
|
|
|||
|
Exit_Alloc_DirBuf2:
|
|||
|
ret
|
|||
|
|
|||
|
Alloc_DirBuf2 endp
|
|||
|
|
|||
|
;=========================================================================
|
|||
|
; Alloc_Cluster_Buf : This routine will allocate a buffer
|
|||
|
; based on a cluster's size. If enough
|
|||
|
; space does not exist, a cluster will
|
|||
|
; be redefined to a smaller size for
|
|||
|
; purposes of sector retries.
|
|||
|
; Note: This buffer is used only for bad
|
|||
|
; tracks on hard disks.
|
|||
|
;
|
|||
|
; Inputs : DeviceParameters.DP_BPB.oldBPB.BPB_BytesPerSector
|
|||
|
; DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerCluster
|
|||
|
;
|
|||
|
; Outputs : ClustBound_Flag - True (space available)
|
|||
|
; False(not enough space)
|
|||
|
; ClustBound_Buffer_Seg - Pointer to buffer
|
|||
|
;=========================================================================
|
|||
|
|
|||
|
Procedure Alloc_Cluster_Buf
|
|||
|
|
|||
|
push AX ; Save regs
|
|||
|
push BX
|
|||
|
|
|||
|
mov AX,(Alloc shl 8) ; Allocate memory
|
|||
|
mov BX,0ffffh ; Get available memory
|
|||
|
int 21h
|
|||
|
|
|||
|
ifdef NEC_98 ; fixed #16585
|
|||
|
mov AX, AllocSectorSize
|
|||
|
else
|
|||
|
mov AX, DeviceParameters.DP_BPB.oldBPB.BPB_BytesPerSector
|
|||
|
endif
|
|||
|
.errnz EDP_BPB NE DP_BPB
|
|||
|
add AX, 0fH
|
|||
|
.386
|
|||
|
shr AX, 4
|
|||
|
.8086
|
|||
|
mul DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerCluster
|
|||
|
|
|||
|
cmp BX,AX ; Enough room
|
|||
|
jna $$IF137 ; Yes
|
|||
|
|
|||
|
mov BX,AX ; Allocate needed memory
|
|||
|
mov AX,(Alloc shl 8)
|
|||
|
int 21h
|
|||
|
mov ClustBound_Buffer_Seg,AX ; Save pointer to buffer
|
|||
|
mov ClustBound_Flag,True ; Signal space available
|
|||
|
jmp SHORT $$EN137 ; Not enough room
|
|||
|
|
|||
|
$$IF137:
|
|||
|
mov ClustBound_Flag,False ; Signal not enough space
|
|||
|
|
|||
|
$$EN137:
|
|||
|
pop BX ; Restore regs
|
|||
|
pop AX
|
|||
|
|
|||
|
ret
|
|||
|
|
|||
|
Alloc_Cluster_Buf ENDP
|
|||
|
|
|||
|
IFDEF OPKBLD
|
|||
|
|
|||
|
;=========================================================================
|
|||
|
;
|
|||
|
; DO_SWITCH_S : This procedure will load the system files into
|
|||
|
; memory (if there's space) if the /s switch is
|
|||
|
; specified.
|
|||
|
;
|
|||
|
; CALLS : ReadDos
|
|||
|
; SysPrm
|
|||
|
; CALLED BY : Global_Init
|
|||
|
; STRATEGY : The largest block of memory available is first
|
|||
|
; determined. The program is aborted if this is zero.
|
|||
|
; This block is then allocated, and the system files
|
|||
|
; are read into it. A prompt for the system disk
|
|||
|
; will be given if the system files are not found.
|
|||
|
;
|
|||
|
;=========================================================================
|
|||
|
|
|||
|
Do_Switch_S proc near
|
|||
|
|
|||
|
test SwitchMap,SWITCH_S
|
|||
|
.386
|
|||
|
jz End_Do_Switch_S ; System files not required
|
|||
|
; allocate memory for system files
|
|||
|
.8086
|
|||
|
mov BX,0ffffh ; This call will actually fail
|
|||
|
mov AH,Alloc ; so that BX returns max block avlbl
|
|||
|
int 21h
|
|||
|
|
|||
|
or BX,BX
|
|||
|
jz MemErr ; No memory
|
|||
|
mov [mSize],BX ; Now allocate the largest block
|
|||
|
mov AH,alloc
|
|||
|
int 21h
|
|||
|
jnc Mem_OK
|
|||
|
|
|||
|
MemErr:
|
|||
|
mov AX, seg data ; Check for memory allocation error
|
|||
|
mov DS, AX
|
|||
|
Message msgOutOfMemory ; call PrintString
|
|||
|
stc ; Let the jump to FatalExit be made
|
|||
|
jmp End_Do_Switch_S ; in the main routine, upon returning
|
|||
|
|
|||
|
Mem_OK:
|
|||
|
mov [mStart],AX ; Save the starting paragraph
|
|||
|
|
|||
|
; =========================================================================
|
|||
|
; This call to ReadDos may not be able to read in all of the DOS files if
|
|||
|
; there is insufficient memory available. In that case the files will
|
|||
|
; be read in after the disk is formatted. If the Drive being formatted is
|
|||
|
; also the boot Drive this function will read the files from that
|
|||
|
; Drive if there is enough memory. If there is insufficent memory it will
|
|||
|
; force the files to be read from Drive A: if the Drive being formatted
|
|||
|
; is also the boot Drive
|
|||
|
; M011; Wrong: Try Boot, Then Default, Then sysprm (usually "A").
|
|||
|
; If not enough memory at boot time, we fail.
|
|||
|
; =========================================================================
|
|||
|
|
|||
|
RdFrst:
|
|||
|
mov AH,GET_DEFAULT_Drive ; Find out default Drive
|
|||
|
int 21h
|
|||
|
push AX ; save default Drive
|
|||
|
mov ax, 3305h ; get startup drive
|
|||
|
int 21h
|
|||
|
mov al,dl
|
|||
|
add AL,40h ; Make it ASCII
|
|||
|
pop BX ; restore default Drive
|
|||
|
ifndef NEC_98
|
|||
|
cmp AL,41h ; Q: Booted from Drive A?
|
|||
|
jnz go_get_Bios ; N: Not a special case
|
|||
|
cmp bl,1 ; Q: is B: current Drive
|
|||
|
jnz go_get_Bios ; N: Not a special case
|
|||
|
jmp short check_default ; check default Drive
|
|||
|
endif
|
|||
|
|
|||
|
go_get_Bios: ; Here to check booted
|
|||
|
call Get_Host_Drive ; Translate to DblSpace host
|
|||
|
mov SystemDriveLetter,AL ; (if necessary)
|
|||
|
|
|||
|
call ReadDos
|
|||
|
jnc CheckAllFilesIn
|
|||
|
|
|||
|
check_default: ; Here to check default
|
|||
|
mov AH,GET_DEFAULT_Drive ; Find out default Drive
|
|||
|
int 21h
|
|||
|
add AL,41h ; Make it ASCII, 1 based
|
|||
|
call Get_Host_Drive ; Translate to DblSpace host
|
|||
|
mov SystemDriveLetter,AL
|
|||
|
|
|||
|
TryThisOne:
|
|||
|
call ReadDos ; Read BIOS and DOS
|
|||
|
jnc CheckAllFilesIn ; Files read in OK
|
|||
|
NeedSys:
|
|||
|
call SysPrm ; Prompt for system disk
|
|||
|
jmp TryThisOne ; Try again
|
|||
|
|
|||
|
CheckAllFilesIn:
|
|||
|
; abort program here if all system files
|
|||
|
; have not been read into memory, since
|
|||
|
; program fails when trying to read them
|
|||
|
; in after formatting is complete
|
|||
|
and FileStat,3fh ; zero out 2 msb
|
|||
|
cmp FileStat,22h ; BIOS and COMMAND in memory?
|
|||
|
jne MemErr ; no - abort program
|
|||
|
;
|
|||
|
; Now we have all of the files in memory, SETBLOCK the allocation
|
|||
|
; block down to the amount required.
|
|||
|
;
|
|||
|
.386
|
|||
|
mov bx,[dos.fileStartSegment]
|
|||
|
mov eax,[dos.fileSizeInBytes]
|
|||
|
cmp bx,[bios.fileStartSegment]
|
|||
|
ja short Skip1
|
|||
|
mov bx,[bios.fileStartSegment]
|
|||
|
mov eax,[bios.fileSizeInBytes]
|
|||
|
Skip1:
|
|||
|
cmp bx,[command.fileStartSegment]
|
|||
|
ja short Skip2
|
|||
|
mov bx,[command.fileStartSegment]
|
|||
|
mov eax,[command.fileSizeInBytes]
|
|||
|
Skip2:
|
|||
|
IFDEF DBLSPACE_HOOKS
|
|||
|
cmp bx,[DblSpaceBin.fileStartSegment]
|
|||
|
ja short Skip3
|
|||
|
mov bx,[DblSpaceBin.fileStartSegment]
|
|||
|
mov eax,[DblSpaceBin.fileSizeInBytes]
|
|||
|
Skip3:
|
|||
|
ENDIF
|
|||
|
add eax,15
|
|||
|
shr eax,4 ; AX = # of paras from SEG BX
|
|||
|
.8086
|
|||
|
add bx,ax ; SEG after end of sys files
|
|||
|
sub bx,[mStart] ; SIZE of sys area
|
|||
|
mov es,[mStart]
|
|||
|
mov AH,setblock
|
|||
|
int 21h
|
|||
|
clc ; yes
|
|||
|
End_Do_Switch_S:
|
|||
|
ret
|
|||
|
|
|||
|
Do_Switch_S ENDP
|
|||
|
|
|||
|
|
|||
|
;******************* START OF SPECIFICATIONS ***********************************
|
|||
|
;Routine name: Get_Host_Drive
|
|||
|
;*******************************************************************************
|
|||
|
;
|
|||
|
;Description: Given a drive letter in AL, check to see if it is a dblspace
|
|||
|
; drive, and if so, translate the drive letter to the host
|
|||
|
; drive letter.
|
|||
|
;
|
|||
|
;Called Procedures: None
|
|||
|
;
|
|||
|
;Input: ASCII drive letter in AL
|
|||
|
;
|
|||
|
;Output: drive letter in AL
|
|||
|
;
|
|||
|
;Change History: Created 11/21/92 MD
|
|||
|
; Cut and paste from SYS command 12/07/92 JEM
|
|||
|
;
|
|||
|
;******************* END OF SPECIFICATIONS *************************************
|
|||
|
|
|||
|
public Get_Host_Drive
|
|||
|
|
|||
|
Get_Host_Drive PROC NEAR
|
|||
|
|
|||
|
push ax
|
|||
|
mov ax,4a11h ; DBLSPACE multiplex number
|
|||
|
xor bx,bx ; inquire version number
|
|||
|
int 2fh
|
|||
|
or ax,ax ; error?
|
|||
|
jnz not_dblspace
|
|||
|
cmp bx,'DM' ; stamp returned correctly?
|
|||
|
jnz not_dblspace
|
|||
|
|
|||
|
; DBLSPACE.BIN is loaded. At this time:
|
|||
|
;
|
|||
|
; (dx & 0x7fff) == driver internal version number
|
|||
|
; high bit of DH set of driver has not yet been permanently placed
|
|||
|
; cl == first disk letter reserved for DBLSPACE
|
|||
|
; ch == number of disk letters reserved for DBLSPACE
|
|||
|
|
|||
|
mov ax,4a11h ; DBLSPACE multiplex number
|
|||
|
mov bx,1 ; inquire drive map
|
|||
|
pop dx
|
|||
|
push dx
|
|||
|
sub dl, 'A' ; convert drv letter to 0 based drv number
|
|||
|
int 2fh
|
|||
|
test bl,80h ; COMPRESSED bit true?
|
|||
|
jz not_dblspace
|
|||
|
|
|||
|
; Drive is compressed. At this time:
|
|||
|
;
|
|||
|
; (bl & 0x7f) == host drive's CURRENT drive number
|
|||
|
; bh == CVF extension number
|
|||
|
;
|
|||
|
mov al,bl
|
|||
|
and al,7Fh
|
|||
|
add al, 'A' ; convert drv number to drv letter
|
|||
|
cbw
|
|||
|
pop dx
|
|||
|
push ax
|
|||
|
|
|||
|
not_dblspace:
|
|||
|
pop ax
|
|||
|
ret
|
|||
|
|
|||
|
Get_Host_Drive ENDP
|
|||
|
|
|||
|
ENDIF ;OPKBLD
|
|||
|
|
|||
|
CODE ENDS
|
|||
|
|
|||
|
END
|
|||
|
|