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

3547 lines
90 KiB
NASM

;/*
; * Microsoft Confidential
; * Copyright (C) Microsoft Corporation 1991
; * All Rights Reserved.
; */
;===========================================================================
;
; FILE: DSKFRMT.ASM
;
;===========================================================================
;===========================================================================
;Declaration of include files
;===========================================================================
debug equ 0
.xlist
INCLUDE DOSEQUS.INC
INCLUDE DOSMAC.INC
INCLUDE SYSCALL.INC
INCLUDE ERROR.INC
INCLUDE BPB.INC
INCLUDE FOREQU.INC
INCLUDE FORMACRO.INC
INCLUDE IOCTL.INC
INCLUDE FORSWTCH.INC
.list
;===========================================================================
; Declarations for all publics in other modules used by this module
;===========================================================================
;===========================================================================
; Data segment
;===========================================================================
DATA SEGMENT PUBLIC PARA 'DATA'
;Bytes
EXTRN fBigFat :BYTE
EXTRN fBig32Fat :BYTE
EXTRN ExitStatus :BYTE
EXTRN DblFlg :BYTE
EXTRN DriveToFormat :BYTE
EXTRN msgFormatFailure :BYTE
EXTRN msgDiskUnusable :BYTE
EXTRN msgNotSystemDisk :BYTE
EXTRN msgParametersNotSupported:BYTE
EXTRN msgParametersNotSupportedByDrive:BYTE
EXTRN msgHardDiskWarning :BYTE
EXTRN msgDiskWarning :BYTE
EXTRN msgInsertDisk :BYTE
EXTRN msgCrLf :BYTE
EXTRN msgCurrentTrack :BYTE
EXTRN msgFormatComplete :BYTE
EXTRN msgVerify :BYTE
EXTRN msgSetBadClus :BYTE
EXTRN msgSetBadClusDone :BYTE
EXTRN ContinueMsg :BYTE
EXTRN Extended_Error_Msg :BYTE
EXTRN Clustbound_Flag :BYTE
EXTRN Fatal_Error :BYTE
EXTRN FATNotAllInMem :BYTE
EXTRN msgInsufficientMemory :BYTE
EXTRN msgWriteFat :BYTE
;Words
EXTRN FirstHead :WORD
EXTRN FirstCylinder :WORD
EXTRN Formatted_Tracks_Low :WORD
EXTRN Formatted_Tracks_High :WORD
EXTRN Paras_Per_Fat :WORD
EXTRN SwitchMap :WORD
EXTRN SwitchMap2 :WORD
EXTRN Relative_Sector_Low :WORD
EXTRN Relative_Sector_High :WORD
EXTRN Clustbound_Buffer_Seg :WORD
EXTRN Clustbound_Adj_Factor :WORD
EXTRN Clustbound_Spt_Count :WORD
; Dwords
EXTRN TracksLeft :DWORD
EXTRN TracksPerDisk :DWORD
EXTRN sector_in_buffer :DWORD
EXTRN sector_to_read :DWORD
EXTRN TotalClusters :DWORD
EXTRN StartSector :DWORD
EXTRN FATSecCntInMem :DWORD
;Pointers
EXTRN SysSiz :DWORD
EXTRN BioSiz :DWORD
EXTRN Msg_Allocation_Unit_Val :DWORD
EXTRN FatSpace :DWORD
EXTRN FatSector :DWORD
;Structures
EXTRN DeviceParameters :BYTE
EXTRN IsExtRAWIODrv :BYTE
EXTRN FormatPacket :BYTE
EXTRN Read_Write_Relative :BYTE
RWPacket a_TrackReadWritePacket <>
ifdef NEC_98
RW_TRF_Area db 2048 dup(0)
else
RW_TRF_Area db 512 dup(0)
endif
fLastChance db FALSE ; Flags reinvocation from
; LastChanceToSaveIt. Used by DskPrm
FormatError db 0
Format_End db FALSE
Track_Action db ? ; Actual operation performed on disk
odd_entry db 1 ; flag for 12-bit FAT entry alignment
entry_offset dw ? ; Offset of entry from start of sector
PercentComplete dw 0FFFFh ; Init non-zero so msg will display
; first time
Fat_Init_Value dw 0 ; initializes the Fat
SysTrks dd ?
Sectors dw ?
CurrentHead dw 0
CurrentCylinder dw 0
Tracks_To_Format dw ?
Track_Count dw ?
CurrentCluster dd ? ; holds the cluster currently being checked
; in QuickFormat
BadClusValue dd ? ; holds FAT entry value for bad cluster
BadClusBitMap dd 0
CurrFATInMemStartSec dd 0
CurrFATInMemSecCnt dd 0
CurrFATInMemStartClus dd 0
FATInMemClusCnt dd 0
DATA ENDS
;===========================================================================
; Executable code segment
;===========================================================================
CODE SEGMENT PUBLIC PARA 'CODE'
ASSUME CS:CODE, DS:DATA, ES:DATA
;===========================================================================
;EXTRNs needed in code segment
;===========================================================================
;Labels
EXTRN FatalExit :NEAR
EXTRN CrLf :NEAR
;Functions
EXTRN DetermineExistingFormatNoMsg :NEAR
EXTRN SetStartSector :NEAR
EXTRN SetfBigFat :NEAR
EXTRN Phase2Initialization :NEAR
EXTRN GetBioSize :NEAR
EXTRN GetDosSize :NEAR
EXTRN GetCmdSize :NEAR
EXTRN LastChanceToSaveIt :NEAR
EXTRN AccessDisk :NEAR
EXTRN Yes? :NEAR
EXTRN ExitProgram :NEAR
EXTRN User_String :NEAR
EXTRN Read_Disk :NEAR
EXTRN Seg_Adj :NEAR
EXTRN IsDblSpaceDisk :NEAR
EXTRN Write_Disk :NEAR
EXTRN ShowFormatSize :NEAR
;Constants
EXTRN EXIT_FATAL :ABS
EXTRN EXIT_NO :ABS
EXTRN EXIT_DRV_NOT_READY :ABS
EXTRN EXIT_WRIT_PROTECT :ABS
EXTRN EXIT_FATAL :ABS
EXTRN EXIT_NO :ABS
;===========================================================================
; Declarations for all publics in this module
;===========================================================================
PUBLIC Disk_Format_Proc
PUBLIC FrmtProb
PUBLIC fLastChance
PUBLIC CurrentHead
PUBLIC CurrentCylinder
PUBLIC Multiply_32_Bits
PUBLIC SetDeviceParameters
PUBLIC PercentComplete
PUBLIC Prompt_User_For_Disk
PUBLIC calc_sector_and_offset
PUBLIC ReadFatSector
PUBLIC GetSetFatEntry
PUBLIC GetFatSectorEntry
PUBLIC DetermineTrackAction
PUBLIC QuickFormat
PUBLIC FlushFATBuf
PUBLIC Fat_Init
PUBLIC Get_Bad_Sector_Hard
PUBLIC Multiply_32_Bits
PUBLIC IsThisClusterBad
PUBLIC WrtEOFMrkInRootClus
;========================================================================
;
; DISK_FORMAT_PROC : This procedure is used to call
; the real DiskFormat procedure.
; Returns : NC --> Format successful
; CY --> Format unsuccessful, go to next disk
;
;========================================================================
Disk_Format_Proc proc near
call DetermineTrackAction ; Check for safe format, and validity
; of quick format switch
call init_fat_with_header ; setup blank FAT image
test SwitchMap,SWITCH_Q ; Check for quick format
jz RegularFormat
test SwitchMap,SWITCH_Z ; Check for sec/clus override
jnz RegularFormat
call ShowFormatSize ; size being formatted message
call QuickFormat
jc FarExit ; procedure exit is too far for direct jump
Message msgFormatComplete
FarExit:
jmp Exit_Disk_Format_Proc
RegularFormat:
test SwitchMap,SWITCH_Z
jnz no_load_old_fat ; don't need to load the old FAT
test SwitchMap2,Switch2_C
jnz no_load_old_fat ; don't need to load the old FAT
call SetUpBadClusTransfer
jnc no_load_old_fat
Message msgInsufficientMemory
stc
jmp short FarExit
no_load_old_fat:
call ShowFormatSize ; size being formatted message
call DiskFormat ; Format the disk
jnc GetTrk ; check for problems
FrmtProb:
test SwitchMap,Switch_Select ; SELECT option?
jnz CheckForMore ; No - display message
Message msgFormatFailure
mov ExitStatus, EXIT_FATAL
CheckForMore:
stc ; Signal error which will be handled
; in main routine (by prompting for
; next disk)
jmp Exit_Disk_Format_Proc
;Mark any bad Sectors in the Fats
;And keep track of how many bytes there are in bad Sectors
GetTrk:
call BadSector ; Do bad track fix-up
jc FrmtProb ; Had an error in Fmt - can't recover
CMP AX,0 ; Are we finished?
jnz TrkFnd ; No - check error conditions
jmp Exit_Disk_Format_Proc ; Yes
;--------------------------------------------------------------------------
; BUG NOTE: The two sections in "if ibmcopyright..." here and below are
; to correct a bug. If one of the Sectors just above the 32M boundary
; were bad, it thought they were in the system area, and hence that the
; disk was unusable.
;
; PYS: IBMCOPYRIGHT removed
;
;--------------------------------------------------------------------------
TrkFnd:
.386
mov EBX,dword ptr Relative_Sector_Low ; get the sector
cmp EBX,StartSector ; Any sec in the sys area bad?
.8086
jae ClrTest ; MZ 2.26 unsigned compare
Message msgDiskUnusable
jmp FrmtProb ; Bad disk -- try again
ClrTest:
mov Sectors,AX ; Save # sectors on the track
test SwitchMap,SWITCH_S ; If sys requested calc size
jz Bad100
cmp BYTE PTR DblFlg,0 ; Is sys space aready calced?
jnz cmpTrks ; Yes - all ready for the compare
inc BYTE PTR DblFlg ; No - set the flag
call GetBioSize ; Get the size of the Bios
mov DX,WORD PTR SysSiz+2
mov AX,WORD PTR SysSiz
mov WORD PTR BioSiz+2,DX
mov WORD PTR BioSiz,AX
call GETDOSSIZE
call GETCMDSIZE
mov DX,WORD PTR BioSiz+2
mov AX,WORD PTR BioSiz
div DeviceParameters.DP_BPB.oldBPB.BPB_BytesPerSector
.errnz EDP_BPB NE DP_BPB
.386
movzx eax,ax
add EAX,StartSector
mov SysTrks,EAX ; Space Fat,Dir,and system
; files require
cmpTrks:
mov EBX,dword ptr Relative_Sector_Low ;get the low word of the sector
cmp EBX,SysTrks
.8086
JA Bad100 ; MZ 2.26 unsigned compare
mov ExitStatus, EXIT_FATAL
Message msgNotSystemDisk
AND SwitchMap,NOT SWITCH_S ; Turn off system transfer switch
mov WORD PTR SysSiz+2,0 ; No system to transfer
mov WORD PTR SysSiz,0 ; No system to transfer
Bad100:
cmp DeviceParameters.DP_DeviceType, DEV_HARDDISK ;hard disk?
.errnz EDP_DEVICETYPE NE DP_DEVICETYPE
jne $$IF5 ; no
call Get_Bad_Sector_Hard ; see if a sector is bad
jmp SHORT $$EN5
$$IF5:
call Get_Bad_Sector_Floppy ; mark entire track bad
$$EN5:
jmp GetTrk
Exit_Disk_Format_Proc:
ret
Disk_Format_Proc ENDP
;==========================================================================
;
; DetermineTrackAction : This procedure sets the value of
; Track_Action based on the setting of
; SWITCH_U, the unconditional format switch.
; Track_Action is the function that is
; actually performed on each track of the
; disk to be formatted.
;
; Inputs : SwitchMap
; Output : SWITCH_U set - Track_Action = Format and Verify
; SWITCH_U clear - Track_Action = Verify only
;
;==========================================================================
DetermineTrackAction proc near
test SwitchMap,SWITCH_U
jz Verify_Only
Format_And_Verify:
mov Track_Action,FORMAT_TRACK ; regular format
ret
Verify_Only:
mov Track_Action,VERIFY_TRACK ; safe format
ret
DetermineTrackAction endp
; =========================================================================
; DiskFormat:
; Format the tracks on the disk
; Since we do our SetDeviceParameters here, we also need to
; detect the legality of /N /T if present and abort with errors
; if not.
; This routine stops as soon as it encounters a bad track
; Then BadSector is called to report the bad track, and it continues
; the format
;
; Algorithm:
; current track = first
; while not done
; if format track fails
; DiskFormatErrors = true
; return
; next track
; =========================================================================
DiskFormat proc near
mov DeviceParameters.DP_SpecialFunctions, (INSTALL_FAKE_BPB or TRACKLAYOUT_IS_GOOD)
.errnz EDP_SPECIALFUNCTIONS NE DP_SPECIALFUNCTIONS
lea DX, DeviceParameters
call SetDeviceParameters
test SwitchMap,switch_8 ; DCL 5/12/86 avoid Naples AH=18h
jnz stdBpB ; lackof support for 8 Secs/track
; DCL 5/12/86 - Always do the Status_FOR_FORMAT test, as we
; don't know if the machine has this support. For 3.2 /N:
; & /T: were not documented & therefore not fully supported
; thru the ROM of Aquarius & Naples & Royal Palm
; test SwitchMap, SWITCH_N or SWITCH_T ; IF (/N or /T)
; jz StdBPB
; THEN check if
; supported
; Check to see if device driver can handle specified parameters
mov FormatPacket.FP_SpecialFunctions, Status_FOR_FORMAT
mov AX, (IOCTL shl 8) or GENERIC_IOCTL
mov BL, DriveToFormat
inc BL
mov CX, (RAWIO shl 8) or FORMAT_TRACK
cmp IsEXTRAWIODrv,0
je DoFrmt
mov CX, (EXTRAWIO shl 8) or FORMAT_TRACK
DoFrmt:
lea DX, FormatPacket
int 21h
;;%out FORQUICK ENABLED!!!!!
;; clc ;**ARR
; switch ( FormatStatusCall)
; cmp FormatPacket.FP_SpecialFunctions, \
; Format_No_ROM_Support
; jb NTSupported ; 0 returned from IBMBIO
; ja IllegalComb ; 2 returned - ROM Support
; Illegal Combination!
cmp FormatPacket.FP_SpecialFunctions,0 ; 0 --> Can support
je NTSupported
cmp FormatPacket.FP_SpecialFunctions,2 ; 2 --> Cannot support
jne $$IF28
Message msgParametersNotSupportedByDrive
mov Fatal_Error,Yes ; Indicate quittin'type err!
jmp SHORT $$EN28
$$IF28:
cmp FormatPacket.FP_SpecialFunctions,3 ; 3 --> No disk in drive
jne $$IF30
mov AX,Error_Not_Ready ; flag not ready
call CheckError ; set error level
jmp FrmtProb ; exit program
jmp SHORT $$EN30 ; DCL No ROM support is okay
$$IF30:
; except for /N: & /T:
test SwitchMap, SWITCH_N or SWITCH_T ; DCL 5/12/86
jz $$IF32
Message msgParametersNotSupported
mov Fatal_Error,Yes ; Indicate quittin 'type err!
$$IF32:
$$EN30:
$$EN28:
cmp Fatal_Error,Yes
jne StdBPB
jmp FatalExit
; We have the support to carry
; out the FORMAT
NTSupported:
StdBPB:
mov FormatPacket.FP_SpecialFunctions, 0
mov AX, FirstHead
mov FormatPacket.FP_Head, AX
mov AX, FirstCylinder
mov FormatPacket.FP_Cylinder, AX
;M018 - begin
mov AX, word ptr TracksPerDisk
mov word ptr TracksLeft, AX
mov AX, word ptr TracksPerDisk+2
mov word ptr TracksLeft+2, AX
;M018 - end
mov Format_End,False ; flag not at end of format
call Calc_MAX_Tracks_To_Format ; Max track count for
; FormatTrack call
FormatLoop:
call Format_Loop ; Format until CY occurs
cmp Format_End,True ; End of Format?
jne $$IF36
mov FormatError,0 ; signal good format
clc ; clear CY
jmp SHORT $$EN36 ; bad format
$$IF36:
call CheckError ; determine type of error
jc $$IF38
call LastChanceToSaveIt ; acceptable error?
jnc $$IF39 ; yes
mov FormatError,1 ; signal error type
clc ; clear CY
jmp SHORT $$EN39 ; not acceptable error
$$IF39:
call SetStartSector ; start from scratch
call SetfBigFat
push AX
call Phase2Initialization
clc
pop AX
jmp DiskFormat ; try again
$$EN39:
$$IF38:
$$EN36:
return
FormatDone:
mov FormatError,0
clc
return
DiskFormat endp
;=========================================================================
; Fat_INIT: This routine initializes the Fat based on the
; number of paragraphs.
;
; input - FatSpace
; FatSpace+2
; paras_per_Fat
; Fat_init_value
; output - Fat space is initialized
;
; Assumes: nothing
;
;=========================================================================
Fat_Init PROC NEAR
push ds
push ES
push DI
push AX
push BX
push CX
mov ax,DATA
mov ds,ax
les di,FatSpace
mov BX,Paras_Per_Fat ; Get number of paras
mov AX,Fat_init_value
push DX
mov DX,ES ; Grab ES into DX
$$DO87:
cmp BX,0 ; do while BX not = 0
je $$EN87 ; exit if 0
mov CX,08h ; Word store of paragraph
rep stosw ; Move the data to Fat
xor DI,DI ; Offset always init to 0
inc DX ; Next paragraph
mov ES,DX ; Put next para in ES
dec BX ; Loop iteration counter
jmp SHORT $$DO87
$$EN87:
pop DX
pop CX
pop BX
pop AX
pop DI
pop ES
pop ds
ret
Fat_Init ENDP
;=========================================================================
; SetDeviceParameters:
; Set the device parameters
;
; Input:
; Drive
; DS:DX - pointer to device parameters
;=========================================================================
SetDeviceParameters proc near
mov CX, (EXTRAWIO shl 8) or SET_DEVICE_PARAMETERS
cmp IsEXTRAWIODrv,0
jne DoIoctl
mov CX, (RAWIO shl 8) or SET_DEVICE_PARAMETERS
DoIoctl:
mov AX, (IOCTL shl 8) or GENERIC_IOCTL
mov bl, DriveToFormat
inc bl
int 21H
return
SetDeviceParameters endp
;=========================================================================
; Prompt_User_For_Disk : This routine prompts the user for the
; disk to be formatted. An appropriate
; message is chosen based on the type
; of switch entered. If the /SELECT
; switch is entered, the disk prompt is
; issued through the int 2fh services
; provided by SELECT.
;
; Inputs : SwitchMap - Switches chosen for format
;
; Outputs : Message printed as appropriate.
;=========================================================================
Procedure Prompt_User_For_Disk
push AX
test SwitchMap, (SWITCH_Backup or SWITCH_Select or SWITCH_AUTOTEST)
jnz $$IF186
ifdef NEC_98
test SwitchMap2,SWITCH2_P
jnz $$IF188
endif
call DskPrm ; prompt user for disk
$$IF186:
test SwitchMap, (Switch_Select) ; /SELECT requested?
jz $$IF188
mov AL, DriveToFormat ; get drv to access for format
call AccessDisk ; access the disk
mov AX,Select_Disk_Message ; display disk prompt
int 2fh ; through int 2fh services
$$IF188:
pop AX
ret
Prompt_User_For_Disk ENDP
;==========================================================================
; DiskPrompt:
;
; This routine prompts for the insertion of the correct diskette
; into the Target Drive, UNLESS we are being re-entrantly invoked
; from LastChanceToSaveIt. If the target is a Hardisk we issue a
; warning message.
;
; INPUTS:
; DeviceParameters.DP_DeviceType
; fLastChance
;
; OUTPUTS:
; Prompt string
; fLastChance := FALSE
;
; Registers affected:
; Flags
;
;==========================================================================
DskPrm PROC NEAR
cmp fLastChance,TRUE
je PrmptRet
cmp DeviceParameters.DP_DeviceType, DEV_HARDDISK
.errnz EDP_DEVICETYPE NE DP_DEVICETYPE
jne GoPrnIt
; DblSpaced floppies return DEV_HARDDISK to the GetDeviceParameters
; call. If this is a DblSpaced drive, go make some additional checks
; before putting out the hard disk warning.
mov al, DriveToFormat
call IsDblSpaceDisk ; sets Z flag if NOT dblspaced disk
jnz MaybeDblSpace
RealHardDisk:
test DeviceParameters.DP_DeviceAttributes, 1
.errnz EDP_DEVICEATTRIBUTES NE DP_DEVICEATTRIBUTES
jnz non_rem
Message msgDiskWarning
jmp short ask_yes
non_rem:
ifndef OPKBLD
Message msgHardDiskWarning
endif ;OPKBLD
ask_yes:
ifndef OPKBLD
call Yes?
pushf
Message msgCrlf
popf
else
clc
endif ;OPKBLD
jnc OkToFormatHardDisk
mov ExitStatus, EXIT_NO
jmp ExitProgram
OkToFormatHardDisk:
call CrLf
call CrLf
return
; Got a DblSpaced drive--see if it's removable or not.
MaybeDblSpace:
mov AX, (IOCTL shl 8) or IOCTL_CHANGEABLE?
mov bl, DriveToFormat
inc bl
int 21H
jc RealHardDisk ; should not happen, but if it does...
or ax, ax ; AX=0 if not removable--quietly exit
jnz PrmptRet ; in this case, later checks will
; stop the format process
GoPrnIt:
mov AL, DriveToFormat
call AccessDisk
Message msgInsertDisk
Message ContinueMsg
call USER_STRING ; Wait for any key
call CrLf
call CrLf
PrmptRet:
mov fLastChance, FALSE
ret
DskPrm ENDP
;=========================================================================
; CheckError:
; Input:
; AX - extended error code
; Ouput:
; carry set if error is Fatal
; Message printed if Not Ready or Write Protect
;=========================================================================
CheckError proc near
cmp AX, error_write_protect
je WriteProtectError
cmp AX, error_not_ready
je NotReadyError
cmp CurrentCylinder, 0
jne CheckRealErrors
cmp CurrentHead, 0
je BadTrackZero
CheckRealErrors:
cmp AX, error_CRC
je JustABadTrack
cmp AX, error_sector_not_found
je JustABadTrack
cmp AX, error_write_fault
je JustABadTrack
cmp AX, error_read_fault
je JustABadTrack
cmp AX, error_gen_failure
je JustABadTrack
stc
ret
JustABadTrack:
clc
ret
WriteProtectError:
test SwitchMap,Switch_SELECT ; SELECT option?
jnz $$IF56 ; no - display messages
Message msgCrLf
Message msgCrLf
Extended_Message
jmp SHORT $$EN56 ; yes - set error level
$$IF56:
mov ExitStatus,EXIT_WRIT_PROTECT ; signal write protect error
$$EN56:
stc ; signal Fatal error
ret ; return to caller
NotReadyError:
test SwitchMap,Switch_SELECT ; SELECT option?
jnz $$IF59 ; no - display messages
Message msgCrLf
Message msgCrLf
Extended_Message
jmp SHORT $$EN59 ; yes - set error level
$$IF59:
mov ExitStatus,EXIT_DRV_NOT_READY ;signal Drive not ready
$$EN59:
stc
ret
BadTrackZero:
Message msgDiskUnusable
stc
ret
CheckError endp
;=========================================================================
;
; Calc_MAX_Tracks_To_Format : This routine determines the maximum
; number of tracks to format at 1 time.
;
; Inputs : DeviceParameters - SectorsPerTrack
; BytesPerSector
;
; Outputs : Track_Count - MAX. # of tracks to format in 1 call
; to FormatTrack
;=========================================================================
Procedure Calc_Max_Tracks_To_Format
push AX ; Save regs
push BX
push DX
mov AX,DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerTrack
mov BX,DeviceParameters.DP_BPB.oldBPB.BPB_BytesPerSector
.errnz EDP_BPB NE DP_BPB
xor DX,DX
mul BX ; Get total byte count
mov BX,AX ; Put count in BX
mov ax, 1
or dx, dx
jnz cmttf_onetrack
mov AX,MAX_Format_Size ; Max bytes to format
div BX ; Get track count
cmttf_onetrack:
mov Track_Count,AX
pop DX
pop BX
pop AX
ret
Calc_Max_Tracks_To_Format ENDP
;=========================================================================
; Format_Loop : This routine provides the main template
; for the formatting of a disk. A disk
; will be formatted as long as there are
; tracks remaining to be formatted.
; This routine can be exited on a carry
; condition; i.e., bad track, last track, etc.
;
; Inputs : none
;
; Outputs : CY - Set on exit from this routine
; AX - Possible error condition code
;
;=========================================================================
Procedure Format_Loop
clc ; Initialize to NC
$$DO173: ; While NC
jc $$EN173 ; Exit on CY
call Calc_Current_Head_Cyl ; Head and cylinder calc.
call Determine_Format_Type ; Floppy/hard media?
call Determine_Track_Count ; How many tracks?
call FormatTrack ; Format track(s)
jnc $$IF175 ; Formattrack failed
pushf ; Save flags
cmp DeviceParameters.DP_DeviceType,Dev_HardDisk ; Harddisk?
.errnz EDP_DEVICETYPE NE DP_DEVICETYPE
jne $$IF176
popf ; Restore flags
call Format_Track_Retry ; Find failing track
jmp SHORT $$EN176
$$IF176:
popf ; Restore flags
$$EN176:
$$IF175:
jnc $$IF180 ; Format error?
pushf ; Yes - save flags
push AX ; Save return code
call CheckRealErrors ; Check error type
jc $$IF181 ; If non-Fatal
call DisplayCurrentTrack ; Display % formatted
$$IF181:
pop AX ; Restore regs
popf
$$IF180:
jc $$EN173 ; Exit on CY
call DisplayCurrentTrack ; Tell how much formatted
call Adj_Track_Count ; Decrease track counter
call NextTrack ; Adjust head and cylinder
jmp SHORT $$DO173
$$EN173:
ret
Format_Loop ENDP
;=========================================================================
; Calc_Current_Head_Cyl : Obtain the current head and cylinder of the
; track being formatted.
;
; Inputs: FP_Cylinder - Cylinder of track being formatted
; FP_Head - Head of track being formatted
;=========================================================================
Procedure Calc_Current_Head_Cyl
push CX ; save CX
mov CX,FormatPacket.FP_Cylinder ; get current cylinder
mov CurrentCylinder,CX ; put into variable
mov CX,FormatPacket.FP_Head ; get current head
mov CurrentHead,CX ; put into variable
pop CX ; restore CX
ret
Calc_Current_Head_Cyl endp
; =========================================================================
; Determine_Format_Type : This routine determines the type of format
; that is to occur based on the media type.
;
; Inputs : Dev_HardDisk - Media type (harddisk)
; Multi_Track_Format - EQU 02h
; Single_Track_Format - EQU 00h
;
; Outputs : FP_SpecialFunctions - Set appropriately for single
; or multi track format
; =========================================================================
Procedure Determine_Format_Type
cmp DeviceParameters.DP_DeviceType,Dev_HardDisk ;harddisk?
.errnz EDP_DEVICETYPE NE DP_DEVICETYPE
jne $$IF158
; Set for multi track format
mov FormatPacket.FP_SpecialFunctions,Multi_Track_Format
jmp SHORT $$EN158
$$IF158: ; Set for single track format
mov FormatPacket.FP_SpecialFunctions,Single_Track_Format
$$EN158:
ret
Determine_Format_Type ENDP
;=========================================================================
;
; Determine_Track_Count : This routine determines the number of
; tracks to be formatted, based on whether
; or not we have a hard disk. If we have
; a hard disk we can use multi-track
; format/verify, otherwise we use the
; single track format/verify.
;
; Inputs : Device_Type - Media type
;
; Outputs : Tracks_To_Format - MAX. number of tracks
; to be formatted in one
; call
;=========================================================================
Procedure Determine_Track_Count
; Harddisk?
cmp DeviceParameters.DP_DeviceType,Dev_HardDisk
.errnz EDP_DEVICETYPE NE DP_DEVICETYPE
jne $$IF163
call Calc_Track_Count ; Calc Tracks_To_Format
jmp SHORT $$EN163 ; Removable media
$$IF163:
mov Tracks_To_Format,0001h ; Default to 1 track
$$EN163:
ret
Determine_Track_Count ENDP
;=========================================================================
;
; Calc_Track_Count : This routine determines if we have enough tracks
; remaining to use the Max. number of tracks
; in the FormatTrack call. If the tracks remaining
; to be formatted is less that the mAX. number of
; allowable tracks for the call, the mAX. number
; of allowable tracks is set to the remaining track
; count.
;
; Inputs : Track_Count - MAX. number of allowable tracks to be
; formatted in 1 FormatTrack call.
; TracksLeft - Track count of remaining tracks to be
; formatted.
;
; Outputs : Tracks_To_Format - Count of the tracks to be formatted
; in the next FormatTrack call.
;
;=========================================================================
Procedure Calc_Track_Count
push AX ; Save regs
mov AX,Track_Count ; Max bytes to format
cmp word ptr TracksLeft+2,0 ; M018; More than 64K of tracks?
jnz $$IF166 ; M018; Then surely use Track_Count
cmp AX,word ptr TracksLeft ; M018; Too many tracks?
JNA $$IF166
mov AX,word ptr TracksLeft ; M018; Format remaining tracks
$$IF166:
mov Tracks_To_Format,AX ; Save track count
pop AX
ret
Calc_Track_Count ENDP
;=========================================================================
; FormatTrack : This routine performs multi track or single
; track formatting based on the state of the
; SpecialFunctions byte.
;
; Inputs : Tracks_To_Format - # of tracks to format in 1 call
; FormatPacket - Parms for IOCTL call
;
; Outputs : NC - formatted track(s)
; CY - error in format
; AX - extended error on CY
;
;=========================================================================
Procedure FormatTrack
mov AX,(IOCTL shl 8) or Generic_IOCTL
mov BL,DriveToFormat ; Get Drive number
inc BL ; Make it 1 based
mov CX,(RawIO shl 8)
cmp IsEXTRAWIODrv,0
je DoFrmt2
mov CX,(EXTRawIO shl 8)
DoFrmt2:
or CL,Track_Action ; Track_Action is either
; Format and Verify, or
; Verify only
mov DX,Tracks_To_Format ; Get track count
mov FormatPacket.FP_TrackCount,DX ; Put count in parms list
lea DX,FormatPacket ; Ptr to parms
int 21h
;;%out FORQUICK ENABLED!!!!!
;; clc ;;** ARR
jnc FormatTrackExit ; Error?
mov AH,59h ; Get extended error
xor BX,BX ; Clear BX
int 21h
cmp AX,67 ; Induced error from ENHDISK?
jne notInducedError
clc
ret
notInducedError:
stc ; Flag an error
FormatTrackExit:
ret
FormatTrack ENDP
;=========================================================================
;
; Format_Track_Retry : This routine performs the retry logic for
; the format multi-track. It will retry each track
; until the failing track is encountered through
; a CY condition.
;
; Inputs : none
;
; Outputs : CY - indicates either a failing track or end of format
;
;
;=========================================================================
Procedure Format_Track_Retry
clc ; Clear existing CY
mov Tracks_To_Format,1 ; Only format 1 track
$$DO168: ; While we have good tracks
jc $$EN168 ; Exit on bad track
call FormatTrack ; Format the track
jc $$IF170 ; Error?
call DisplayCurrentTrack ; Adjust percent counter
call Adj_Track_Count
call NextTrack ; Calc next track
$$IF170:
jmp SHORT $$DO168
$$EN168:
ret
Format_Track_Retry ENDP
;=========================================================================
;
; DisplayCurrentTrack : This procedure prints the percentage of disk
; formatted so far.
; If /select is present, format can be exited
; by returning AX!=0 from the int 2fh call, which
; is handled by install. This is to permit the
; user to stop the format in progress.
;
;===========================================================================
DisplayCurrentTrack proc near
push DX
push CX
push AX
push DI ; M018
push SI ; M018
mov AX,Tracks_To_Format ;get track count
add Formatted_Tracks_Low,AX ;Indicate formatted a track
adc Formatted_Tracks_High,0
mov AX,Formatted_Tracks_Low
mov BX,Formatted_Tracks_High
mov CX,100 ;Make integer calc for div
call Multiply_32_Bits ; BX:AX = (Cyl * Head *100)
mov DX,BX ;Set up divide
; DX:AX: 100*head*cylinder (should not be bigger than a DWORD)
; DI:SI: heads*cylinder
; We need to assure a word division
mov DI,word ptr TracksPerDisk+2
mov SI,word ptr TracksPerDisk
SetUpDivide:
or DI,DI
jz DivideOK
shr DI,1 ; shift DI:SI 1 bit right
rcr SI,1
shr DX,1 ; shift DX:AX 1 bit right
rcr AX,1
jmp short SetUpDivide
DivideOk:
div SI
cmp AX,PercentComplete ;Only print message when change
je ScreenUpdateDone
; johnhe 02-27-90
; Change added here for DOS 5.0 install program to do a
; special interrupt to display the percent complete on
; a gage. AX == special code, BX == percent complete
test SwitchMap,Switch_Select ; Was format spawned by the
jz NormalDisplay ; install program?
mov BX,AX ; BX == percent completed
mov AX,GAGE_UPDATE ; AX == special function code
clc
int 2fh ; Multiplex interrupt
or AX,AX ; AX <> 0 --> user wants to exit
jz ScreenUpdateDone
jmp FatalExit
; End of code added for DOS 5.0 install program
NormalDisplay:
mov PercentComplete,AX ; Save it if changed
Message msgCurrentTrack
ScreenUpdateDone:
pop SI ; M018
pop DI ; M018
pop AX
pop CX ; Restore register
pop DX
return
DisplayCurrentTrack endp
;=========================================================================
; Adj_Track_Count : This routine adjusts the track count by the
; number of tracks that have been formatted
; in one FormatTrack call.
;
; Inputs : TracksLeft - # of tracks remaining to be formatted
; Tracks_To_Format - Tracks formatted in 1 call
;
; Outputs : TracksLeft - # of tracks remaining to be formatted
;=========================================================================
Procedure Adj_Track_Count
push AX ; save regs
push DX ; M018
mov DX,word ptr TracksLeft+2; get tracks remaining
mov AX,word ptr TracksLeft
sub AX,Tracks_To_Format ; subtract amount formatted
sbb DX,0
mov word ptr TracksLeft,AX ; save new tracks remaining value
mov word ptr TracksLeft+2,DX
pop DX ; M018
pop AX ; restore regs
ret
Adj_Track_Count endp
;=========================================================================
;
; NextTrack : This routine determines the next track to be
; formatted.
;
; Inputs : TracksLeft - # of tracks remaining
; Tracks_To_Format - # of tracks to format in 1 call
; FP_Head - disk head
; FP_Cylinder - disk cylinder
;
; Outputs : TracksLeft - # of tracks remaining
; FP_Head - disk head
; FP_Cylinder - disk cylinder
; CY - no tracks left to format
; NC - tracks left to format
;
;=========================================================================
Procedure NextTrack
mov CX,word ptr TracksLeft+2
or CX,Word ptr TracksLeft
jne $$IF149 ; Yes
stc ; Signal end of format
mov Format_End,True
jmp SHORT $$EN149
$$IF149:
mov CX,Tracks_To_Format ; Get mAX track count for call
$$DO151: ; While tracks remain
cmp CX,00 ; End of head/cyl. adjustment?
je $$EN151 ; Yes
inc FormatPacket.FP_Head ; Next head
mov AX,FormatPacket.FP_Head ; Get head for comp
cmp AX,DeviceParameters.DP_BPB.oldBPB.BPB_Heads ; Exceeded head count?
.errnz EDP_BPB NE DP_BPB
jne $$IF154 ; Yes
mov FormatPacket.FP_Head,00 ; Reinit. head
inc FormatPacket.FP_Cylinder ; Next cylinder
$$IF154:
dec CX ; Decrease counter
jmp SHORT $$DO151
$$EN151:
clc ; Clear CY
$$EN149:
ret
NextTrack ENDP
;=========================================================================
; CurrentLogicalSector:
; Get the current logical sector number
;
; Input:
; current track = TracksPerDisk - TracksLeft
; SectorsPerTrack
;
; Output:
; BX = logical sector number of the first sector in the track we
; just tried to format
;=========================================================================
CurrentLogicalSector PROC NEAR
push AX ; Save regs
push BX
push DX
.386
mov EAX, TracksPerDisk
sub EAX, TracksLeft
movzx ebx, DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerTrack
.errnz EDP_BPB NE DP_BPB
mul ebx
mov dword ptr Relative_Sector_Low,EAX ; Save sector #
.8086
pop DX ; Restore regs
pop BX
pop AX
return
CurrentLogicalSector ENDP
;=========================================================================
;
; BadSector:
; Reports the bad Sectors.
; Reports the track where DiskFormat stopped.
; From then on it formats until it reaches a bad track, or end,
; and reports that.
;
; Output:
; Carry: set --> Fatal error
; if Carry not set
; AX - The number of consecutive bad Sectors encountered
; AX == 0 --> no More bad Sectors
; BX - The logical sector number of the first bad sector
;
; Algorithm:
; if DiskFormatErrors
; DiskFormatErrors = false
; return current track
; else
; next track
; while not done
; if format track fails
; return current track
; next track
; return 0
;=========================================================================
BadSector proc near
; Don't bother to do the format
; /c was given
test FormatError, 0ffH
jz ContinueFormat
mov FormatError, 0
jmp SHORT ReportBadTrack
ContinueFormat:
call Adj_Track_Count ; Decrease track counter
call NextTrack ; Adjust head and cylinder
cmp Format_End,True ; End of format?
je $$IF44 ; No
call Format_Loop ; Format until CY
cmp Format_End,True ; End of format?
je $$IF45 ; No
call CheckError ; Must be error - which error?
jc $$IF46 ; Non-Fatal error?
call CurrentLogicalSector ; Yes - get position
; set tracksize
mov AX,DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerTrack
.errnz EDP_BPB NE DP_BPB
clc ; Signal O.K. to continue
$$IF46:
jmp SHORT $$EN45
$$IF45:
jmp SHORT NoMoreTracks ;End of format
$$EN45:
jmp SHORT $$EN44
$$IF44:
jmp SHORT NoMoreTracks ; end of format
$$EN44:
return
ReportBadTrack:
call CurrentLogicalSector
mov AX, DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerTrack
clc
return
NoMoreTracks: ; Don't display done msg
test SwitchMap,(Switch_Select or SWITCH_AUTOtest)
jnz $$IF52 ; if EXEC'd by SELECT
Message msgFormatComplete
$$IF52:
mov AX, 0
clc
return
BadSector endp
;=========================================================================
; Get_Bad_Sector_Hard : Determine the bad sector.
;
; Inputs :
; Head of failing track
; Cylinder of failing track
; Relative_Sector_Low - 1st. sector in track
; Relative_Sector_High
;
; ClustBound_Adj_Factor - The number of Sectors
; that are to be read
; at one time.
; ClustBound_SPT_Count - Used by Calc_Cluster_Boundary
; to track how many Sectors
; have been read.
; ClustBound_Flag - True (Use cluster buffer)
; - False (Use internal buffer)
; ClustBound_Buffer_Seg - Segment of buffer
;
; Outputs : Marked cluster as bad
;=========================================================================
Procedure Get_Bad_Sector_Hard
push CX ; Save CX
mov CX,0001h ; Set counter to start at 1
mov ClustBound_SPT_Count,00h ; Clear sector counter
mov ClustBound_Adj_Factor,01h ; Default value
; DO WHILE Sectors left
$$DO115:
cmp CX,DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerTrack ;At end?
.errnz EDP_BPB NE DP_BPB
ja $$EN115 ; Yes,exit
push CX ; Save CX
cmp ClustBound_Flag,True ; Full buffer there?
jne $$IF117 ; Yes
call Calc_ClustBound_ ; See if on boundary
mov AX,ClustBound_Buffer_Seg
; Point to transfer area
mov WORD PTR RWPacket.TRWP_Transferaddress[0],0
mov WORD PTR RWPacket.TRWP_Transferaddress[2],AX
jmp SHORT $$EN117 ; Default to internal buffer
$$IF117: ; Point to transfer area
mov WORD PTR RWPacket.TRWP_Transferaddress[0],offset RW_TRF_Area
mov WORD PTR RWPacket.TRWP_Transferaddress[2],DS
$$EN117:
call Verify_Structure_Set_Up ; Set up verify vars
mov AX,(IOCTL shl 8) or GENERIC_IOCTL
xor BX,BX ; Clear BX
mov BL,DriveToFormat ; Get Drive
inc BL ; Adjust it
mov CX,(IOC_DC shl 8) or READ_TRACK
cmp IsEXTRAWIODrv,0
je DoIOCTL2
mov CX,(IOC_EDC shl 8) or READ_TRACK
; Buffer is only cluster size
DoIOCTL2:
lea DX,RWPacket ; Point to parms
int 21h
pop CX ; Restore CX
push CX ; Save CX
jnc $$IF120 ; An error occurred
call Calc_Cluster_Position ; Determine which cluster
call BadClus ; Mark the cluster as bad
$$IF120:
pop CX
add CX,ClustBound_Adj_Factor ; Adjust loop counter
mov AX,ClustBound_Adj_Factor ; Get adjustment factor
xor DX,DX
add AX,Relative_Sector_Low ; Add in low word
adc DX,Relative_Sector_High ; Pick up carry in high word
mov Relative_Sector_Low,AX ; Save low word
mov Relative_Sector_High,DX ; Save high word
jmp SHORT $$DO115
$$EN115:
pop CX
ret
Get_Bad_Sector_Hard ENDP
;=========================================================================
; Get_Bad_Sector_Floppy : This routine marks an entire track as bad
; since it is a floppy disk.
;
; Inputs : Relative_Sector_Low - first sector
;
; Outputs : Fat marked with bad Sectors
;=========================================================================
Procedure Get_Bad_Sector_Floppy
push BX ; Save regs
push CX
; Get Sectors/track
mov CX,DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerTrack
.errnz EDP_BPB NE DP_BPB
$$DO123: ; While Sectors left
cmp CX,00 ; At end
je $$EN123 ; Yes
push BX ; Save BX we destroy it
push CX ; Save CX we destroy it
call Calc_Cluster_Position ; Get cluster position
call BadClus ; Mark it as bad
pop CX ; Restore regs
pop BX
dec CX ; Decrease loop counter
.386
inc dword ptr Relative_Sector_Low ; Next sector
.8086
jmp SHORT $$DO123
$$EN123:
pop CX ; Restore regs
pop BX
ret
Get_Bad_Sector_Floppy ENDP
;=========================================================================
;
; Inputs: DX:AX - Cluster number
; Outputs: The given cluster is marked as invalid
; Zero flag is set if the cluster was already marked bad
;
; Registers modified: DX
; SI
;
; BADCLUS : Marks off a bad cluster in the Fat
; If a cluster has already been marked bad it
; will return with ZR.
;
; Inputs : DX:AX - Cluster Number
;
; Outputs : Cluster is marked invalid
; ZR set if cluster already marked bad
;
;=========================================================================
BadClus proc near ; mark bad clusters
push DI ; save affected regs
push AX
push BX
push CX
push DX
push ES
cmp FATNotAllInMem,0
je DoBadInMem
push dx ; Save cluster number
push ax
call AllocInitBadClusBitmap ; Does nothing if BadClusBitmap
; already exists
.386
;; Manual assemble to prevent compile warning
;; pop eax ; recover cluster #
db 066h,058h
;;
jc $$EN10
push ds
push si
lds si,BadClusBitMap
mov ebx,eax
shr ebx,19 ; (E)BX is "64k index" of this bit
and eax,00007FFFFh ; bit index in that 64k
mov cx,ds
add cx,bx ; Go to correct 64k piece
mov ds,cx
bts dword ptr [si],eax ; Set the bit
pop si
pop ds
jnc short ClrZr
xor ax,ax ; Set zero flag, cluster already marked
jmp $$EN10
ClrZr:
inc eax ; Clear the zero flag
.8086
jmp $$EN10
DoBadInMem:
mov ES, WORD PTR FatSpace + 2 ; obtain seg of Fat
cmp fBig32Fat,TRUE ; 32 bit Fat?
je $$IF8a ; yes
cmp fBigFat,TRUE ; 16 bit Fat?
je $$IF8 ; yes
; 12-bit FAT
mov CX,2 ; divide by 2
push AX ; saves low cluster number
mov SI,DX ; pick up high word of cluster
mov DI,AX ; pick up low word of cluster
call divide_32_Bits ; 32 bit divide
add AX,DI ; add in low word of result
adc DX,SI ; pick up low word carry
; cluster = cluster * 1.5
add AX,WORD PTR FatSpace ; add 0
adc DX,0 ; pick up carry
mov BX,DX ; get high word for adjust
mov CX,ES ; place seg in AX
call BadClus_address_Adjust ; adjust segment offset
mov ES,CX ; new segment
mov SI,AX ; new offset
mov DX,0ff7h ; bad cluster flag
mov AX,0fffh ; mask value
pop CX ; restore AX in CX - low cluster #
test CX,1 ; is old clus num odd?
jz $$IF9 ; yes
.386
shl AX,4 ; get only 12 bits - fff0
shl DX,4 ; get 12 bits - ff70
.8086
$$IF9:
jmp SHORT $$EN8
$$IF8a:
; 32-bit FAT
xor SI,SI ; clear si
mov BX,DX ; get high word for multiply
mov CX,4 ; multiply by 4
call Multiply_32_Bits ; 32 bit multiply due to 4 bytes per
; Fat cell. This gives us an offset
; into the FAT
mov CX,ES ; place seg in CX
call BadClus_Address_Adjust ; adjust segment:offset
mov ES,CX ; new segment
mov SI,AX ; new offset
.386
mov ecx,dword ptr ES:[SI] ; Get previous value
and ecx,00FFFFFFFh ; Discard high 4 bits
mov edx,00ffffff7h
mov dword ptr ES:[SI],edx ; flag it a bad cluster
cmp EDX,ECX ; return op == badval;
.8086
jmp short $$EN10
$$IF8:
; 16-bit FAT
xor SI,SI ; clear si
mov BX,DX ; get high word for multiply
mov CX,2 ; multiply by 2
call Multiply_32_Bits ; 32 bit multiply due to 2 bytes per
; Fat cell. This gives us an offset
; into the FAT
mov CX,ES ; place seg in CX
call BadClus_Address_Adjust ; adjust segment:offset
mov ES,CX ; new segment
mov SI,AX ; new offset
mov DX,0fff7h ; bad cluster value
mov AX,0ffffh ; mask value
$$EN8:
mov CX,ES:[SI] ; get contents of Fat cell
and CX,AX ; make it 12 or 16 bit
; depending on value in AX
not AX ; set AX to 0
and ES:[SI],AX ; clear Fat entry
or ES:[SI],DX ; flag it a bad cluster
cmp DX,CX ; return op == badval;
$$EN10:
pop ES
pop DX
pop CX
pop BX
pop AX
pop DI
return
badclus endp
;=========================================================================
; Verify_Structure_Set_Up : Set up the fields for the Read IOCTL
; to verify the Sectors in a failing
; track. Also, it displays the
; message notifying the user of the
; Sectors it is verifying.
;=========================================================================
Procedure Verify_Structure_Set_Up ; Set up verify structure
mov RWPacket.TRWP_SpecialFunctions,00h ; Reset special functions
mov AX,FormatPacket.FP_Head ; Get current head
mov RWPacket.TRWP_Head,AX ; Get current head
mov AX,FormatPacket.FP_Cylinder ; Get current cylinder
mov RWPacket.TRWP_Cylinder,AX ; Get current cylinder
dec CX ; Make sector 0 based
mov RWPacket.TRWP_FirstSector,CX ; Get sector to read
mov AX,ClustBound_Adj_Factor ; Get # of Sectors to read
mov RWPacket.TRWP_SectorsToReadWrite,AX ; Read only # sector(s)
call Calc_Cluster_Position ; Determine cluster number
mov WORD PTR Msg_Allocation_Unit_Val[+2],DX ; Save high word of cluster
mov WORD PTR Msg_Allocation_Unit_Val[+0],AX ; Save low word of cluster
message msgVerify
ret
Verify_Structure_Set_Up ENDP
;=========================================================================
; Calc_Cluster_Position : This routine calculates which cluster the
; failing sector falls in.
;
; Inputs : Relative_Sector_High - high word of sector position
; Relative_Sector_Low - low word of sector position
;
; Outputs : DX:AX - Cluster number
;=========================================================================
Procedure Calc_Cluster_Position
push CX ; Save regs
push DI
push SI
mov DX,WORD PTR Relative_Sector_High ; Get the high sector word
mov AX,WORD PTR Relative_Sector_Low ; Get the low sector word
sub AX,word ptr StartSector ; Get relative sector #
sbb DX,word ptr StartSector+2 ; Pick up borrow
mov SI,DX ; Get high word
mov DI,AX ; Get low word
xor CX,CX ; Clear CX
; Get Sectors/cluster
mov CL,DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerCluster
.errnz EDP_BPB NE DP_BPB
call divide_32_Bits ; 32 bit division
mov DX,SI ; Get high word of result
mov AX,DI ; Get low word of result
add AX,2 ; Adjust for cluster bias
adc DX,0 ; Pick up carry
pop SI ; Restore regs
pop DI
pop CX
ret
Calc_Cluster_Position ENDP
;=========================================================================
; Calc_ClustBound_ : This routine will determine where, within a
; cluster, a sector resides.
;
; Inputs : Relative_Sector_Low - Sector
; Relative_Sector_High
;
; Outputs : ClustBound_Adj_Factor - The number of Sectors
; remaining in the cluster.
; ClustBound_SPT_Count - The count of Sectors
; having been accessed for
; a track.
;=========================================================================
Procedure Calc_ClustBound_
push AX ; Save regs
push BX
push CX
push DX
push SI
push DI
xor DX,DX ; Clear high word
mov DX,WORD PTR Relative_Sector_High
mov AX,WORD PTR Relative_Sector_Low
sub AX,word ptr StartSector ; Get relative sector #
sbb DX,word ptr StartSector+2 ; Pick up borrow
mov SI,DX ; Get high word
mov DI,AX ; Get low word
xor CX,CX ; Clear CX
mov CL,DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerCluster
.errnz EDP_BPB NE DP_BPB
call divide_32_Bits ; 32 bit division
or CX,CX ; See if remainder exists
jz $$IF132 ; Remainder exists
xor BX,BX
mov BL,DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerCluster
sub BX,CX
mov ClustBound_Adj_Factor,BX ; Remainder = sector count
jmp SHORT $$EN132 ; Noremainder
$$IF132:
xor BX,BX ; Clear BX
mov BL,DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerCluster
mov ClustBound_Adj_Factor,BX
$$EN132:
mov AX,ClustBound_SPT_Count ; Get current sector count
xor DX,DX ; Clear high word
add AX,ClustBound_Adj_Factor ; Get next sector count
; Exceeded Sectors/track?
cmp AX,DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerTrack
jna $$IF135 ; Yes
; only use difference
mov AX,DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerTrack
sub AX,ClustBound_SPT_Count ; Get next sector count
mov ClustBound_Adj_Factor,AX
$$IF135:
mov AX,ClustBound_SPT_Count ; Get sector count
xor DX,DX ; Clear high word
add AX,ClustBound_Adj_Factor ; Get new sector count
mov ClustBound_SPT_Count,AX ; Save it
pop DI ; Restore regs
pop SI
pop DX
pop CX
pop BX
pop AX
ret
Calc_ClustBound_ ENDP
;=========================================================================
;
; BadClus_address_Adjust - This routine adjusts the segment and
; offset to provide addressibility into
; the Fat table.
;
; Inputs : BX - high word to adjust segment for
; AX - low word to adjust segment for
; CX - segment to be adjusted
;
; Outputs : CX - new segment value
; AX - new offset value
;
;=========================================================================
Procedure BadClus_address_Adjust
push BX ; Save regs
push DX
push DI
push SI
mov DX,CX ; Save segment value
mov SI,BX ; Get high word for divide
mov DI,AX ; Get low word for divide
xor CX,CX ; Clear CX
mov CL,Paragraph_Size ; Divide by 16
call divide_32_Bits ; Perform division
add DX,DI ; Adjust segment for result
mov AX,CX ; Pick up the remainder
mov CX,DX ; Pass back new segment
pop SI ; Restore regs
pop DI
pop DX
pop BX
ret
BadClus_address_Adjust ENDP
;=========================================================================
;Routine name: Multiply_32_Bits
;=========================================================================
;Description: A real sleazy 32 bit x 16 bit multiply routine. Works by adding
; the 32 bit number to itself for each power of 2 contained in the
; 16 bit number. Whenever a bit that is set in the multiplier (CX)
; gets shifted to the bit 0 spot, it means that that amount has
; been multiplied so far, and it should be added into the total
; value. Take the example CX = 12 (1100). Using the associative
; rule, this is the same as CX = 8+4 (1000 + 0100). The
; multiply is done on this principle - whenever a bit that is set
; is shifted down to the bit 0 location, the value in BX:AX is
; added to the running total in DI:SI. The multiply is continued
; until CX = 0. The routine will exit with CY set if overflow
; occurs.
;
;
;Called Procedures: None
;
;Change History: Created 7/23/87 MT
;
;Input: BX:AX = 32 bit number to be multiplied
; CX = 16 bit number to be multiplied. (Must be even number)
;
;Output: BX:AX = output.
; CY set if overflow
;
;=========================================================================
Multiply_32_Bits PROC
.386
push ax
mov ax,bx
shl eax,16
pop ax
movzx ecx,cx
mul ecx
mov ebx,eax
shr ebx,16
or edx,edx ; Overflow?
jz short OkRet ; No, carry clear
stc
OkRet:
.8086
ret
Multiply_32_Bits endp
;=========================================================================
; divide_32_Bits - This routine will perform 32bit division
; It works by first dividing the high word
; and leaving the remainder in DX and then
; dividing the low word with the remainder
; still in DX
;
; Inputs : SI:DI - value to be divided
; CX - divisor
;
; Outputs : SI:DI - result
; CX - remainder
;=========================================================================
Procedure divide_32_Bits
push AX ; Save regs
push BX
push DX
xor DX,DX ; clear DX
mov AX,SI ; get high word
div CX ; get high word result
mov SI,AX ; save high word result
mov AX,DI ; get low word
div CX ; get low word result
mov DI,AX ; save low word result
mov CX,DX ; pick up remainder
pop DX ; restore regs
pop BX
pop AX
ret
divide_32_Bits endp
;=========================================================================
;
; QuickFormat : This procedure will perform a Quick format by
; simply copying any bad cluster markers from the
; old FAT on the disk to the new FAT. The old FAT is
; read in one sector at a time using FatSector buffer.
; The new FAT is held in FatSpace buffer.
;
; Registers Destroyed : SI,AX,BX,CX
;
; Assumes: DS:DATA,ES:Nothing
;
;=========================================================================
QuickFormat proc near
mov SI,DATA
mov ES,SI ; Set ES to data segment
assume ES:DATA,DS:Nothing ; Assembler directive
; Set device parameters here
mov ES:DeviceParameters.DP_SpecialFunctions,(INSTALL_FAKE_BPB or TRACKLAYOUT_IS_GOOD)
.errnz EDP_SPECIALFUNCTIONS NE DP_SPECIALFUNCTIONS
lea DX,ES:DeviceParameters
call SetDeviceParameters
.386
mov ES:sector_in_buffer,0ffffffffh ; force first read to ensure
; buffer validity
test ES:fBig32FAT,0ffh ; See if 32 bit fat
jz short Test16BitEntry ; If zero then 16 or 12 bit fat
mov EBX,00ffffff7h ; Set 32 bit value for bad cluster
jmp SHORT InitClusCount
Test16BitEntry:
test ES:fBigFAT,0ffh ; See if 16 bit fat
jz short Set12BitEntry ; If zero then 12 bit fat
mov EBX,0000fff7h ; Set 16 bit value for bad cluster
jmp SHORT InitClusCount
Set12BitEntry:
mov EBX,00000ff7h ; Set 12 bit value for bad cluster
InitClusCount:
mov ES:CurrentCluster,2 ; M015; No need to do the first 2
mov ES:BadClusValue,EBX
cmp es:FATNotAllInMem,0
je short QuickLoop
push ds
push es
pop ds
message msgSetBadClus
pop ds
QuickLoop:
mov EAX,ES:CurrentCluster
call calc_sector_and_offset ; determine location of this entry
mov EBX,ES:sector_in_buffer
mov EAX,ES:sector_to_read
cmp EAX,EBX ; check if required sector is in buffer
je short DontHaveToRead
cmp es:FATNotAllInMem,0
je short DoRead
push ds
push es
pop ds
call DisplayFatDonePcnt
pop ds
DoRead:
call ReadFatSector ; read a sector of the FAT into buffer
jc ExitQuickFormatCRLFErr ; check for error
DontHaveToRead:
mov EAX,ES:CurrentCluster ; EAX = current cluster
xor CX,CX ; ECX = get cluster contents signal
lds SI,ES:FatSector ; DS:SI --> FAT buffer
call GetFatSectorEntry ; EAX = contents of FAT entry
mov EBX,ES:BadClusValue ; Restore bad cluster value
cmp EAX,EBX ; Is this cluster marked bad?
jne short NextCluster ; If EAX<>EBX good cluster
MarkInFormatBuffer:
cmp es:FATNotAllInMem,0
je short DoBadInMem2
push es
pop ds
call AllocInitBadClusBitmap ; Does nothing if BadClusBitmap
; already exists
jc short ExitQuickFormatCRLFErr ; check for error
mov EAX,ES:CurrentCluster ; EAX = current cluster
lds si,es:BadClusBitMap
mov ebx,eax
shr ebx,19 ; (E)BX is "64k index" of this bit
and eax,00007FFFFh ; bit index in that 64k
mov cx,ds
add cx,bx ; Go to correct 64k piece
mov ds,cx
bts dword ptr [si],eax ; Set the bit
jmp short NextCluster
DoBadInMem2:
mov ECX,EBX ; ECX = value to set in FAT buffer
mov EAX,ES:CurrentCluster ; EAX = this cluster number
lds SI,ES:FatSpace ; DS:SI --> Format's FAT buffer
call GetSetFatEntry ; Set the cluster in Format's buffer
NextCluster:
inc ES:CurrentCluster ; go to next cluster
mov EAX,ES:CurrentCluster
mov EBX,ES:TotalClusters
cmp EAX,EBX ; check for last cluster in FAT
jna QuickLoop
.8086
mov BX,DATA
mov DS,BX ; restore DS to DATA segment
assume DS:DATA,ES:DATA ; Assembler directive
cmp ds:FATNotAllInMem,0
je ExitQuickFormatRet ; Carry clear if jmp
message msgSetBadClusDone
clc
ExitQuickFormatRet:
ret
ExitQuickFormatCRLFErr:
mov BX,DATA
mov DS,BX
cmp ds:FATNotAllInMem,0
stc
je ExitQuickFormatRet
Message msgCrLf
stc
jmp short ExitQuickFormatRet
QuickFormat endp
;===========================================================================
;
; calc_sector_and_offset : This procedure computes the logical sector
; number the given FAT entry is in, and its
; offset from the start of the sector.
;
; Inputs : EAX = entry number
; fBigFat = flag for 12- or 16-bit FAT entries
; Number of reserved sectors
;
; Output : sector_to_read = logical disk sector holding FAT entry
; entry_offset = offset from start of sector
; odd_entry = flag for 12-bit entry alignment (1=odd,0=even)
;
; Registers Destroyed : AX,BX,CX,DX
;
; Strategy : This procedure assumes the sector size is 512 bytes.
; The byte offset from the start of the FAT is first
; calculated. This is then divided by 512, so that
; required sector = quotient
; offset = remainder
; The logical sector number is obtained by adding on the
; number of reserved sectors.
;
; M017: The code does not assume 512 BytesPerSector (it is even simpler!)
;============================================================================
calc_sector_and_offset proc near
assume DS:NOTHING,ES:DATA
.386
xor EDX,EDX
test ES:fBig32Fat,0ffh ; See if 32 bit FAT
jz short TestOffset16 ; If not do 16 or 12 bit FAT
shl EAX,1 ; EAX *= 2
rcl EDX,1
jmp short FindOffset16 ; Now mult by 2 again for * 4
TestOffset16:
test ES:fBigFat,0ffh ; See if 16 bit FAT
jz short FindOffset12 ; If not do 12 bit FAT
FindOffset16:
shl EAX,1 ; EAX *= 2
rcl EDX,1
; Now offset from start of FAT is
; in EDX:EAX
jmp SHORT FindSector
FindOffset12:
mov BX,AX ; BX = cluster number
shl AX,1
add AX,BX ; AX *= 3
mov ES:odd_entry,AL ; lsb of AX determines even or odd
and ES:odd_entry,1
shr AX,1 ; Divide by 2
; Now offset from start of FAT is
; in EDX:EAX
FindSector:
movzx ebx,ES:DeviceParameters.DP_BPB.oldBPB.BPB_BytesPerSector
.errnz EDP_BPB NE DP_BPB
div ebx
mov ES:entry_offset,DX
movzx ebx,ES:DeviceParameters.DP_BPB.oldBPB.BPB_RESERVEDSECTORS
add EAX,ebx
mov ES:sector_to_read,EAX
.8086
ret
calc_sector_and_offset endp
;===========================================================================
;
; ReadFatSector : This procedure will read in a sector of the FAT
; into the FatSector buffer. This is done by loading
; the required parameters and calling ReadWriteSectors.
;
; Input : sector_to_read
; Output: loaded buffer
; sector_in_buffer
;
; Registers destroyed: AX,BX,CX,DX
;
;===========================================================================
ReadFatSector proc near
assume DS:NOTHING,ES:DATA
push DS ; Preserve DS
.386
mov EDX,ES:sector_to_read ; EDX = starting sector
;; Manual assemble to prevent compile warning
;; push edx
db 066h,052h
;;
cmp ES:DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerFAT,0
.errnz EDP_BPB NE DP_BPB
je short NotBigFAT1
mov cx,ES:DeviceParameters.DP_BPB.BGBPB_ExtFlags
test cx,BGBPB_F_NoFATMirror
jz short NotBigFAT1
and ecx,NOT BGBPB_F_ActiveFATMsk
jz short NotBigFAT1
mov ebx,edx
mov eax,dword ptr ES:DeviceParameters.DP_BPB.BGBPB_BigSectorsPerFat
mul ecx
add eax,ebx
mov edx,eax
NotBigFAT1:
mov ebx,edx
shr ebx,16
mov ES:Read_Write_Relative.Start_Sector_High,bx
mov AL,ES:DriveToFormat ; AL = DOS drive number
mov CX,1 ; 1 sector only
lds BX,ES:FatSector ; DS:BX --> read buffer
call Read_Disk ; perform read
;; Manual assemble to prevent compile warning
;; pop edx
db 066h,05Ah
;;
mov ES:sector_in_buffer,EDX ; update sector in memory
.8086
pop DS ; Restore DS
ret
ReadFatSector endp
;=========================================================================
; WriteFatSector : This routine writes the logical sector count requested.
; of the FAT
;
; Inputs : AL - Drive letter
; DS:BX - Segment:offset of transfer address
; ECX - Sector count
; EDX - 1st. sector
; ES -> Data
;
; Outputs : Logical Sectors written
;=========================================================================
procedure WriteFatSector
assume DS:NOTHING,ES:DATA
.386
$$DO67:
or ECX,ECX ; any Sectors?
jz short $$EN67 ; no
ifdef NEC_98
cmp ECX,10h ; Single write?
jna short $$IF69 ; yes
;; Manual assemble to prevent compilke warning
;; push ECX ; save count left
db 066h,051h
;;
mov CX,10h
push AX ; save AX
mov eax,edx
shr eax,16
mov Read_Write_Relative.Start_Sector_High,ax
pop ax ; Recover drive
push ax
;; Manual assemble to prevent compile warning
;; push EDX
db 066h,052h
;;
call write_disk ; write it
;; Manual assemble to prevent compile warning
;; pop EDX
db 066h,05Ah
;;
pop AX ; restore AX
;; Manual assemble to prevent compile warning
;; pop ECX ; restore count
db 066h,059h
;;
jc short Write_Exit ; exit if fail
mov SI,ES:DeviceParameters.DP_BPB.oldBPB.BPB_BytesPerSector
shl SI,1
shl SI,1
shl SI,1
shl SI,1 ; * 10h
call seg_adj ; adjust segment
mov BX,SI ; new offset
add EDX,10h
sub ECX,10h
jmp SHORT $$DO67
else
cmp ECX,40h ; Single write?
jna short $$IF69 ; yes
;; Manual assemble to prevent compilke warning
;; push ECX ; save count left
db 066h,051h
;;
mov CX,40h
push AX ; save AX
mov eax,edx
shr eax,16
mov Read_Write_Relative.Start_Sector_High,ax
pop ax ; Recover drive
push ax
;; Manual assemble to prevent compile warning
;; push EDX
db 066h,052h
;;
call write_disk ; write it
;; Manual assemble to prevent compile warning
;; pop EDX
db 066h,05Ah
;;
pop AX ; restore AX
;; Manual assemble to prevent compile warning
;; pop ECX ; restore count
db 066h,059h
;;
jc short Write_Exit ; exit if fail
mov SI,8000h
call seg_adj ; adjust segment
mov BX,SI ; new offset
add EDX,40h
sub ECX,40h
jmp SHORT $$DO67
endif
$$IF69:
push AX ; save drive
mov eax,edx
shr eax,16
mov Read_Write_Relative.Start_Sector_High,ax
pop ax ; Recover drive
push ax
call write_disk ; write it
pop AX ; restore AX
mov ECX,0 ; set CX to 0 - last read
.8086 ; DO NOT XOR!!!
$$EN67:
Write_Exit:
ret
WriteFatSector ENDP
FlushCurrInMemFATBuf proc near
ASSUME DS:DATA,ES:NOTHING
push ds
push ds
pop es
.386
assume ES:DATA
movzx cx,DeviceParameters.DP_BPB.oldBPB.BPB_NumberOfFats ;loop control
.errnz EDP_BPB NE DP_BPB
or CX,CX ;check for zero
stc
jz short FFMExit
movzx eax,DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerFat
or ax,ax
jnz short GtFatSz
mov eax,dword ptr DeviceParameters.DP_BPB.BGBPB_BigSectorsPerFat
GtFatSz:
mov edx,CurrFATInMemStartSec
lds BX,FatSpace ;DS:BX --> FatSpace
assume DS:NOTHING,ES:DATA
mov SI,BX ;Set up for add. calc
call SEG_ADJ ;Get adjusted seg:off
mov BX,SI ;Get new offset
WriteFatLp: ;loop while FATs > 0
;; Manual assemble to prevent compile warning
;; push eax ;Save FAT size
db 066h,050h
;;
push BX ;save Fat offset
push DS ;save Fat segment
push CX ;save Fat count
;; Manual assemble to prevent compile warning
;; push EDX ;Fat start sector
db 066h,052h
;;
mov ecx,CurrFATInMemSecCnt
mov AL,DriveToFormat
call WriteFatSector ;write the Fat
;; Manual assemble to prevent compile warning
;; pop EDX ;get 1st. Fat sector
db 066h,05Ah
;;
pop CX ;get Fat count
pop DS ;restore Fat segment
pop BX ;restore Fat offset
;; Manual assemble to prevent compile warning
;; pop eax ;restore FAT size
db 066h,058h
;;
jc short FFMExit ;check for errors
add EDX,EAX ;next FAT start sector
loop WriteFatLp ;write all FATs
clc
FFMExit:
pop ds
ret
FlushCurrInMemFATBuf endp
DisplayFatDonePcnt proc near
assume DS:DATA,ES:NOTHING
.386
mov eax,CurrentCluster
dec eax
dec eax
mov ecx,100
mul ecx
mov ecx,TotalClusters
dec ecx
dec ecx
div ecx
.8086
cmp ax,100
jbe PcntOk
mov ax,100
PcntOk:
cmp AX,PercentComplete ;Only print message when change
je NoUpd
mov PercentComplete,AX ; Save it if changed
Message msgCurrentTrack
NoUpd:
ret
DisplayFatDonePcnt endp
;===========================================================================
; Routine name: FlushFATBuf
;===========================================================================
;
; Description: Flush the in memory FAT buffer out to the disk
;
; Arguments: None
; ----------------------------------------------------------------
; Returns: carry set if error
; -----------------------------------------------------
; Registers destroyed: EAX EBX ECX EDX
; ----------------------------------------
; Strategy
; --------
;===========================================================================
FlushFATBuf PROC near
assume DS:DATA,ES:NOTHING
push DS ;preserve DS
cmp FATNotAllInMem,0
.386
je WrtWholeFat
;
; What we have at this point is the first part of the FAT in the in
; memory FAT buf and the BadClusBitMap which indicates what clusters
; we want to mark bad.
;
; Go through the in memory FAT buf (currently first FATSecCntInMem
; sectors of the FAT) marking any bad clusters, and/or write an EOF
; mark in the root directory start cluster, write it out.
;
; For rest of FAT, zero init in memory FAT buf, mark bad clusters
; and/or put EOF mark in root directory start cluster and write it out.
;
; We had better be talking 32-bit FAT here!!!!!! 12-bit and 16-bit
; FATs always fit in memory.
;
test fBig32FAT,0ffh
stc ; Set error
jz ExitWriteFat ; Not 32-bit FAT !!!?????
; Set the bad clus value
mov BadClusValue,00ffffff7h ; Set 32 bit value for bad cluster
; Set the current in mem FAT start sector
movzx eax,DeviceParameters.DP_BPB.oldBPB.BPB_ReservedSectors
.errnz EDP_BPB NE DP_BPB
mov CurrFATInMemStartSec,eax
; Calculate how many clusters fit in memory NOTE that since we are
; restricted to 32-bit FAT here we KNOW clusters do not span sector
; boundaries in the FAT.
mov eax,FATSecCntInMem
mov CurrFATInMemSecCnt,eax
movzx ecx,DeviceParameters.DP_BPB.oldBPB.BPB_BytesPerSector
mul ecx
mov ecx,4 ; 4 bytes per cluster
div ecx
mov FATInMemClusCnt,eax
; Init the in memory buffer with the start of the fat
call init_fat_with_header
message msgWriteFat
; Set the rest of the buffer control variables for the in memory buffer
mov CurrentCluster,2
mov CurrFATInMemStartClus,0
ClusLoop2:
call DisplayFatDonePcnt
mov eax,dword ptr DeviceParameters.DP_BPB.BGBPB_RootDirStrtClus
cmp eax,2
jb short NoRootEOF
cmp EAX,TotalClusters
ja short NoRootEOF
sub eax,CurrFATInMemStartClus
jc short NoRootEOF
cmp eax,FATInMemClusCnt
jae short NoRootEOF
shl eax,2 ; * 4 bytes per cluster
mov ecx,00FFFFFFFh
push ds
lds BX,FatSpace ;DS:BX --> FatSpace
assume DS:nothing
ror eax,16 ; AX is high 16 bits of EAX
; which is "64k index" of
; this DWORD
shl ax,12 ; Convert to SEGMENT value
push cx ; Save low 16 of bad mark
mov cx,ds
add cx,ax
mov ds,cx
pop cx
ror eax,16 ; Get back offset in this 64k
add bx,ax ; Index this cluster entry
mov dword ptr ds:[bx],ecx ; Mark as BAD
pop ds
assume DS:data
NoRootEOF:
cmp BadClusBitMap,0 ; Anything to mark?
je short NoBadToMark ; Nope
mov eax,CurrentCluster
push ds
lds si,BadClusBitMap
assume DS:nothing
mov ebx,eax
shr ebx,19 ; (E)BX is "64k index" of this bit
and eax,00007FFFFh ; bit index in that 64k
mov cx,ds
add cx,bx ; Go to correct 64k piece
mov ds,cx
bt dword ptr [si],eax ; Bad cluster?
pop ds
assume DS:data
jnc short NxtClus ; No
mov eax,CurrentCluster
sub eax,CurrFATInMemStartClus
shl eax,2 ; * 4 bytes per cluster
mov ecx,BadClusValue
push ds
lds BX,FatSpace ;DS:BX --> FatSpace
assume DS:nothing
ror eax,16 ; AX is high 16 bits of EAX
; which is "64k index" of
; this DWORD
shl ax,12 ; Convert to SEGMENT value
push cx ; Save low 16 of bad mark
mov cx,ds
add cx,ax
mov ds,cx
pop cx
ror eax,16 ; Get back offset in this 64k
add bx,ax ; Index this cluster entry
mov dword ptr ds:[bx],ecx ; Mark as BAD
pop ds
assume DS:data
NxtClus:
inc CurrentCluster
mov eax,CurrentCluster
sub eax,CurrFATInMemStartClus
cmp eax,FATInMemClusCnt
jae short NextBuf
jmp short TestDone
NoBadToMark:
mov eax,FATInMemClusCnt
cmp CurrentCluster,2 ; Special case?
ja short OkayAdd ; No
sub eax,2 ; Special case for first FAT buffer
OkayAdd:
add CurrentCluster,eax
NextBuf:
call FlushCurrInMemFATBuf
jc short ExitWriteFatCRLFErr
call Fat_Init ; Zero init the in mem buffer
mov eax,CurrFATInMemSecCnt
add eax,CurrFATInMemStartSec
movzx ECX,DeviceParameters.DP_BPB.oldBPB.BPB_ReservedSectors
movzx edx,DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerFat
or dx,dx
jnz short GtFatSz2
mov edx,dword ptr DeviceParameters.DP_BPB.BGBPB_BigSectorsPerFat
GtFatSz2:
add ecx,edx ; First sector after the FAT
cmp eax,ecx ; Start sector in the FAT area?
jae short DoMsgDone ; No, all done, carry clear if jump
mov CurrFATInMemStartSec,eax
sub ecx,eax ; ecx is count of sectors left in FAT
cmp ecx,CurrFATInMemSecCnt ; Partial buffer at end of FAT?
jae short CntOK ; No.
mov CurrFATInMemSecCnt,ecx ; Last part of FAT < buffer size
CntOK:
mov eax,CurrentCluster
mov CurrFATInMemStartClus,eax
TestDone:
mov EAX,CurrentCluster
cmp EAX,TotalClusters ; check for last cluster in FAT
jna ClusLoop2
.8086
call FlushCurrInMemFATBuf ; Sets carry for return
jc ExitWriteFatCRLFErr
DoMsgDone:
message msgSetBadClusDone
clc
jmp short ExitWriteFat
ExitWriteFatCRLFErr:
Message msgCrLf
stc
jmp short ExitWriteFat
WrtWholeFat:
.386
push ds
pop es
assume ES:DATA
movzx cx,DeviceParameters.DP_BPB.oldBPB.BPB_NumberOfFats ;loop control
or CX,CX ;check for zero
jz short ExitWriteFat
movzx eax,ES:DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerFat
or ax,ax
jnz short GotFatSz
mov eax,dword ptr ES:DeviceParameters.DP_BPB.BGBPB_BigSectorsPerFat
GotFatSz:
movzx EDX,DeviceParameters.DP_BPB.oldBPB.BPB_ReservedSectors ;starting sector
lds BX,FatSpace ;DS:BX --> FatSpace
assume DS:NOTHING,ES:DATA
mov SI,BX ;Set up for add. calc
call SEG_ADJ ;Get adjusted seg:off
mov BX,SI ;Get new offset
WriteFatLoop: ;loop while FATs > 0
;; Manual assemble to prevent compile warning
;; push eax ;Save FAT size
db 066h,050h
;;
push BX ;save Fat offset
push DS ;save Fat segment
push CX ;save Fat count
;; Manual assemble to prevent compile warning
;; push EDX ;Fat start sector
db 066h,052h
;;
mov ecx,eax
mov AL,DriveToFormat
call WriteFatSector ;write the Fat
;; Manual assemble to prevent compile warning
;; pop EDX ;get 1st. Fat sector
db 066h,05Ah
;;
pop CX ;get Fat count
pop DS ;restore Fat segment
pop BX ;restore Fat offset
;; Manual assemble to prevent compile warning
;; pop eax ;restore FAT size
db 066h,058h
;;
jc short ExitWriteFat ;check for errors
add EDX,EAX ;next FAT start sector
loop WriteFatLoop ;write all FATs
ExitWriteFatOK:
clc ;signal success
ExitWriteFat:
pop DS ;restore DS
assume DS:DATA
ret
FlushFATBuf endp
;===========================================================================
; Routine name: GetSetFatEntry
;===========================================================================
;
; Description: Returns or sets the contents of the specified fat
; entry from a specified buffer. The buffer may be
; up to 128K in length for 16 bit FATs and 16K for
; 12 bit FATs.
;
; WARNING: Do not call this if FATNotAllInMem is TRUE
;
; Arguments: DS:SI --> Start of FAT buffer
; EAX = Cluster number
; ECX = 0fffffffeh if get cluster else set cluster from ECX
; ----------------------------------------------------------------
; Returns: EAX --> Contents of FAT entry
; -----------------------------------------------------
; Registers destroyed: EAX EBX ECX EDX
; ----------------------------------------
; Strategy
; --------
;===========================================================================
GetSetFatEntry PROC near
assume DS:NOTHING,ES:DATA
push SI ; Save regs for 'C' compatibility
push DS
push ES
; mov BX,DATA ; ES = DATA
; mov ES,BX
call Seg_Adj ; Normalize the pointer
.386
xor edx,edx
test ES:fBig32Fat,0ffh ; See if 32 bit FAT
jz short TestEntry16 ; If not do 12 bit FAT
shl EAX,1 ; EDX:EAX *= 2
rcl edx,1
jmp short FindEntry16 ; and *2 again for total of *4
TestEntry16:
test ES:fBigFat,0ffh ; See if 16 bit FAT
jz short FindEntry12 ; If not do 12 bit FAT
FindEntry16:
shl EAX,1 ; EDX:EAX *= 2
rcl edx,1
AddToBufStart: ; Offset in EAX may be > 64K
mov EBX,16 ; Convert EDX:EAX to paragraphs
div EBX ; (AX = DX:AX / 16) (DX = DX:AX % 16)
mov BX,DS ; Add paragraphs to DS
add AX,BX
mov DS,AX
add SI,DX ; Add remaining offset in DX to SI
cmp ecx,0fffffffeh ; if ECX == 0fffffffeh, then get entry
jz short GetFatEntry16
test ES:fBig32Fat,0ffh ; See if 32 bit FAT
jz short DoEntry16a
mov dword ptr [SI],ECX ; Set the entry
jmp SHORT GetSetEntryExit
DoEntry16a:
mov word ptr [SI],CX ; Set the entry
jmp SHORT GetSetEntryExitzx ; AX = FAT entry
GetFatEntry16:
test ES:fBig32Fat,0ffh ; See if 32 bit FAT
jz short DoEntry16
mov EAX,dword ptr [SI] ; Move the entry into EAX
and EAX,00FFFFFFFh ; Discard high 4 bits
jmp SHORT GetSetEntryExit ; EAX = FAT entry
.8086
DoEntry16:
mov AX,word ptr [SI] ; Move the entry into AX
jmp SHORT GetSetEntryExitzx ; AX = FAT entry
FindEntry12:
mov BX,AX ; BX = cluster number
shl AX,1
add AX,BX ; AX *= 3
test AX,1 ; Test lsb of AX to see if even or odd
pushf ; Save zero flag
shr AX,1 ; Divid by 2
add SI,AX ; Address the cluster
mov AX,[SI] ; AX = entry + part of another entry
popf ; Get bit test off the stack
jnz OddCluster ; If not zero then it's an odd cluster
EvenCluster:
cmp cx,0fffeh
jz GetEvenCluster ; Check for get or set
and AX,0f000h ; Zero out the value in 12 lsb
or AX,CX ; Set the new value
mov [SI],AX
jmp SHORT GetSetEntryExitzx ; AX = FAT entry
GetEvenCluster:
and AX,0fffh ; Mask off high 4 bits
jmp SHORT GetSetEntryExitzx ; AX = FAT entry
OddCluster:
cmp cx,0fffeh
jz GetOddCluster ; Check for get or set
.386
shl CX,4 ; Set the value Shift left 4 bits
.8086
and AX,0fh ; Zero out existing value in 12 msb
or AX,CX ; Insert new value
mov [SI],AX
jmp SHORT GetSetEntryExitzx ; AX = FAT entry
GetOddCluster:
.386
shr AX,4 ; Shift over 4 bits to get entry
GetSetEntryExitzx:
movzx eax,ax
.8086
GetSetEntryExit:
pop ES ; Restore regs for 'C' compatibility
pop DS
pop SI
ret
GetSetFatEntry ENDP
;===========================================================================
; Routine name: GetFatSectorEntry
;===========================================================================
;
; Description: Returns the contents of the specified fat
; entry from the FatSector buffer. It is assumed that the
; required sector is in the buffer already. If the entry is
; 12 bits and overlaps sectors, the next sector will be read
; into the FatSector buffer.
;
; Arguments: DS:SI --> Start of FatSector buffer
; entry_offset = offset of entry from start of buffer
; fBigFat fBig32Fat = flags for 12-16-32-bit FAT
; -------------------------------------------------------------------------
; Returns: EAX --> Contents of FAT entry
; -----------------------------------------------------
; Registers destroyed: EAX BX CX DX
; ---------------------------------
;
;===========================================================================
GetFatSectorEntry PROC near
assume DS:NOTHING,ES:DATA
push SI ; Save regs for 'C' compatibility
push DS
push ES
; mov BX,DATA ; ES = DATA
; mov ES,BX
call Seg_Adj ; Normalize the pointer
mov DX,ES:entry_offset
add SI,DX ; Add offset value in DX to SI
test ES:fBig32Fat,0ffh ; See if 32 bit FAT
jz short TestEntry16x ; If not do 12-16 bit FAT
.386
mov EAX,dword ptr [SI] ; Move the entry into AX
and EAX,00FFFFFFFh ; Discard high 4 bits
.8086
jmp SHORT GetFatSectorEntryExit ; EAX = FAT entry
TestEntry16x:
test ES:fBigFat,0ffh ; See if 16 bit FAT
jz Get12 ; If not do 12 bit FAT
Get16:
.386
movzx EAX,word ptr [SI] ; Move the entry into AX
.8086
jmp SHORT GetFatSectorEntryExit ; EAX = FAT entry
Get12:
ifdef NEC_98
;;; mov AX,DeviceParameters.DP_BPB.BPB_BytesPerSector
; Fix for B#7418
mov AX,ES:DeviceParameters.DP_BPB.oldBPB.BPB_BytesPerSector
dec AX
cmp DX,AX
else
cmp DX,1ffh ; Entry straddles two FAT sectors if
endif
jb Does_Not_Straddle ; offset is 511 (last byte in sector)
Does_Straddle:
call ReadInTwoParts ; Handle special case by reading in
jmp SHORT GetFatSectorEntryExit ; the next FAT sector also
Does_Not_Straddle:
.386
movzx EAX,word ptr [SI] ; EAX = entry + part of another entry
.8086
test ES:odd_entry,1
jnz IsOddCluster ; If not zero then it's an odd cluster
IsEvenCluster:
and AX,0fffh ; Mask off high 4 bits
jmp SHORT GetFatSectorEntryExit ; EAX = FAT entry
IsOddCluster:
.386
shr AX,4 ; Shift over 4 bits to get entry
.8086
GetFatSectorEntryExit:
pop ES ; Restore regs for 'C' compatibility
pop DS
pop SI
ret
GetFatSectorEntry ENDP
;===========================================================================
;
; ReadInTwoParts : This procedure will determine the value of a 12-bit
; FAT entry which is straddled across two consecutive
; sectors in the FAT. The value is found by reading
; the available part from the first sector, and then
; loading the next sector in the FatSector buffer and
; reading and combining the next part.
; Input : odd_entry
; sector_to_read
; DS:SI pointer to required entry location in buffer (this is
; offset 511 from start of FatSector buffer)
;
; Output: AX = entry value
; FatSector contains next FAT sector
;
; Registers destroyed : AX
;
; Strategy : The position of the entry is different for odd and even
; entries. The 3-byte layout for two consecutive 12-bit entries
; is shown below. (The byte |X1 X0| or |Y2 X2| is the last
; in the buffer).
;
; |X1 X0|Y0 X2|Y2 Y1|
;
; Even entries start at X1 (thus need to get X2X1X0).
; Odd entries start at Y0 (thus need to get Y2Y1Y0).
;
;===========================================================================
ReadInTwoParts proc near
assume DS:NOTHING,ES:DATA
push BX ; Save registers
push CX
push DX
push SI
xor AX,AX
mov AL,BYTE PTR [SI] ; read last byte of buffer
.386
mov EBX,ES:sector_to_read ; compute next sector to read
inc EBX
mov ES:sector_to_read,EBX
.8086
test ES:odd_entry,1
jnz OddEntry
EvenEntry:
push AX
call ReadFatSector ; read in the next FAT sector
pop AX
lds SI,ES:FatSector ; set DS:SI to start of buffer
xor BX,BX
mov BH,BYTE PTR [SI] ; read first byte of buffer
and BH,0fh ; mask out Y2
or AX,BX ; place X2 into position
jmp SHORT EndReadInTwo
OddEntry:
and AL,0f0h ; mask out X2
mov CL,4
shr AX,CL ; place Y0 into position
push AX
call ReadFatSector ; read in the next FAT sector
pop AX
lds SI,ES:FatSector ; set DS:SI to start of buffer
xor BX,BX
mov BL,BYTE PTR [SI] ; read first byte of buffer
mov CL,4
shl BX,CL
or AX,BX ; place Y2 Y1 into position
EndReadInTwo:
pop SI ; Restore registers
pop DX
pop CX
pop BX
ret
ReadInTwoParts endp
AllocInitBadClusBitmap proc near
assume DS:DATA,ES:NOTHING
push es
.386
cmp BadClusBitMap,0
jne short DoneOk
mov ebx, TotalClusters
add ebx,32 ; 32 not 31
shr ebx,5 ; Cnt of DWORDs
shl ebx,2 ; Cnt of BYTEs
add ebx,15
shr ebx,4 ; Cnt of paras
%out BUG limit on bad cluster bitmap
cmp ebx,00005000h ; Bitmap > 320k? (2,621,440 clusters)
ja short ErrExit_Alloc_BCB ; Yes
mov AH, Alloc
int 21h
jc short Exit_Alloc_BCB
mov WORD PTR BadClusBitMap+2,AX
mov es,ax
xor di,di
xor AX,AX
mov WORD PTR BadClusBitMap,AX
movzx ecx,bx ; Cnt of paras
shl ecx,3 ; 8 words per para
mov edx,ecx
NxtBlock:
cmp ecx,8000h ; More than 64k left?
jbe short ZotIt ; Nope, last one
mov ecx,00008000h ; Do at most 64k at a time
ZotIt:
mov ebx,ecx ; Save count
cld
rep stosw ; zero init bit array
sub edx,ebx ; Sub off amount done
jbe short DoneOk ; All done
mov ax,es
add ax,1000h ; Next 64k
mov es,ax
xor di,di
mov ecx,edx
jmp short NxtBlock
.8086
DoneOk:
clc
Exit_Alloc_BCB:
pop es
ret
ErrExit_Alloc_BCB:
stc
jmp short Exit_Alloc_BCB
AllocInitBadClusBitmap endp
;*** SetUpBadClusTransfer - Set things up to copy bad clusters from the current
; disk to the new disk.
;
; DS -> data
;
SetUpBadClusTransfer proc near
assume DS:DATA,ES:NOTHING
mov word ptr BadClusBitMap,0
mov word ptr BadClusBitMap+2,0
cmp FATNotAllInMem,0
.386
je InitFatWithBad
call DetermineExistingFormatNoMsg
cmc
jnc AllDone6 ; Drive isn't valid so no bad sectors
;
; Build a bit map of the bad clusters
;
message msgSetBadClus
.386
test fBig32FAT,0ffh ; See if 32 bit fat
jz short Tst16BitEntry ; If zero then 16 or 12 bit fat
mov EBX,00ffffff7h ; Set 32 bit value for bad cluster
jmp SHORT SetBdVal
Tst16BitEntry:
mov EBX,00000ff7h ; Set 12 bit value for bad cluster
test fBigFAT,0ffh ; See if 16 bit fat
jz short SetBdVal ; If zero then 12 bit fat
mov EBX,0000fff7h ; Set 16 bit value for bad cluster
SetBdVal:
mov CurrentCluster,2
mov BadClusValue,EBX
push ds
pop es ; following routines expect ES->DATA
ClusLoop:
call DisplayFatDonePcnt
mov EAX,CurrentCluster
call calc_sector_and_offset ; determine location of this entry
mov EBX,sector_in_buffer
mov EAX,sector_to_read
cmp EAX,EBX ; check if required sector is in buffer
je short DontHaveToRead2
call ReadFatSector ; read a sector of the FAT into buffer
jc short AllDoneDoCRLFErr
DontHaveToRead2:
mov EAX,CurrentCluster ; EAX = current cluster
xor CX,CX ; ECX = get cluster contents signal
push ds
lds SI,FatSector ; DS:SI --> FAT buffer
call GetFatSectorEntry ; EAX = contents of FAT entry
pop ds
cmp EAX,BadClusValue ; Is this cluster marked bad?
jne short NextClus ; If EAX<>EBX good cluster
call AllocInitBadClusBitmap ; Does nothing if BadClusBitMap
jc short AllDoneDoCRLFErr ; already exists
mov EAX,CurrentCluster ; EAX = this cluster number
push ds
lds si,BadClusBitMap
mov ebx,eax
shr ebx,19 ; (E)BX is "64k index" of this bit
and eax,00007FFFFh ; bit index in that 64k
mov cx,ds
add cx,bx ; Go to correct 64k piece
mov ds,cx
bts dword ptr [si],eax ; Set the bit
pop ds
NextClus:
inc CurrentCluster ; go to next cluster
mov EAX,CurrentCluster
mov EBX,TotalClusters
cmp EAX,EBX ; check for last cluster in FAT
.8086
jna ClusLoop
message msgSetBadClusDone
jmp short AllDone6Ok
AllDoneDoCRLFErr:
message msgCRLF
stc
jmp short AllDone6
InitFatWithBad:
call load_old_fat ; load old FAT
call mark_non_bad_as_free ; mark anything not BAD as FREE
AllDone6Ok:
clc
AllDone6:
ret
SetUpBadClusTransfer endp
;*** mark_non_bad_as_free -- mark all non-bad blocks as free in the FAT
;
; WARNING: Do not call this if FATNotAllInMem is TRUE
;
mark_non_bad_as_free proc near
assume ds:nothing, es:DATA
.386
test fBig32FAT,0ffh ; See if 32 bit fat
mov ebx,00ffffff7h ; get bad value for 32-bit
jnz short mark_non_bad_1
test fBigFAT,0ffh ; See if 16 bit fat
mov ebx,0fff7h ; get bad value for 16-bit
jnz short mark_non_bad_1
and bh,0fh ; get the 0ff7h for 12-bit
mark_non_bad_1:
mov eax,2 ; cluster number
mark_non_bad_2:
push ds
lds si,FatSpace ; point to the FAT buffer
mov ecx,0fffffffeh ; get the FAT entry
;; Manual to disable compile warning
;; push eax ; save cluster number
db 066h,050h
;; push ebx ; save bad value
db 066h,053h
call GetSetFatEntry ; read the cluster value
;; pop ebx
db 066h,05Bh
cmp eax,ebx ; is it bad?
jz short mark_non_bad_3 ; skip if so (leave alone)
;; pop eax ; get the cluster number
db 066h,058h
;; push eax
db 066h,050h
;; push ebx
db 066h,053h
xor ecx,ecx ; set FAT entry to FREE
call GetSetFatEntry
;; pop ebx
db 066h,05Bh
mark_non_bad_3:
;; pop eax ; restore cluster number
db 066h,058h
pop ds
inc eax
cmp eax,TotalClusters
jna mark_non_bad_2
.8086
ret
mark_non_bad_as_free endp
;*** load_old_fat -- loads the FAT from the existing media into FatSpace
;
; WARNING: Do not call this if FATNotAllInMem is TRUE
;
; If not properly formatted, will just terminate.
; If any error reading FAT, will zero entire FAT via Fat_Init.
load_old_fat proc near
mov SI,DATA
mov ES,SI ; Set ES to data segment
assume ES:DATA,DS:Nothing ; Assembler directive
push ds
call DetermineExistingFormatNoMsg
jc load_old_fat_error
call init_fat_with_header
mov CX, DeviceParameters.DP_BPB.oldBPB.BPB_SectorsPerFat
.errnz EDP_BPB NE DP_BPB
or cx, cx
jnz short GotFSz
;
; We KNOW the high word must be 0 since whole FAT fits in memory
;
mov cx, DeviceParameters.DP_BPB.BGBPB_BigSectorsPerFat
GotFSz:
mov DX,DeviceParameters.DP_BPB.oldBPB.BPB_ReservedSectors ;starting sector
mov AL,DriveToFormat
lds bx,FatSpace
load_old_fat_1:
jcxz load_old_fat_done ; exit when done
ifdef NEC_98
push cx ; save size to write
cmp cx,10h ; need more than 40h secs?
jb load_old_fat_2 ; skip if we can get it all
mov cx,10h ; clip to max
load_old_fat_2:
push AX ; save AX
call Read_Disk ; read it
pop AX ; restore AX
pop CX ; restore CX
jc load_old_fat_error ; exit if error
push ax
mov ax,ds
add ax,es:DeviceParameters.DP_BPB.BPB_BytesPerSector; adjust segment
mov ds,ax
pop ax
add DX,10h
sub CX,10h
ja load_old_fat_1 ; loop until done
else
push cx ; save size to write
cmp cx,40h ; need more than 40h secs?
jb load_old_fat_2 ; skip if we can get it all
mov cx,40h ; clip to max
load_old_fat_2:
push AX ; save AX
call Read_Disk ; read it
pop AX ; restore AX
pop CX ; restore CX
jc load_old_fat_error ; exit if error
push ax
mov ax,ds
add ax,40h*512/16 ; adjust segment
mov ds,ax
pop ax
add DX,40h
sub CX,40h
ja load_old_fat_1 ; loop until done
endif
load_old_fat_done:
pop ds
jmp short init_fat_header
load_old_fat_error:
pop ds
if2
.errnz (offset $) - (offset init_fat_with_header) ; fall thru!
endif
load_old_fat endp
;*** init_fat_with_header -- Do a Fat_Init, and then put bytes at front
;
; alternate entry point: init_fat_header (just fill in header)
init_fat_with_header proc near
assume ds:nothing, es:DATA
mov DI,DATA
mov ES,DI
call Fat_Init
init_fat_header:
push ES
push es
pop ds ; get ds -> DATA
assume ds:DATA
les DI, FatSpace ; ES:DI --> FatSpace buffer
assume es:nothing ; Store media descriptor byte
mov AL, DeviceParameters.DP_BPB.oldBPB.BPB_MediaDescriptor
.errnz EDP_BPB NE DP_BPB
ifdef NEC_98
cmp AL,Fixed_Disk ;FAT-ID twisted (for HD and 5"MO)
jne @F ; F8h --> FEh (Only FAT-ID)
mov AL,Single_8_Media ;it is NEC_98 local.
@@:
endif
mov AH, 0ffH
stosw ; Cluster 0 start
mov al,ah
stosb ; cluster 0 if 32 bit else cluster 1
test fBig32Fat, TRUE
jz Not32
stosb ; finish cluster 0 dword
stosw ; cluster 1 dword
and ah,0Fh ; High nibble reserved, zero it.
stosw
sub di,5 ; Go back to high byte of cluster 0
mov al,ah
stosb ; High nibble reserved, zero it.
jmp short NotBig
Not32:
test fBigFat, TRUE
jz NotBig
stosb ; finish cluster 1 word
NotBig:
pop ES
ret
init_fat_with_header endp
;**** IsThisClusterBad
;
; ENTRY: EAX is cluster #
;
; EXIT: Zero set if cluster bad
;
; USES: EBX,ECX,EDX,FLAGS
;
IsThisClusterBad proc near
assume ds:data, es:data
push ds
push si
.386
;; push eax
db 066h,050h
cmp FATNotAllInMem,0
je short ChkInMemFat
cmp BadClusBitMap,0 ; Anything marked?
je short NotBadRet ; Nope
lds si,BadClusBitMap
assume DS:nothing
mov ebx,eax
shr ebx,19 ; (E)BX is "64k index" of this bit
and eax,00007FFFFh ; bit index in that 64k
mov cx,ds
add cx,bx ; Go to correct 64k piece
mov ds,cx
bt dword ptr [si],eax ; Bad cluster?
jnc short NotBadRet ; No
mov bx,0FFFFh ; Set so following inc sets ZERO
NotBadRet:
inc bx
PopRet:
;; pop eax ; restore cluster number
db 066h,058h
.8086
pop si
pop ds
ret
ChkInMemFat:
.386
assume DS:data
test fBig32FAT,0ffh
jz short Tst16BitEnt
mov EBX,00ffffff7h
jmp SHORT SetBVal
Tst16BitEnt:
mov EBX,00000ff7h
test fBigFAT,0ffh
jz short SetBVal
mov EBX,0000fff7h
SetBVal:
mov BadClusValue,EBX
mov ECX,0fffffffeh ; ECX = Get value
lds SI,FatSpace
assume DS:nothing
call GetSetFatEntry
cmp eax,es:BadClusValue
.8086
jmp short PopRet
IsThisClusterBad endp
;**** WrtEOFMrkInRootClus
;
; ENTRY: None
;
; EXIT: EOF mark written into root directory staring cluster
;
; USES: EAX,EBX,ECX,EDX,FLAGS
;
WrtEOFMrkInRootClus proc near
assume ds:data, es:data
push ds
push si
;
; In FATNotAllInMem case FlushFatBuf will take care of putting
; an eof mark in the root clus
;
cmp FATNotAllInMem,0
jne SetDone
test fBig32FAT,0ffh ; See if 32 bit fat
jz SetDone ; No
.386
mov eax,dword ptr DeviceParameters.DP_BPB.BGBPB_RootDirStrtClus
.errnz EDP_BPB NE DP_BPB
cmp eax,2
jb short SetDone
cmp eax,TotalClusters
ja short SetDone
mov ECX,00fffffffh
.8086
lds SI,FatSpace
assume DS:nothing
call GetSetFatEntry
SetDone:
pop si
pop ds
ret
WrtEOFMrkInRootClus endp
CODE ENDS
END