ordos/msdos.asm

4031 lines
112 KiB
NASM

; 86-DOS High-performance operating system for the 8086 version 1.25
; by Tim Paterson
; ****************** Revision History *************************
; >> EVERY change must noted below!! <<
;
; 0.34 12/29/80 General release, updating all past customers
; 0.42 02/25/81 32-byte directory entries added
; 0.56 03/23/81 Variable record and sector sizes
; 0.60 03/27/81 Ctrl-C exit changes, including register save on user stack
; 0.74 04/15/81 Recognize I/O devices with file names
; 0.75 04/17/81 Improve and correct buffer handling
; 0.76 04/23/81 Correct directory size when not 2^N entries
; 0.80 04/27/81 Add console input without echo, Functions 7 & 8
; 1.00 04/28/81 Renumber for general release
; 1.01 05/12/81 Fix bug in `STORE'
; 1.10 07/21/81 Fatal error trapping, NUL device, hidden files, date & time,
; RENAME fix, general cleanup
; 1.11 09/03/81 Don't set CURRENT BLOCK to 0 on open; fix SET FILE SIZE
; 1.12 10/09/81 Zero high half of CURRENT BLOCK after all (CP/M programs don't)
; 1.13 10/29/81 Fix classic "no write-through" error in buffer handling
; 1.20 12/31/81 Add time to FCB; separate FAT from DPT; Kill SMALLDIR;
; Add FLUSH and MAPDEV calls; allow disk mapping in DSKCHG;
; Lots of smaller improvements
; 1.21 01/06/82 HIGHMEM switch to run DOS in high memory
; 1.22 01/12/82 Add VERIFY system call to enable/disable verify after write
; 1.23 02/11/82 Add defaulting to parser; use variable escape character
; Don't zero extent field in IBM version (back to 1.01!)
; 1.24 03/01/82 Restore fcn. 27 to 1.0 level; add fcn. 28
; 1.25 03/03/82 Put marker (00) at end of directory to speed searches
;
; *************************************************************
; Interrupt Entry Points:
; INTBASE: ABORT
; INTBASE+4: COMMAND
; INTBASE+8: BASE EXIT ADDRESS
; INTBASE+C: CONTROL-C ABORT
; INTBASE+10H: FATAL ERROR ABORT
; INTBASE+14H: BIOS DISK READ
; INTBASE+18H: BIOS DISK WRITE
; INTBASE+40H: Long jump to CALL entry point
IF IBM
ESCCH EQU 0
CANCEL EQU 1BH ;Cancel with ESC
TOGLINS EQU TRUE ;One key toggles insert mode
TOGLPRN EQU TRUE ;One key toggles printer echo
NUMDEV EQU 6 ;Include "COM1" as I/O device name
ZEROEXT EQU TRUE
ELSE
ESCCH EQU 1BH
CANCEL EQU "X"-"@" ;Cancel with Ctrl-X
TOGLINS EQU FALSE ;Separate keys for insert mode on and off
TOGLPRN EQU FALSE ;Separate keys for printer echo on and off
NUMDEV EQU 5 ;Number of I/O device names
ZEROEXT EQU FALSE
ENDIF
MAXCALL EQU 36
MAXCOM EQU 46
INTBASE EQU 80H
INTTAB EQU 20H
ENTRYPOINTSEG EQU 0CH
ENTRYPOINT EQU INTBASE+40H
CONTC EQU INTTAB+3
EXIT EQU INTBASE+8
LONGJUMP EQU 0EAH
LONGCALL EQU 9AH
MAXDIF EQU 0FFFH
SAVEXIT EQU 10
; Field definition for FCBs
FCBLOCK STRUC
DB 12 DUP (?) ;Drive code and name
EXTENT DW ?
RECSIZ DW ? ;Size of record (user settable)
FILSIZ DW ? ;Size of file in bytes
DRVBP DW ? ;BP for SEARCH FIRST and SEARCH NEXT
FDATE DW ? ;Date of last writing
FTIME DW ? ;Time of last writing
DEVID DB ? ;Device ID number, bits 0-5
;bit 7=0 for file, bit 7=1 for I/O device
;If file, bit 6=0 if dirty
;If I/O device, bit 6=0 if EOF (input)
FIRCLUS DW ? ;First cluster of file
LSTCLUS DW ? ;Last cluster accessed
CLUSPOS DW ? ;Position of last cluster accessed
DB ? ;Forces NR to offset 32
NR DB ? ;Next record
RR DB 3 DUP (?) ;Random record
FCBLOCK ENDS
FILDIRENT = FILSIZ ;Used only by SEARCH FIRST and SEARCH NEXT
; Description of 32-byte directory entry (same as returned by SEARCH FIRST
; and SEARCH NEXT, functions 17 and 18).
;
; Location bytes Description
;
; 0 11 File name and extension ( 0E5H if empty)
; 11 1 Attributes. Bits 1 or 2 make file hidden
; 12 10 Zero field (for expansion)
; 22 2 Time. Bits 0-4=seconds/2, bits 5-10=minute, 11-15=hour
; 24 2 Date. Bits 0-4=day, bits 5-8=month, bits 9-15=year-1980
; 26 2 First allocation unit ( < 4080 )
; 28 4 File size, in bytes (LSB first, 30 bits max.)
;
; The File Allocation Table uses a 12-bit entry for each allocation unit on
; the disk. These entries are packed, two for every three bytes. The contents
; of entry number N is found by 1) multiplying N by 1.5; 2) adding the result
; to the base address of the Allocation Table; 3) fetching the 16-bit word at
; this address; 4) If N was odd (so that N*1.5 was not an integer), shift the
; word right four bits; 5) mask to 12 bits (AND with 0FFF hex). Entry number
; zero is used as an end-of-file trap in the OS and as a flag for directory
; entry size (if SMALLDIR selected). Entry 1 is reserved for future use. The
; first available allocation unit is assigned entry number two, and even
; though it is the first, is called cluster 2. Entries greater than 0FF8H are
; end of file marks; entries of zero are unallocated. Otherwise, the contents
; of a FAT entry is the number of the next cluster in the file.
; Field definition for Drive Parameter Block
DPBLOCK STRUC
DEVNUM DB ? ;I/O driver number
DRVNUM DB ? ;Physical Unit number
SECSIZ DW ? ;Size of physical sector in bytes
CLUSMSK DB ? ;Sectors/cluster - 1
CLUSSHFT DB ? ;Log2 of sectors/cluster
FIRFAT DW ? ;Starting record of FATs
FATCNT DB ? ;Number of FATs for this drive
MAXENT DW ? ;Number of directory entries
FIRREC DW ? ;First sector of first cluster
MAXCLUS DW ? ;Number of clusters on drive + 1
FATSIZ DB ? ;Number of records occupied by FAT
FIRDIR DW ? ;Starting record of directory
FAT DW ? ;Pointer to start of FAT
DPBLOCK ENDS
DPBSIZ EQU 20 ;Size of the structure in bytes
DIRSEC = FIRREC ;Number of dir. sectors (init temporary)
DSKSIZ = MAXCLUS ;Size of disk (temp used during init only)
;The following are all of the segments used
;They are declared in the order that they should be placed in the executable
CODE SEGMENT
CODE ENDS
CONSTANTS SEGMENT BYTE
CONSTANTS ENDS
DATA SEGMENT WORD
DATA ENDS
DOSGROUP GROUP CODE,CONSTANTS,DATA
SEGBIOS SEGMENT
SEGBIOS ENDS
; BOIS entry point definitions
IF IBM
BIOSSEG EQU 60H
ENDIF
IF NOT IBM
BIOSSEG EQU 40H
ENDIF
SEGBIOS SEGMENT AT BIOSSEG
ORG 0
DB 3 DUP (?) ;Reserve room for jump to init code
BIOSSTAT DB 3 DUP (?) ;Console input status check
BIOSIN DB 3 DUP (?) ;Get console character
BIOSOUT DB 3 DUP (?) ;Output console character
BIOSPRINT DB 3 DUP (?) ;Output to printer
BIOSAUXIN DB 3 DUP (?) ;Get byte from auxilliary
BIOSAUXOUT DB 3 DUP (?) ;Output byte to auxilliary
BIOSREAD DB 3 DUP (?) ;Disk read
BIOSWRITE DB 3 DUP (?) ;Disk write
BIOSDSKCHG DB 3 DUP (?) ;Dsik-change status
BIOSSETDATE DB 3 DUP (?) ;Set date
BIOSSETTIME DB 3 DUP (?) ;Set time
BIOSGETTIME DB 3 DUP (?) ;Get time and date
BIOSFLUSH DB 3 DUP (?) ;Clear console input buffer
BIOSMAPDEV DB 3 DUP (?) ;Dynamic disk table mapper
SEGBIOS ENDS
; Location of user registers relative user stack pointer
STKPTRS STRUC
AXSAVE DW ?
BXSAVE DW ?
CXSAVE DW ?
DXSAVE DW ?
SISAVE DW ?
DISAVE DW ?
BPSAVE DW ?
DSSAVE DW ?
ESSAVE DW ?
IPSAVE DW ?
CSSAVE DW ?
FSAVE DW ?
STKPTRS ENDS
; Start of code
CODE SEGMENT
ASSUME CS:DOSGROUP,DS:DOSGROUP,ES:DOSGROUP,SS:DOSGROUP
ORG 0
CODSTRT EQU $
JMP DOSINIT
ESCCHAR DB ESCCH ;Lead-in character for escape sequences
ESCTAB:
IF NOT IBM
DB "S" ;Copy one char
DB "V" ;Skip one char
DB "T" ;Copy to char
DB "W" ;Skip to char
DB "U" ;Copy line
DB "E" ;Kill line (no change in template)
DB "J" ;Reedit line (new template)
DB "D" ;Backspace
DB "P" ;Enter insert mode
DB "Q" ;Exit insert mode
DB "R" ;Escape character
DB "R" ;End of table
ENDIF
IF IBM
DB 64 ;Crtl-Z - F6
DB 77 ;Copy one char - -->
DB 59 ;Copy one char - F1
DB 83 ;Skip one char - DEL
DB 60 ;Copy to char - F2
DB 62 ;Skip to char - F4
DB 61 ;Copy line - F3
DB 61 ;Kill line (no change to template ) - Not used
DB 63 ;Reedit line (new template) - F5
DB 75 ;Backspace - <--
DB 82 ;Enter insert mode - INS (toggle)
DB 65 ;Escape character - F7
DB 65 ;End of table
ENDIF
ESCTABLEN EQU $-ESCTAB
IF NOT IBM
HEADER DB 13,10,"MS-DOS version 1.25"
IF HIGHMEM
DB "H"
ENDIF
IF DSKTEST
DB "D"
ENDIF
DB 13,10
DB "Copyright 1981,82 Microsoft, Inc.",13,10,"$"
ENDIF
QUIT:
MOV AH,0
JMP SHORT SAVREGS
COMMAND: ;Interrupt call entry point
CMP AH,MAXCOM
JBE SAVREGS
BADCALL:
MOV AL,0
IRET: IRET
ENTRY: ;System call entry point and dispatcher
POP AX ;IP from the long call at 5
POP AX ;Segment from the long call at 5
POP CS:[TEMP] ;IP from the CALL 5
PUSHF ;Start re-ordering the stack
CLI
PUSH AX ;Save segment
PUSH CS:[TEMP] ;Stack now ordered as if INT had been used
CMP CL,MAXCALL ;This entry point doesn't get as many calls
JA BADCALL
MOV AH,CL
SAVREGS:
PUSH ES
PUSH DS
PUSH BP
PUSH DI
PUSH SI
PUSH DX
PUSH CX
PUSH BX
PUSH AX
IF DSKTEST
MOV AX,CS:[SPSAVE]
MOV CS:[NSP],AX
MOV AX,CS:[SSSAVE]
MOV CS:[NSS],AX
POP AX
PUSH AX
ENDIF
MOV CS:[SPSAVE],SP
MOV CS:[SSSAVE],SS
MOV SP,CS
MOV SS,SP
REDISP:
MOV SP,OFFSET DOSGROUP:IOSTACK
STI ;Stack OK now
MOV BL,AH
MOV BH,0
SHL BX,1
CLD
CMP AH,12
JLE SAMSTK
MOV SP,OFFSET DOSGROUP:DSKSTACK
SAMSTK:
CALL CS:[BX+DISPATCH]
LEAVE:
CLI
MOV SP,CS:[SPSAVE]
MOV SS,CS:[SSSAVE]
MOV BP,SP
MOV BYTE PTR [BP.AXSAVE],AL
IF DSKTEST
MOV AX,CS:[NSP]
MOV CS:[SPSAVE],AX
MOV AX,CS:[NSS]
MOV CS:[SSSAVE],AX
ENDIF
POP AX
POP BX
POP CX
POP DX
POP SI
POP DI
POP BP
POP DS
POP ES
IRET
; Standard Functions
DISPATCH DW ABORT ;0
DW CONIN
DW CONOUT
DW READER
DW PUNCH
DW LIST ;5
DW RAWIO
DW RAWINP
DW IN
DW PRTBUF
DW BUFIN ;10
DW CONSTAT
DW FLUSHKB
DW DSKRESET
DW SELDSK
DW OPEN ;15
DW CLOSE
DW SRCHFRST
DW SRCHNXT
DW DELETE
DW SEQRD ;20
DW SEQWRT
DW CREATE
DW RENAME
DW INUSE
DW GETDRV ;25
DW SETDMA
DW GETFATPT
DW GETFATPTDL
DW GETRDONLY
DW SETATTRIB ;30
DW GETDSKPT
DW USERCODE
DW RNDRD
DW RNDWRT
DW FILESIZE ;35
DW SETRNDREC
; Extended Functions
DW SETVECT
DW NEWBASE
DW BLKRD
DW BLKWRT ;40
DW MAKEFCB
DW GETDATE
DW SETDATE
DW GETTIME
DW SETTIME ;45
DW VERIFY
INUSE:
GETIO:
SETIO:
GETRDONLY:
SETATTRIB:
USERCODE:
MOV AL,0
RET
VERIFY:
AND AL,1
MOV CS:VERFLG,AL
RET
FLUSHKB:
PUSH AX
CALL FAR PTR BIOSFLUSH
POP AX
MOV AH,AL
CMP AL,1
JZ REDISPJ
CMP AL,6
JZ REDISPJ
CMP AL,7
JZ REDISPJ
CMP AL,8
JZ REDISPJ
CMP AL,10
JZ REDISPJ
MOV AL,0
RET
REDISPJ:JMP REDISP
READER:
AUXIN:
CALL STATCHK
CALL FAR PTR BIOSAUXIN
RET
PUNCH:
MOV AL,DL
AUXOUT:
PUSH AX
CALL STATCHK
POP AX
CALL FAR PTR BIOSAUXOUT
RET
UNPACK:
; Inputs:
; DS = CS
; BX = Cluster number
; BP = Base of drive parameters
; SI = Pointer to drive FAT
; Outputs:
; DI = Contents of FAT for given cluster
; Zero set means DI=0 (free cluster)
; No other registers affected. Fatal error if cluster too big.
CMP BX,[BP.MAXCLUS]
JA HURTFAT
LEA DI,[SI+BX]
SHR BX,1
MOV DI,[DI+BX]
JNC HAVCLUS
SHR DI,1
SHR DI,1
SHR DI,1
SHR DI,1
STC
HAVCLUS:
RCL BX,1
AND DI,0FFFH
RET
HURTFAT:
PUSH AX
MOV AH,80H ;Signal Bad FAT to INT 24H handler
MOV DI,0FFFH ;In case INT 24H returns (it shouldn't)
CALL FATAL
POP AX ;Try to ignore bad FAT
RET
PACK:
; Inputs:
; DS = CS
; BX = Cluster number
; DX = Data
; SI = Pointer to drive FAT
; Outputs:
; The data is stored in the FAT at the given cluster.
; BX,DX,DI all destroyed
; No other registers affected
MOV DI,BX
SHR BX,1
ADD BX,SI
ADD BX,DI
SHR DI,1
MOV DI,[BX]
JNC ALIGNED
SHL DX,1
SHL DX,1
SHL DX,1
SHL DX,1
AND DI,0FH
JMP SHORT PACKIN
ALIGNED:
AND DI,0F000H
PACKIN:
OR DI,DX
MOV [BX],DI
RET
DEVNAME:
MOV SI,OFFSET DOSGROUP:IONAME ;List of I/O devices with file names
MOV BH,NUMDEV ;BH = number of device names
LOOKIO:
MOV DI,OFFSET DOSGROUP:NAME1
MOV CX,4 ;All devices are 4 letters
REPE CMPSB ;Check for name in list
JZ IOCHK ;If first 3 letters OK, check for the rest
ADD SI,CX ;Point to next device name
DEC BH
JNZ LOOKIO
CRET:
STC ;Not found
RET
IOCHK:
IF IBM
CMP BH,NUMDEV ;Is it the first device?
JNZ NOTCOM1
MOV BH,2 ;Make it the same as AUX
NOTCOM1:
ENDIF
NEG BH
MOV CX,2 ;Check rest of name but not extension
MOV AX,2020H
REPE SCASW ;Make sure rest of name is blanks
JNZ CRET
RET1: RET ;Zero set so CREATE works
GETFILE:
; Same as GETNAME except ES:DI points to FCB on successful return
CALL MOVNAME
JC RET1
PUSH DX
PUSH DS
CALL FINDNAME
POP ES
POP DI
RET2: RET
GETNAME:
; Inputs:
; DS,DX point to FCB
; Function:
; Find file name in disk directory. First byte is
; drive number (0=current disk). "?" matches any
; character.
; Outputs:
; Carry set if file not found
; ELSE
; Zero set if attributes match (always except when creating)
; BP = Base of drive parameters
; DS = CS
; ES = CS
; BX = Pointer into directory buffer
; SI = Pointer to First Cluster field in directory entry
; [DIRBUF] has directory record with match
; [NAME1] has file name
; All other registers destroyed.
CALL MOVNAME
JC RET2 ;Bad file name?
FINDNAME:
MOV AX,CS
MOV DS,AX
CALL DEVNAME
JNC RET2
CALL STARTSRCH
CONTSRCH:
CALL GETENTRY
JC RET2
SRCH:
MOV AH,BYTE PTR [BX]
OR AH,AH ;End of directory?
JZ FREE
CMP AH,[DELALL] ;Free entry?
JZ FREE
MOV SI,BX
MOV DI,OFFSET DOSGROUP:NAME1
MOV CX,11
WILDCRD:
REPE CMPSB
JZ FOUND
CMP BYTE PTR [DI-1],"?"
JZ WILDCRD
NEXTENT:
CALL NEXTENTRY
JNC SRCH
RET3: RET
FREE:
CMP [ENTFREE],-1 ;Found a free entry before?
JNZ TSTALL ;If so, ignore this one
MOV CX,[LASTENT]
MOV [ENTFREE],CX
TSTALL:
CMP AH,[DELALL] ;At end of directory?
JZ NEXTENT ;No - continue search
STC ;Report not found
RET
FOUND:
;Check if attributes allow finding it
MOV AH,[ATTRIB] ;Attributes of search
NOT AH
AND AH,[SI] ;Compare with attributes of file
ADD SI,15
AND AH,6 ;Only look at bits 1 and 2
JZ RET3
TEST BYTE PTR [CREATING],-1 ;Pass back mismatch if creating
JZ NEXTENT ;Otherwise continue searching
RET
GETENTRY:
; Inputs:
; [LASTENT] has previously searched directory entry
; Function:
; Locates next sequential directory entry in preparation for search
; Outputs:
; Carry set if none
; ELSE
; AL = Current directory block
; BX = Pointer to next directory entry in [DIRBUF]
; DX = Pointer to first byte after end of DIRBUF
; [LASTENT] = New directory entry number
MOV AX,[LASTENT]
INC AX ;Start with next entry
CMP AX,[BP.MAXENT]
JAE NONE
GETENT:
MOV [LASTENT],AX
MOV CL,4
SHL AX,CL
XOR DX,DX
SHL AX,1
RCL DX,1 ;Account for overflow in last shift
MOV BX,[BP.SECSIZ]
AND BL,255-31 ;Must be multiple of 32
DIV BX
MOV BX,DX ;Position within sector
MOV AH,[BP.DEVNUM] ;AL=Directory sector no.
CMP AX,[DIRBUFID]
JZ HAVDIRBUF
PUSH BX
CALL DIRREAD
POP BX
HAVDIRBUF:
MOV DX,OFFSET DOSGROUP:DIRBUF
ADD BX,DX
ADD DX,[BP.SECSIZ]
RET
NEXTENTRY:
; Inputs:
; Same as outputs of GETENTRY, above
; Function:
; Update AL, BX, and [LASTENT] for next directory entry.
; Carry set if no more.
MOV DI,[LASTENT]
INC DI
CMP DI,[BP.MAXENT]
JAE NONE
MOV [LASTENT],DI
ADD BX,32
CMP BX,DX
JB HAVIT
INC AL ;Next directory sector
PUSH DX ;Save limit
CALL DIRREAD
POP DX
MOV BX,OFFSET DOSGROUP:DIRBUF
HAVIT:
CLC
RET
NONE:
CALL CHKDIRWRITE
STC
RET4: RET
DELETE: ; System call 19
CALL MOVNAME
MOV AL,-1
JC RET4
MOV AL,CS:[ATTRIB]
AND AL,6 ;Look only at hidden bits
CMP AL,6 ;Both must be set
JNZ NOTALL
MOV CX,11
MOV AL,"?"
MOV DI,OFFSET DOSGROUP:NAME1
REPE SCASB ;See if name is *.*
JNZ NOTALL
MOV BYTE PTR CS:[DELALL],0 ;DEL *.* - flag deleting all
NOTALL:
CALL FINDNAME
MOV AL,-1
JC RET4
OR BH,BH ;Check if device name
JS RET4 ;Can't delete I/O devices
DELFILE:
MOV BYTE PTR [DIRTYDIR],-1
MOV AH,[DELALL]
MOV BYTE PTR [BX],AH
MOV BX,[SI]
MOV SI,[BP.FAT]
OR BX,BX
JZ DELNXT
CMP BX,[BP.MAXCLUS]
JA DELNXT
CALL RELEASE
DELNXT:
CALL CONTSRCH
JNC DELFILE
CALL FATWRT
CALL CHKDIRWRITE
XOR AL,AL
RET
RENAME: ;System call 23
CALL MOVNAME
JC ERRET
ADD SI,5
MOV DI,OFFSET DOSGROUP:NAME2
CALL LODNAME
JC ERRET ;Report error if second name invalid
CALL FINDNAME
JC ERRET
OR BH,BH ;Check if I/O device name
JS ERRET ;If so, can't rename it
MOV SI,OFFSET DOSGROUP:NAME1
MOV DI,OFFSET DOSGROUP:NAME3
MOV CX,6 ;6 words (12 bytes)--include attribute byte
REP MOVSW ;Copy name to search for
RENFIL:
MOV DI,OFFSET DOSGROUP:NAME1
MOV SI,OFFSET DOSGROUP:NAME2
MOV CX,11
NEWNAM:
LODSB
CMP AL,"?"
JNZ NOCHG
MOV AL,[BX]
NOCHG:
STOSB
INC BX
LOOP NEWNAM
MOV BYTE PTR [DI],6 ;Stop duplicates with any attributes
CALL DEVNAME ;Check if giving it a device name
JNC RENERR
PUSH [LASTENT] ;Save position of match
MOV [LASTENT],-1 ;Search entire directory for duplicate
CALL CONTSRCH ;See if new name already exists
POP AX
JNC RENERR ;Error if found
CALL GETENT ;Re-read matching entry
MOV DI,BX
MOV SI,OFFSET DOSGROUP:NAME1
MOV CX,5
MOVSB
REP MOVSW ;Replace old name with new one
MOV BYTE PTR [DIRTYDIR],-1 ;Flag change in directory
MOV SI,OFFSET DOSGROUP:NAME3
MOV DI,OFFSET DOSGROUP:NAME1
MOV CX,6 ;Include attribute byte
REP MOVSW ;Copy name back into search buffer
CALL CONTSRCH
JNC RENFIL
CALL CHKDIRWRITE
XOR AL,AL
RET
RENERR:
CALL CHKDIRWRITE
ERRET:
MOV AL,-1
RET5: RET
MOVNAME:
; Inputs:
; DS, DX point to FCB or extended FCB
; Outputs:
; DS:DX point to normal FCB
; ES = CS
; If file name OK:
; BP has base of driver parameters
; [NAME1] has name in upper case
; All registers except DX destroyed
; Carry set if bad file name or drive
MOV CS:WORD PTR [CREATING],0E500H ;Not creating, not DEL *.*
MOV AX,CS
MOV ES,AX
MOV DI,OFFSET DOSGROUP:NAME1
MOV SI,DX
LODSB
MOV CS:[EXTFCB],AL ;Set flag if extended FCB in use
MOV AH,0 ;Set default attributes
CMP AL,-1 ;Is it an extended FCB?
JNZ HAVATTRB
ADD DX,7 ;Adjust to point to normal FCB
ADD SI,6 ;Point to drive select byte
MOV AH,[SI-1] ;Get attribute byte
LODSB ;Get drive select byte
HAVATTRB:
MOV CS:[ATTRIB],AH ;Save attributes
CALL GETTHISDRV
LODNAME:
; This entry point copies a file name from DS,SI
; to ES,DI converting to upper case.
CMP BYTE PTR [SI]," " ;Don't allow blank as first letter
STC ;In case of error
JZ RET5
MOV CX,11
MOVCHK:
CALL GETLET
JB RET5
JNZ STOLET ;Is it a delimiter?
CMP AL," " ;This is the only delimiter allowed
STC ;In case of error
JNZ RET5
STOLET:
STOSB
LOOP MOVCHK
CLC ;Got through whole name - no error
RET6: RET
GETTHISDRV:
CMP CS:[NUMDRV],AL
JC RET6
DEC AL
JNS PHYDRV
MOV AL,CS:[CURDRV]
PHYDRV:
MOV CS:[THISDRV],AL
RET
OPEN: ;System call 15
CALL GETFILE
DOOPEN:
; Enter here to perform OPEN on file already found
; in directory. DS=CS, BX points to directory
; entry in DIRBUF, SI points to First Cluster field, and
; ES:DI point to the FCB to be opened. This entry point
; is used by CREATE.
JC ERRET
OR BH,BH ;Check if file is I/O device
JS OPENDEV ;Special handler if so
MOV AL,[THISDRV]
INC AX
STOSB
XOR AX,AX
IF ZEROEXT
ADD DI,11
STOSW ;Zero low byte of extent field if IBM only
ENDIF
IF NOT ZEROEXT
ADD DI,12 ;Point to high half of CURRENT BLOCK field
STOSB ;Set it to zero (CP/M programs set low byte)
ENDIF
MOV AL,128 ;Default record size
STOSW ;Set record size
LODSW ;Get starting cluster
MOV DX,AX ;Save it for the moment
MOVSW ;Transfer size to FCB
MOVSW
MOV AX,[SI-8] ;Get date
STOSW ;Save date in FCB
MOV AX,[SI-10] ;Get time
STOSW ;Save it in FCB
MOV AL,[BP.DEVNUM]
OR AL,40H
STOSB
MOV AX,DX ;Restore starting cluster
STOSW ; first cluster
STOSW ; last cluster accessed
XOR AX,AX
STOSW ; position of last cluster
RET
OPENDEV:
ADD DI,13 ;point to 2nd half of extent field
XOR AX,AX
STOSB ;Set it to zero
MOV AL,128
STOSW ;Set record size to 128
XOR AX,AX
STOSW
STOSW ;Set current size to zero
CALL DATE16
STOSW ;Date is todays
XCHG AX,DX
STOSW ;Use current time
MOV AL,BH ;Get device number
STOSB
XOR AL,AL ;No error
RET
FATERR:
XCHG AX,DI ;Put error code in DI
MOV AH,2 ;While trying to read FAT
MOV AL,[THISDRV] ;Tell which drive
CALL FATAL1
JMP SHORT FATREAD
STARTSRCH:
MOV AX,-1
MOV [LASTENT],AX
MOV [ENTFREE],AX
FATREAD:
; Inputs:
; DS = CS
; Function:
; If disk may have been changed, FAT is read in and buffers are
; flagged invalid. If not, no action is taken.
; Outputs:
; BP = Base of drive parameters
; Carry set if invalid drive returned by MAPDEV
; All other registers destroyed
MOV AL,[THISDRV]
XOR AH,AH ;Set default response to zero & clear carry
CALL FAR PTR BIOSDSKCHG ;See what BIOS has to say
JC FATERR
CALL GETBP
MOV AL,[THISDRV] ;Use physical unit number
MOV SI,[BP.FAT]
OR AH,[SI-1] ;Dirty byte for FAT
JS NEWDSK ;If either say new disk, then it's so
JNZ MAPDRV
MOV AH,1
CMP AX,WORD PTR [BUFDRVNO] ;Does buffer have dirty sector of this drive?
JZ MAPDRV
NEWDSK:
CMP AL,[BUFDRVNO] ;See if buffer is for this drive
JNZ BUFOK ;If not, don't touch it
MOV [BUFSECNO],0 ;Flag buffers invalid
MOV WORD PTR [BUFDRVNO],00FFH
BUFOK:
MOV [DIRBUFID],-1
CALL FIGFAT
NEXTFAT:
PUSH AX
CALL DSKREAD
POP AX
JC BADFAT
SUB AL,[BP.FATCNT]
JZ NEWFAT
CALL FATWRT
NEWFAT:
MOV SI,[BP.FAT]
MOV AL,[BP.DEVNUM]
MOV AH,[SI] ;Get first byte of FAT
OR AH,0F8H ;Put in range
CALL FAR PTR BIOSMAPDEV
MOV AH,0
MOV [SI-2],AX ;Set device no. and reset dirty bit
MAPDRV:
MOV AL,[SI-2] ;Get device number
GETBP:
MOV BP,[DRVTAB] ;Just in case drive isn't valid
AND AL,3FH ;Mask out dirty bit
CMP AL,[NUMIO]
CMC
JC RET7
PUSH AX
MOV AH,DPBSIZ
MUL AH
ADD BP,AX
POP AX
RET7: RET
BADFAT:
MOV CX,DI
ADD DX,CX
DEC AL
JNZ NEXTFAT
CALL FIGFAT ;Reset registers
CALL DREAD ;Try first FAT once more
JMP SHORT NEWFAT
OKRET1:
MOV AL,0
RET
CLOSE: ;System call 16
MOV DI,DX
CMP BYTE PTR [DI],-1 ;Check for extended FCB
JNZ NORMFCB3
ADD DI,7
NORMFCB3:
TEST BYTE PTR [DI.DEVID],0C0H ;Allow only dirty files
JNZ OKRET1 ;can't close if I/O device, or not writen
MOV AL,[DI] ;Get physical unit number
DEC AL ;Make zero = drive A
MOV AH,1 ;Look for dirty buffer
CMP AX,CS:WORD PTR [BUFDRVNO]
JNZ FNDDIR
;Write back dirty buffer if on same drive
PUSH DX
PUSH DS
PUSH CS
POP DS
MOV BYTE PTR [DIRTYBUF],0
MOV BX,[BUFFER]
MOV CX,1
MOV DX,[BUFSECNO]
MOV BP,[BUFDRVBP]
CALL DWRITE
POP DS
POP DX
FNDDIR:
CALL GETFILE
BADCLOSEJ:
JC BADCLOSE
MOV CX,ES:[DI.FIRCLUS]
MOV [SI],CX
MOV DX,ES:WORD PTR [DI.FILSIZ]
MOV [SI+2],DX
MOV DX,ES:WORD PTR [DI.FILSIZ+2]
MOV [SI+4],DX
MOV DX,ES:[DI.FDATE]
MOV [SI-2],DX
MOV DX,ES:[DI.FTIME]
MOV [SI-4],DX
CALL DIRWRITE
CHKFATWRT:
; Do FATWRT only if FAT is dirty and uses same I/O driver
MOV SI,[BP.FAT]
MOV AL,[BP.DEVNUM]
MOV AH,1
CMP [SI-2],AX ;See if FAT dirty and uses same driver
JNZ OKRET
FATWRT:
; Inputs:
; DS = CS
; BP = Base of drive parameter table
; Function:
; Write the FAT back to disk and reset FAT
; dirty bit.
; Outputs:
; AL = 0
; BP unchanged
; All other registers destroyed
CALL FIGFAT
MOV BYTE PTR [BX-1],0
EACHFAT:
PUSH DX
PUSH CX
PUSH BX
PUSH AX
CALL DWRITE
POP AX
POP BX
POP CX
POP DX
ADD DX,CX
DEC AL
JNZ EACHFAT
OKRET:
MOV AL,0
RET
BADCLOSE:
MOV SI,[BP.FAT]
MOV BYTE PTR [SI-1],0
MOV AL,-1
RET
FIGFAT:
; Loads registers with values needed to read or
; write a FAT.
MOV AL,[BP.FATCNT]
MOV BX,[BP.FAT]
MOV CL,[BP.FATSIZ] ;No. of records occupied by FAT
MOV CH,0
MOV DX,[BP.FIRFAT] ;Record number of start of FATs
RET
DIRCOMP:
; Prepare registers for directory read or write
CBW
ADD AX,[BP.FIRDIR]
MOV DX,AX
MOV BX,OFFSET DOSGROUP:DIRBUF
MOV CX,1
RET
CREATE: ;System call 22
CALL MOVNAME
JC ERRET3
MOV DI,OFFSET DOSGROUP:NAME1
MOV CX,11
MOV AL,"?"
REPNE SCASB
JZ ERRET3
MOV CS:BYTE PTR [CREATING],-1
PUSH DX
PUSH DS
CALL FINDNAME
JNC EXISTENT
MOV AX,[ENTFREE] ;First free entry found in FINDNAME
CMP AX,-1
JZ ERRPOP
CALL GETENT ;Point at that free entry
JMP SHORT FREESPOT
ERRPOP:
POP DS
POP DX
ERRET3:
MOV AL,-1
RET
EXISTENT:
JNZ ERRPOP ;Error if attributes don't match
OR BH,BH ;Check if file is I/O device
JS OPENJMP ;If so, no action
MOV CX,[SI] ;Get pointer to clusters
JCXZ FREESPOT
CMP CX,[BP.MAXCLUS]
JA FREESPOT
PUSH BX
MOV BX,CX
MOV SI,[BP.FAT]
CALL RELEASE ;Free any data already allocated
CALL FATWRT
POP BX
FREESPOT:
MOV DI,BX
MOV SI,OFFSET DOSGROUP:NAME1
MOV CX,5
MOVSB
REP MOVSW
MOV AL,[ATTRIB]
STOSB
XOR AX,AX
MOV CL,5
REP STOSW
CALL DATE16
XCHG AX,DX
STOSW
XCHG AX,DX
STOSW
XOR AX,AX
PUSH DI
MOV CL,6
SMALLENT:
REP STOSB
PUSH BX
CALL DIRWRITE
POP BX
POP SI
OPENJMP:
CLC ;Clear carry so OPEN won't fail
POP ES
POP DI
JMP DOOPEN
DIRREAD:
; Inputs:
; DS = CS
; AL = Directory block number
; BP = Base of drive parameters
; Function:
; Read the directory block into DIRBUF.
; Outputs:
; AX,BP unchanged
; All other registers destroyed.
PUSH AX
CALL CHKDIRWRITE
POP AX
PUSH AX
MOV AH,[BP.DEVNUM]
MOV [DIRBUFID],AX
CALL DIRCOMP
CALL DREAD
POP AX
RET8: RET
DREAD:
; Inputs:
; BX,DS = Transfer address
; CX = Number of sectors
; DX = Absolute record number
; BP = Base of drive parameters
; Function:
; Calls BIOS to perform disk read. If BIOS reports
; errors, will call HARDERR for further action.
; BP preserved. All other registers destroyed.
CALL DSKREAD
JNC RET8
MOV CS:BYTE PTR [READOP],0
CALL HARDERR
CMP AL,1 ;Check for retry
JZ DREAD
RET ;Ignore otherwise
HARDERR:
;Hard disk error handler. Entry conditions:
; DS:BX = Original disk transfer address
; DX = Original logical sector number
; CX = Number of sectors to go (first one gave the error)
; AX = Hardware error code
; DI = Original sector transfer count
; BP = Base of drive parameters
; [READOP] = 0 for read, 1 for write
XCHG AX,DI ;Error code in DI, count in AX
SUB AX,CX ;Number of sectors successfully transferred
ADD DX,AX ;First sector number to retry
PUSH DX
MUL [BP.SECSIZ] ;Number of bytes transferred
POP DX
ADD BX,AX ;First address for retry
MOV AH,0 ;Flag disk section in error
CMP DX,[BP.FIRFAT] ;In reserved area?
JB ERRINT
INC AH ;Flag for FAT
CMP DX,[BP.FIRDIR] ;In FAT?
JB ERRINT
INC AH
CMP DX,[BP.FIRREC] ;In directory?
JB ERRINT
INC AH ;Must be in data area
ERRINT:
SHL AH,1 ;Make room for read/write bit
OR AH,CS:[READOP]
FATAL:
MOV AL,[BP.DRVNUM] ;Get drive number
FATAL1:
PUSH BP ;The only thing we preserve
MOV CS:[CONTSTK],SP
CLI ;Prepare to play with stack
MOV SS,CS:[SSSAVE]
MOV SP,CS:[SPSAVE] ;User stack pointer restored
INT 24H ;Fatal error interrupt vector
MOV CS:[SPSAVE],SP
MOV CS:[SSSAVE],SS
MOV SP,CS
MOV SS,SP
MOV SP,CS:[CONTSTK]
STI
POP BP
CMP AL,2
JZ ERROR
RET
DSKREAD:
MOV AL,[BP.DEVNUM]
PUSH BP
PUSH BX
PUSH CX
PUSH DX
CALL FAR PTR BIOSREAD
POP DX
POP DI
POP BX
POP BP
RET9: RET
CHKDIRWRITE:
TEST BYTE PTR [DIRTYDIR],-1
JZ RET9
DIRWRITE:
; Inputs:
; DS = CS
; AL = Directory block number
; BP = Base of drive parameters
; Function:
; Write the directory block into DIRBUF.
; Outputs:
; BP unchanged
; All other registers destroyed.
MOV BYTE PTR [DIRTYDIR],0
MOV AL,BYTE PTR [DIRBUFID]
CALL DIRCOMP
DWRITE:
; Inputs:
; BX,DS = Transfer address
; CX = Number of sectors
; DX = Absolute record number
; BP = Base of drive parameters
; Function:
; Calls BIOS to perform disk write. If BIOS reports
; errors, will call HARDERR for further action.
; BP preserved. All other registers destroyed.
MOV AL,[BP.DEVNUM]
MOV AH,CS:VERFLG
PUSH BP
PUSH BX
PUSH CX
PUSH DX
CALL FAR PTR BIOSWRITE
POP DX
POP DI
POP BX
POP BP
JNC RET9
MOV CS:BYTE PTR [READOP],1
CALL HARDERR
CMP AL,1 ;Check for retry
JZ DWRITE
RET
ABORT:
LDS SI,CS:DWORD PTR [SPSAVE]
MOV DS,[SI.CSSAVE]
XOR AX,AX
MOV ES,AX
MOV SI,SAVEXIT
MOV DI,EXIT
MOVSW
MOVSW
MOVSW
MOVSW
MOVSW
MOVSW
ERROR:
MOV AX,CS
MOV DS,AX
MOV ES,AX
CALL WRTFATS
XOR AX,AX
CLI
MOV SS,[SSSAVE]
MOV SP,[SPSAVE]
MOV DS,AX
MOV SI,EXIT
MOV DI,OFFSET DOSGROUP:EXITHOLD
MOVSW
MOVSW
POP AX
POP BX
POP CX
POP DX
POP SI
POP DI
POP BP
POP DS
POP ES
STI ;Stack OK now
JMP CS:DWORD PTR [EXITHOLD]
SEQRD: ;System call 20
CALL GETREC
CALL LOAD
JMP SHORT FINSEQ
SEQWRT: ;System call 21
CALL GETREC
CALL STORE
FINSEQ:
JCXZ SETNREX
ADD AX,1
ADC DX,0
JMP SHORT SETNREX
RNDRD: ;System call 33
CALL GETRRPOS1
CALL LOAD
JMP SHORT FINRND
RNDWRT: ;System call 34
CALL GETRRPOS1
CALL STORE
JMP SHORT FINRND
BLKRD: ;System call 39
CALL GETRRPOS
CALL LOAD
JMP SHORT FINBLK
BLKWRT: ;System call 40
CALL GETRRPOS
CALL STORE
FINBLK:
LDS SI,DWORD PTR [SPSAVE]
MOV [SI.CXSAVE],CX
JCXZ FINRND
ADD AX,1
ADC DX,0
FINRND:
MOV ES:WORD PTR [DI.RR],AX
MOV ES:[DI.RR+2],DL
OR DH,DH
JZ SETNREX
MOV ES:[DI.RR+3],DH ;Save 4 byte of RECPOS only if significant
SETNREX:
MOV CX,AX
AND AL,7FH
MOV ES:[DI.NR],AL
AND CL,80H
SHL CX,1
RCL DX,1
MOV AL,CH
MOV AH,DL
MOV ES:[DI.EXTENT],AX
MOV AL,CS:[DSKERR]
RET
GETRRPOS1:
MOV CX,1
GETRRPOS:
MOV DI,DX
CMP BYTE PTR [DI],-1
JNZ NORMFCB1
ADD DI,7
NORMFCB1:
MOV AX,WORD PTR [DI.RR]
MOV DX,WORD PTR [DI.RR+2]
RET
NOFILERR:
XOR CX,CX
MOV BYTE PTR [DSKERR],4
POP BX
RET
SETUP:
; Inputs:
; DS:DI point to FCB
; DX:AX = Record position in file of disk transfer
; CX = Record count
; Outputs:
; DS = CS
; ES:DI point to FCB
; BL = DEVID from FCB
; CX = No. of bytes to transfer
; BP = Base of drive parameters
; SI = FAT pointer
; [RECCNT] = Record count
; [RECPOS] = Record position in file
; [FCB] = DI
; [NEXTADD] = Displacement of disk transfer within segment
; [SECPOS] = Position of first sector
; [BYTPOS] = Byte position in file
; [BYTSECPOS] = Byte position in first sector
; [CLUSNUM] = First cluster
; [SECCLUSPOS] = Sector within first cluster
; [DSKERR] = 0 (no errors yet)
; [TRANS] = 0 (No transfers yet)
; [THISDRV] = Physical drive unit number
; If SETUP detects no records will be transfered, it returns 1 level up
; with CX = 0.
PUSH AX
MOV AL,[DI]
DEC AL
MOV CS:[THISDRV],AL
MOV AL,[DI.DEVID]
MOV SI,[DI.RECSIZ]
OR SI,SI
JNZ HAVRECSIZ
MOV SI,128
MOV [DI.RECSIZ],SI
HAVRECSIZ:
PUSH DS
POP ES ;Set ES to DS
PUSH CS
POP DS ;Set DS to CS
OR AL,AL ;Is it a device?
JNS NOTDEVICE
MOV AL,0 ;Fake in drive 0 so we can get SP
NOTDEVICE:
CALL GETBP
POP AX
JC NOFILERR
CMP SI,64 ;Check if highest byte of RECPOS is significant
JB SMALREC
MOV DH,0 ;Ignore MSB if record >= 64 bytes
SMALREC:
MOV [RECCNT],CX
MOV WORD PTR [RECPOS],AX
MOV WORD PTR [RECPOS+2],DX
MOV [FCB],DI
MOV BX,[DMAADD]
MOV [NEXTADD],BX
MOV BYTE PTR [DSKERR],0
MOV BYTE PTR [TRANS],0
MOV BX,DX
MUL SI
MOV WORD PTR [BYTPOS],AX
PUSH DX
MOV AX,BX
MUL SI
POP BX
ADD AX,BX
ADC DX,0 ;Ripple carry
JNZ EOFERR
MOV WORD PTR [BYTPOS+2],AX
MOV DX,AX
MOV AX,WORD PTR [BYTPOS]
MOV BX,[BP.SECSIZ]
CMP DX,BX ;See if divide will overflow
JNC EOFERR
DIV BX
MOV [SECPOS],AX
MOV [BYTSECPOS],DX
MOV DX,AX
AND AL,[BP.CLUSMSK]
MOV [SECCLUSPOS],AL
MOV AX,CX ;Record count
MOV CL,[BP.CLUSSHFT]
SHR DX,CL
MOV [CLUSNUM],DX
MUL SI ;Multiply by bytes per record
MOV CX,AX
ADD AX,[DMAADD] ;See if it will fit in one segment
ADC DX,0
JZ OK ;Must be less than 64K
MOV AX,[DMAADD]
NEG AX ;Amount of room left in segment
JNZ PARTSEG ;All 64K available?
DEC AX ;If so, reduce by one
PARTSEG:
XOR DX,DX
DIV SI ;How many records will fit?
MOV [RECCNT],AX
MUL SI ;Translate that back into bytes
MOV BYTE PTR [DSKERR],2 ;Flag that trimming took place
MOV CX,AX
JCXZ NOROOM
OK:
MOV BL,ES:[DI.DEVID]
MOV SI,[BP.FAT]
RET
EOFERR:
MOV BYTE PTR [DSKERR],1
XOR CX,CX
NOROOM:
POP BX ;Kill return address
RET
BREAKDOWN:
;Inputs:
; DS = CS
; CX = Length of disk transfer in bytes
; BP = Base of drive parameters
; [BYTSECPOS] = Byte position witin first sector
;Outputs:
; [BYTCNT1] = Bytes to transfer in first sector
; [SECCNT] = No. of whole sectors to transfer
; [BYTCNT2] = Bytes to transfer in last sector
;AX, BX, DX destroyed. No other registers affected.
MOV AX,[BYTSECPOS]
MOV BX,CX
OR AX,AX
JZ SAVFIR ;Partial first sector?
SUB AX,[BP.SECSIZ]
NEG AX ;Max number of bytes left in first sector
SUB BX,AX ;Subtract from total length
JAE SAVFIR
ADD AX,BX ;Don't use all of the rest of the sector
XOR BX,BX ;And no bytes are left
SAVFIR:
MOV [BYTCNT1],AX
MOV AX,BX
XOR DX,DX
DIV [BP.SECSIZ] ;How many whole sectors?
MOV [SECCNT],AX
MOV [BYTCNT2],DX ;Bytes remaining for last sector
RET10: RET
FNDCLUS:
; Inputs:
; DS = CS
; CX = No. of clusters to skip
; BP = Base of drive parameters
; SI = FAT pointer
; ES:DI point to FCB
; Outputs:
; BX = Last cluster skipped to
; CX = No. of clusters remaining (0 unless EOF)
; DX = Position of last cluster
; DI destroyed. No other registers affected.
MOV BX,ES:[DI.LSTCLUS]
MOV DX,ES:[DI.CLUSPOS]
OR BX,BX
JZ NOCLUS
SUB CX,DX
JNB FINDIT
ADD CX,DX
XOR DX,DX
MOV BX,ES:[DI.FIRCLUS]
FINDIT:
JCXZ RET10
SKPCLP:
CALL UNPACK
CMP DI,0FF8H
JAE RET10
XCHG BX,DI
INC DX
LOOP SKPCLP
RET
NOCLUS:
INC CX
DEC DX
RET
BUFSEC:
; Inputs:
; AL = 0 if buffer must be read, 1 if no pre-read needed
; BP = Base of drive parameters
; [CLUSNUM] = Physical cluster number
; [SECCLUSPOS] = Sector position of transfer within cluster
; [BYTCNT1] = Size of transfer
; Function:
; Insure specified sector is in buffer, flushing buffer before
; read if necessary.
; Outputs:
; SI = Pointer to buffer
; DI = Pointer to transfer address
; CX = Number of bytes
; [NEXTADD] updated
; [TRANS] set to indicate a transfer will occur
MOV DX,[CLUSNUM]
MOV BL,[SECCLUSPOS]
CALL FIGREC
MOV [PREREAD],AL
CMP DX,[BUFSECNO]
JNZ GETSEC
MOV AL,[BUFDRVNO]
CMP AL,[THISDRV]
JZ FINBUF ;Already have it?
GETSEC:
XOR AL,AL
XCHG [DIRTYBUF],AL ;Read dirty flag and reset it
OR AL,AL
JZ RDSEC
PUSH DX
PUSH BP
MOV BP,[BUFDRVBP]
MOV BX,[BUFFER]
MOV CX,1
MOV DX,[BUFSECNO]
CALL DWRITE
POP BP
POP DX
RDSEC:
TEST BYTE PTR [PREREAD],-1
JNZ SETBUF
XOR AX,AX
MOV [BUFSECNO],AX ;Set buffer valid in case of disk error
DEC AX
MOV [BUFDRVNO],AL
MOV BX,[BUFFER]
MOV CX,1
PUSH DX
CALL DREAD
POP DX
SETBUF:
MOV [BUFSECNO],DX
MOV AL,[THISDRV]
MOV [BUFDRVNO],AL
MOV [BUFDRVBP],BP
FINBUF:
MOV BYTE PTR [TRANS],1 ;A transfer is taking place
MOV DI,[NEXTADD]
MOV SI,DI
MOV CX,[BYTCNT1]
ADD SI,CX
MOV [NEXTADD],SI
MOV SI,[BUFFER]
ADD SI,[BYTSECPOS]
RET
BUFRD:
XOR AL,AL ;Pre-read necessary
CALL BUFSEC
PUSH ES
MOV ES,[DMAADD+2]
SHR CX,1
JNC EVENRD
MOVSB
EVENRD:
REP MOVSW
POP ES
RET
BUFWRT:
MOV AX,[SECPOS]
INC AX ;Set for next sector
MOV [SECPOS],AX
CMP AX,[VALSEC] ;Has sector been written before?
MOV AL,1
JA NOREAD ;Skip preread if SECPOS>VALSEC
MOV AL,0
NOREAD:
CALL BUFSEC
XCHG DI,SI
PUSH DS
PUSH ES
PUSH CS
POP ES
MOV DS,[DMAADD+2]
SHR CX,1
JNC EVENWRT
MOVSB
EVENWRT:
REP MOVSW
POP ES
POP DS
MOV BYTE PTR [DIRTYBUF],1
RET
NEXTSEC:
TEST BYTE PTR [TRANS],-1
JZ CLRET
MOV AL,[SECCLUSPOS]
INC AL
CMP AL,[BP.CLUSMSK]
JBE SAVPOS
MOV BX,[CLUSNUM]
CMP BX,0FF8H
JAE NONEXT
MOV SI,[BP.FAT]
CALL UNPACK
MOV [CLUSNUM],DI
INC [LASTPOS]
MOV AL,0
SAVPOS:
MOV [SECCLUSPOS],AL
CLRET:
CLC
RET
NONEXT:
STC
RET
TRANBUF:
LODSB
STOSB
CMP AL,13 ;Check for carriage return
JNZ NORMCH
MOV BYTE PTR [SI],10
NORMCH:
CMP AL,10
LOOPNZ TRANBUF
JNZ ENDRDCON
CALL OUT ;Transmit linefeed
XOR SI,SI
OR CX,CX
JNZ GETBUF
OR AL,1 ;Clear zero flag--not end of file
ENDRDCON:
MOV [CONTPOS],SI
ENDRDDEV:
MOV [NEXTADD],DI
POP ES
JNZ SETFCBJ ;Zero set if Ctrl-Z found in input
MOV DI,[FCB]
AND ES:BYTE PTR [DI.DEVID],0FFH-40H ;Mark as no more data available
SETFCBJ:
JMP SETFCB
READDEV:
PUSH ES
LES DI,DWORD PTR [DMAADD]
INC BL
JZ READCON
INC BL
JNZ ENDRDDEV
READAUX:
CALL AUXIN
STOSB
CMP AL,1AH
LOOPNZ READAUX
JMP SHORT ENDRDDEV
READCON:
PUSH CS
POP DS
MOV SI,[CONTPOS]
OR SI,SI
JNZ TRANBUF
CMP BYTE PTR [CONBUF],128
JZ GETBUF
MOV WORD PTR [CONBUF],0FF80H ;Set up 128-byte buffer with no template
GETBUF:
PUSH CX
PUSH ES
PUSH DI
MOV DX,OFFSET DOSGROUP:CONBUF
CALL BUFIN ;Get input buffer
POP DI
POP ES
POP CX
MOV SI,2 + OFFSET DOSGROUP:CONBUF
CMP BYTE PTR [SI],1AH ;Check for Ctrl-Z in first character
JNZ TRANBUF
MOV AL,1AH
STOSB
MOV AL,10
CALL OUT ;Send linefeed
XOR SI,SI
JMP SHORT ENDRDCON
RDERR:
XOR CX,CX
JMP WRTERR
RDLASTJ:JMP RDLAST
LOAD:
; Inputs:
; DS:DI point to FCB
; DX:AX = Position in file to read
; CX = No. of records to read
; Outputs:
; DX:AX = Position of last record read
; CX = No. of bytes read
; ES:DI point to FCB
; LSTCLUS, CLUSPOS fields in FCB set
CALL SETUP
OR BL,BL ;Check for named device I/O
JS READDEV
MOV AX,ES:WORD PTR [DI.FILSIZ]
MOV BX,ES:WORD PTR [DI.FILSIZ+2]
SUB AX,WORD PTR [BYTPOS]
SBB BX,WORD PTR [BYTPOS+2]
JB RDERR
JNZ ENUF
OR AX,AX
JZ RDERR
CMP AX,CX
JAE ENUF
MOV CX,AX
ENUF:
CALL BREAKDOWN
MOV CX,[CLUSNUM]
CALL FNDCLUS
OR CX,CX
JNZ RDERR
MOV [LASTPOS],DX
MOV [CLUSNUM],BX
CMP [BYTCNT1],0
JZ RDMID
CALL BUFRD
RDMID:
CMP [SECCNT],0
JZ RDLASTJ
CALL NEXTSEC
JC SETFCB
MOV BYTE PTR [TRANS],1 ;A transfer is taking place
ONSEC:
MOV DL,[SECCLUSPOS]
MOV CX,[SECCNT]
MOV BX,[CLUSNUM]
RDLP:
CALL OPTIMIZE
PUSH DI
PUSH AX
PUSH DS
MOV DS,[DMAADD+2]
PUSH DX
PUSH BX
PUSHF ;Save carry flag
CALL DREAD
POPF ;Restore carry flag
POP DI ;Initial transfer address
POP AX ;First sector transfered
POP DS
JC NOTBUFFED ;Was one of those sectors in the buffer?
CMP BYTE PTR [DIRTYBUF],0 ;Is buffer dirty?
JZ NOTBUFFED ;If not no problem
;We have transfered in a sector from disk when a dirty copy of it is in the buffer.
;We must transfer the sector from the buffer to correct memory address
SUB AX,[BUFSECNO] ;How many sectors into the transfer?
NEG AX
MOV CX,[BP.SECSIZ]
MUL CX ;How many bytes into the transfer?
ADD DI,AX
MOV SI,[BUFFER]
PUSH ES
MOV ES,[DMAADD+2] ;Get disk transfer segment
SHR CX,1
REP MOVSW
JNC EVENMOV
MOVSB
EVENMOV:
POP ES
NOTBUFFED:
POP CX
POP BX
JCXZ RDLAST
CMP BX,0FF8H
JAE SETFCB
MOV DL,0
INC [LASTPOS] ;We'll be using next cluster
JMP SHORT RDLP
SETFCB:
MOV SI,[FCB]
MOV AX,[NEXTADD]
MOV DI,AX
SUB AX,[DMAADD] ;Number of bytes transfered
XOR DX,DX
MOV CX,ES:[SI.RECSIZ]
DIV CX ;Number of records
CMP AX,[RECCNT] ;Check if all records transferred
JZ FULLREC
MOV BYTE PTR [DSKERR],1
OR DX,DX
JZ FULLREC ;If remainder 0, then full record transfered
MOV BYTE PTR [DSKERR],3 ;Flag partial last record
SUB CX,DX ;Bytes left in last record
PUSH ES
MOV ES,[DMAADD+2]
XCHG AX,BX ;Save the record count temporarily
XOR AX,AX ;Fill with zeros
SHR CX,1
JNC EVENFIL
STOSB
EVENFIL:
REP STOSW
XCHG AX,BX ;Restore record count to AX
POP ES
INC AX ;Add last (partial) record to total
FULLREC:
MOV CX,AX
MOV DI,SI ;ES:DI point to FCB
SETCLUS:
MOV AX,[CLUSNUM]
MOV ES:[DI.LSTCLUS],AX
MOV AX,[LASTPOS]
MOV ES:[DI.CLUSPOS],AX
ADDREC:
MOV AX,WORD PTR [RECPOS]
MOV DX,WORD PTR [RECPOS+2]
JCXZ RET28 ;If no records read, don't change position
DEC CX
ADD AX,CX ;Update current record position
ADC DX,0
INC CX
RET28: RET
RDLAST:
MOV AX,[BYTCNT2]
OR AX,AX
JZ SETFCB
MOV [BYTCNT1],AX
CALL NEXTSEC
JC SETFCB
MOV [BYTSECPOS],0
CALL BUFRD
JMP SHORT SETFCB
WRTDEV:
PUSH DS
LDS SI,DWORD PTR [DMAADD]
OR BL,40H
INC BL
JZ WRTCON
INC BL
JZ WRTAUX
INC BL
JZ ENDWRDEV ;Done if device is NUL
WRTLST:
LODSB
CMP AL,1AH
JZ ENDWRDEV
CALL LISTOUT
LOOP WRTLST
JMP SHORT ENDWRDEV
WRTAUX:
LODSB
CALL AUXOUT
CMP AL,1AH
LOOPNZ WRTAUX
JMP SHORT ENDWRDEV
WRTCON:
LODSB
CMP AL,1AH
JZ ENDWRDEV
CALL OUT
LOOP WRTCON
ENDWRDEV:
POP DS
MOV CX,[RECCNT]
MOV DI,[FCB]
JMP SHORT ADDREC
HAVSTART:
MOV CX,AX
CALL SKPCLP
JCXZ DOWRTJ
CALL ALLOCATE
JNC DOWRTJ
WRTERR:
MOV BYTE PTR [DSKERR],1
LVDSK:
MOV AX,WORD PTR [RECPOS]
MOV DX,WORD PTR [RECPOS+2]
MOV DI,[FCB]
RET
DOWRTJ: JMP DOWRT
WRTEOFJ:
JMP WRTEOF
STORE:
; Inputs:
; DS:DI point to FCB
; DX:AX = Position in file of disk transfer
; CX = Record count
; Outputs:
; DX:AX = Position of last record written
; CX = No. of records written
; ES:DI point to FCB
; LSTCLUS, CLUSPOS fields in FCB set
CALL SETUP
CALL DATE16
MOV ES:[DI.FDATE],AX
MOV ES:[DI.FTIME],DX
OR BL,BL
JS WRTDEV
AND BL,3FH ;Mark file as dirty
MOV ES:[DI.DEVID],BL
CALL BREAKDOWN
MOV AX,WORD PTR [BYTPOS]
MOV DX,WORD PTR [BYTPOS+2]
JCXZ WRTEOFJ
DEC CX
ADD AX,CX
ADC DX,0 ;AX:DX=last byte accessed
DIV [BP.SECSIZ] ;AX=last sector accessed
MOV CL,[BP.CLUSSHFT]
SHR AX,CL ;Last cluster to be accessed
PUSH AX
MOV AX,ES:WORD PTR [DI.FILSIZ]
MOV DX,ES:WORD PTR [DI.FILSIZ+2]
DIV [BP.SECSIZ]
OR DX,DX
JZ NORNDUP
INC AX ;Round up if any remainder
NORNDUP:
MOV [VALSEC],AX ;Number of sectors that have been written
POP AX
MOV CX,[CLUSNUM] ;First cluster accessed
CALL FNDCLUS
MOV [CLUSNUM],BX
MOV [LASTPOS],DX
SUB AX,DX ;Last cluster minus current cluster
JZ DOWRT ;If we have last clus, we must have first
JCXZ HAVSTART ;See if no more data
PUSH CX ;No. of clusters short of first
MOV CX,AX
CALL ALLOCATE
POP AX
JC WRTERR
MOV CX,AX
MOV DX,[LASTPOS]
INC DX
DEC CX
JZ NOSKIP
CALL SKPCLP
NOSKIP:
MOV [CLUSNUM],BX
MOV [LASTPOS],DX
DOWRT:
CMP [BYTCNT1],0
JZ WRTMID
MOV BX,[CLUSNUM]
CALL BUFWRT
WRTMID:
MOV AX,[SECCNT]
OR AX,AX
JZ WRTLAST
ADD [SECPOS],AX
CALL NEXTSEC
MOV BYTE PTR [TRANS],1 ;A transfer is taking place
MOV DL,[SECCLUSPOS]
MOV BX,[CLUSNUM]
MOV CX,[SECCNT]
WRTLP:
CALL OPTIMIZE
JC NOTINBUF ;Is one of the sectors buffered?
MOV [BUFSECNO],0 ;If so, invalidate the buffer since we're
MOV WORD PTR [BUFDRVNO],0FFH ; completely rewritting it
NOTINBUF:
PUSH DI
PUSH AX
PUSH DS
MOV DS,[DMAADD+2]
CALL DWRITE
POP DS
POP CX
POP BX
JCXZ WRTLAST
MOV DL,0
INC [LASTPOS] ;We'll be using next cluster
JMP SHORT WRTLP
WRTLAST:
MOV AX,[BYTCNT2]
OR AX,AX
JZ FINWRT
MOV [BYTCNT1],AX
CALL NEXTSEC
MOV [BYTSECPOS],0
CALL BUFWRT
FINWRT:
MOV AX,[NEXTADD]
SUB AX,[DMAADD]
ADD AX,WORD PTR [BYTPOS]
MOV DX,WORD PTR [BYTPOS+2]
ADC DX,0
MOV CX,DX
MOV DI,[FCB]
CMP AX,ES:WORD PTR [DI.FILSIZ]
SBB CX,ES:WORD PTR [DI.FILSIZ+2]
JB SAMSIZ
MOV ES:WORD PTR [DI.FILSIZ],AX
MOV ES:WORD PTR [DI.FILSIZ+2],DX
SAMSIZ:
MOV CX,[RECCNT]
JMP SETCLUS
WRTERRJ:JMP WRTERR
WRTEOF:
MOV CX,AX
OR CX,DX
JZ KILLFIL
SUB AX,1
SBB DX,0
DIV [BP.SECSIZ]
MOV CL,[BP.CLUSSHFT]
SHR AX,CL
MOV CX,AX
CALL FNDCLUS
JCXZ RELFILE
CALL ALLOCATE
JC WRTERRJ
UPDATE:
MOV DI,[FCB]
MOV AX,WORD PTR [BYTPOS]
MOV ES:WORD PTR [DI.FILSIZ],AX
MOV AX,WORD PTR [BYTPOS+2]
MOV ES:WORD PTR [DI.FILSIZ+2],AX
XOR CX,CX
JMP ADDREC
RELFILE:
MOV DX,0FFFH
CALL RELBLKS
SETDIRT:
MOV BYTE PTR [SI-1],1
JMP SHORT UPDATE
KILLFIL:
XOR BX,BX
XCHG BX,ES:[DI.FIRCLUS]
OR BX,BX
JZ UPDATE
CALL RELEASE
JMP SHORT SETDIRT
OPTIMIZE:
; Inputs:
; DS = CS
; BX = Physical cluster
; CX = No. of records
; DL = sector within cluster
; BP = Base of drives parameters
; [NEXTADD] = transfer address
; Outputs:
; AX = No. of records remaining
; BX = Transfer address
; CX = No. or records to be transferred
; DX = Physical sector address
; DI = Next cluster
; Carry clear if a sector to transfer is in the buffer
; Carry set otherwise
; [CLUSNUM] = Last cluster accessed
; [NEXTADD] updated
; BP unchanged. Note that segment of transfer not set.
PUSH DX
PUSH BX
MOV AL,[BP.CLUSMSK]
INC AL ;Number of sectors per cluster
MOV AH,AL
SUB AL,DL ;AL = Number of sectors left in first cluster
MOV DX,CX
MOV SI,[BP.FAT]
MOV CX,0
OPTCLUS:
;AL has number of sectors available in current cluster
;AH has number of sectors available in next cluster
;BX has current physical cluster
;CX has number of sequential sectors found so far
;DX has number of sectors left to transfer
;SI has FAT pointer
CALL UNPACK
ADD CL,AL
ADC CH,0
CMP CX,DX
JAE BLKDON
MOV AL,AH
INC BX
CMP DI,BX
JZ OPTCLUS
DEC BX
FINCLUS:
MOV [CLUSNUM],BX ;Last cluster accessed
SUB DX,CX ;Number of sectors still needed
PUSH DX
MOV AX,CX
MUL [BP.SECSIZ] ;Number of sectors times sector size
MOV SI,[NEXTADD]
ADD AX,SI ;Adjust by size of transfer
MOV [NEXTADD],AX
POP AX ;Number of sectors still needed
POP DX ;Starting cluster
SUB BX,DX ;Number of new clusters accessed
ADD [LASTPOS],BX
POP BX ;BL = sector postion within cluster
CALL FIGREC
MOV BX,SI
;Now let's see if any of these sectors are already in the buffer
CMP [BUFSECNO],DX
JC RET100 ;If DX > [BUFSECNO] then not in buffer
MOV SI,DX
ADD SI,CX ;Last sector + 1
CMP [BUFSECNO],SI
CMC
JC RET100 ;If SI <= [BUFSECNO] then not in buffer
PUSH AX
MOV AL,[BP.DEVNUM]
CMP AL,[BUFDRVNO] ;Is buffer for this drive?
POP AX
JZ RET100 ;If so, then we match
STC ;No match
RET100: RET
BLKDON:
SUB CX,DX ;Number of sectors in cluster we don't want
SUB AH,CL ;Number of sectors in cluster we accepted
DEC AH ;Adjust to mean position within cluster
MOV [SECCLUSPOS],AH
MOV CX,DX ;Anyway, make the total equal to the request
JMP SHORT FINCLUS
FIGREC:
;Inputs:
; DX = Physical cluster number
; BL = Sector postion within cluster
; BP = Base of drive parameters
;Outputs:
; DX = physical sector number
;No other registers affected.
PUSH CX
MOV CL,[BP.CLUSSHFT]
DEC DX
DEC DX
SHL DX,CL
OR DL,BL
ADD DX,[BP.FIRREC]
POP CX
RET
GETREC:
; Inputs:
; DS:DX point to FCB
; Outputs:
; CX = 1
; DX:AX = Record number determined by EXTENT and NR fields
; DS:DI point to FCB
; No other registers affected.
MOV DI,DX
CMP BYTE PTR [DI],-1 ;Check for extended FCB
JNZ NORMFCB2
ADD DI,7
NORMFCB2:
MOV CX,1
MOV AL,[DI.NR]
MOV DX,[DI.EXTENT]
SHL AL,1
SHR DX,1
RCR AL,1
MOV AH,DL
MOV DL,DH
MOV DH,0
RET
ALLOCATE:
; Inputs:
; DS = CS
; ES = Segment of FCB
; BX = Last cluster of file (0 if null file)
; CX = No. of clusters to allocate
; DX = Position of cluster BX
; BP = Base of drive parameters
; SI = FAT pointer
; [FCB] = Displacement of FCB within segment
; Outputs:
; IF insufficient space
; THEN
; Carry set
; CX = max. no. of records that could be added to file
; ELSE
; Carry clear
; BX = First cluster allocated
; FAT is fully updated including dirty bit
; FIRCLUS field of FCB set if file was null
; SI,BP unchanged. All other registers destroyed.
PUSH [SI]
PUSH DX
PUSH CX
PUSH BX
MOV AX,BX
ALLOC:
MOV DX,BX
FINDFRE:
INC BX
CMP BX,[BP.MAXCLUS]
JLE TRYOUT
CMP AX,1
JG TRYIN
POP BX
MOV DX,0FFFH
CALL RELBLKS
POP AX ;No. of clusters requested
SUB AX,CX ;AX=No. of clusters allocated
POP DX
POP [SI]
INC DX ;Position of first cluster allocated
ADD AX,DX ;AX=max no. of cluster in file
MOV DL,[BP.CLUSMSK]
MOV DH,0
INC DX ;DX=records/cluster
MUL DX ;AX=max no. of records in file
MOV CX,AX
SUB CX,WORD PTR [RECPOS] ;CX=max no. of records that could be written
JA MAXREC
XOR CX,CX ;If CX was negative, zero it
MAXREC:
STC
RET11: RET
TRYOUT:
CALL UNPACK
JZ HAVFRE
TRYIN:
DEC AX
JLE FINDFRE
XCHG AX,BX
CALL UNPACK
JZ HAVFRE
XCHG AX,BX
JMP SHORT FINDFRE
HAVFRE:
XCHG BX,DX
MOV AX,DX
CALL PACK
MOV BX,AX
LOOP ALLOC
MOV DX,0FFFH
CALL PACK
MOV BYTE PTR [SI-1],1
POP BX
POP CX ;Don't need this stuff since we're successful
POP DX
CALL UNPACK
POP [SI]
XCHG BX,DI
OR DI,DI
JNZ RET11
MOV DI,[FCB]
MOV ES:[DI.FIRCLUS],BX
RET12: RET
RELEASE:
; Inputs:
; DS = CS
; BX = Cluster in file
; SI = FAT pointer
; BP = Base of drive parameters
; Function:
; Frees cluster chain starting with [BX]
; AX,BX,DX,DI all destroyed. Other registers unchanged.
XOR DX,DX
RELBLKS:
; Enter here with DX=0FFFH to put an end-of-file mark
; in the first cluster and free the rest in the chain.
CALL UNPACK
JZ RET12
MOV AX,DI
CALL PACK
CMP AX,0FF8H
MOV BX,AX
JB RELEASE
RET13: RET
GETEOF:
; Inputs:
; BX = Cluster in a file
; SI = Base of drive FAT
; DS = CS
; Outputs:
; BX = Last cluster in the file
; DI destroyed. No other registers affected.
CALL UNPACK
CMP DI,0FF8H
JAE RET13
MOV BX,DI
JMP SHORT GETEOF
SRCHFRST: ;System call 17
CALL GETFILE
SAVPLCE:
; Search-for-next enters here to save place and report
; findings.
JC KILLSRCH
OR BH,BH
JS SRCHDEV
MOV AX,[LASTENT]
MOV ES:[DI.FILDIRENT],AX
MOV ES:[DI.DRVBP],BP
;Information in directory entry must be copied into the first
; 33 bytes starting at the disk transfer address.
MOV SI,BX
LES DI,DWORD PTR [DMAADD]
MOV AX,00FFH
CMP AL,[EXTFCB]
JNZ NORMFCB
STOSW
INC AL
STOSW
STOSW
MOV AL,[ATTRIB]
STOSB
NORMFCB:
MOV AL,[THISDRV]
INC AL
STOSB ;Set drive number
MOV CX,16
REP MOVSW ;Copy remaining 10 characters of name
XOR AL,AL
RET
KILLSRCH:
KILLSRCH1 EQU KILLSRCH+1
;The purpose of the KILLSRCH1 label is to provide a jump label to the following
; instruction which leaves out the segment override.
MOV WORD PTR ES:[DI.FILDIRENT],-1
MOV AL,-1
RET
SRCHDEV:
MOV ES:[DI.FILDIRENT],BX
LES DI,DWORD PTR [DMAADD]
XOR AX,AX
STOSB ;Zero drive byte
SUB SI,4 ;Point to device name
MOVSW
MOVSW
MOV AX,2020H
STOSB
STOSW
STOSW
STOSW ;Fill with 8 blanks
XOR AX,AX
MOV CX,10
REP STOSW
STOSB
RET14: RET
SRCHNXT: ;System call 18
CALL MOVNAME
MOV DI,DX
JC NEAR PTR KILLSRCH1
MOV BP,[DI.DRVBP]
MOV AX,[DI.FILDIRENT]
OR AX,AX
JS NEAR PTR KILLSRCH1
PUSH DX
PUSH DS
PUSH CS
POP DS
MOV [LASTENT],AX
CALL CONTSRCH
POP ES
POP DI
JMP SAVPLCE
FILESIZE: ;System call 35
CALL GETFILE
MOV AL,-1
JC RET14
ADD DI,33 ;Write size in RR field
MOV CX,ES:[DI.RECSIZ-33]
OR CX,CX
JNZ RECOK
MOV CX,128
RECOK:
XOR AX,AX
XOR DX,DX ;Intialize size to zero
OR BH,BH ;Check for named I/O device
JS DEVSIZ
INC SI
INC SI ;Point to length field
MOV AX,[SI+2] ;Get high word of size
DIV CX
PUSH AX ;Save high part of result
LODSW ;Get low word of size
DIV CX
OR DX,DX ;Check for zero remainder
POP DX
JZ DEVSIZ
INC AX ;Round up for partial record
JNZ DEVSIZ ;Propagate carry?
INC DX
DEVSIZ:
STOSW
MOV AX,DX
STOSB
MOV AL,0
CMP CX,64
JAE RET14 ;Only 3-byte field if RECSIZ >= 64
MOV ES:[DI],AH
RET
SETDMA: ;System call 26
MOV CS:[DMAADD],DX
MOV CS:[DMAADD+2],DS
RET
NOSUCHDRV:
MOV AL,-1
RET
GETFATPT: ;System call 27
MOV DL,0 ;Use default drive
GETFATPTDL: ;System call 28
PUSH CS
POP DS
MOV AL,DL
CALL GETTHISDRV
JC NOSUCHDRV
CALL FATREAD
MOV BX,[BP.FAT]
MOV AL,[BP.CLUSMSK]
INC AL
MOV DX,[BP.MAXCLUS]
DEC DX
MOV CX,[BP.SECSIZ]
LDS SI,DWORD PTR [SPSAVE]
MOV [SI.BXSAVE],BX
MOV [SI.DXSAVE],DX
MOV [SI.CXSAVE],CX
MOV [SI.DSSAVE],CS
RET
GETDSKPT: ;System call 31
PUSH CS
POP DS
MOV AL,[CURDRV]
MOV [THISDRV],AL
CALL FATREAD
LDS SI,DWORD PTR [SPSAVE]
MOV [SI.BXSAVE],BP
MOV [SI.DSSAVE],CS
RET
DSKRESET: ;System call 13
PUSH CS
POP DS
WRTFATS:
; DS=CS. Writes back all dirty FATs. All registers destroyed.
XOR AL,AL
XCHG AL,[DIRTYBUF]
OR AL,AL
JZ NOBUF
MOV BP,[BUFDRVBP]
MOV DX,[BUFSECNO]
MOV BX,[BUFFER]
MOV CX,1
CALL DWRITE
NOBUF:
MOV CL,[NUMIO]
MOV CH,0
MOV BP,[DRVTAB]
WRTFAT:
PUSH CX
CALL CHKFATWRT
POP CX
ADD BP,DPBSIZ
LOOP WRTFAT
RET
GETDRV: ;System call 25
MOV AL,CS:[CURDRV]
RET15: RET
SETRNDREC: ;System call 36
CALL GETREC
MOV [DI+33],AX
MOV [DI+35],DL
CMP [DI.RECSIZ],64
JAE RET15
MOV [DI+36],DH ;Set 4th byte only if record size < 64
RET16: RET
SELDSK: ;System call 14
MOV AL,CS:[NUMDRV]
CMP DL,AL
JNB RET17
MOV CS:[CURDRV],DL
RET17: RET
BUFIN: ;System call 10
MOV AX,CS
MOV ES,AX
MOV SI,DX
MOV CH,0
LODSW
OR AL,AL
JZ RET17
MOV BL,AH
MOV BH,CH
CMP AL,BL
JBE NOEDIT
CMP BYTE PTR [BX+SI],0DH
JZ EDITON
NOEDIT:
MOV BL,CH
EDITON:
MOV DL,AL
DEC DX
NEWLIN:
MOV AL,CS:[CARPOS]
MOV CS:[STARTPOS],AL
PUSH SI
MOV DI,OFFSET DOSGROUP:INBUF
MOV AH,CH
MOV BH,CH
MOV DH,CH
GETCH:
CALL IN
CMP AL,"F"-"@" ;Ignore ^F
JZ GETCH
CMP AL,CS:ESCCHAR
JZ ESC
CMP AL,7FH
JZ BACKSP
CMP AL,8
JZ BACKSP
CMP AL,13
JZ ENDLIN
CMP AL,10
JZ PHYCRLF
CMP AL,CANCEL
JZ KILNEW
SAVCH:
CMP DH,DL
JAE BUFFUL
STOSB
INC DH
CALL BUFOUT
OR AH,AH
JNZ GETCH
CMP BH,BL
JAE GETCH
INC SI
INC BH
JMP SHORT GETCH
BUFFUL:
MOV AL,7
CALL OUT
JMP SHORT GETCH
ESC:
CALL IN
MOV CL,ESCTABLEN
PUSH DI
MOV DI,OFFSET DOSGROUP:ESCTAB
REPNE SCASB
POP DI
SHL CX,1
MOV BP,CX
JMP [BP+OFFSET DOSGROUP:ESCFUNC]
ENDLIN:
STOSB
CALL OUT
POP DI
MOV [DI-1],DH
INC DH
COPYNEW:
MOV BP,ES
MOV BX,DS
MOV ES,BX
MOV DS,BP
MOV SI,OFFSET DOSGROUP:INBUF
MOV CL,DH
REP MOVSB
RET
CRLF:
MOV AL,13
CALL OUT
MOV AL,10
JMP OUT
PHYCRLF:
CALL CRLF
JMP SHORT GETCH
KILNEW:
MOV AL,"\"
CALL OUT
POP SI
PUTNEW:
CALL CRLF
MOV AL,CS:[STARTPOS]
CALL TAB
JMP NEWLIN
BACKSP:
OR DH,DH
JZ OLDBAK
CALL BACKUP
MOV AL,ES:[DI]
CMP AL," "
JAE OLDBAK
CMP AL,9
JZ BAKTAB
CALL BACKMES
OLDBAK:
OR AH,AH
JNZ GETCH1
OR BH,BH
JZ GETCH1
DEC BH
DEC SI
GETCH1:
JMP GETCH
BAKTAB:
PUSH DI
DEC DI
STD
MOV CL,DH
MOV AL," "
PUSH BX
MOV BL,7
JCXZ FIGTAB
FNDPOS:
SCASB
JNA CHKCNT
CMP ES:BYTE PTR [DI+1],9
JZ HAVTAB
DEC BL
CHKCNT:
LOOP FNDPOS
FIGTAB:
SUB BL,CS:[STARTPOS]
HAVTAB:
SUB BL,DH
ADD CL,BL
AND CL,7
CLD
POP BX
POP DI
JZ OLDBAK
TABBAK:
CALL BACKMES
LOOP TABBAK
JMP SHORT OLDBAK
BACKUP:
DEC DH
DEC DI
BACKMES:
MOV AL,8
CALL OUT
MOV AL," "
CALL OUT
MOV AL,8
JMP OUT
TWOESC:
MOV AL,ESCCH
JMP SAVCH
COPYLIN:
MOV CL,BL
SUB CL,BH
JMP SHORT COPYEACH
COPYSTR:
CALL FINDOLD
JMP SHORT COPYEACH
COPYONE:
MOV CL,1
COPYEACH:
MOV AH,0
CMP DH,DL
JZ GETCH2
CMP BH,BL
JZ GETCH2
LODSB
STOSB
CALL BUFOUT
INC BH
INC DH
LOOP COPYEACH
GETCH2:
JMP GETCH
SKIPONE:
CMP BH,BL
JZ GETCH2
INC BH
INC SI
JMP GETCH
SKIPSTR:
CALL FINDOLD
ADD SI,CX
ADD BH,CL
JMP GETCH
FINDOLD:
CALL IN
MOV CL,BL
SUB CL,BH
JZ NOTFND
DEC CX
JZ NOTFND
PUSH ES
PUSH DS
POP ES
PUSH DI
MOV DI,SI
INC DI
REPNE SCASB
POP DI
POP ES
JNZ NOTFND
NOT CL
ADD CL,BL
SUB CL,BH
RET30: RET
NOTFND:
POP BP
JMP GETCH
REEDIT:
MOV AL,"@"
CALL OUT
POP DI
PUSH DI
PUSH ES
PUSH DS
CALL COPYNEW
POP DS
POP ES
POP SI
MOV BL,DH
JMP PUTNEW
ENTERINS:
IF TOGLINS
NOT AH
JMP GETCH
ENDIF
IF NOT TOGLINS
MOV AH,-1
JMP GETCH
EXITINS:
MOV AH,0
JMP GETCH
ENDIF
ESCFUNC DW GETCH
DW TWOESC
IF NOT TOGLINS
DW EXITINS
ENDIF
DW ENTERINS
DW BACKSP
DW REEDIT
DW KILNEW
DW COPYLIN
DW SKIPSTR
DW COPYSTR
DW SKIPONE
DW COPYONE
IF IBM
DW COPYONE
DW CTRLZ
CTRLZ:
MOV AL,"Z"-"@"
JMP SAVCH
ENDIF
BUFOUT:
CMP AL," "
JAE OUT
CMP AL,9
JZ OUT
PUSH AX
MOV AL,"^"
CALL OUT
POP AX
OR AL,40H
JMP SHORT OUT
NOSTOP:
CMP AL,"P"-"@"
JZ INCHK
IF NOT TOGLPRN
CMP AL,"N"-"@"
JZ INCHK
ENDIF
CMP AL,"C"-"@"
JZ INCHK
RET
CONOUT: ;System call 2
MOV AL,DL
OUT:
CMP AL,20H
JB CTRLOUT
CMP AL,7FH
JZ OUTCH
INC CS:BYTE PTR [CARPOS]
OUTCH:
PUSH AX
CALL STATCHK
POP AX
CALL FAR PTR BIOSOUT
TEST CS:BYTE PTR [PFLAG],-1
JZ RET18
CALL FAR PTR BIOSPRINT
RET18: RET
STATCHK:
CALL FAR PTR BIOSSTAT
JZ RET18
CMP AL,'S'-'@'
JNZ NOSTOP
CALL FAR PTR BIOSIN ;Eat Cntrl-S
INCHK:
CALL FAR PTR BIOSIN
CMP AL,'P'-'@'
JZ PRINTON
IF NOT TOGLPRN
CMP AL,'N'-'@'
JZ PRINTOFF
ENDIF
CMP AL,'C'-'@'
JNZ RET18
; Ctrl-C handler.
; "^C" and CR/LF is printed. Then the user registers are restored and the
; user CTRL-C handler is executed. At this point the top of the stack has
; 1) the interrupt return address should the user CTRL-C handler wish to
; allow processing to continue; 2) the original interrupt return address
; to the code that performed the function call in the first place. If the
; user CTRL-C handler wishes to continue, it must leave all registers
; unchanged and IRET. The function that was interrupted will simply be
; repeated.
MOV AL,3 ;Display "^C"
CALL BUFOUT
CALL CRLF
CLI ;Prepare to play with stack
MOV SS,CS:[SSSAVE]
MOV SP,CS:[SPSAVE] ;User stack now restored
POP AX
POP BX
POP CX
POP DX
POP SI
POP DI
POP BP
POP DS
POP ES ;User registers now restored
INT CONTC ;Execute user Ctrl-C handler
JMP COMMAND ;Repeat command otherwise
PRINTON:
IF TOGLPRN
NOT CS:BYTE PTR [PFLAG]
RET
ENDIF
IF NOT TOGLPRN
MOV CS:BYTE PTR [PFLAG],1
RET
PRINTOFF:
MOV CS:BYTE PTR [PFLAG],0
RET
ENDIF
CTRLOUT:
CMP AL,13
JZ ZERPOS
CMP AL,8
JZ BACKPOS
CMP AL,9
JNZ OUTCHJ
MOV AL,CS:[CARPOS]
OR AL,0F8H
NEG AL
TAB:
PUSH CX
MOV CL,AL
MOV CH,0
JCXZ POPTAB
TABLP:
MOV AL," "
CALL OUT
LOOP TABLP
POPTAB:
POP CX
RET19: RET
ZERPOS:
MOV CS:BYTE PTR [CARPOS],0
OUTCHJ: JMP OUTCH
BACKPOS:
DEC CS:BYTE PTR [CARPOS]
JMP OUTCH
CONSTAT: ;System call 11
CALL STATCHK
MOV AL,0
JZ RET19
OR AL,-1
RET
CONIN: ;System call 1
CALL IN
PUSH AX
CALL OUT
POP AX
RET
IN: ;System call 8
CALL INCHK
JZ IN
RET29: RET
RAWIO: ;System call 6
MOV AL,DL
CMP AL,-1
JNZ RAWOUT
LDS SI,DWORD PTR CS:[SPSAVE] ;Get pointer to register save area
CALL FAR PTR BIOSSTAT
JNZ RESFLG
OR BYTE PTR [SI.FSAVE],40H ;Set user's zero flag
XOR AL,AL
RET
RESFLG:
AND BYTE PTR [SI.FSAVE],0FFH-40H ;Reset user's zero flag
RAWINP: ;System call 7
CALL FAR PTR BIOSIN
RET
RAWOUT:
CALL FAR PTR BIOSOUT
RET
LIST: ;System call 5
MOV AL,DL
LISTOUT:
PUSH AX
CALL STATCHK
POP AX
CALL FAR PTR BIOSPRINT
RET20: RET
PRTBUF: ;System call 9
MOV SI,DX
OUTSTR:
LODSB
CMP AL,"$"
JZ RET20
CALL OUT
JMP SHORT OUTSTR
OUTMES: ;String output for internal messages
LODS CS:BYTE PTR [SI]
CMP AL,"$"
JZ RET20
CALL OUT
JMP SHORT OUTMES
MAKEFCB: ;Interrupt call 41
DRVBIT EQU 2
NAMBIT EQU 4
EXTBIT EQU 8
MOV DL,0 ;Flag--not ambiguous file name
TEST AL,DRVBIT ;Use current drive field if default?
JNZ DEFDRV
MOV BYTE PTR ES:[DI],0 ;No - use default drive
DEFDRV:
INC DI
MOV CX,8
TEST AL,NAMBIT ;Use current name fiels as defualt?
XCHG AX,BX ;Save bits in BX
MOV AL," "
JZ FILLB ;If not, go fill with blanks
ADD DI,CX
XOR CX,CX ;Don't fill any
FILLB:
REP STOSB
MOV CL,3
TEST BL,EXTBIT ;Use current extension as default
JZ FILLB2
ADD DI,CX
XOR CX,CX
FILLB2:
REP STOSB
XCHG AX,CX ;Put zero in AX
STOSW
STOSW ;Initialize two words after to zero
SUB DI,16 ;Point back at start
TEST BL,1 ;Scan off separators if not zero
JZ SKPSPC
CALL SCANB ;Peel off blanks and tabs
CALL DELIM ;Is it a one-time-only delimiter?
JNZ NOSCAN
INC SI ;Skip over the delimiter
SKPSPC:
CALL SCANB ;Always kill preceding blanks and tabs
NOSCAN:
CALL GETLET
JBE NODRV ;Quit if termination character
CMP BYTE PTR[SI],":" ;Check for potential drive specifier
JNZ NODRV
INC SI ;Skip over colon
SUB AL,"@" ;Convert drive letter to binary drive number
JBE BADDRV ;Valid drive numbers are 1-15
CMP AL,CS:[NUMDRV]
JBE HAVDRV
BADDRV:
MOV DL,-1
HAVDRV:
STOSB ;Put drive specifier in first byte
INC SI
DEC DI ;Counteract next two instructions
NODRV:
DEC SI ;Back up
INC DI ;Skip drive byte
MOV CX,8
CALL GETWORD ;Get 8-letter file name
CMP BYTE PTR [SI],"."
JNZ NODOT
INC SI ;Skip over dot if present
MOV CX,3 ;Get 3-letter extension
CALL MUSTGETWORD
NODOT:
LDS BX,CS:DWORD PTR [SPSAVE]
MOV [BX.SISAVE],SI
MOV AL,DL
RET
NONAM:
ADD DI,CX
DEC SI
RET
GETWORD:
CALL GETLET
JBE NONAM ;Exit if invalid character
DEC SI
MUSTGETWORD:
CALL GETLET
JBE FILLNAM
JCXZ MUSTGETWORD
DEC CX
CMP AL,"*" ;Check for ambiguous file specifier
JNZ NOSTAR
MOV AL,"?"
REP STOSB
NOSTAR:
STOSB
CMP AL,"?"
JNZ MUSTGETWORD
OR DL,1 ;Flag ambiguous file name
JMP MUSTGETWORD
FILLNAM:
MOV AL," "
REP STOSB
DEC SI
RET21: RET
SCANB:
LODSB
CALL SPCHK
JZ SCANB
DEC SI
RET
GETLET:
;Get a byte from [SI], convert it to upper case, and compare for delimiter.
;ZF set if a delimiter, CY set if a control character (other than TAB).
LODSB
AND AL,7FH
CMP AL,"a"
JB CHK
CMP AL,"z"
JA CHK
SUB AL,20H ;Convert to upper case
CHK:
CMP AL,"."
JZ RET21
CMP AL,'"'
JZ RET21
CMP AL,"/"
JZ RET21
CMP AL,"["
JZ RET21
CMP AL,"]"
JZ RET21
IF IBM
DELIM:
ENDIF
CMP AL,":" ;Allow ":" as separator in IBM version
JZ RET21
IF NOT IBM
DELIM:
ENDIF
CMP AL,"+"
JZ RET101
CMP AL,"="
JZ RET101
CMP AL,";"
JZ RET101
CMP AL,","
JZ RET101
SPCHK:
CMP AL,9 ;Filter out tabs too
JZ RET101
;WARNING! " " MUST be the last compare
CMP AL," "
RET101: RET
SETVECT: ; Interrupt call 37
XOR BX,BX
MOV ES,BX
MOV BL,AL
SHL BX,1
SHL BX,1
MOV ES:[BX],DX
MOV ES:[BX+2],DS
RET
NEWBASE: ; Interrupt call 38
MOV ES,DX
LDS SI,CS:DWORD PTR [SPSAVE]
MOV DS,[SI.CSSAVE]
XOR SI,SI
MOV DI,SI
MOV AX,DS:[2]
MOV CX,80H
REP MOVSW
SETMEM:
; Inputs:
; AX = Size of memory in paragraphs
; DX = Segment
; Function:
; Completely prepares a program base at the
; specified segment.
; Outputs:
; DS = DX
; ES = DX
; [0] has INT 20H
; [2] = First unavailable segment ([ENDMEM])
; [5] to [9] form a long call to the entry point
; [10] to [13] have exit address (from INT 22H)
; [14] to [17] have ctrl-C exit address (from INT 23H)
; [18] to [21] have fatal error address (from INT 24H)
; DX,BP unchanged. All other registers destroyed.
XOR CX,CX
MOV DS,CX
MOV ES,DX
MOV SI,EXIT
MOV DI,SAVEXIT
MOVSW
MOVSW
MOVSW
MOVSW
MOVSW
MOVSW
MOV ES:[2],AX
SUB AX,DX
CMP AX,MAXDIF
JBE HAVDIF
MOV AX,MAXDIF
HAVDIF:
MOV BX,ENTRYPOINTSEG
SUB BX,AX
SHL AX,1
SHL AX,1
SHL AX,1
SHL AX,1
MOV DS,DX
MOV DS:[6],AX
MOV DS:[8],BX
MOV DS:[0],20CDH ;"INT INTTAB"
MOV DS:(BYTE PTR [5]),LONGCALL
RET
DATE16:
PUSH CX
CALL READTIME
SHL CL,1 ;Minutes to left part of byte
SHL CL,1
SHL CX,1 ;Push hours and minutes to left end
SHL CX,1
SHL CX,1
SHR DH,1 ;Count every two seconds
OR CL,DH ;Combine seconds with hours and minutes
MOV DX,CX
POP CX
MOV AX,WORD PTR [MONTH] ;Fetch month and year
SHL AL,1 ;Push month to left to make room for day
SHL AL,1
SHL AL,1
SHL AL,1
SHL AX,1
OR AL,[DAY]
RET22: RET
FOURYEARS EQU 3*365+366
READTIME:
;Gets time in CX:DX. Figures new date if it has changed.
;Uses AX, CX, DX.
CALL FAR PTR BIOSGETTIME
CMP AX,[DAYCNT] ;See if day count is the same
JZ RET22
CMP AX,FOURYEARS*30 ;Number of days in 120 years
JAE RET22 ;Ignore if too large
MOV [DAYCNT],AX
PUSH SI
PUSH CX
PUSH DX ;Save time
XOR DX,DX
MOV CX,FOURYEARS ;Number of days in 4 years
DIV CX ;Compute number of 4-year units
SHL AX,1
SHL AX,1
SHL AX,1 ;Multiply by 8 (no. of half-years)
MOV CX,AX ;<240 implies AH=0
MOV SI,OFFSET DOSGROUP:YRTAB ;Table of days in each year
CALL DSLIDE ;Find out which of four years we're in
SHR CX,1 ;Convert half-years to whole years
JNC SK ;Extra half-year?
ADD DX,200
SK:
CALL SETYEAR
MOV CL,1 ;At least at first month in year
MOV SI,OFFSET DOSGROUP:MONTAB ;Table of days in each month
CALL DSLIDE ;Find out which month we're in
MOV [MONTH],CL
INC DX ;Remainder is day of month (start with one)
MOV [DAY],DL
CALL WKDAY ;Set day of week
POP DX
POP CX
POP SI
RET23: RET
DSLIDE:
MOV AH,0
DSLIDE1:
LODSB ;Get count of days
CMP DX,AX ;See if it will fit
JB RET23 ;If not, done
SUB DX,AX
INC CX ;Count one more month/year
JMP SHORT DSLIDE1
SETYEAR:
;Set year with value in CX. Adjust length of February for this year.
MOV BYTE PTR [YEAR],CL
CHKYR:
TEST CL,3 ;Check for leap year
MOV AL,28
JNZ SAVFEB ;28 days if no leap year
INC AL ;Add leap day
SAVFEB:
MOV [MONTAB+1],AL ;Store for February
RET
;Days in year
YRTAB DB 200,166 ;Leap year
DB 200,165
DB 200,165
DB 200,165
;Days of each month
MONTAB DB 31 ;January
DB 28 ;February--reset each time year changes
DB 31 ;March
DB 30 ;April
DB 31 ;May
DB 30 ;June
DB 31 ;July
DB 31 ;August
DB 30 ;September
DB 31 ;October
DB 30 ;November
DB 31 ;December
GETDATE: ;Function call 42
PUSH CS
POP DS
CALL READTIME ;Check for rollover to next day
MOV AX,[YEAR]
MOV BX,WORD PTR [DAY]
LDS SI,DWORD PTR [SPSAVE] ;Get pointer to user registers
MOV [SI.DXSAVE],BX ;DH=month, DL=day
ADD AX,1980 ;Put bias back
MOV [SI.CXSAVE],AX ;CX=year
MOV AL,CS:[WEEKDAY]
RET24: RET
SETDATE: ;Function call 43
MOV AL,-1 ;Be ready to flag error
SUB CX,1980 ;Fix bias in year
JC RET24 ;Error if not big enough
CMP CX,119 ;Year must be less than 2100
JA RET24
OR DH,DH
JZ RET24
OR DL,DL
JZ RET24 ;Error if either month or day is 0
CMP DH,12 ;Check against max. month
JA RET24
PUSH CS
POP DS
CALL CHKYR ;Set Feb. up for new year
MOV AL,DH
MOV BX,OFFSET DOSGROUP:MONTAB-1
XLAT ;Look up days in month
CMP AL,DL
MOV AL,-1 ;Restore error flag, just in case
JB RET24 ;Error if too many days
CALL SETYEAR
MOV WORD PTR [DAY],DX ;Set both day and month
SHR CX,1
SHR CX,1
MOV AX,FOURYEARS
MOV BX,DX
MUL CX
MOV CL,BYTE PTR [YEAR]
AND CL,3
MOV SI,OFFSET DOSGROUP:YRTAB
MOV DX,AX
SHL CX,1 ;Two entries per year, so double count
CALL DSUM ;Add up the days in each year
MOV CL,BH ;Month of year
MOV SI,OFFSET DOSGROUP:MONTAB
DEC CX ;Account for months starting with one
CALL DSUM ;Add up days in each month
MOV CL,BL ;Day of month
DEC CX ;Account for days starting with one
ADD DX,CX ;Add in to day total
XCHG AX,DX ;Get day count in AX
MOV [DAYCNT],AX
CALL FAR PTR BIOSSETDATE
WKDAY:
MOV AX,[DAYCNT]
XOR DX,DX
MOV CX,7
INC AX
INC AX ;First day was Tuesday
DIV CX ;Compute day of week
MOV [WEEKDAY],DL
XOR AL,AL ;Flag OK
RET25: RET
DSUM:
MOV AH,0
JCXZ RET25
DSUM1:
LODSB
ADD DX,AX
LOOP DSUM1
RET
GETTIME: ;Function call 44
PUSH CS
POP DS
CALL READTIME
LDS SI,DWORD PTR [SPSAVE] ;Get pointer to user registers
MOV [SI.DXSAVE],DX
MOV [SI.CXSAVE],CX
XOR AL,AL
RET26: RET
SETTIME: ;Function call 45
;Time is in CX:DX in hours, minutes, seconds, 1/100 sec.
MOV AL,-1 ;Flag in case of error
CMP CH,24 ;Check hours
JAE RET26
CMP CL,60 ;Check minutes
JAE RET26
CMP DH,60 ;Check seconds
JAE RET26
CMP DL,100 ;Check 1/100's
JAE RET26
CALL FAR PTR BIOSSETTIME
XOR AL,AL
RET
; Default handler for division overflow trap
DIVOV:
PUSH SI
PUSH AX
MOV SI,OFFSET DOSGROUP:DIVMES
CALL OUTMES
POP AX
POP SI
INT 23H ;Use Ctrl-C abort on divide overflow
IRET
CODSIZ EQU $-CODSTRT ;Size of code segment
CODE ENDS
;***** DATA AREA *****
CONSTANTS SEGMENT BYTE
ORG 0
CONSTRT EQU $ ;Start of constants segment
IONAME:
IF NOT IBM
DB "PRN ","LST ","NUL ","AUX ","CON "
ENDIF
IF IBM
DB "COM1","PRN ","LPT1","NUL ","AUX ","CON "
ENDIF
DIVMES DB 13,10,"Divide overflow",13,10,"$"
CARPOS DB 0
STARTPOS DB 0
PFLAG DB 0
DIRTYDIR DB 0 ;Dirty buffer flag
NUMDRV DB 0 ;Number of drives
NUMIO DB ? ;Number of disk tables
VERFLG DB 0 ;Initialize with verify off
CONTPOS DW 0
DMAADD DW 80H ;User's disk transfer address (disp/seg)
DW ?
ENDMEM DW ?
MAXSEC DW 0
BUFFER DW ?
BUFSECNO DW 0
BUFDRVNO DB -1
DIRTYBUF DB 0
BUFDRVBP DW ?
DIRBUFID DW -1
DAY DB 0
MONTH DB 0
YEAR DW 0
DAYCNT DW -1
WEEKDAY DB 0
CURDRV DB 0 ;Default to drive A
DRVTAB DW 0 ;Address of start of DPBs
DOSLEN EQU CODSIZ+($-CONSTRT) ;Size of CODE + CONSTANTS segments
CONSTANTS ENDS
DATA SEGMENT WORD
; Init code overlaps with data area below
ORG 0
INBUF DB 128 DUP (?)
CONBUF DB 131 DUP (?) ;The rest of INBUF and console buffer
LASTENT DW ?
EXITHOLD DB 4 DUP (?)
FATBASE DW ?
NAME1 DB 11 DUP (?) ;File name buffer
ATTRIB DB ?
NAME2 DB 11 DUP (?)
NAME3 DB 12 DUP (?)
EXTFCB DB ?
;WARNING - the following two items are accessed as a word
CREATING DB ?
DELALL DB ?
TEMP LABEL WORD
SPSAVE DW ?
SSSAVE DW ?
CONTSTK DW ?
SECCLUSPOS DB ? ;Position of first sector within cluster
DSKERR DB ?
TRANS DB ?
PREREAD DB ? ;0 means preread; 1 means optional
READOP DB ?
THISDRV DB ?
EVEN
FCB DW ? ;Address of user FCB
NEXTADD DW ?
RECPOS DB 4 DUP (?)
RECCNT DW ?
LASTPOS DW ?
CLUSNUM DW ?
SECPOS DW ? ;Position of first sector accessed
VALSEC DW ? ;Number of valid (previously written) sectors
BYTSECPOS DW ? ;Position of first byte within sector
BYTPOS DB 4 DUP (?) ;Byte position in file of access
BYTCNT1 DW ? ;No. of bytes in first sector
BYTCNT2 DW ? ;No. of bytes in last sector
SECCNT DW ? ;No. of whole sectors
ENTFREE DW ?
DB 80H DUP (?) ;Stack space
IOSTACK LABEL BYTE
DB 80H DUP (?)
DSKSTACK LABEL BYTE
IF DSKTEST
NSS DW ?
NSP DW ?
ENDIF
DIRBUF LABEL WORD
;Init code below overlaps with data area above
ORG 0
MOVFAT:
;This section of code is safe from being overwritten by block move
REP MOVS BYTE PTR [DI],[SI]
CLD
MOV ES:[DMAADD+2],DX
MOV SI,[DRVTAB] ;Address of first DPB
MOV AL,-1
MOV CL,[NUMIO] ;Number of DPBs
FLGFAT:
MOV DI,ES:[SI.FAT] ;get pointer to FAT
DEC DI ;Point to dirty byte
STOSB ;Flag as unused
ADD SI,DPBSIZ ;Point to next DPB
LOOP FLGFAT
MOV AX,[ENDMEM]
CALL SETMEM ;Set up segment
XXX PROC FAR
RET
XXX ENDP
DOSINIT:
CLI
CLD
PUSH CS
POP ES
MOV ES:[ENDMEM],DX
LODSB ;Get no. of drives & no. of I/O drivers
MOV ES:[NUMIO],AL
MOV DI,OFFSET DOSGROUP:MEMSTRT
PERDRV:
MOV BP,DI
MOV AL,ES:[DRVCNT]
STOSB ;DEVNUM
LODSB ;Physical unit no.
STOSB ;DRVNUM
CMP AL,15
JA BADINIT
CBW ;Index into FAT size table
SHL AX,1
ADD AX,OFFSET DOSGROUP:FATSIZTAB
XCHG BX,AX
LODSW ;Pointer to DPT
PUSH SI
MOV SI,AX
LODSW
STOSW ;SECSIZ
MOV DX,AX
CMP AX,ES:[MAXSEC]
JBE NOTMAX
MOV ES:[MAXSEC],AX
NOTMAX:
LODSB
DEC AL
STOSB ;CLUSMSK
JZ HAVSHFT
CBW
FIGSHFT:
INC AH
SAR AL,1
JNZ FIGSHFT
MOV AL,AH
HAVSHFT:
STOSB ;CLUSSHFT
MOVSW ;FIRFAT (= number of reserved sectors)
MOVSB ;FATCNT
MOVSW ;MAXENT
MOV AX,DX ;SECSIZ again
MOV CL,5
SHR AX,CL
MOV CX,AX ;Directory entries per sector
DEC AX
ADD AX,ES:[BP.MAXENT]
XOR DX,DX
DIV CX
STOSW ;DIRSEC (temporarily)
MOVSW ;DSKSIZ (temporarily)
FNDFATSIZ:
MOV AL,1
MOV DX,1
GETFATSIZ:
PUSH DX
CALL FIGFATSIZ
POP DX
CMP AL,DL ;Compare newly computed FAT size with trial
JZ HAVFATSIZ ;Has sequence converged?
CMP AL,DH ;Compare with previous trial
MOV DH,DL
MOV DL,AL ;Shuffle trials
JNZ GETFATSIZ ;Continue iterations if not oscillating
DEC WORD PTR ES:[BP.DSKSIZ] ;Damp those oscillations
JMP SHORT FNDFATSIZ ;Try again
BADINIT:
MOV SI,OFFSET DOSGROUP:BADMES
CALL OUTMES
STI
HLT
HAVFATSIZ:
STOSB ;FATSIZ
MUL ES:BYTE PTR[BP.FATCNT] ;Space occupied by all FATs
ADD AX,ES:[BP.FIRFAT]
STOSW ;FIRDIR
ADD AX,ES:[BP.DIRSEC]
MOV ES:[BP.FIRREC],AX ;Destroys DIRSEC
CALL FIGMAX
MOV ES:[BP.MAXCLUS],CX
MOV AX,BX ;Pointer into FAT size table
STOSW ;Allocate space for FAT pointer
MOV AL,ES:[BP.FATSIZ]
XOR AH,AH
MUL ES:[BP.SECSIZ]
CMP AX,ES:[BX] ;Bigger than already allocated
JBE SMFAT
MOV ES:[BX],AX
SMFAT:
POP SI ;Restore pointer to init. table
MOV AL,ES:[DRVCNT]
INC AL
MOV ES:[DRVCNT],AL
CMP AL,ES:[NUMIO]
JAE CONTINIT
JMP PERDRV
BADINITJ:
JMP BADINIT
CONTINIT:
PUSH CS
POP DS
;Calculate true address of buffers, FATs, free space
MOV BP,[MAXSEC]
MOV AX,OFFSET DOSGROUP:DIRBUF
ADD AX,BP
MOV [BUFFER],AX ;Start of buffer
ADD AX,BP
MOV [DRVTAB],AX ;Start of DPBs
SHL BP,1 ;Two sectors - directory and buffer
ADD BP,DI ;Allocate buffer space
ADD BP,ADJFAC ;True address of FATs
PUSH BP
MOV SI,OFFSET DOSGROUP:FATSIZTAB
MOV DI,SI
MOV CX,16
TOTFATSIZ:
INC BP ;Add one for Dirty byte
INC BP ;Add one for I/O device number
LODSW ;Get size of this FAT
XCHG AX,BP
STOSW ;Save address of this FAT
ADD BP,AX ;Compute size of next FAT
CMP AX,BP ;If size was zero done
LOOPNZ TOTFATSIZ
MOV AL,15
SUB AL,CL ;Compute number of FATs used
MOV [NUMDRV],AL
XOR AX,AX ;Set zero flag
REPZ SCASW ;Make sure all other entries are zero
JNZ BADINITJ
ADD BP,15 ;True start of free space
MOV CL,4
SHR BP,CL ;First free segment
MOV DX,CS
ADD DX,BP
MOV BX,0FH
MOV CX,[ENDMEM]
CMP CX,1 ;Use memory scan?
JNZ SETEND
MOV CX,DX ;Start scanning just after DOS
MEMSCAN:
INC CX
JZ SETEND
MOV DS,CX
MOV AL,[BX]
NOT AL
MOV [BX],AL
CMP AL,[BX]
NOT AL
MOV [BX],AL
JZ MEMSCAN
SETEND:
IF HIGHMEM
SUB CX,BP
MOV BP,CX ;Segment of DOS
MOV DX,CS ;Program segment
ENDIF
IF NOT HIGHMEM
MOV BP,CS
ENDIF
; BP has segment of DOS (whether to load high or run in place)
; DX has program segment (whether after DOS or overlaying DOS)
; CX has size of memory in paragraphs (reduced by DOS size if HIGHMEM)
MOV CS:[ENDMEM],CX
IF HIGHMEM
MOV ES,BP
XOR SI,SI
MOV DI,SI
MOV CX,(DOSLEN+1)/2
PUSH CS
POP DS
REP MOVSW ;Move DOS to high memory
ENDIF
XOR AX,AX
MOV DS,AX
MOV ES,AX
MOV DI,INTBASE
MOV AX,OFFSET DOSGROUP:QUIT
STOSW ;Set abort address--displacement
MOV AX,BP
MOV BYTE PTR DS:[ENTRYPOINT],LONGJUMP
MOV WORD PTR DS:[ENTRYPOINT+1],OFFSET DOSGROUP:ENTRY
MOV WORD PTR DS:[ENTRYPOINT+3],AX
MOV WORD PTR DS:[0],OFFSET DOSGROUP:DIVOV ;Set default divide trap address
MOV DS:[2],AX
MOV CX,9
REP STOSW ;Set 5 segments (skip 2 between each)
MOV WORD PTR DS:[INTBASE+4],OFFSET DOSGROUP:COMMAND
MOV WORD PTR DS:[INTBASE+12],OFFSET DOSGROUP:IRET ;Ctrl-C exit
MOV WORD PTR DS:[INTBASE+16],OFFSET DOSGROUP:IRET ;Fatal error exit
MOV AX,OFFSET BIOSREAD
STOSW
MOV AX,BIOSSEG
STOSW
STOSW ;Add 2 to DI
STOSW
MOV WORD PTR DS:[INTBASE+18H],OFFSET BIOSWRITE
MOV WORD PTR DS:[EXIT],100H
MOV WORD PTR DS:[EXIT+2],DX
IF NOT IBM
MOV SI,OFFSET DOSGROUP:HEADER
CALL OUTMES
ENDIF
PUSH CS
POP DS
PUSH CS
POP ES
;Move the FATs into position
MOV AL,[NUMIO]
CBW
XCHG AX,CX
MOV DI,OFFSET DOSGROUP:MEMSTRT.FAT
FATPOINT:
MOV SI,WORD PTR [DI] ;Get address within FAT address table
MOVSW ;Set address of this FAT
ADD DI,DPBSIZ-2 ;Point to next DPB
LOOP FATPOINT
POP CX ;True address of first FAT
MOV SI,OFFSET DOSGROUP:MEMSTRT ;Place to move DPBs from
MOV DI,[DRVTAB] ;Place to move DPBs to
SUB CX,DI ;Total length of DPBs
CMP DI,SI
JBE MOVJMP ;Are we moving to higher or lower memory?
DEC CX ;Move backwards to higher memory
ADD DI,CX
ADD SI,CX
INC CX
STD
MOVJMP:
MOV ES,BP
JMP MOVFAT
FIGFATSIZ:
MUL ES:BYTE PTR[BP.FATCNT]
ADD AX,ES:[BP.FIRFAT]
ADD AX,ES:[BP.DIRSEC]
FIGMAX:
;AX has equivalent of FIRREC
SUB AX,ES:[BP.DSKSIZ]
NEG AX
MOV CL,ES:[BP.CLUSSHFT]
SHR AX,CL
INC AX
MOV CX,AX ;MAXCLUS
INC AX
MOV DX,AX
SHR DX,1
ADC AX,DX ;Size of FAT in bytes
MOV SI,ES:[BP.SECSIZ]
ADD AX,SI
DEC AX
XOR DX,DX
DIV SI
RET
BADMES:
DB 13,10,"INIT TABLE BAD",13,10,"$"
FATSIZTAB:
DW 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0
DRVCNT DB 0
MEMSTRT LABEL WORD
ADJFAC EQU DIRBUF-MEMSTRT
DATA ENDS
END