mirror of https://github.com/microsoft/MS-DOS.git
876 lines
23 KiB
NASM
876 lines
23 KiB
NASM
|
TITLE RECOVER MS-DOS File/Disk Recovery Utility
|
|||
|
;----------------------------------------------------------
|
|||
|
;
|
|||
|
; Recover - Program to rebuild an ms.dos directory
|
|||
|
;
|
|||
|
; Copyright 1982 by Microsoft Corporation
|
|||
|
; Written by Chris Peters, April 1982
|
|||
|
;
|
|||
|
;-----------------------------------------------------------
|
|||
|
;
|
|||
|
;REV 1.5 added header message ARR
|
|||
|
;
|
|||
|
|
|||
|
FALSE EQU 0
|
|||
|
TRUE EQU NOT FALSE
|
|||
|
|
|||
|
|
|||
|
IBMVER EQU true
|
|||
|
KANJI EQU FALSE
|
|||
|
|
|||
|
bdos equ 21h
|
|||
|
boot equ 20h
|
|||
|
aread equ 25h
|
|||
|
awrite equ 26h
|
|||
|
|
|||
|
INCLUDE DOSSYM.ASM
|
|||
|
|
|||
|
;
|
|||
|
cr equ 0dh
|
|||
|
lf equ 0ah
|
|||
|
;
|
|||
|
fcb equ 5ch
|
|||
|
|
|||
|
code segment public
|
|||
|
code ends
|
|||
|
|
|||
|
const segment public byte
|
|||
|
const ends
|
|||
|
|
|||
|
data segment public byte
|
|||
|
data ends
|
|||
|
|
|||
|
|
|||
|
dg group code,const,data
|
|||
|
|
|||
|
code segment public
|
|||
|
assume cs:dg,ds:dg,es:dg,ss:dg
|
|||
|
|
|||
|
PUBLIC PCRLF,PRINT,INT_23,convert
|
|||
|
EXTRN dskwrt:NEAR,dskrd:NEAR,DSKERR:NEAR,report:NEAR
|
|||
|
|
|||
|
org 100h
|
|||
|
|
|||
|
recover:jmp rec_start
|
|||
|
|
|||
|
HEADER DB "Vers 1.50"
|
|||
|
|
|||
|
;-----------------------------------------------------------------------;
|
|||
|
hardch dd ?
|
|||
|
|
|||
|
the_root db 0 ;root directory flag
|
|||
|
|
|||
|
fudge db 0 ;directory changed flag
|
|||
|
user_drive db 0
|
|||
|
drive db 0
|
|||
|
|
|||
|
|
|||
|
dirchar db "/",0
|
|||
|
|
|||
|
|
|||
|
userdir db "/",0
|
|||
|
db (dirstrlen) dup(0)
|
|||
|
|
|||
|
fname_buffer db 128 dup(0)
|
|||
|
;-----------------------------------------------------------------------;
|
|||
|
|
|||
|
pcrlf: mov dx,offset dg: crlf
|
|||
|
print: mov ah,STD_CON_STRING_OUTPUT
|
|||
|
int bdos
|
|||
|
pret: ret
|
|||
|
;
|
|||
|
convert:push bx
|
|||
|
xor ax,ax
|
|||
|
mov bx,ax
|
|||
|
mov bp,ax
|
|||
|
mov cx,32
|
|||
|
convlp: shl si,1
|
|||
|
rcl di,1
|
|||
|
xchg ax,bp
|
|||
|
call convwrd
|
|||
|
xchg ax,bp
|
|||
|
xchg ax,bx
|
|||
|
call convwrd
|
|||
|
xchg ax,bx
|
|||
|
adc al,0
|
|||
|
loop convlp
|
|||
|
|
|||
|
mov cx,1810h
|
|||
|
xchg dx,ax
|
|||
|
call digit
|
|||
|
xchg ax,bx
|
|||
|
call outword
|
|||
|
mov ax,bp
|
|||
|
call outword
|
|||
|
pop dx
|
|||
|
call print
|
|||
|
ret2: ret
|
|||
|
;
|
|||
|
outword:push ax
|
|||
|
mov dl,ah
|
|||
|
call outbyte
|
|||
|
pop dx
|
|||
|
outbyte:mov dh,dl
|
|||
|
shr dl,1
|
|||
|
shr dl,1
|
|||
|
shr dl,1
|
|||
|
shr dl,1
|
|||
|
call digit
|
|||
|
mov dl,dh
|
|||
|
digit: and dl,0fh
|
|||
|
jz blankzer
|
|||
|
xor cl,cl
|
|||
|
blankzer:
|
|||
|
dec ch
|
|||
|
and cl,ch
|
|||
|
or dl,30h
|
|||
|
sub dl,cl
|
|||
|
cmp dl,30h
|
|||
|
jl ret2
|
|||
|
mov ah,STD_CON_OUTPUT
|
|||
|
int bdos
|
|||
|
ret
|
|||
|
;
|
|||
|
convwrd:adc al,al
|
|||
|
daa
|
|||
|
xchg al,ah
|
|||
|
adc al,al
|
|||
|
daa
|
|||
|
xchg al,ah
|
|||
|
ret
|
|||
|
;
|
|||
|
; bx = fat[ax]
|
|||
|
;
|
|||
|
getfat: mov bx,offset dg: fattbl
|
|||
|
push ax
|
|||
|
mov si,ax
|
|||
|
sar ax,1
|
|||
|
pushf
|
|||
|
add si,ax
|
|||
|
mov bx,word ptr [bx][si]
|
|||
|
popf
|
|||
|
jnc getfat1
|
|||
|
mov cl,4
|
|||
|
shr bx,cl
|
|||
|
getfat1:and bh,00001111b
|
|||
|
pop ax
|
|||
|
mov cx,secsiz
|
|||
|
ret
|
|||
|
;
|
|||
|
; fat[ax] = dx
|
|||
|
;
|
|||
|
setfat: mov bx,offset dg: fattbl
|
|||
|
push ax
|
|||
|
push dx
|
|||
|
mov si,ax
|
|||
|
sar ax,1
|
|||
|
pushf
|
|||
|
add si,ax
|
|||
|
mov ax,word ptr [bx][si]
|
|||
|
popf
|
|||
|
jnc setfat2
|
|||
|
and ax,000fh
|
|||
|
mov cl,4
|
|||
|
shl dx,cl
|
|||
|
setfat1:or ax,dx
|
|||
|
mov word ptr [bx][si],ax
|
|||
|
pop dx
|
|||
|
pop ax
|
|||
|
ret
|
|||
|
|
|||
|
setfat2:and ax,0f000h
|
|||
|
jmp setfat1
|
|||
|
|
|||
|
load: mov dx,firfat
|
|||
|
mov al,byte ptr fatnum
|
|||
|
mov byte ptr fatcnt,al
|
|||
|
mov al,byte ptr drive
|
|||
|
mov cx,fatsiz
|
|||
|
mov bx,offset dg: fattbl
|
|||
|
ret66: ret
|
|||
|
|
|||
|
readft: call load
|
|||
|
readit: call dskrd
|
|||
|
cmp [fndfat],0 ;save location of readable fat sector
|
|||
|
jnz fdfat
|
|||
|
mov [fndfat],dx
|
|||
|
fdfat: cmp word ptr [bx+1],-1
|
|||
|
jz ret66
|
|||
|
|
|||
|
add dx,cx ;try to read the other fats
|
|||
|
dec byte ptr fatcnt
|
|||
|
jnz readit
|
|||
|
|
|||
|
mov dx,[fndfat] ;see if any readable at all
|
|||
|
or dx,dx
|
|||
|
jz readft ;if not disk is blown, keep trying
|
|||
|
call dskrd
|
|||
|
ret
|
|||
|
|
|||
|
wrtfat: call load
|
|||
|
wrtit: push ax
|
|||
|
push bx
|
|||
|
push cx
|
|||
|
push dx
|
|||
|
call dskwrt
|
|||
|
pop dx
|
|||
|
pop cx
|
|||
|
pop bx
|
|||
|
pop ax
|
|||
|
|
|||
|
wrtok: add dx,cx
|
|||
|
dec byte ptr fatcnt
|
|||
|
jnz wrtit
|
|||
|
ret
|
|||
|
|
|||
|
printerr:
|
|||
|
call print
|
|||
|
jmp rabort
|
|||
|
|
|||
|
|
|||
|
rec_start:
|
|||
|
|
|||
|
;Code to print header
|
|||
|
; PUSH AX
|
|||
|
; MOV DX,OFFSET DG:HEADER
|
|||
|
; CALL print
|
|||
|
; POP AX
|
|||
|
|
|||
|
DOSVER_HIGH EQU 0200H ;2.00 in hex
|
|||
|
PUSH AX ;Save DRIVE validity info
|
|||
|
MOV AH,GET_VERSION
|
|||
|
INT 21H
|
|||
|
XCHG AH,AL ;Turn it around to AH.AL
|
|||
|
CMP AX,DOSVER_HIGH
|
|||
|
JAE OKDOS
|
|||
|
GOTBADDOS:
|
|||
|
MOV DX,OFFSET DG:BADVER
|
|||
|
CALL PRINT
|
|||
|
INT 20H
|
|||
|
|
|||
|
OKDOS: POP AX
|
|||
|
|
|||
|
cmp al,0ffH
|
|||
|
JZ BADDRVSPECJ
|
|||
|
mov si,80h
|
|||
|
lodsb
|
|||
|
or al,al
|
|||
|
jz noparm
|
|||
|
look: lodsb
|
|||
|
cmp al," "
|
|||
|
jz look
|
|||
|
cmp al,9
|
|||
|
jz look
|
|||
|
cmp al,13
|
|||
|
jnz gotparm
|
|||
|
noparm:
|
|||
|
jmp noname
|
|||
|
|
|||
|
BADDRVSPECJ: JMP BADDRVSPEC
|
|||
|
|
|||
|
gotparm:
|
|||
|
mov ah,DISK_RESET
|
|||
|
int bdos ;empty buffer queue
|
|||
|
|
|||
|
mov ah,get_default_drive ;save current drive
|
|||
|
int 21h
|
|||
|
mov [user_drive],al
|
|||
|
|
|||
|
mov bx,fcb ;determine input command
|
|||
|
mov al,[bx]
|
|||
|
dec al
|
|||
|
cmp al,-1
|
|||
|
jnz drvok1
|
|||
|
mov al,[user_drive]
|
|||
|
drvok1:
|
|||
|
mov [drive],al
|
|||
|
add [drvlet],al
|
|||
|
add [drvlet1],al
|
|||
|
mov dx,offset dg: askmsg
|
|||
|
call print
|
|||
|
mov ah,STD_CON_INPUT_FLUSH
|
|||
|
mov al,1 ;wait for a key
|
|||
|
int bdos
|
|||
|
|
|||
|
cmp al,17h
|
|||
|
jnz drvok2
|
|||
|
mov dx,offset dg: egomes
|
|||
|
jmp printerr
|
|||
|
egomes: db "Chris Peters helped with the new dos!",cr,lf
|
|||
|
db "Microsoft rules ok$"
|
|||
|
|
|||
|
drvok2:
|
|||
|
IF IBMVER
|
|||
|
MOV AL,DRIVE ;This is for ibm's single drive sys
|
|||
|
PUSH DS
|
|||
|
MOV BX,50H
|
|||
|
MOV DS,BX
|
|||
|
MOV DS:(BYTE PTR 4),AL ;Indicate drive changed
|
|||
|
POP DS
|
|||
|
ENDIF
|
|||
|
|
|||
|
;----- Process Pathnames -----------------------------------------------;
|
|||
|
mov ax,(char_oper shl 8) ;get switch character
|
|||
|
int 21h
|
|||
|
cmp dl,"/"
|
|||
|
jnz slashok ;if not / , then not PC
|
|||
|
mov [dirchar],"\" ;in PC, dir separator = \
|
|||
|
mov [userdir],"\"
|
|||
|
|
|||
|
slashok:
|
|||
|
mov si,81h ;point to cammand line
|
|||
|
mov di,offset dg: fname_buffer
|
|||
|
xor cx,cx ;zero pathname length
|
|||
|
|
|||
|
kill_bl:
|
|||
|
lodsb ;get rid of blanks
|
|||
|
cmp al,9
|
|||
|
je kill_bl
|
|||
|
cmp al,' '
|
|||
|
je kill_bl
|
|||
|
cmp al,13 ;A carriage return?
|
|||
|
jne next_char
|
|||
|
jmp noname ;yes, file name missing
|
|||
|
|
|||
|
next_char:
|
|||
|
stosb ;put patname in buffer
|
|||
|
inc cx
|
|||
|
lodsb
|
|||
|
cmp al,' '
|
|||
|
je name_copied
|
|||
|
cmp al,9
|
|||
|
je name_copied
|
|||
|
cmp al,13 ; a CR ?
|
|||
|
jne next_char
|
|||
|
|
|||
|
name_copied:
|
|||
|
mov byte ptr [di],0 ;nul terminate the pathname
|
|||
|
dec di ;adjust to the end of the pathname
|
|||
|
|
|||
|
;----- Scan for directory ----------------------------------------------;
|
|||
|
|
|||
|
IF KANJI
|
|||
|
mov dx,offset dg: [fname_buffer]
|
|||
|
PUSH DX
|
|||
|
PUSH DI
|
|||
|
MOV BX,DI
|
|||
|
MOV DI,DX
|
|||
|
DELLOOP:
|
|||
|
CMP DI,BX
|
|||
|
JZ GOTDELE
|
|||
|
MOV AL,[DI]
|
|||
|
INC DI
|
|||
|
CALL TESTKANJ
|
|||
|
JZ NOTKANJ11
|
|||
|
INC DI
|
|||
|
JMP DELLOOP
|
|||
|
|
|||
|
NOTKANJ11:
|
|||
|
cmp al,[dirchar]
|
|||
|
JNZ DELLOOP
|
|||
|
MOV DX,DI ;Point to char after '/'
|
|||
|
DEC DX
|
|||
|
DEC DX ;Point to char before '/'
|
|||
|
JMP DELLOOP
|
|||
|
|
|||
|
GOTDELE:
|
|||
|
MOV DI,DX
|
|||
|
POP AX ;Initial DI
|
|||
|
POP DX
|
|||
|
SUB AX,DI ;Distance moved
|
|||
|
SUB CX,AX ;Set correct CX
|
|||
|
CMP DX,DI
|
|||
|
JB sja ;Found a pathsep
|
|||
|
JA sjb ;Started with a pathsep, root
|
|||
|
MOV AX,[DI]
|
|||
|
CALL TESTKANJ
|
|||
|
JNZ same_dirj
|
|||
|
XCHG AH,AL
|
|||
|
cmp al,[dirchar]
|
|||
|
jz sja ;One character directory
|
|||
|
same_dirj:
|
|||
|
ELSE
|
|||
|
mov al,[dirchar] ;get directory separator character
|
|||
|
std ;scan backwards
|
|||
|
repnz scasb ;(cx has the pathname length)
|
|||
|
cld ;reset direction, just in case
|
|||
|
jz sja
|
|||
|
ENDIF
|
|||
|
|
|||
|
jmp same_dir ;no dir separator char. found, the
|
|||
|
; file is in the current directory
|
|||
|
; of the corresponding drive. Ergo,
|
|||
|
; the FCB contains the data already.
|
|||
|
|
|||
|
sja:
|
|||
|
jcxz sjb ;no more chars left, it refers to root
|
|||
|
cmp byte ptr [di],':' ;is the prvious character a disk def?
|
|||
|
jne not_root
|
|||
|
sjb:
|
|||
|
mov [the_root],01h ;file is in the root
|
|||
|
not_root:
|
|||
|
inc di ;point to dir separator char.
|
|||
|
mov al,0
|
|||
|
stosb ;nul terminate directory name
|
|||
|
pop ax
|
|||
|
push di ;save pointer to file name
|
|||
|
mov [fudge],01h ;remember that the current directory
|
|||
|
; has been changed.
|
|||
|
|
|||
|
;----- Save current directory for exit ---------------------------------;
|
|||
|
mov dl,byte ptr ds:[fcb] ;get specified drive if any
|
|||
|
or dl,dl ;default disk?
|
|||
|
jz same_drive
|
|||
|
dec dl ;adjust to real drive (a=0,b=1,...)
|
|||
|
mov ah,set_default_drive ;change disks
|
|||
|
int 21h
|
|||
|
cmp al,-1 ;error?
|
|||
|
jne same_drive
|
|||
|
BADDRVSPEC:
|
|||
|
mov dx,offset dg: baddrv
|
|||
|
jmp printerr
|
|||
|
|
|||
|
same_drive:
|
|||
|
mov ah,get_default_dpb
|
|||
|
int 21h
|
|||
|
|
|||
|
assume ds:nothing
|
|||
|
|
|||
|
cmp al,-1 ;bad drive? (should always be ok)
|
|||
|
jne drvisok
|
|||
|
mov dx,offset dg: baddrv
|
|||
|
jmp printerr
|
|||
|
|
|||
|
drvisok:
|
|||
|
cmp [bx.dpb_current_dir],0
|
|||
|
je curr_is_root
|
|||
|
mov si,bx
|
|||
|
add si,dpb_dir_text
|
|||
|
mov di,offset dg: userdir + 1
|
|||
|
|
|||
|
dir_save_loop:
|
|||
|
lodsb
|
|||
|
stosb
|
|||
|
or al,al
|
|||
|
jnz dir_save_loop
|
|||
|
|
|||
|
curr_is_root:
|
|||
|
push cs
|
|||
|
pop ds
|
|||
|
|
|||
|
assume ds:dg
|
|||
|
|
|||
|
|
|||
|
;----- Change directories ----------------------------------------------;
|
|||
|
cmp [the_root],01h
|
|||
|
mov dx,offset dg: [dirchar] ;assume the root
|
|||
|
je sj1
|
|||
|
mov dx,offset dg: [fname_buffer]
|
|||
|
sj1:
|
|||
|
mov ah,chdir ;change directory
|
|||
|
int 21h
|
|||
|
mov dx,offset dg: baddrv
|
|||
|
jnc no_errors
|
|||
|
jmp printerr
|
|||
|
no_errors:
|
|||
|
|
|||
|
;----- Set Up int 24 intercept -----------------------------------------;
|
|||
|
|
|||
|
mov ax,(get_interrupt_vector shl 8) or 24h
|
|||
|
int 21h
|
|||
|
mov word ptr [hardch],bx
|
|||
|
mov word ptr [hardch+2],es
|
|||
|
mov ax,(set_interrupt_vector shl 8) or 23h
|
|||
|
mov dx,offset dg: int_23
|
|||
|
int 21h
|
|||
|
mov ax,(set_interrupt_vector shl 8) or 24h
|
|||
|
mov dx,offset dg: int_24
|
|||
|
int 21h
|
|||
|
push cs
|
|||
|
pop es
|
|||
|
|
|||
|
;----- Parse filename to FCB -------------------------------------------;
|
|||
|
pop si
|
|||
|
mov di,fcb
|
|||
|
mov ax,(parse_file_descriptor shl 8) or 1
|
|||
|
int 21h
|
|||
|
push ax
|
|||
|
;-----------------------------------------------------------------------;
|
|||
|
same_dir:
|
|||
|
pop ax
|
|||
|
|
|||
|
mov bx,fcb
|
|||
|
cmp byte ptr [bx+1],' ' ;must specify file name
|
|||
|
jnz drvok
|
|||
|
cmp byte ptr [bx],0 ;or drive specifier
|
|||
|
jnz drvok
|
|||
|
noname: mov dx,offset dg: drverr
|
|||
|
call print
|
|||
|
jmp int_23
|
|||
|
|
|||
|
drvok: push ds
|
|||
|
mov dl,drive
|
|||
|
inc dl
|
|||
|
mov ah,GET_DPB
|
|||
|
int bdos
|
|||
|
mov ax,word ptr [bx+2] ;get physical sector size
|
|||
|
mov cl,byte ptr [bx+4] ;get sectors/cluster - 1
|
|||
|
xor ch,ch
|
|||
|
inc cx
|
|||
|
mov cs:secall,cx ;save sectors per cluster
|
|||
|
mul cx ;ax = bytes per cluster
|
|||
|
mov bp,word ptr [bx+11] ;get record of first sector
|
|||
|
mov dx,word ptr [bx+16] ;get record of first directory entry
|
|||
|
mov si,word ptr [bx+6] ;get record of first fat
|
|||
|
mov cl,byte ptr [bx+15] ;get size of fat
|
|||
|
mov di,word ptr [bx+13] ;get number of clusters
|
|||
|
mov ch,byte ptr [bx+8] ;get number of fats on drive
|
|||
|
mov bx,word ptr [bx+9] ;get max number of dir entries
|
|||
|
pop ds
|
|||
|
|
|||
|
mov maxent,bx
|
|||
|
mov firfat,si
|
|||
|
mov firrec,bp
|
|||
|
mov firdir,dx
|
|||
|
mov byte ptr fatsiz,cl
|
|||
|
mov lastfat,di ;number of fat entries
|
|||
|
mov byte ptr fatnum,ch ;save number of fats on disk
|
|||
|
|
|||
|
mov secsiz,ax
|
|||
|
|
|||
|
mov di,table ;di points into constructed directory
|
|||
|
mov ax,0e5e5h ;deleted file magic number
|
|||
|
shl bx,1 ;16 words in a dir entry
|
|||
|
shl bx,1
|
|||
|
shl bx,1
|
|||
|
shl bx,1
|
|||
|
mov cx,bx
|
|||
|
rep stosw
|
|||
|
|
|||
|
call readft
|
|||
|
mov bx,fcb
|
|||
|
cmp byte ptr [bx+1],' '
|
|||
|
jz recdsk
|
|||
|
jmp recfil
|
|||
|
|
|||
|
recdsk: mov di,table
|
|||
|
mov fatptr,2
|
|||
|
mov ax,fatptr
|
|||
|
step1: call getfat
|
|||
|
cmp bx,0fffh
|
|||
|
jz step1a
|
|||
|
jmp step6
|
|||
|
step1a: mov filsiz,0
|
|||
|
mov word ptr filsiz+2,0
|
|||
|
mov dx,lastfat
|
|||
|
mov target,ax
|
|||
|
step2: mov ax,2
|
|||
|
add filsiz,cx
|
|||
|
adc word ptr filsiz+2,0
|
|||
|
step3: call getfat
|
|||
|
cmp bx,target
|
|||
|
jne step4
|
|||
|
mov target,ax
|
|||
|
jmp step2
|
|||
|
step4: inc ax
|
|||
|
cmp ax,dx
|
|||
|
jle step3
|
|||
|
;
|
|||
|
; at this point target = head of list, filsiz = file size
|
|||
|
;
|
|||
|
inc filcnt ;increment file count
|
|||
|
mov ax,maxent
|
|||
|
cmp filcnt,ax ;compare with max number of entries
|
|||
|
ja direrr
|
|||
|
|
|||
|
mov si,(offset dg: dirent)+7
|
|||
|
nam0: inc byte ptr [si] ;increment file name
|
|||
|
cmp byte ptr [si],'9'
|
|||
|
jle nam1
|
|||
|
mov byte ptr [si],'0'
|
|||
|
dec si
|
|||
|
jmp nam0
|
|||
|
|
|||
|
nam1: mov ah,GET_DATE
|
|||
|
int bdos ;set the date
|
|||
|
sub cx,1980
|
|||
|
add dh,dh
|
|||
|
add dh,dh
|
|||
|
add dh,dh
|
|||
|
add dh,dh
|
|||
|
add dh,dh
|
|||
|
rcl cl,1
|
|||
|
or dh,dl
|
|||
|
mov byte ptr dirent+24,dh
|
|||
|
mov byte ptr dirent+25,cl
|
|||
|
mov ah,GET_TIME
|
|||
|
int bdos ;set the time
|
|||
|
shr dh,1
|
|||
|
add cl,cl
|
|||
|
add cl,cl
|
|||
|
add cl,cl
|
|||
|
rcl ch,1
|
|||
|
add cl,cl
|
|||
|
rcl ch,1
|
|||
|
add cl,cl
|
|||
|
rcl ch,1
|
|||
|
or dh,cl
|
|||
|
mov byte ptr dirent+22,dh
|
|||
|
mov byte ptr dirent+23,ch
|
|||
|
|
|||
|
mov ax,filsiz ;set file size
|
|||
|
mov word ptr dirent+28,ax
|
|||
|
mov ax,word ptr filsiz+2
|
|||
|
mov word ptr dirent+30,ax
|
|||
|
mov ax,target ;set first cluster location
|
|||
|
mov word ptr dirent+26,ax
|
|||
|
|
|||
|
mov si,offset dg: dirent ;copy in new dir entry
|
|||
|
mov cx,32
|
|||
|
rep movsb
|
|||
|
|
|||
|
step6: inc fatptr ;keep looking for eof's
|
|||
|
mov ax,fatptr
|
|||
|
cmp ax,lastfat
|
|||
|
jg step7
|
|||
|
jmp step1
|
|||
|
|
|||
|
direrr: dec filcnt
|
|||
|
mov dx,offset dg: dirmsg
|
|||
|
call print
|
|||
|
|
|||
|
step7:
|
|||
|
mov al,drive
|
|||
|
mov dx,firdir ;write out constructed directory
|
|||
|
mov cx,firrec
|
|||
|
sub cx,dx
|
|||
|
mov bx,table
|
|||
|
call dskwrt
|
|||
|
call pcrlf
|
|||
|
mov dx,offset dg: recmsg_pre
|
|||
|
call print
|
|||
|
mov bx,offset dg: recmsg_post
|
|||
|
mov si,filcnt
|
|||
|
xor di,di ;output number of files created
|
|||
|
call convert
|
|||
|
jmp rexit
|
|||
|
recfil: mov dx,fcb
|
|||
|
mov ah,FCB_OPEN
|
|||
|
int bdos
|
|||
|
inc al
|
|||
|
jnz recfil0
|
|||
|
mov dx,offset dg: opnerr
|
|||
|
call print
|
|||
|
jmp rexit
|
|||
|
|
|||
|
recfil0:mov lastfat,1 ;indicate location of list head
|
|||
|
mov di,fcb
|
|||
|
mov ax,[di+16] ;get file size
|
|||
|
mov filsiz,ax
|
|||
|
mov siztmp,ax
|
|||
|
mov ax,[di+18]
|
|||
|
mov filsiz+2,ax
|
|||
|
mov siztmp+2,ax
|
|||
|
mov ax,[di+25] ;get list head
|
|||
|
or ax,ax
|
|||
|
mov fatptr,ax
|
|||
|
jnz recfil1
|
|||
|
recvec: jmp recfil6
|
|||
|
|
|||
|
recfil1:cmp fatptr,0fffh
|
|||
|
jz recvec ;terminate loop at e-o-f
|
|||
|
|
|||
|
mov cx,secall
|
|||
|
mov ax,fatptr
|
|||
|
dec ax
|
|||
|
dec ax
|
|||
|
mul cx
|
|||
|
add ax,firrec
|
|||
|
mov dx,ax
|
|||
|
mov bx,table
|
|||
|
mov al,drive
|
|||
|
int aread
|
|||
|
pop di ;restore stack pointer
|
|||
|
mov di,fcb ;restore pointer to fcb
|
|||
|
jnc recfil4 ;if no error continue reading
|
|||
|
|
|||
|
mov ax,fatptr
|
|||
|
call getfat
|
|||
|
cmp lastfat,1
|
|||
|
jnz recfil2
|
|||
|
|
|||
|
cmp bx,0fffh
|
|||
|
jnz noteof
|
|||
|
xor bx,bx
|
|||
|
noteof: mov word ptr [di+25],bx
|
|||
|
jmp recfil3
|
|||
|
|
|||
|
recfil2:mov dx,bx ;jump around bad sector
|
|||
|
mov ax,lastfat
|
|||
|
call setfat
|
|||
|
|
|||
|
recfil3:mov ax,fatptr ;mark sector bad
|
|||
|
mov dx,0ff7h
|
|||
|
call setfat
|
|||
|
mov ax,secsiz ;prepare to dec filsiz by secsiz
|
|||
|
cmp siztmp+2,0
|
|||
|
jnz recfilx
|
|||
|
cmp siztmp,ax
|
|||
|
ja recfilx
|
|||
|
mov ax,siztmp
|
|||
|
|
|||
|
recfilx:sub word ptr [di+16],ax
|
|||
|
sbb word ptr [di+18],0
|
|||
|
sub siztmp,ax
|
|||
|
sbb siztmp,0
|
|||
|
|
|||
|
and byte ptr [di+24],10111111b ;mark file dirty
|
|||
|
|
|||
|
mov ax,lastfat ;point to next sector to check
|
|||
|
jmp recfil5
|
|||
|
|
|||
|
recfil4:
|
|||
|
mov ax,secsiz ;set bytes remaining to be read
|
|||
|
sub siztmp,ax
|
|||
|
sbb siztmp+2,0
|
|||
|
jnc recok
|
|||
|
xor ax,ax ;if < 0, then set to zero
|
|||
|
mov siztmp,ax
|
|||
|
mov siztmp+2,ax
|
|||
|
|
|||
|
recok: mov ax,fatptr ;get next sector to test
|
|||
|
mov lastfat,ax
|
|||
|
recfil5:call getfat
|
|||
|
mov fatptr,bx
|
|||
|
jmp recfil1
|
|||
|
|
|||
|
recfil6: ;all done
|
|||
|
mov dx,fcb
|
|||
|
mov ah,FCB_CLOSE
|
|||
|
int bdos ;close the file
|
|||
|
call pcrlf
|
|||
|
call report
|
|||
|
|
|||
|
;
|
|||
|
rexit: mov ah,DISK_RESET
|
|||
|
int bdos
|
|||
|
call wrtfat ;save the fat
|
|||
|
int_23: call rest_dir
|
|||
|
rabort: int boot ;home, james...
|
|||
|
|
|||
|
;----- Restore INT 24 vector and old current directory -----------------;
|
|||
|
rest_dir:
|
|||
|
cmp [fudge],0
|
|||
|
je no_fudge
|
|||
|
|
|||
|
mov ax,(set_interrupt_vector shl 8) or 24h
|
|||
|
lds dx,[hardch]
|
|||
|
int 21h
|
|||
|
push cs
|
|||
|
pop ds
|
|||
|
|
|||
|
mov dx,offset dg: userdir ;restore directory
|
|||
|
mov ah,chdir
|
|||
|
int 21h
|
|||
|
mov dl,[user_drive] ;restore old current drive
|
|||
|
mov ah,set_default_drive
|
|||
|
int 21h
|
|||
|
|
|||
|
no_fudge:
|
|||
|
ret
|
|||
|
|
|||
|
;----- INT 24 Processing -----------------------------------------------;
|
|||
|
|
|||
|
int_24_retaddr dw int_24_back
|
|||
|
|
|||
|
int_24 proc far
|
|||
|
assume ds:nothing,es:nothing,ss:nothing
|
|||
|
|
|||
|
pushf
|
|||
|
push cs
|
|||
|
push [int_24_retaddr]
|
|||
|
push word ptr [hardch+2]
|
|||
|
push word ptr [hardch]
|
|||
|
ret
|
|||
|
int_24 endp
|
|||
|
|
|||
|
int_24_back:
|
|||
|
cmp al,2 ;abort?
|
|||
|
jnz ireti
|
|||
|
push cs
|
|||
|
pop ds
|
|||
|
|
|||
|
assume ds:dg
|
|||
|
|
|||
|
call rest_dir
|
|||
|
int 20h
|
|||
|
ireti:
|
|||
|
iret
|
|||
|
|
|||
|
IF KANJI
|
|||
|
TESTKANJ:
|
|||
|
CMP AL,81H
|
|||
|
JB NOTLEAD
|
|||
|
CMP AL,9FH
|
|||
|
JBE ISLEAD
|
|||
|
CMP AL,0E0H
|
|||
|
JB NOTLEAD
|
|||
|
CMP AL,0FCH
|
|||
|
JBE ISLEAD
|
|||
|
NOTLEAD:
|
|||
|
PUSH AX
|
|||
|
XOR AX,AX ;Set zero
|
|||
|
POP AX
|
|||
|
RET
|
|||
|
|
|||
|
ISLEAD:
|
|||
|
PUSH AX
|
|||
|
XOR AX,AX ;Set zero
|
|||
|
INC AX ;Reset zero
|
|||
|
POP AX
|
|||
|
RET
|
|||
|
ENDIF
|
|||
|
|
|||
|
code ends
|
|||
|
|
|||
|
const segment public byte
|
|||
|
|
|||
|
EXTRN BADVER:BYTE,askmsg:BYTE,drvlet:BYTE,dirmsg:BYTE
|
|||
|
EXTRN recmsg_pre:BYTE,DRVLET1:BYTE,recmsg_post:BYTE
|
|||
|
EXTRN crlf:BYTE,drverr:BYTE,baddrv:BYTE,opnerr:BYTE
|
|||
|
|
|||
|
const ends
|
|||
|
|
|||
|
data segment byte
|
|||
|
|
|||
|
PUBLIC filsiz
|
|||
|
|
|||
|
dirent db 'FILE0000REC'
|
|||
|
db 21 dup (00)
|
|||
|
|
|||
|
fndfat dw 0000 ;sector of first good fat
|
|||
|
filcnt dw 0000
|
|||
|
fatcnt db 00
|
|||
|
fatnum db 00
|
|||
|
fatsiz dw 0000
|
|||
|
firfat dw 0000
|
|||
|
fatptr dw 0000
|
|||
|
secall dw 0000 ;sectors per cluster
|
|||
|
target dw 0000
|
|||
|
maxent dw 0000
|
|||
|
firrec dw 0000
|
|||
|
firdir dw 0000
|
|||
|
secsiz dw 0000
|
|||
|
siztmp dw 0000
|
|||
|
dw 0000
|
|||
|
filsiz dw 0000
|
|||
|
dw 0000
|
|||
|
lastfat dw 0000
|
|||
|
;
|
|||
|
table dw offset dg:fattbl + 6 * 1024
|
|||
|
fattbl db 0
|
|||
|
|
|||
|
data ends
|
|||
|
|
|||
|
end recover
|
|||
|
|