mirror of https://github.com/microsoft/MS-DOS.git
749 lines
20 KiB
NASM
749 lines
20 KiB
NASM
;
|
||
; system call entry points MSDOS
|
||
;
|
||
|
||
INCLUDE DOSSEG.ASM
|
||
|
||
CODE SEGMENT BYTE PUBLIC 'CODE'
|
||
ASSUME SS:DOSGROUP,CS:DOSGROUP
|
||
|
||
.xlist
|
||
.xcref
|
||
INCLUDE DOSSYM.ASM
|
||
INCLUDE DEVSYM.ASM
|
||
.cref
|
||
.list
|
||
|
||
|
||
i_need YEAR,WORD
|
||
i_need DAY,BYTE
|
||
i_need WeekDay,BYTE
|
||
i_need TimeBuf,6
|
||
i_need BCLOCK,DWORD
|
||
i_need DskErr,BYTE
|
||
i_need Attrib,BYTE
|
||
i_need Name1,BYTE
|
||
i_need Name2,BYTE
|
||
i_need Name3,BYTE
|
||
i_need DelAll,BYTE
|
||
i_need ThisDPB,DWORD
|
||
i_need CurBuf,DWORD
|
||
i_need LastEnt,WORD
|
||
i_need ThisDrv,BYTE
|
||
i_need DirStart,WORD
|
||
i_need DevPt,DWORD
|
||
i_need Creating,BYTE
|
||
i_need VolID,BYTE
|
||
i_need FoundDel,BYTE
|
||
|
||
SUBTTL DATE AND TIME - SYSTEM CALLS 42,43,44,45; S/G DATE,TIME
|
||
PAGE
|
||
procedure $GET_DATE,NEAR ;System call 42
|
||
ASSUME DS:NOTHING,ES:NOTHING
|
||
|
||
; Inputs:
|
||
; None
|
||
; Function:
|
||
; Return current date
|
||
; Returns:
|
||
; Date in CX:DX
|
||
|
||
PUSH SS
|
||
POP DS
|
||
ASSUME DS:DOSGROUP
|
||
invoke READTIME ;Check for rollover to next day
|
||
MOV AX,[YEAR]
|
||
MOV BX,WORD PTR [DAY]
|
||
invoke get_user_stack ;Get pointer to user registers
|
||
ASSUME DS:NOTHING
|
||
MOV [SI.user_DX],BX ;DH=month, DL=day
|
||
ADD AX,1980 ;Put bias back
|
||
MOV [SI.user_CX],AX ;CX=year
|
||
MOV AL,BYTE PTR [WEEKDAY]
|
||
RET
|
||
$GET_DATE ENDP
|
||
|
||
procedure $SET_DATE,NEAR ;System call 43
|
||
ASSUME DS:NOTHING,ES:NOTHING
|
||
|
||
; Inputs:
|
||
; CX:DX valid date
|
||
; Function:
|
||
; Set current date
|
||
; Returns:
|
||
; AL = -1 date bad, = 0 OK
|
||
|
||
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 SS
|
||
POP DS
|
||
ASSUME DS:DOSGROUP
|
||
invoke DODATE
|
||
RET24: RET
|
||
$SET_DATE ENDP
|
||
|
||
procedure $GET_TIME,NEAR ;System call 44
|
||
ASSUME DS:NOTHING,ES:NOTHING
|
||
|
||
; Inputs:
|
||
; None
|
||
; Function:
|
||
; Get current time
|
||
; Returns:
|
||
; Time in CX:DX
|
||
|
||
PUSH SS
|
||
POP DS
|
||
ASSUME DS:DOSGROUP
|
||
invoke READTIME
|
||
invoke get_user_stack ;Get pointer to user registers
|
||
MOV [SI.user_DX],DX
|
||
MOV [SI.user_CX],CX
|
||
XOR AL,AL
|
||
RET26: RET
|
||
$GET_TIME ENDP
|
||
|
||
procedure $SET_TIME,NEAR ;System call 45
|
||
;Time is in CX:DX in hours, minutes, seconds, 1/100 sec.
|
||
ASSUME DS:NOTHING,ES:NOTHING
|
||
|
||
; Inputs:
|
||
; CX:DX = Time
|
||
; Function:
|
||
; Set time
|
||
; Returns:
|
||
; AL = -1 time bad, = 0 OK
|
||
|
||
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
|
||
PUSH CX
|
||
PUSH DX
|
||
PUSH SS
|
||
POP DS
|
||
ASSUME DS:DOSGROUP
|
||
MOV BX,OFFSET DOSGROUP:TIMEBUF
|
||
MOV CX,6
|
||
XOR DX,DX
|
||
MOV AX,DX
|
||
PUSH BX
|
||
invoke SETREAD
|
||
ASSUME ES:DOSGROUP
|
||
PUSH DS
|
||
LDS SI,[BCLOCK]
|
||
ASSUME DS:NOTHING
|
||
invoke DEVIOCALL2 ;Get correct day count
|
||
POP DS
|
||
ASSUME DS:DOSGROUP
|
||
POP BX
|
||
invoke SETWRITE
|
||
POP WORD PTR [TIMEBUF+4]
|
||
POP WORD PTR [TIMEBUF+2]
|
||
LDS SI,[BCLOCK]
|
||
ASSUME DS:NOTHING
|
||
invoke DEVIOCALL2 ;Set the time
|
||
XOR AL,AL
|
||
RET
|
||
$SET_TIME ENDP
|
||
|
||
SUBTTL DISK R/W ROUTINES
|
||
PAGE
|
||
procedure $FCB_SEQ_READ,NEAR ; System call 20
|
||
ASSUME DS:NOTHING,ES:NOTHING
|
||
|
||
; Inputs:
|
||
; DS:DX Points to openned FCB
|
||
; Function:
|
||
; Read next record from file to disk transfer address
|
||
; Returns:
|
||
; AL = 1 EOF record is empty
|
||
; AL = 3 EOF record is partial zero filled
|
||
; AL = 2 No room at disk transfer address
|
||
; AL = 0 All OK
|
||
|
||
invoke GETREC
|
||
invoke LOAD
|
||
JMP SHORT FINSEQ
|
||
|
||
entry $FCB_SEQ_WRITE ; System call 21
|
||
ASSUME DS:NOTHING,ES:NOTHING
|
||
|
||
; Inputs:
|
||
; DS:DX Points to openned FCB
|
||
; Function:
|
||
; Write next record to file from disk transfer address
|
||
; Returns:
|
||
; AL = 1 Disk full
|
||
; AL = 2 No room in disk transfer segment
|
||
; AL = 0 All OK
|
||
|
||
invoke GETREC
|
||
invoke STORE
|
||
FINSEQ:
|
||
JCXZ SETNREX
|
||
ADD AX,1
|
||
ADC DX,0
|
||
JMP SHORT SETNREX
|
||
|
||
entry $FCB_RANDOM_READ ; System call 33
|
||
ASSUME DS:NOTHING,ES:NOTHING
|
||
|
||
; Inputs:
|
||
; DS:DX Points to openned FCB
|
||
; Function:
|
||
; Read record addressed by random record field from file to
|
||
; disk transfer address
|
||
; Returns:
|
||
; AL = 1 EOF record is empty
|
||
; AL = 3 EOF record is partial zero filled
|
||
; AL = 2 No room at disk transfer address
|
||
; AL = 0 All OK
|
||
|
||
invoke GETRRPOS1
|
||
invoke LOAD
|
||
JMP SHORT FINRND
|
||
|
||
entry $FCB_RANDOM_WRITE ; System call 34
|
||
ASSUME DS:NOTHING,ES:NOTHING
|
||
|
||
; Inputs:
|
||
; DS:DX Points to openned FCB
|
||
; Function:
|
||
; Write record addressed by random record field to file from
|
||
; disk transfer address
|
||
; Returns:
|
||
; AL = 1 Disk full
|
||
; AL = 2 No room in disk transfer segment
|
||
; AL = 0 All OK
|
||
|
||
invoke GETRRPOS1
|
||
invoke STORE
|
||
JMP SHORT FINRND
|
||
|
||
entry $FCB_RANDOM_READ_BLOCK ; System call 39
|
||
ASSUME DS:NOTHING,ES:NOTHING
|
||
|
||
; Inputs:
|
||
; DS:DX Points to openned FCB
|
||
; CX = Record count
|
||
; Function:
|
||
; Read CX records starting at random record field from file
|
||
; to disk transfer address
|
||
; Returns:
|
||
; AL = 1 EOF record is empty
|
||
; AL = 3 EOF record is partial zero filled
|
||
; AL = 2 No room at disk transfer address
|
||
; AL = 0 All OK
|
||
; CX = Actual number of records read
|
||
|
||
invoke GETRRPOS
|
||
invoke LOAD
|
||
JMP SHORT FINBLK
|
||
|
||
entry $FCB_RANDOM_WRITE_BLOCK ; System call 40
|
||
ASSUME DS:NOTHING,ES:NOTHING
|
||
|
||
; Inputs:
|
||
; DS:DX Points to openned FCB
|
||
; CX = Record count
|
||
; Function:
|
||
; Write CX records starting at random record field to file
|
||
; from disk transfer address
|
||
; If CX = 0 File is set to length determined from random record field
|
||
; Returns:
|
||
; AL = 1 Disk full
|
||
; AL = 2 No room in disk transfer segment
|
||
; AL = 0 All OK
|
||
; CX = Actual number of records written
|
||
|
||
invoke GETRRPOS
|
||
invoke STORE
|
||
FINBLK:
|
||
invoke get_user_stack
|
||
MOV [SI.user_CX],CX
|
||
entry FINNOSAV
|
||
JCXZ FINRND
|
||
ADD AX,1
|
||
ADC DX,0
|
||
FINRND:
|
||
MOV WORD PTR ES:[DI.fcb_RR],AX
|
||
MOV ES:[DI.fcb_RR+2],DL
|
||
OR DH,DH
|
||
JZ SETNREX
|
||
MOV ES:[DI.fcb_RR+3],DH ; Save 4 byte of RECPOS only if significant
|
||
SETNREX:
|
||
MOV CX,AX
|
||
AND AL,7FH
|
||
MOV ES:[DI.fcb_NR],AL
|
||
AND CL,80H
|
||
SHL CX,1
|
||
RCL DX,1
|
||
MOV AL,CH
|
||
MOV AH,DL
|
||
MOV ES:[DI.fcb_EXTENT],AX
|
||
MOV AL,BYTE PTR [DSKERR]
|
||
RET4:
|
||
RET
|
||
$FCB_SEQ_READ ENDP
|
||
|
||
SUBTTL $FCB_DELETE -- SYSTEM CALL 19
|
||
PAGE
|
||
procedure $FCB_DELETE,NEAR ; System call 19
|
||
ASSUME DS:NOTHING,ES:NOTHING
|
||
|
||
; Inputs:
|
||
; DS:DX point to unopened FCB
|
||
; Function:
|
||
; Delete all matching entries
|
||
; Returns:
|
||
; AL = -1 if no entries matched, otherwise 0
|
||
|
||
invoke MOVNAME
|
||
ASSUME ES:DOSGROUP
|
||
MOV AL,-1
|
||
MOV BYTE PTR [FoundDel],AL
|
||
JC RET4
|
||
MOV AL,BYTE PTR [ATTRIB]
|
||
AND AL,attr_hidden+attr_system+attr_directory+attr_volume_id+attr_read_only
|
||
; Look only at hidden bits
|
||
CMP AL,attr_hidden+attr_system+attr_directory+attr_volume_id+attr_read_only
|
||
; All 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 [DELALL],0 ; DEL *.* - flag deleting all
|
||
NOTALL:
|
||
invoke FINDNAME
|
||
ASSUME DS:DOSGROUP
|
||
MOV AL,-1
|
||
JC RET4
|
||
OR AH,AH ; Check if device name
|
||
JS RET4 ; Can't delete I/O devices
|
||
DELFILE:
|
||
LES BP,[THISDPB]
|
||
MOV AH,BYTE PTR [DELALL]
|
||
PUSH DS
|
||
LDS DI,[CURBUF]
|
||
ASSUME DS:NOTHING
|
||
TEST [Attrib],attr_read_only ; are we deleting RO files too?
|
||
JNZ DoDelete ; yes
|
||
TEST DS:[BX.dir_attr],attr_read_only
|
||
JZ DoDelete ; not read only
|
||
POP DS
|
||
JMP SHORT DelNxt
|
||
DoDelete:
|
||
MOV BYTE PTR [FoundDel],0
|
||
MOV [DI.BUFDIRTY],1
|
||
MOV BYTE PTR [BX],AH
|
||
MOV BX,[SI]
|
||
POP DS
|
||
ASSUME DS:DOSGROUP
|
||
OR BX,BX
|
||
JZ DELNXT
|
||
CMP BX,ES:[BP.dpb_max_cluster]
|
||
JA DELNXT
|
||
invoke RELEASE
|
||
DELNXT:
|
||
invoke GETENTRY ; Registers need to be reset
|
||
invoke NEXTENT
|
||
JNC DELFILE
|
||
CALL FLUSHRET1
|
||
MOV AL,BYTE PTR [FoundDel]
|
||
RET
|
||
|
||
$FCB_DELETE ENDP
|
||
|
||
SUBTTL $FCB_RENAME -- SYSTEM CALL 23; RENAME FILES
|
||
PAGE
|
||
ERRETJ: JMP ERRET
|
||
|
||
procedure $FCB_RENAME,NEAR ; System call 23
|
||
ASSUME DS:NOTHING,ES:NOTHING
|
||
|
||
; Inputs:
|
||
; DS:DX point to a modified FCB (DS:DX+11H points to destination
|
||
; name)
|
||
; Function:
|
||
; Rename all matching entries to indicated name
|
||
; Returns:
|
||
; AL = -1 if no entries matched, otherwise 0
|
||
|
||
invoke MOVNAME
|
||
ASSUME ES:DOSGROUP
|
||
JC ERRETJ
|
||
ADD SI,5
|
||
MOV DI,OFFSET DOSGROUP:NAME2
|
||
invoke LODNAME
|
||
JC ERRETJ ; Report error if second name invalid
|
||
invoke FINDNAME
|
||
ASSUME DS:DOSGROUP
|
||
JC ERRETJ
|
||
OR AH,AH ; Check if I/O device name
|
||
JS ERRETJ ; If so, can't rename it
|
||
MOV SI,OFFSET DOSGROUP:NAME1
|
||
MOV DI,OFFSET DOSGROUP:NAME3
|
||
MOV CX,13
|
||
REP MOVSB ; Copy name to search for --include attribute byte
|
||
RENFIL:
|
||
MOV DI,OFFSET DOSGROUP:NAME1
|
||
MOV SI,OFFSET DOSGROUP:NAME2
|
||
MOV CX,11
|
||
NEWNAM:
|
||
LODSB
|
||
CMP AL,"?"
|
||
JNZ NOCHG
|
||
PUSH DS
|
||
MOV DS,WORD PTR [CURBUF+2]
|
||
MOV AL,[BX]
|
||
POP DS
|
||
NOCHG:
|
||
STOSB
|
||
INC BX
|
||
LOOP NEWNAM
|
||
INC DI
|
||
MOV BYTE PTR [DI],attr_all ;Sets ATTRIB
|
||
; Stop duplicates with any attributes
|
||
invoke DEVNAME ; Check if giving it a device name
|
||
JNC RENERR
|
||
XOR AX,AX
|
||
PUSH [LASTENT]
|
||
invoke FINDENTRY ; See if new name already exists
|
||
POP AX
|
||
JNC RENERR ; Error if found
|
||
LES BP,[THISDPB]
|
||
invoke GETENT ; Re-read matching entry
|
||
MOV DI,BX ; Leave BX,DX until call to NEXTENT
|
||
MOV ES,WORD PTR [CURBUF+2]
|
||
MOV SI,OFFSET DOSGROUP:NAME1
|
||
MOV CX,11
|
||
REP MOVSB ; Replace old name with new one
|
||
MOV DI,WORD PTR [CURBUF]
|
||
MOV ES:[DI.BUFDIRTY],1 ; Directory changed
|
||
PUSH SS
|
||
POP ES
|
||
MOV SI,OFFSET DOSGROUP:NAME3
|
||
MOV DI,OFFSET DOSGROUP:NAME1
|
||
MOV CX,13 ; Include attribute byte
|
||
REP MOVSB ; Copy name back into search buffer
|
||
invoke NEXTENT
|
||
JNC RENFIL
|
||
JMP FLUSHRET1
|
||
|
||
RENERR:
|
||
CALL FLUSHRET1
|
||
ERRET:
|
||
MOV AL,-1
|
||
RET
|
||
$FCB_RENAME ENDP
|
||
|
||
SUBTTL $FCB_OPEN -- SYSTEM CALL 15; OPEN A FILE
|
||
PAGE
|
||
procedure $FCB_OPEN,NEAR ; System call 15
|
||
ASSUME DS:NOTHING,ES:NOTHING
|
||
|
||
; Inputs:
|
||
; DS:DX point to an unopened FCB
|
||
; Function:
|
||
; Open indicated file and fill in FCB
|
||
; Returns:
|
||
; AL = -1 if no entries matched, otherwise 0
|
||
; FOR INTERNAL USE
|
||
; [CURBUF+2]:SI and [CURBUF+2]:BX Preserved
|
||
|
||
invoke GETFILE
|
||
ASSUME DS:DOSGROUP,ES:NOTHING
|
||
|
||
entry DOOPEN
|
||
|
||
; Enter here to perform $FCB_OPEN on file already found
|
||
; in directory. AH=device ID number, DS=CS, BX points to directory
|
||
; entry in [CURBUF], SI points to First Cluster field, and
|
||
; ES:DI point to the FCB to be opened. This entry point
|
||
; is used by $FCB_CREATE.
|
||
JC ERRET
|
||
PUSH SI
|
||
PUSH AX ; Save I/O driver number
|
||
XOR AL,AL
|
||
OR AH,AH
|
||
JS OPENDEV
|
||
MOV AL,[THISDRV]
|
||
MOV DS,WORD PTR [CURBUF+2]
|
||
ASSUME DS:NOTHING
|
||
INC AX
|
||
OPENDEV:
|
||
STOSB
|
||
XOR AX,AX
|
||
IF ZEROEXT
|
||
ADD DI,11
|
||
STOSW ; Zero low byte of extent field if ZERPEXT only
|
||
ELSE
|
||
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
|
||
POP AX ; Restore I/O driver number
|
||
POP SI
|
||
MOV AL,AH
|
||
OR AL,40H ; Not dirty
|
||
STOSB
|
||
JS SAVDEVPT ; If device, go save pointer to it
|
||
MOV AX,DX ; Restore starting cluster
|
||
STOSW ; first cluster
|
||
PUSH AX ; save cluster
|
||
XOR AX,AX
|
||
STOSW ; clus pos
|
||
POP AX ; last cluster
|
||
STOSB
|
||
MOV AL,AH
|
||
MOV AH,BYTE PTR [DIRSTART]
|
||
PUSH CX
|
||
MOV CL,4
|
||
SHL AH,CL
|
||
OR AL,AH
|
||
STOSB
|
||
MOV AX,[DIRSTART]
|
||
MOV CL,4
|
||
SHL AX,CL
|
||
POP CX
|
||
MOV AL,AH
|
||
STOSB
|
||
OPEN_RET:
|
||
XOR AX,AX
|
||
RET
|
||
|
||
SAVDEVPT:
|
||
ASSUME DS:DOSGROUP
|
||
LDS AX,[DEVPT]
|
||
ASSUME DS:NOTHING
|
||
STOSW
|
||
MOV ES:[DI],DS
|
||
JMP SHORT OPEN_RET
|
||
$FCB_OPEN ENDP
|
||
|
||
SUBTTL $FCB_CLOSE -- SYSTEM CALL 16; CLOSE FILE
|
||
PAGE
|
||
procedure $FCB_CLOSE,NEAR ; System call 16
|
||
ASSUME DS:NOTHING,ES:NOTHING
|
||
|
||
; Inputs:
|
||
; DS:DX point to an opened FCB
|
||
; Function:
|
||
; Close the indicated file
|
||
; Returns:
|
||
; AL = -1 if disk has been changed, otherwise 0
|
||
|
||
MOV DI,DX
|
||
CMP BYTE PTR [DI],-1 ; Check for extended FCB
|
||
JNZ NORMFCB3
|
||
ADD DI,7
|
||
NORMFCB3:
|
||
TEST [DI.fcb_DEVID],devid_file_clean+devid_device
|
||
; Allow only dirty files
|
||
JNZ OKRET1 ; can't close I/O device or not written
|
||
invoke MOVNAMENOSET
|
||
JC BADCLOSE ; Bad file name
|
||
entry FCB_CLOSE_INNER
|
||
PUSH DX
|
||
PUSH DS
|
||
MOV SI,DX
|
||
MOV BX,[SI.fcb_LSTCLUS+1]
|
||
MOV CL,4
|
||
SHR BX,CL
|
||
PUSH BX
|
||
PUSH SS
|
||
POP DS
|
||
ASSUME DS:DOSGROUP
|
||
invoke FATREAD
|
||
POP BX
|
||
invoke SETDIRSRCH
|
||
invoke FINDENTRY
|
||
POP ES
|
||
POP DI
|
||
JC BADCLOSE
|
||
LDS BX,[CURBUF]
|
||
ASSUME DS:NOTHING
|
||
|
||
; note that SI points to dir_first...
|
||
|
||
OR BYTE PTR [SI-dir_first+dir_attr],attr_archive
|
||
MOV CX,ES:[DI.fcb_FIRCLUS]
|
||
MOV [SI-dir_first+dir_first],CX
|
||
MOV DX,ES:WORD PTR [DI.fcb_FILSIZ]
|
||
MOV [SI-dir_first+dir_size_l],DX
|
||
MOV DX,ES:WORD PTR [DI.fcb_FILSIZ+2]
|
||
MOV [SI-dir_first+dir_size_h],DX
|
||
MOV DX,ES:[DI.fcb_FDATE]
|
||
MOV [SI-dir_first+dir_date],DX
|
||
MOV DX,ES:[DI.fcb_FTIME]
|
||
MOV [SI-dir_first+dir_time],DX
|
||
MOV [BX.BUFDIRTY],1
|
||
PUSH SS
|
||
POP DS
|
||
ASSUME DS:DOSGROUP
|
||
FLUSHRET1:
|
||
LES BP,[THISDPB]
|
||
MOV AL,ES:[BP.dpb_drive]
|
||
invoke FLUSHBUF
|
||
OKRET1:
|
||
XOR AL,AL
|
||
RET
|
||
|
||
BADCLOSE:
|
||
MOV AL,-1
|
||
RET
|
||
$FCB_CLOSE ENDP
|
||
|
||
SUBTTL $FCB_CREATE -- SYSTEM CALL 22; MAKE AND OPEN A NEW FILE
|
||
PAGE
|
||
procedure $FCB_CREATE,NEAR ; System call 22
|
||
ASSUME DS:NOTHING,ES:NOTHING
|
||
|
||
; Inputs:
|
||
; DS:DX point to an unopened FCB
|
||
; Function:
|
||
; If file does not exist, create it and open it
|
||
; If file exists, free up its contents and open the file
|
||
; Returns:
|
||
; AL = -1 if file cannot be created, otherwise 0
|
||
|
||
invoke MOVNAME
|
||
ASSUME ES:DOSGROUP
|
||
JC ERRET3
|
||
MOV DI,OFFSET DOSGROUP:NAME1
|
||
MOV CX,11
|
||
MOV AL,"?"
|
||
REPNE SCASB
|
||
JZ ERRET3
|
||
MOV BYTE PTR [CREATING],-1
|
||
PUSH DX
|
||
PUSH DS
|
||
invoke FINDNAME
|
||
ASSUME DS:DOSGROUP
|
||
NWENTY:
|
||
LES BP,[THISDPB]
|
||
ASSUME ES:NOTHING
|
||
JNC EXISTENT
|
||
invoke BUILDDIR
|
||
JC ERRPOP
|
||
invoke GETENT ; Point at that free entry
|
||
JMP SHORT FREESPOT
|
||
ERRPOP:
|
||
POP DS
|
||
POP DX
|
||
ASSUME DS:NOTHING
|
||
ERRET3:
|
||
JMP SHORT BADCLOSE
|
||
|
||
entry NEWENTRY
|
||
POP DX ; Return address
|
||
POP ES ; ES
|
||
POP CX ; DI
|
||
PUSH DX
|
||
PUSH CX
|
||
PUSH ES
|
||
JMP NWENTY
|
||
|
||
EXISTENT:
|
||
ASSUME DS:DOSGROUP
|
||
JNZ ERRPOP ; Error if attributes don't match
|
||
OR AH,AH ; Check if file is I/O device
|
||
JS OPENJMP ; If so, no action
|
||
PUSH DS
|
||
LDS DI,[CURBUF]
|
||
ASSUME DS:NOTHING
|
||
MOV CX,[SI] ; Get pointer to clusters
|
||
MOV SI,[DI.BUFSECNO]
|
||
POP DS
|
||
ASSUME DS:DOSGROUP
|
||
JCXZ FREESPOT
|
||
CMP CX,ES:[BP.dpb_max_cluster]
|
||
JA FREESPOT
|
||
SUB BX,DI
|
||
PUSH BX
|
||
PUSH SI ; Save sector number
|
||
MOV BX,CX
|
||
invoke RELEASE ; Free any data already allocated
|
||
POP DX
|
||
XOR AL,AL
|
||
invoke GETBUFFR
|
||
POP BX
|
||
ADD BX,WORD PTR [CURBUF]
|
||
FREESPOT:
|
||
TEST BYTE PTR [ATTRIB],attr_volume_id
|
||
JZ NOTVOLID
|
||
CMP BYTE PTR [VOLID],0
|
||
JNZ ERRPOP ; Can't create a second volume ID
|
||
NOTVOLID:
|
||
MOV ES,WORD PTR [CURBUF+2]
|
||
MOV DI,BX
|
||
MOV SI,OFFSET DOSGROUP:NAME1
|
||
MOV CX,5
|
||
MOVSB
|
||
REP MOVSW
|
||
MOV AL,[ATTRIB]
|
||
STOSB
|
||
MOV CL,5
|
||
XOR AX,AX
|
||
REP STOSW
|
||
invoke DATE16
|
||
XCHG AX,DX
|
||
STOSW
|
||
XCHG AX,DX
|
||
STOSW
|
||
XOR AX,AX
|
||
PUSH DI
|
||
STOSW
|
||
STOSW
|
||
STOSW
|
||
MOV SI,WORD PTR [CURBUF]
|
||
MOV ES:[SI.BUFDIRTY],1
|
||
LES BP,[THISDPB]
|
||
MOV AL,ES:[BP.dpb_drive]
|
||
PUSH AX
|
||
PUSH BX
|
||
invoke FLUSHBUF
|
||
POP BX
|
||
POP AX
|
||
POP SI
|
||
MOV AH,AL ; Get I/O driver number back
|
||
OPENJMP:
|
||
CLC ; Clear carry so OPEN won't fail
|
||
POP ES
|
||
POP DI
|
||
ASSUME ES:NOTHING
|
||
JMP DOOPEN
|
||
$FCB_CREATE ENDP
|
||
|
||
do_ext
|
||
|
||
CODE ENDS
|
||
END
|
||
|
||
|