mirror of
https://github.com/microsoft/MS-DOS.git
synced 2025-01-22 16:50:21 +00:00
507 lines
14 KiB
NASM
507 lines
14 KiB
NASM
|
TITLE DIRCALL - Directory manipulation internal calls
|
|||
|
NAME DIRCALL
|
|||
|
|
|||
|
; $MKDIR
|
|||
|
; $CHDIR
|
|||
|
; $RMDIR
|
|||
|
|
|||
|
.xlist
|
|||
|
INCLUDE DOSSEG.ASM
|
|||
|
|
|||
|
CODE SEGMENT BYTE PUBLIC 'CODE'
|
|||
|
ASSUME SS:DOSGROUP,CS:DOSGROUP
|
|||
|
|
|||
|
.xcref
|
|||
|
INCLUDE DOSSYM.ASM
|
|||
|
INCLUDE DEVSYM.ASM
|
|||
|
.cref
|
|||
|
.list
|
|||
|
|
|||
|
ifndef Kanji
|
|||
|
Kanji equ 0
|
|||
|
endif
|
|||
|
|
|||
|
i_need AUXSTACK,BYTE
|
|||
|
i_need NoSetDir,BYTE
|
|||
|
i_need CURBUF, DWORD
|
|||
|
i_need DIRSTART,WORD
|
|||
|
i_need THISDPB,DWORD
|
|||
|
i_need NAME1,BYTE
|
|||
|
i_need LASTENT,WORD
|
|||
|
i_need ATTRIB,BYTE
|
|||
|
i_need THISFCB,DWORD
|
|||
|
i_need AUXSTACK,BYTE
|
|||
|
i_need CREATING,BYTE
|
|||
|
i_need DRIVESPEC,BYTE
|
|||
|
i_need ROOTSTART,BYTE
|
|||
|
i_need SWITCH_CHARACTER,BYTE
|
|||
|
|
|||
|
extrn sys_ret_ok:near,sys_ret_err:near
|
|||
|
|
|||
|
|
|||
|
; XENIX CALLS
|
|||
|
BREAK <$MkDir - Make a directory entry>
|
|||
|
MKNERRJ: JMP MKNERR
|
|||
|
NODEEXISTSJ: JMP NODEEXISTS
|
|||
|
procedure $MKDIR,NEAR
|
|||
|
ASSUME DS:NOTHING,ES:NOTHING
|
|||
|
|
|||
|
; Inputs:
|
|||
|
; DS:DX Points to asciz name
|
|||
|
; Function:
|
|||
|
; Make a new directory
|
|||
|
; Returns:
|
|||
|
; STD XENIX Return
|
|||
|
; AX = mkdir_path_not_found if path bad
|
|||
|
; AX = mkdir_access_denied If
|
|||
|
; Directory cannot be created
|
|||
|
; Node already exists
|
|||
|
; Device name given
|
|||
|
; Disk or directory(root) full
|
|||
|
invoke validate_path
|
|||
|
JC MKNERRJ
|
|||
|
MOV SI,DX
|
|||
|
MOV WORD PTR [THISFCB+2],SS
|
|||
|
MOV WORD PTR [THISFCB],OFFSET DOSGROUP:AUXSTACK-40 ; Scratch space
|
|||
|
MOV AL,attr_directory
|
|||
|
MOV WORD PTR [CREATING],0E500h
|
|||
|
invoke MAKENODE
|
|||
|
ASSUME DS:DOSGROUP
|
|||
|
MOV AL,mkdir_path_not_found
|
|||
|
JC MKNERRJ
|
|||
|
JNZ NODEEXISTSJ
|
|||
|
LDS DI,[CURBUF]
|
|||
|
ASSUME DS:NOTHING
|
|||
|
SUB SI,DI
|
|||
|
PUSH SI ; Pointer to fcb_FIRCLUS
|
|||
|
PUSH [DI.BUFSECNO] ; Sector of new node
|
|||
|
PUSH SS
|
|||
|
POP DS
|
|||
|
ASSUME DS:DOSGROUP
|
|||
|
PUSH [DIRSTART] ; Parent for .. entry
|
|||
|
XOR AX,AX
|
|||
|
MOV [DIRSTART],AX ; Null directory
|
|||
|
invoke NEWDIR
|
|||
|
JC NODEEXISTSPOPDEL ; No room
|
|||
|
invoke GETENT ; First entry
|
|||
|
LES DI,[CURBUF]
|
|||
|
MOV ES:[DI.BUFDIRTY],1
|
|||
|
ADD DI,BUFINSIZ ; Point at buffer
|
|||
|
MOV AX,202EH ; ". "
|
|||
|
STOSW
|
|||
|
MOV DX,[DIRSTART] ; Point at itself
|
|||
|
invoke SETDOTENT
|
|||
|
MOV AX,2E2EH ; ".."
|
|||
|
STOSW
|
|||
|
POP DX ; Parent
|
|||
|
invoke SETDOTENT
|
|||
|
LES BP,[THISDPB]
|
|||
|
POP DX ; Entry sector
|
|||
|
XOR AL,AL ; Pre read
|
|||
|
invoke GETBUFFR
|
|||
|
MOV DX,[DIRSTART]
|
|||
|
LDS DI,[CURBUF]
|
|||
|
ASSUME DS:NOTHING
|
|||
|
ZAPENT:
|
|||
|
POP SI ; fcb_Firclus pointer
|
|||
|
ADD SI,DI
|
|||
|
MOV [SI],DX
|
|||
|
XOR DX,DX
|
|||
|
MOV [SI+2],DX
|
|||
|
MOV [SI+4],DX
|
|||
|
DIRUP:
|
|||
|
MOV [DI.BUFDIRTY],1
|
|||
|
PUSH SS
|
|||
|
POP DS
|
|||
|
ASSUME DS:DOSGROUP
|
|||
|
MOV AL,ES:[BP.dpb_drive]
|
|||
|
invoke FLUSHBUF
|
|||
|
SYS_RET_OKJ:
|
|||
|
JMP SYS_RET_OK
|
|||
|
|
|||
|
NODEEXISTSPOPDEL:
|
|||
|
POP DX ; Parent
|
|||
|
POP DX ; Entry sector
|
|||
|
LES BP,[THISDPB]
|
|||
|
XOR AL,AL ; Pre read
|
|||
|
invoke GETBUFFR
|
|||
|
LDS DI,[CURBUF]
|
|||
|
ASSUME DS:NOTHING
|
|||
|
POP SI ; dir_first pointer
|
|||
|
ADD SI,DI
|
|||
|
SUB SI,dir_first ; Point back to start of dir entry
|
|||
|
MOV BYTE PTR [SI],0E5H ; Free the entry
|
|||
|
CALL DIRUP
|
|||
|
NODEEXISTS:
|
|||
|
MOV AL,mkdir_access_denied
|
|||
|
MKNERR:
|
|||
|
JMP SYS_RET_ERR
|
|||
|
$MKDIR ENDP
|
|||
|
|
|||
|
BREAK <$ChDir -- Change current directory on a drive>
|
|||
|
procedure $CHDIR,NEAR
|
|||
|
ASSUME DS:NOTHING,ES:NOTHING
|
|||
|
|
|||
|
; Inputs:
|
|||
|
; DS:DX Points to asciz name
|
|||
|
; Function:
|
|||
|
; Change current directory
|
|||
|
; Returns:
|
|||
|
; STD XENIX Return
|
|||
|
; AX = chdir_path_not_found if error
|
|||
|
|
|||
|
invoke validate_path
|
|||
|
JC PathTooLong
|
|||
|
|
|||
|
PUSH DS
|
|||
|
PUSH DX
|
|||
|
MOV SI,DX
|
|||
|
invoke GETPATH
|
|||
|
JC PATHNOGOOD
|
|||
|
JNZ PATHNOGOOD
|
|||
|
ASSUME DS:DOSGROUP
|
|||
|
MOV AX,[DIRSTART]
|
|||
|
MOV BX,AX
|
|||
|
XCHG BX,ES:[BP.dpb_current_dir]
|
|||
|
OR AX,AX
|
|||
|
POP SI
|
|||
|
POP DS
|
|||
|
ASSUME DS:NOTHING
|
|||
|
JZ SYS_RET_OKJ
|
|||
|
MOV DI,BP
|
|||
|
ADD DI,dpb_dir_text
|
|||
|
MOV DX,DI
|
|||
|
CMP [DRIVESPEC],0
|
|||
|
JZ NODRIVESPEC
|
|||
|
INC SI
|
|||
|
INC SI
|
|||
|
NODRIVESPEC:
|
|||
|
MOV CX,SI
|
|||
|
CMP [ROOTSTART],0
|
|||
|
JZ NOTROOTPATH
|
|||
|
INC SI
|
|||
|
INC CX
|
|||
|
JMP SHORT COPYTHESTRINGBXZ
|
|||
|
NOTROOTPATH:
|
|||
|
OR BX,BX ; Previous path root?
|
|||
|
JZ COPYTHESTRING ; Yes
|
|||
|
XOR BX,BX
|
|||
|
ENDLOOP:
|
|||
|
CMP BYTE PTR ES:[DI],0
|
|||
|
JZ PATHEND
|
|||
|
INC DI
|
|||
|
INC BX
|
|||
|
JMP SHORT ENDLOOP
|
|||
|
PATHEND:
|
|||
|
MOV AL,'/'
|
|||
|
CMP AL,[switch_character]
|
|||
|
JNZ SLASHOK
|
|||
|
MOV AL,'\' ; Use the alternate character
|
|||
|
SLASHOK:
|
|||
|
STOSB
|
|||
|
INC BX
|
|||
|
JMP SHORT CHECK_LEN
|
|||
|
|
|||
|
PATHNOGOOD:
|
|||
|
POP AX
|
|||
|
POP AX
|
|||
|
PATHTOOLONG:
|
|||
|
error error_path_not_found
|
|||
|
|
|||
|
ASSUME DS:NOTHING
|
|||
|
|
|||
|
INCBXCHK:
|
|||
|
INC BX
|
|||
|
BXCHK:
|
|||
|
CMP BX,DIRSTRLEN
|
|||
|
return
|
|||
|
|
|||
|
COPYTHESTRINGBXZ:
|
|||
|
XOR BX,BX
|
|||
|
COPYTHESTRING:
|
|||
|
LODSB
|
|||
|
OR AL,AL
|
|||
|
|
|||
|
JNZ FOOB
|
|||
|
JMP CPSTDONE
|
|||
|
FOOB:
|
|||
|
CMP AL,'.'
|
|||
|
JZ SEEDOT
|
|||
|
CALL COPYELEM
|
|||
|
CHECK_LEN:
|
|||
|
CMP BX,DIRSTRLEN
|
|||
|
JB COPYTHESTRING
|
|||
|
MOV AL,ES:[DI-1]
|
|||
|
invoke PATHCHRCMP
|
|||
|
JNZ OK_DI
|
|||
|
DEC DI
|
|||
|
OK_DI:
|
|||
|
XOR AL,AL
|
|||
|
STOSB ; Correctly terminate the path
|
|||
|
MOV ES:[BP.dpb_current_dir],-1 ; Force re-validation
|
|||
|
JMP SHORT PATHTOOLONG
|
|||
|
|
|||
|
SEEDOT:
|
|||
|
LODSB
|
|||
|
OR AL,AL ; Check for null
|
|||
|
JZ CPSTDONEDEC
|
|||
|
CMP AL,'.'
|
|||
|
JNZ COPYTHESTRING ; eat ./
|
|||
|
CALL DELELMES ; have ..
|
|||
|
LODSB ; eat the /
|
|||
|
OR AL,AL ; Check for null
|
|||
|
JZ CPSTDONEDEC
|
|||
|
JMP SHORT COPYTHESTRING
|
|||
|
|
|||
|
; Copy one element from DS:SI to ES:DI include trailing / not trailing null
|
|||
|
; LODSB has already been done
|
|||
|
COPYELEM:
|
|||
|
PUSH DI ; Save in case too long
|
|||
|
PUSH CX
|
|||
|
MOV CX,800h ; length of filename
|
|||
|
MOV AH,'.' ; char to stop on
|
|||
|
CALL CopyPiece ; go for it!
|
|||
|
CALL BXCHK ; did we go over?
|
|||
|
JAE POPCXDI ; yep, go home
|
|||
|
CMP AH,AL ; did we stop on .?
|
|||
|
JZ CopyExt ; yes, go copy ext
|
|||
|
OR AL,AL ; did we end on nul?
|
|||
|
JZ DECSIRet ; yes, bye
|
|||
|
CopyPathEnd:
|
|||
|
STOSB ; save the path char
|
|||
|
CALL INCBXCHK ; was there room for it?
|
|||
|
JAE POPCXDI ; Nope
|
|||
|
INC SI ; guard against following dec
|
|||
|
DECSIRET:
|
|||
|
DEC SI ; point back at null
|
|||
|
POP CX
|
|||
|
POP AX ; toss away saved DI
|
|||
|
return
|
|||
|
POPCXDI:
|
|||
|
POP CX ; restore
|
|||
|
POP DI ; point back...
|
|||
|
return
|
|||
|
CopyExt:
|
|||
|
STOSB ; save the dot
|
|||
|
CALL INCBXCHK ; room?
|
|||
|
JAE POPCXDI ; nope.
|
|||
|
LODSB ; get next char
|
|||
|
XOR AH,AH ; NUL here
|
|||
|
MOV CX,300h ; at most 3 chars
|
|||
|
CALL CopyPiece ; go copy it
|
|||
|
CALL BXCHK ; did we go over
|
|||
|
JAE POPCXDI ; yep
|
|||
|
OR AL,AL ; sucessful end?
|
|||
|
JZ DECSIRET ; yes
|
|||
|
JMP CopyPathEnd ; go stash path char
|
|||
|
|
|||
|
DELELMES:
|
|||
|
; Delete one path element from ES:DI
|
|||
|
DEC DI ; the '/'
|
|||
|
DEC BX
|
|||
|
|
|||
|
IF KANJI
|
|||
|
PUSH AX
|
|||
|
PUSH CX
|
|||
|
PUSH DI
|
|||
|
PUSH DX
|
|||
|
MOV CX,DI
|
|||
|
MOV DI,DX
|
|||
|
DELLOOP:
|
|||
|
CMP DI,CX
|
|||
|
JZ GOTDELE
|
|||
|
MOV AL,ES:[DI]
|
|||
|
INC DI
|
|||
|
invoke TESTKANJ
|
|||
|
JZ NOTKANJ11
|
|||
|
INC DI
|
|||
|
JMP DELLOOP
|
|||
|
|
|||
|
NOTKANJ11:
|
|||
|
invoke PATHCHRCMP
|
|||
|
JNZ DELLOOP
|
|||
|
MOV DX,DI ; Point to char after '/'
|
|||
|
JMP DELLOOP
|
|||
|
|
|||
|
GOTDELE:
|
|||
|
MOV DI,DX
|
|||
|
POP DX
|
|||
|
POP AX ; Initial DI
|
|||
|
SUB AX,DI ; Distance moved
|
|||
|
SUB BX,AX ; Set correct BX
|
|||
|
POP CX
|
|||
|
POP AX
|
|||
|
return
|
|||
|
ELSE
|
|||
|
DELLOOP:
|
|||
|
CMP DI,DX
|
|||
|
retz
|
|||
|
PUSH AX
|
|||
|
MOV AL,ES:[DI-1]
|
|||
|
invoke PATHCHRCMP
|
|||
|
POP AX
|
|||
|
retz
|
|||
|
DEC DI
|
|||
|
DEC BX
|
|||
|
JMP SHORT DELLOOP
|
|||
|
ENDIF
|
|||
|
|
|||
|
CPSTDONEDEC:
|
|||
|
DEC DI ; Back up over trailing /
|
|||
|
CPSTDONE:
|
|||
|
STOSB ; The NUL
|
|||
|
JMP SYS_RET_OK
|
|||
|
|
|||
|
; copy a piece CH chars max until the char in AH (or path or NUL)
|
|||
|
CopyPiece:
|
|||
|
STOSB ; store the character
|
|||
|
INC CL ; moved a byte
|
|||
|
CALL INCBXCHK ; room enough?
|
|||
|
JAE CopyPieceRet ; no, pop CX and DI
|
|||
|
OR AL,AL ; end of string?
|
|||
|
JZ CopyPieceRet ; yes, dec si and return
|
|||
|
|
|||
|
IF KANJI
|
|||
|
CALL TestKanj ; was it kanji?
|
|||
|
JZ NotKanj ; nope
|
|||
|
MOVSB ; move the next byte
|
|||
|
CALL INCBXCHK ; room for it?
|
|||
|
JAE CopyPieceRet ; nope
|
|||
|
INC CL ; moved a byte
|
|||
|
NotKanj:
|
|||
|
ENDIF
|
|||
|
|
|||
|
CMP CL,CH ; move too many?
|
|||
|
JBE CopyPieceNext ; nope
|
|||
|
|
|||
|
IF KANJI
|
|||
|
CALL TestKanj ; was the last byte kanji
|
|||
|
JZ NotKanj2 ; no only single byte backup
|
|||
|
DEC DI ; back up a char
|
|||
|
DEC BX
|
|||
|
NotKanj2:
|
|||
|
ENDIF
|
|||
|
|
|||
|
DEC DI ; back up a char
|
|||
|
DEC BX
|
|||
|
CopyPieceNext:
|
|||
|
LODSB ; get next character
|
|||
|
invoke PathChrCmp ; end of road?
|
|||
|
JZ CopyPieceRet ; yep, return and don't dec SI
|
|||
|
CMP AL,AH ; end of filename?
|
|||
|
JNZ CopyPiece ; go do name
|
|||
|
CopyPieceRet:
|
|||
|
return ; bye!
|
|||
|
|
|||
|
$CHDIR ENDP
|
|||
|
|
|||
|
BREAK <$RmDir -- Remove a directory>
|
|||
|
NOPATHJ: JMP NOPATH
|
|||
|
|
|||
|
procedure $RMDIR,NEAR ; System call 47
|
|||
|
ASSUME DS:NOTHING,ES:NOTHING
|
|||
|
|
|||
|
; Inputs:
|
|||
|
; DS:DX Points to asciz name
|
|||
|
; Function:
|
|||
|
; Delete directory if empty
|
|||
|
; Returns:
|
|||
|
; STD XENIX Return
|
|||
|
; AX = rmdir_path_not_found If path bad
|
|||
|
; AX = rmdir_access_denied If
|
|||
|
; Directory not empty
|
|||
|
; Path not directory
|
|||
|
; Root directory specified
|
|||
|
; Directory malformed (. and .. not first two entries)
|
|||
|
; AX = rmdir_current_directory
|
|||
|
|
|||
|
invoke Validate_path
|
|||
|
JC NoPathJ
|
|||
|
MOV SI,DX
|
|||
|
invoke GETPATH
|
|||
|
JC NOPATHJ
|
|||
|
ASSUME DS:DOSGROUP
|
|||
|
JNZ NOTDIRPATH
|
|||
|
MOV DI,[DIRSTART]
|
|||
|
OR DI,DI
|
|||
|
JZ NOTDIRPATH
|
|||
|
MOV CX,ES:[BP.dpb_current_dir]
|
|||
|
CMP CX,-1
|
|||
|
JNZ rmdir_current_dir_check
|
|||
|
invoke GetCurrDir
|
|||
|
invoke Get_user_stack
|
|||
|
MOV DX,[SI.user_DX]
|
|||
|
MOV DS,[SI.user_DS]
|
|||
|
JMP $RMDIR
|
|||
|
|
|||
|
NOTDIRPATHPOP:
|
|||
|
POP AX
|
|||
|
POP AX
|
|||
|
NOTDIRPATH:
|
|||
|
error error_access_denied
|
|||
|
|
|||
|
rmdir_current_dir_check:
|
|||
|
CMP DI,CX
|
|||
|
JNZ rmdir_get_buf
|
|||
|
error error_current_directory
|
|||
|
|
|||
|
rmdir_get_buf:
|
|||
|
LDS DI,[CURBUF]
|
|||
|
ASSUME DS:NOTHING
|
|||
|
SUB BX,DI
|
|||
|
PUSH BX ; Save entry pointer
|
|||
|
PUSH [DI.BUFSECNO] ; Save sector number
|
|||
|
PUSH SS
|
|||
|
POP DS
|
|||
|
ASSUME DS:DOSGROUP
|
|||
|
PUSH SS
|
|||
|
POP ES
|
|||
|
MOV DI,OFFSET DOSGROUP:NAME1
|
|||
|
MOV AL,'?'
|
|||
|
MOV CX,11
|
|||
|
REP STOSB
|
|||
|
XOR AL,AL
|
|||
|
STOSB
|
|||
|
invoke STARTSRCH
|
|||
|
invoke GETENTRY
|
|||
|
MOV DS,WORD PTR [CURBUF+2]
|
|||
|
ASSUME DS:NOTHING
|
|||
|
MOV SI,BX
|
|||
|
LODSW
|
|||
|
CMP AX,(' ' SHL 8) OR '.'
|
|||
|
JNZ NOTDIRPATHPOP
|
|||
|
ADD SI,32-2
|
|||
|
LODSW
|
|||
|
CMP AX,('.' SHL 8) OR '.'
|
|||
|
JNZ NOTDIRPATHPOP
|
|||
|
PUSH SS
|
|||
|
POP DS
|
|||
|
ASSUME DS:DOSGROUP
|
|||
|
MOV [LASTENT],2 ; Skip . and ..
|
|||
|
invoke GETENTRY
|
|||
|
MOV [ATTRIB],attr_directory+attr_hidden+attr_system
|
|||
|
invoke SRCH
|
|||
|
JNC NOTDIRPATHPOP
|
|||
|
LES BP,[THISDPB]
|
|||
|
MOV BX,[DIRSTART]
|
|||
|
invoke RELEASE
|
|||
|
POP DX
|
|||
|
XOR AL,AL
|
|||
|
invoke GETBUFFR
|
|||
|
LDS DI,[CURBUF]
|
|||
|
ASSUME DS:NOTHING
|
|||
|
POP BX
|
|||
|
ADD BX,DI
|
|||
|
MOV BYTE PTR [BX],0E5H ; Free the entry
|
|||
|
JMP DIRUP
|
|||
|
|
|||
|
NOPATH:
|
|||
|
error error_path_not_found
|
|||
|
|
|||
|
$RMDIR ENDP
|
|||
|
|
|||
|
do_ext
|
|||
|
|
|||
|
CODE ENDS
|
|||
|
END
|
|||
|
|