mirror of
https://github.com/microsoft/MS-DOS.git
synced 2025-01-22 16:50:21 +00:00
522 lines
13 KiB
NASM
522 lines
13 KiB
NASM
|
TITLE PART2 - COMMAND Transient routines.
|
|||
|
|
|||
|
INCLUDE COMSW.ASM
|
|||
|
|
|||
|
.xlist
|
|||
|
.xcref
|
|||
|
INCLUDE DOSSYM.ASM
|
|||
|
INCLUDE DEVSYM.ASM
|
|||
|
INCLUDE COMSEG.ASM
|
|||
|
.list
|
|||
|
.cref
|
|||
|
|
|||
|
INCLUDE COMEQU.ASM
|
|||
|
|
|||
|
CODERES SEGMENT PUBLIC
|
|||
|
EXTRN LODCOM1:NEAR
|
|||
|
CODERES ENDS
|
|||
|
|
|||
|
DATARES SEGMENT PUBLIC
|
|||
|
EXTRN PARENT:WORD,IO_SAVE:WORD,PERMCOM:BYTE
|
|||
|
EXTRN PIPEFLAG:BYTE,ENVIRSEG:WORD
|
|||
|
if ibmver
|
|||
|
EXTRN SYS_CALL:DWORD
|
|||
|
endif
|
|||
|
DATARES ENDS
|
|||
|
|
|||
|
TRANDATA SEGMENT PUBLIC
|
|||
|
|
|||
|
EXTRN PATH_TEXT:BYTE,PROMPT_TEXT:BYTE
|
|||
|
EXTRN BADDEV:BYTE,SYNTMES:BYTE,ENVERR:BYTE
|
|||
|
TRANDATA ENDS
|
|||
|
|
|||
|
TRANSPACE SEGMENT PUBLIC
|
|||
|
|
|||
|
EXTRN CURDRV:BYTE,DIRCHAR:BYTE,PWDBUF:BYTE
|
|||
|
EXTRN INTERNATVARS:BYTE,RESSEG:WORD,TPA:WORD
|
|||
|
|
|||
|
TRANSPACE ENDS
|
|||
|
|
|||
|
|
|||
|
TRANCODE SEGMENT PUBLIC BYTE
|
|||
|
ASSUME CS:TRANGROUP,DS:NOTHING,ES:NOTHING,SS:NOTHING
|
|||
|
|
|||
|
EXTRN CERROR:NEAR,ZPRINT:NEAR
|
|||
|
EXTRN CRLF2:NEAR,SCANOFF:NEAR,FREE_TPA:NEAR,ALLOC_TPA:NEAR
|
|||
|
EXTRN OUT:NEAR,DRVBAD:NEAR,SETPATH:NEAR,PRINT:NEAR
|
|||
|
EXTRN FCB_TO_ASCZ:NEAR
|
|||
|
|
|||
|
PUBLIC PRINT_DRIVE,$EXIT,MOVE_NAME
|
|||
|
PUBLIC UPCONV,ADD_PROMPT,CTTY,PRINT_DEFAULT_DIRECTORY
|
|||
|
PUBLIC ADD_NAME_TO_ENVIRONMENT,PWD,SCAN_DOUBLE_NULL
|
|||
|
PUBLIC FIND_NAME_IN_ENVIRONMENT,STORE_CHAR
|
|||
|
PUBLIC FIND_PATH,DELETE_PATH,FIND_PROMPT
|
|||
|
PUBLIC SCASB2
|
|||
|
|
|||
|
IF KANJI
|
|||
|
PUBLIC TESTKANJ
|
|||
|
ENDIF
|
|||
|
|
|||
|
BREAK <Environment utilities>
|
|||
|
ASSUME DS:TRANGROUP
|
|||
|
|
|||
|
ADD_PROMPT:
|
|||
|
CALL DELETE_PROMPT ; DELETE ANY EXISTING PROMPT
|
|||
|
CALL SCAN_DOUBLE_NULL
|
|||
|
ADD_PROMPT2:
|
|||
|
PUSH SI
|
|||
|
CALL GETARG
|
|||
|
POP SI
|
|||
|
retz ; PRE SCAN FOR ARGUMENTS
|
|||
|
CALL MOVE_NAME ; MOVE IN NAME
|
|||
|
CALL GETARG
|
|||
|
JMP SHORT ADD_NAME
|
|||
|
;
|
|||
|
; Input: DS:SI points to a CR terminated string
|
|||
|
; Output: carry flag is set if no room
|
|||
|
; otherwise name is added to environment
|
|||
|
;
|
|||
|
ADD_NAME_TO_ENVIRONMENT:
|
|||
|
CALL GETARG
|
|||
|
JZ DISP_ENV
|
|||
|
;
|
|||
|
; check if line contains exactly one equals sign
|
|||
|
;
|
|||
|
XOR BX,BX ;= COUNT IS 0
|
|||
|
PUSH SI ;SAVE POINTER TO BEGINNING OF LINE
|
|||
|
EQLP:
|
|||
|
LODSB ;GET A CHAR
|
|||
|
CMP AL,13 ;IF CR WE'RE ALL DONE
|
|||
|
JZ QUEQ
|
|||
|
CMP AL,"=" ;LOOK FOR = SIGN
|
|||
|
JNZ EQLP ;NOT THERE, GET NEXT CHAR
|
|||
|
INC BL ;OTHERWISE INCREMENT EQ COUNT
|
|||
|
CMP BYTE PTR [SI],13 ;LOOK FOR CR FOLLOWING = SIGN
|
|||
|
JNZ EQLP
|
|||
|
INC BH ;SET BH=1 MEANS NO PARAMETERS
|
|||
|
JMP EQLP ;AND LOOK FOR MORE
|
|||
|
QUEQ:
|
|||
|
POP SI ;RESTORE BEGINNING OF LINE
|
|||
|
DEC BL ;ZERO FLAG MEANS ONLY ONE EQ
|
|||
|
JZ ONEQ ;GOOD LINE
|
|||
|
MOV DX,OFFSET TRANGROUP:SYNTMES
|
|||
|
JMP CERROR
|
|||
|
|
|||
|
ONEQ:
|
|||
|
PUSH BX
|
|||
|
CALL DELETE_NAME_IN_ENVIRONMENT
|
|||
|
POP BX
|
|||
|
DEC BH
|
|||
|
retz
|
|||
|
|
|||
|
CALL SCAN_DOUBLE_NULL
|
|||
|
CALL MOVE_NAME
|
|||
|
ADD_NAME:
|
|||
|
LODSB
|
|||
|
CMP AL,13
|
|||
|
retz
|
|||
|
CALL STORE_CHAR
|
|||
|
JMP ADD_NAME
|
|||
|
|
|||
|
DISP_ENV:
|
|||
|
MOV DS,[RESSEG]
|
|||
|
ASSUME DS:RESGROUP
|
|||
|
MOV DS,[ENVIRSEG]
|
|||
|
ASSUME DS:NOTHING
|
|||
|
XOR SI,SI
|
|||
|
PENVLP:
|
|||
|
CMP BYTE PTR [SI],0
|
|||
|
retz
|
|||
|
|
|||
|
MOV DX,SI
|
|||
|
CALL ZPRINT
|
|||
|
CALL CRLF2
|
|||
|
PENVLP2:
|
|||
|
LODSB
|
|||
|
OR AL,AL
|
|||
|
JNZ PENVLP2
|
|||
|
JMP PENVLP
|
|||
|
|
|||
|
ASSUME DS:TRANGROUP
|
|||
|
DELETE_PATH:
|
|||
|
MOV SI,OFFSET TRANGROUP:PATH_TEXT
|
|||
|
JMP SHORT DELETE_NAME_IN_environment
|
|||
|
|
|||
|
DELETE_PROMPT:
|
|||
|
MOV SI,OFFSET TRANGROUP:PROMPT_TEXT
|
|||
|
|
|||
|
DELETE_NAME_IN_environment:
|
|||
|
;
|
|||
|
; Input: DS:SI points to a "=" terminated string
|
|||
|
; Output: carry flag is set if name not found
|
|||
|
; otherwise name is deleted
|
|||
|
;
|
|||
|
PUSH SI
|
|||
|
PUSH DS
|
|||
|
CALL FIND ; ES:DI POINTS TO NAME
|
|||
|
JC DEL1
|
|||
|
MOV SI,DI ; SAVE IT
|
|||
|
CALL SCASB2 ; SCAN FOR THE NUL
|
|||
|
XCHG SI,DI
|
|||
|
CALL GETENVSIZ
|
|||
|
SUB CX,SI
|
|||
|
PUSH ES
|
|||
|
POP DS ; ES:DI POINTS TO NAME, DS:SI POINTS TO NEXT NAME
|
|||
|
REP MOVSB ; DELETE THE NAME
|
|||
|
DEL1:
|
|||
|
POP DS
|
|||
|
POP SI
|
|||
|
return
|
|||
|
|
|||
|
FIND_PATH:
|
|||
|
MOV SI,OFFSET TRANGROUP:PATH_TEXT
|
|||
|
JMP SHORT FIND_NAME_IN_environment
|
|||
|
|
|||
|
FIND_PROMPT:
|
|||
|
MOV SI,OFFSET TRANGROUP:PROMPT_TEXT
|
|||
|
|
|||
|
FIND_NAME_IN_environment:
|
|||
|
;
|
|||
|
; Input: DS:SI points to a "=" terminated string
|
|||
|
; Output: ES:DI points to the arguments in the environment
|
|||
|
; zero is set if name not found
|
|||
|
; carry flag is set if name not valid format
|
|||
|
;
|
|||
|
CALL FIND ; FIND THE NAME
|
|||
|
retc ; CARRY MEANS NOT FOUND
|
|||
|
JMP SCASB1 ; SCAN FOR = SIGN
|
|||
|
;
|
|||
|
; On return of FIND1, ES:DI points to beginning of name
|
|||
|
;
|
|||
|
FIND:
|
|||
|
CLD
|
|||
|
CALL COUNT0 ; CX = LENGTH OF NAME
|
|||
|
MOV ES,[RESSEG]
|
|||
|
ASSUME ES:RESGROUP
|
|||
|
MOV ES,[ENVIRSEG]
|
|||
|
ASSUME ES:NOTHING
|
|||
|
XOR DI,DI
|
|||
|
FIND1:
|
|||
|
PUSH CX
|
|||
|
PUSH SI
|
|||
|
PUSH DI
|
|||
|
FIND11:
|
|||
|
LODSB
|
|||
|
|
|||
|
IF KANJI
|
|||
|
CALL TESTKANJ
|
|||
|
JZ NOTKANJ3
|
|||
|
DEC SI
|
|||
|
LODSW
|
|||
|
INC DI
|
|||
|
INC DI
|
|||
|
CMP AX,ES:[DI-2]
|
|||
|
JNZ FIND12
|
|||
|
DEC CX
|
|||
|
LOOP FIND11
|
|||
|
JMP SHORT FIND12
|
|||
|
|
|||
|
NOTKANJ3:
|
|||
|
ENDIF
|
|||
|
|
|||
|
CALL UPCONV
|
|||
|
INC DI
|
|||
|
CMP AL,ES:[DI-1]
|
|||
|
JNZ FIND12
|
|||
|
LOOP FIND11
|
|||
|
FIND12:
|
|||
|
POP DI
|
|||
|
POP SI
|
|||
|
POP CX
|
|||
|
retz
|
|||
|
PUSH CX
|
|||
|
CALL SCASB2 ; SCAN FOR A NUL
|
|||
|
POP CX
|
|||
|
CMP BYTE PTR ES:[DI],0
|
|||
|
JNZ FIND1
|
|||
|
STC ; INDICATE NOT FOUND
|
|||
|
return
|
|||
|
|
|||
|
COUNT0:
|
|||
|
PUSH DS
|
|||
|
POP ES
|
|||
|
MOV DI,SI
|
|||
|
|
|||
|
COUNT1:
|
|||
|
PUSH DI ; COUNT NUMBER OF CHARS UNTIL "="
|
|||
|
CALL SCASB1
|
|||
|
JMP SHORT COUNTX
|
|||
|
COUNT2:
|
|||
|
PUSH DI ; COUNT NUMBER OF CHARS UNTIL NUL
|
|||
|
CALL SCASB2
|
|||
|
COUNTX:
|
|||
|
POP CX
|
|||
|
SUB DI,CX
|
|||
|
XCHG DI,CX
|
|||
|
return
|
|||
|
|
|||
|
MOVE_NAME:
|
|||
|
CMP BYTE PTR DS:[SI],13
|
|||
|
retz
|
|||
|
LODSB
|
|||
|
|
|||
|
IF KANJI
|
|||
|
CALL TESTKANJ
|
|||
|
JZ NOTKANJ1
|
|||
|
CALL STORE_CHAR
|
|||
|
LODSB
|
|||
|
CALL STORE_CHAR
|
|||
|
JMP SHORT MOVE_NAME
|
|||
|
|
|||
|
NOTKANJ1:
|
|||
|
ENDIF
|
|||
|
|
|||
|
CALL UPCONV
|
|||
|
CALL STORE_CHAR
|
|||
|
CMP AL,"="
|
|||
|
JNZ MOVE_NAME
|
|||
|
return
|
|||
|
|
|||
|
GETARG:
|
|||
|
MOV SI,80H
|
|||
|
LODSB
|
|||
|
OR AL,AL
|
|||
|
retz
|
|||
|
CALL SCANOFF
|
|||
|
CMP AL,13
|
|||
|
return
|
|||
|
|
|||
|
SCAN_DOUBLE_NULL:
|
|||
|
MOV ES,[RESSEG]
|
|||
|
ASSUME ES:RESGROUP
|
|||
|
MOV ES,[ENVIRSEG]
|
|||
|
ASSUME ES:NOTHING
|
|||
|
XOR DI,DI
|
|||
|
SDN1:
|
|||
|
CALL SCASB2
|
|||
|
CMP BYTE PTR ES:[DI],0
|
|||
|
JNZ SDN1
|
|||
|
return
|
|||
|
|
|||
|
SCASB1:
|
|||
|
MOV AL,"=" ; SCAN FOR AN =
|
|||
|
JMP SHORT SCASBX
|
|||
|
SCASB2:
|
|||
|
XOR AL,AL ; SCAN FOR A NUL
|
|||
|
SCASBX:
|
|||
|
MOV CX,100H
|
|||
|
REPNZ SCASB
|
|||
|
return
|
|||
|
|
|||
|
IF KANJI
|
|||
|
TESTKANJ:
|
|||
|
CMP AL,81H
|
|||
|
JB NOTLEAD
|
|||
|
CMP AL,9FH
|
|||
|
JBE ISLEAD
|
|||
|
CMP AL,0E0H
|
|||
|
JB NOTLEAD
|
|||
|
CMP AL,0FCH
|
|||
|
JBE ISLEAD
|
|||
|
NOTLEAD:
|
|||
|
PUSH AX
|
|||
|
XOR AX,AX ;Set zero
|
|||
|
POP AX
|
|||
|
return
|
|||
|
|
|||
|
ISLEAD:
|
|||
|
PUSH AX
|
|||
|
XOR AX,AX ;Set zero
|
|||
|
INC AX ;Reset zero
|
|||
|
POP AX
|
|||
|
return
|
|||
|
ENDIF
|
|||
|
|
|||
|
UPCONV:
|
|||
|
CMP AL,"a"
|
|||
|
JB RET22C
|
|||
|
CMP AL,"z"
|
|||
|
JA RET22C
|
|||
|
SUB AL,20H ; Lower-case changed to upper-case
|
|||
|
RET22C:
|
|||
|
CALL DWORD PTR CS:[INTERNATVARS.Map_call]
|
|||
|
return
|
|||
|
;
|
|||
|
; STORE A CHAR IN environment, GROWING IT IF NECESSARY
|
|||
|
;
|
|||
|
STORE_CHAR:
|
|||
|
PUSH CX
|
|||
|
PUSH BX
|
|||
|
CALL GETENVSIZ
|
|||
|
MOV BX,CX
|
|||
|
SUB BX,2 ; SAVE ROOM FOR DOUBLE NULL
|
|||
|
CMP DI,BX
|
|||
|
JB STORE1
|
|||
|
|
|||
|
PUSH AX
|
|||
|
PUSH CX
|
|||
|
PUSH BX ; Save Size of environment
|
|||
|
CALL FREE_TPA
|
|||
|
POP BX
|
|||
|
ADD BX,2 ; Recover true environment size
|
|||
|
MOV CL,4
|
|||
|
SHR BX,CL ; Convert back to paragraphs
|
|||
|
INC BX ; Try to grow environment by one para
|
|||
|
MOV AH,SETBLOCK
|
|||
|
INT int_command
|
|||
|
PUSHF
|
|||
|
PUSH ES
|
|||
|
MOV ES,[RESSEG]
|
|||
|
CALL ALLOC_TPA
|
|||
|
POP ES
|
|||
|
POPF
|
|||
|
POP CX
|
|||
|
POP AX
|
|||
|
JNC STORE1
|
|||
|
MOV DX,OFFSET TRANGROUP:ENVERR
|
|||
|
JMP CERROR
|
|||
|
STORE1:
|
|||
|
STOSB
|
|||
|
MOV WORD PTR ES:[DI],0 ; NULL IS AT END
|
|||
|
POP BX
|
|||
|
POP CX
|
|||
|
return
|
|||
|
|
|||
|
GETENVSIZ:
|
|||
|
;Get size of environment in bytes, rounded up to paragraph boundry
|
|||
|
;ES has environment segment
|
|||
|
;Size returned in CX, all other registers preserved
|
|||
|
|
|||
|
PUSH ES
|
|||
|
PUSH AX
|
|||
|
MOV AX,ES
|
|||
|
DEC AX ;Point at arena
|
|||
|
MOV ES,AX
|
|||
|
MOV AX,ES:[arena_size]
|
|||
|
MOV CL,4
|
|||
|
SHL AX,CL ;Convert to bytes
|
|||
|
MOV CX,AX
|
|||
|
POP AX
|
|||
|
POP ES
|
|||
|
return
|
|||
|
|
|||
|
PRINT_DRIVE:
|
|||
|
MOV AH,GET_DEFAULT_DRIVE
|
|||
|
INT int_command
|
|||
|
ADD AL,"A"
|
|||
|
JMP OUT
|
|||
|
|
|||
|
ASSUME DS:TRANGROUP,ES:TRANGROUP
|
|||
|
PWD:
|
|||
|
CALL PRINT_DIRECTORY
|
|||
|
CALL CRLF2
|
|||
|
return
|
|||
|
|
|||
|
PRINT_DEFAULT_DIRECTORY:
|
|||
|
MOV BYTE PTR DS:[FCB],0
|
|||
|
PRINT_DIRECTORY:
|
|||
|
MOV DL,DS:[FCB]
|
|||
|
MOV AL,DL
|
|||
|
ADD AL,'@'
|
|||
|
CMP AL,'@'
|
|||
|
JNZ GOTDRIVE
|
|||
|
ADD AL,[CURDRV]
|
|||
|
INC AL
|
|||
|
GOTDRIVE:
|
|||
|
PUSH AX
|
|||
|
MOV SI,OFFSET TRANGROUP:PWDBUF+3
|
|||
|
MOV AH,CURRENT_DIR
|
|||
|
INT int_command
|
|||
|
JNC DPBISOK
|
|||
|
PUSH CS
|
|||
|
POP DS
|
|||
|
JMP DRVBAD
|
|||
|
DPBISOK:
|
|||
|
MOV DI,OFFSET TRANGROUP:PWDBUF
|
|||
|
MOV DX,DI
|
|||
|
POP AX
|
|||
|
MOV AH,DRVCHAR
|
|||
|
STOSW
|
|||
|
MOV AL,[DIRCHAR]
|
|||
|
STOSB
|
|||
|
JMP ZPRINT
|
|||
|
|
|||
|
$EXIT:
|
|||
|
PUSH ES
|
|||
|
MOV ES,[RESSEG]
|
|||
|
ASSUME ES:RESGROUP
|
|||
|
MOV AX,[PARENT]
|
|||
|
MOV WORD PTR ES:[PDB_Parent_PID],AX
|
|||
|
|
|||
|
IF IBM
|
|||
|
CMP [PERMCOM],0
|
|||
|
JNZ NORESETVEC ;Don't reset the vector if a PERMCOM
|
|||
|
LDS DX,DWORD PTR ES:[SYS_CALL]
|
|||
|
ASSUME DS:NOTHING
|
|||
|
MOV AX,(SET_INTERRUPT_VECTOR SHL 8) + INT_COMMAND
|
|||
|
INT int_command
|
|||
|
NORESETVEC:
|
|||
|
ENDIF
|
|||
|
|
|||
|
POP ES
|
|||
|
ASSUME ES:TRANGROUP
|
|||
|
MOV ES,[TPA]
|
|||
|
MOV AH,DEALLOC
|
|||
|
INT int_command ; Now running in "free" space
|
|||
|
MOV AX,(EXIT SHL 8)
|
|||
|
INT int_command
|
|||
|
|
|||
|
CTTY:
|
|||
|
CALL SETPATH ; Get spec
|
|||
|
MOV AX,(OPEN SHL 8) OR 2 ; Read and write
|
|||
|
INT int_command ; Open new device
|
|||
|
JC ISBADDEV
|
|||
|
MOV BX,AX
|
|||
|
MOV AX,IOCTL SHL 8
|
|||
|
INT int_command
|
|||
|
TEST DL,80H
|
|||
|
JNZ DEVISOK
|
|||
|
MOV AH,CLOSE ; Close initial handle
|
|||
|
INT int_command
|
|||
|
ISBADDEV:
|
|||
|
MOV DX,OFFSET TRANGROUP:BADDEV
|
|||
|
CALL PRINT
|
|||
|
JMP RESRET
|
|||
|
|
|||
|
DEVISOK:
|
|||
|
XOR DH,DH
|
|||
|
OR DL,3 ; Make sure has CON attributes
|
|||
|
MOV AX,(IOCTL SHL 8) OR 1
|
|||
|
INT int_command
|
|||
|
PUSH BX ; Save handle
|
|||
|
MOV CX,3
|
|||
|
XOR BX,BX
|
|||
|
ICLLOOP: ; Close basic handles
|
|||
|
MOV AH,CLOSE
|
|||
|
INT int_command
|
|||
|
INC BX
|
|||
|
LOOP ICLLOOP
|
|||
|
POP BX ; Get handle
|
|||
|
MOV AH,XDUP
|
|||
|
INT int_command ; Dup it to 0
|
|||
|
MOV AH,XDUP
|
|||
|
INT int_command ; Dup to 1
|
|||
|
MOV AH,XDUP
|
|||
|
INT int_command ; Dup to 2
|
|||
|
MOV AH,CLOSE ; Close initial handle
|
|||
|
INT int_command
|
|||
|
RESRET:
|
|||
|
MOV DS,[RESSEG]
|
|||
|
ASSUME DS:RESGROUP
|
|||
|
PUSH DS
|
|||
|
MOV AX,WORD PTR DS:[PDB_JFN_Table] ; Get new 0 and 1
|
|||
|
MOV [IO_SAVE],AX
|
|||
|
MOV AX,OFFSET RESGROUP:LODCOM1
|
|||
|
PUSH AX
|
|||
|
ZMMMM PROC FAR
|
|||
|
RET ; Force header to be checked
|
|||
|
ZMMMM ENDP
|
|||
|
|
|||
|
TRANCODE ENDS
|
|||
|
END
|
|||
|
|