mirror of https://github.com/microsoft/MS-DOS.git
788 lines
23 KiB
NASM
788 lines
23 KiB
NASM
;
|
||
; This version of COMMAND is divided into three distinct parts. First is the
|
||
; resident portion, which includes handlers for interrupts 22H (terminate),
|
||
; 23H (Cntrl-C), 24H (fatal error), and 27H (stay resident); it also has code
|
||
; to test and, if necessary, reload the transient portion. Following the
|
||
; resident is the init code, which is overwritten after use. Then comes the
|
||
; transient portion, which includes all command processing (whether internal
|
||
; or external). The transient portion loads at the end of physical memory,
|
||
; and it may be overlayed by programs that need as much memory as possible.
|
||
; When the resident portion of command regains control from a user program, a
|
||
; checksum is performed on the transient portion to see if it must be
|
||
; reloaded. Thus programs which do not need maximum memory will save the time
|
||
; required to reload COMMAND when they terminate.
|
||
|
||
;
|
||
; REV 1.17
|
||
; 05/19/82 Fixed bug in BADEXE error (relocation error must return to
|
||
; resident since the EXELOAD may have overwritten the transient.
|
||
; REV 1.18
|
||
; 05/21/82 IBM version always looks on drive A
|
||
; MSVER always looks on default drive
|
||
;
|
||
; REV 1.19
|
||
; 06/03/82 Drive spec now entered in command line
|
||
; 06/07/82 Added VER command (print DOS version number) and VOL command
|
||
; (print volume label)
|
||
; REV 1.20
|
||
; 06/09/82 Prints "directory" after directories
|
||
; 06/13/82 MKDIR, CHDIR, PWD, RMDIR added
|
||
; REV 1.50
|
||
; Some code for new 2.0 DOS, sort of HACKey. Not enough time to
|
||
; do it right.
|
||
; REV 1.70
|
||
; EXEC used to fork off new processes
|
||
; REV 1.80
|
||
; C switch for single command execution
|
||
; REV 1.90
|
||
; Batch uses XENIX
|
||
; Rev 2.00
|
||
; Lots of neato stuff
|
||
; IBM 2.00 level
|
||
; Rev 2.01
|
||
; 'D' switch for date time suppression
|
||
; Rev 2.02
|
||
; Default userpath is NUL rather than BIN
|
||
; same as IBM
|
||
; COMMAND split into pieces
|
||
; Rev 2.10
|
||
; INTERNATIONAL SUPPORT
|
||
; Rev 2.11 COMMAND split into more pieces
|
||
|
||
INCLUDE DOSSYM.ASM
|
||
INCLUDE DEVSYM.ASM
|
||
INCLUDE COMSW.ASM
|
||
INCLUDE COMEQU.ASM
|
||
|
||
CODERES SEGMENT PUBLIC
|
||
CODERES ENDS
|
||
|
||
DATARES SEGMENT PUBLIC BYTE
|
||
EXTRN COMBAD:BYTE,NEEDCOM:BYTE,DRVMSG:BYTE
|
||
EXTRN DEFMSG:BYTE,PROMPT:BYTE,EXECEMES:BYTE,EXEBAD:BYTE
|
||
EXTRN TOOBIG:BYTE,NOCOM:BYTE,RBADNAM:BYTE,INT_2E_RET:DWORD
|
||
EXTRN NOHANDMES:BYTE,BMEMMES:BYTE,HALTMES:BYTE,FRETMES:BYTE
|
||
EXTRN PARENT:WORD,HANDLE01:WORD,LOADING:BYTE,BATCH:WORD
|
||
EXTRN TRNSEG:WORD,COMDRV:BYTE,MEMSIZ:WORD,SUM:WORD,EXTCOM:BYTE
|
||
EXTRN IO_SAVE:WORD,PERMCOM:BYTE,SINGLECOM:WORD,VERVAL:WORD
|
||
EXTRN PIPEFLAG:BYTE,SAVE_PDB:WORD,COMSPEC:BYTE,TRANS:WORD
|
||
EXTRN TRANVARS:BYTE,LTPA:WORD,RSWITCHAR:BYTE,RDIRCHAR:BYTE
|
||
EXTRN RETCODE:WORD,FORFLAG:BYTE
|
||
|
||
IF IBMVER
|
||
EXTRN SYS_CALL:DWORD,ZEXEC:WORD,EXESEG:WORD,EXESUM:WORD
|
||
EXTRN USER_SS:WORD,USER_SP:WORD
|
||
ENDIF
|
||
|
||
DATARES ENDS
|
||
|
||
ENVIRONMENT SEGMENT PUBLIC PARA ; Default COMMAND environment
|
||
ENVIRONMENT ENDS
|
||
|
||
INIT SEGMENT PUBLIC PARA
|
||
EXTRN CONPROC:NEAR
|
||
INIT ENDS
|
||
|
||
TAIL SEGMENT PUBLIC PARA
|
||
TAIL ENDS
|
||
|
||
TRANCODE SEGMENT PUBLIC PARA
|
||
TRANCODE ENDS
|
||
|
||
TRANDATA SEGMENT PUBLIC BYTE
|
||
EXTRN TRANDATAEND:BYTE
|
||
TRANDATA ENDS
|
||
|
||
TRANSPACE SEGMENT PUBLIC BYTE
|
||
EXTRN TRANSPACEEND:BYTE,HEADCALL:DWORD
|
||
TRANSPACE ENDS
|
||
|
||
TRANTAIL SEGMENT PUBLIC PARA
|
||
TRANTAIL ENDS
|
||
|
||
ZEXEC_CODE SEGMENT PUBLIC PARA
|
||
ZEXEC_CODE ENDS
|
||
|
||
ZEXEC_DATA SEGMENT PUBLIC BYTE
|
||
ZEXEC_DATA ENDS
|
||
|
||
RESGROUP GROUP CODERES,DATARES,ENVIRONMENT,INIT,TAIL
|
||
TRANGROUP GROUP TRANCODE,TRANDATA,TRANSPACE,TRANTAIL
|
||
EGROUP GROUP ZEXEC_CODE,ZEXEC_DATA
|
||
|
||
ENVIRONMENT SEGMENT PUBLIC PARA ; Default COMMAND environment
|
||
|
||
PUBLIC ECOMSPEC,ENVIREND,PATHSTRING
|
||
|
||
ORG 0
|
||
ENVARENA DB 10H DUP (?) ; Pad for mem arena
|
||
PATHSTRING DB "PATH="
|
||
USERPATH LABEL BYTE
|
||
|
||
DB 0 ; Null path
|
||
DB "COMSPEC="
|
||
ECOMSPEC DB "/COMMAND.COM"
|
||
DB 134 DUP (0)
|
||
|
||
ENVIREND LABEL BYTE
|
||
|
||
ENVIRONSIZ EQU $-PATHSTRING
|
||
ENVIRONSIZ2 EQU $-ECOMSPEC
|
||
ENVIRONMENT ENDS
|
||
|
||
|
||
; START OF RESIDENT PORTION
|
||
|
||
CODERES SEGMENT PUBLIC
|
||
|
||
PUBLIC GETCOMDSK2,LODCOM,THEADFIX,CONTCTERM,LOADCOM,INT_2E,LODCOM1
|
||
PUBLIC CHKSUM,SETVECT,EXT_EXEC,TREMCHECK,RESTHAND,CONTC,RSTACK
|
||
PUBLIC SAVHAND
|
||
|
||
IF IBMVER
|
||
PUBLIC EXECHK,SYSCALL,EXEC_WAIT
|
||
ENDIF
|
||
|
||
ASSUME CS:RESGROUP,DS:NOTHING,ES:NOTHING,SS:NOTHING
|
||
|
||
EXTRN RPRINT:NEAR,ASKEND:NEAR,DSKERR:NEAR
|
||
|
||
|
||
ORG 0
|
||
ZERO = $
|
||
|
||
ORG 100H
|
||
|
||
PROGSTART:
|
||
JMP RESGROUP:CONPROC
|
||
|
||
DB (80H - 3) DUP (?)
|
||
RSTACK LABEL WORD
|
||
|
||
IF IBMVER
|
||
SYSCALL:
|
||
CMP AH,EXEC
|
||
JZ do_exec
|
||
JMP DWORD PTR [SYS_CALL]
|
||
|
||
do_exec:
|
||
PUSH ES
|
||
PUSH DS
|
||
PUSH BP
|
||
PUSH DI
|
||
PUSH SI
|
||
PUSH DX
|
||
PUSH CX
|
||
PUSH BX
|
||
PUSH AX
|
||
MOV [user_ss],SS
|
||
MOV [user_sp],SP
|
||
;
|
||
; are we running on RSTACK already?
|
||
;
|
||
PUSH CS
|
||
POP BX ; BX <- CS
|
||
PUSH SS
|
||
POP AX ; AX <- SS
|
||
CMP AX,BX ; IF AX == BX then no stack switch!
|
||
JZ Get_mem
|
||
MOV SS,BX
|
||
ASSUME SS:RESGROUP
|
||
MOV SP,OFFSET RESGROUP:RSTACK
|
||
|
||
Get_mem:
|
||
MOV BX,0FFFFH ; allocate all of memory
|
||
MOV AH,ALLOC
|
||
INT int_command
|
||
MOV AX,OFFSET EGROUP:ZEXECDATAEND + 15
|
||
MOV CL,4
|
||
SHR AX,CL
|
||
MOV CX,AX ; Save in CX
|
||
CMP BX,AX ; enough for EXEC?
|
||
JB EXECMER ; nope... cry
|
||
MOV AH,ALLOC
|
||
INT int_command
|
||
JC EXECMER ; Memory arenas probably trashed
|
||
ADD BX,AX
|
||
MOV [MEMSIZ],BX
|
||
SUB BX,CX
|
||
MOV [EXESEG],BX ; exec
|
||
MOV ES,AX
|
||
MOV AH,DEALLOC
|
||
INT int_command
|
||
PUSH CS
|
||
POP DS
|
||
ASSUME DS:RESGROUP
|
||
CALL EXECHK
|
||
CMP DX,[EXESUM]
|
||
JZ HAVEXEC ; EXEC OK
|
||
MOV DX,OFFSET RESGROUP:COMSPEC
|
||
MOV AX,OPEN SHL 8
|
||
INT int_command ; Open COMMAND.COM
|
||
JC EXECMER
|
||
MOV BX,AX ; Handle
|
||
MOV DX,OFFSET RESGROUP:TRANSTART
|
||
ADD DX,OFFSET TRANGROUP:EXECSTART - 100H
|
||
XOR CX,CX ; Seek loc
|
||
MOV AX,LSEEK SHL 8
|
||
INT int_command
|
||
MOV CX,OFFSET EGROUP:ZEXECCODEEND
|
||
MOV DS,[EXESEG]
|
||
ASSUME DS:NOTHING
|
||
MOV AH,READ
|
||
INT int_command
|
||
PUSH AX
|
||
MOV AH,CLOSE
|
||
INT int_command ; Close COMMAND.COM
|
||
POP CX
|
||
CMP CX,OFFSET EGROUP:ZEXECCODEEND
|
||
JNZ EXECMER ; Size matched
|
||
|
||
CALL EXECHK
|
||
CMP DX,[EXESUM]
|
||
JNZ EXECMER
|
||
HAVEXEC:
|
||
MOV [LOADING],0 ; Flag to DSKERR
|
||
CALL DWORD PTR [ZEXEC]
|
||
JMP SHORT EXECRET
|
||
execmer:
|
||
LDS SI,DWORD PTR [user_Sp]
|
||
MOV [SI.user_AX],exec_not_enough_memory
|
||
PUSH [SI.user_F]
|
||
POPF
|
||
STC
|
||
PUSHF
|
||
POP [SI.user_F]
|
||
execret:
|
||
MOV SS,[user_SS]
|
||
ASSUME SS:NOTHING
|
||
MOV SP,[user_SP]
|
||
POP AX ; PUSH ES
|
||
POP BX ; PUSH DS
|
||
POP CX ; PUSH BP
|
||
POP DX ; PUSH DI
|
||
POP SI ; PUSH SI
|
||
POP DI ; PUSH DX
|
||
POP BP ; PUSH CX
|
||
POP DS ; PUSH BX
|
||
POP ES ; PUSH AX
|
||
IRET
|
||
|
||
EXECHK:
|
||
ASSUME DS:NOTHING,ES:NOTHING,SS:NOTHING
|
||
PUSH DS
|
||
MOV DS,[EXESEG]
|
||
MOV CX,OFFSET EGROUP:ZEXECCODEEND
|
||
XOR SI,SI
|
||
JMP CHECK_SUM
|
||
ENDIF
|
||
|
||
EXEC_ERR: ; Select the correct error message
|
||
MOV DX,OFFSET RESGROUP:RBADNAM
|
||
CMP AX,exec_file_not_found
|
||
JZ GOTEXECEMES
|
||
CMP AX,error_access_denied
|
||
JZ GOTEXECEMES
|
||
MOV DX,OFFSET RESGROUP:TOOBIG
|
||
CMP AX,exec_not_enough_memory
|
||
JZ GOTEXECEMES
|
||
MOV DX,OFFSET RESGROUP:EXEBAD
|
||
CMP AX,exec_bad_format
|
||
JZ GOTEXECEMES
|
||
MOV DX,OFFSET RESGROUP:EXECEMES
|
||
GOTEXECEMES:
|
||
PUSH CS
|
||
POP DS
|
||
CALL RPRINT
|
||
JMP SHORT NOEXEC
|
||
|
||
EXT_EXEC:
|
||
;
|
||
; we are now running in free space. anything we do from here
|
||
; on may get trashed. Move the stack (also in free space) to
|
||
; allocated space because since EXEC restores the stack,
|
||
; somebody may trash what is on the stack.
|
||
;
|
||
MOV CX,CS
|
||
MOV SS,CX
|
||
MOV SP,OFFSET RESGROUP:RSTACK
|
||
;
|
||
; Oops!! We have to make sure that the EXEC code doesn't blop a newstack!
|
||
;
|
||
;
|
||
INT int_command ; Do the EXEC
|
||
JC EXEC_ERR ; EXEC failed
|
||
EXEC_WAIT:
|
||
MOV AH,WAIT
|
||
INT int_command ; Get the return code
|
||
MOV [RETCODE],AX
|
||
NOEXEC:
|
||
JMP LODCOM
|
||
|
||
CONTC:
|
||
STI
|
||
MOV AX,CS
|
||
MOV DS,AX
|
||
ASSUME DS:RESGROUP
|
||
MOV AH,DISK_RESET
|
||
INT int_command ; Reset disks in case files were open
|
||
TEST [BATCH],-1
|
||
JZ CONTCTERM
|
||
JMP ASKEND ; See if user wants to terminate batch
|
||
CONTCTERM:
|
||
XOR BP,BP ; Indicate no read
|
||
MOV [FORFLAG],0 ; Turn off for processing
|
||
MOV [PIPEFLAG],0 ; Turn off any pipe
|
||
CMP [SINGLECOM],0 ; See if we need to set SINGLECOM
|
||
JZ NOSETSING
|
||
MOV [SINGLECOM],-1 ; Cause termination on pipe, batch, for
|
||
NOSETSING:
|
||
CMP [EXTCOM],0
|
||
JNZ DODAB ; Internal ^C
|
||
JMP LODCOM1
|
||
DODAB:
|
||
STC ; Tell DOS to abort
|
||
ZZY PROC FAR
|
||
RET ; Leave flags on stack
|
||
ZZY ENDP
|
||
|
||
BADMEMERR: ; Allocation error loading transient
|
||
MOV DX,OFFSET RESGROUP:BMEMMES
|
||
FATALC:
|
||
PUSH CS
|
||
POP DS
|
||
CALL RPRINT
|
||
CMP [PERMCOM],0
|
||
JZ FATALRET
|
||
CMP [SINGLECOM],0 ; If PERMCOM and SINGLECOM
|
||
JNZ FATALRET ; Must take INT_2E exit
|
||
MOV DX,OFFSET RESGROUP:HALTMES
|
||
CALL RPRINT
|
||
STALL:
|
||
JMP STALL ; Crash the system nicely
|
||
|
||
FATALRET:
|
||
MOV DX,OFFSET RESGROUP:FRETMES
|
||
CALL RPRINT
|
||
FATALRET2:
|
||
CMP [PERMCOM],0 ; If we get here and PERMCOM,
|
||
JNZ RET_2E ; must be INT_2E
|
||
IF IBM
|
||
LDS DX,DWORD PTR [SYS_CALL]
|
||
ASSUME DS:NOTHING
|
||
MOV AX,(SET_INTERRUPT_VECTOR SHL 8) + INT_COMMAND
|
||
INT int_command
|
||
ENDIF
|
||
MOV AX,[PARENT]
|
||
MOV WORD PTR CS:[PDB_Parent_PID],AX
|
||
MOV AX,(EXIT SHL 8) ; Return to lower level
|
||
INT int_command
|
||
|
||
RET_2E:
|
||
PUSH CS
|
||
POP DS
|
||
ASSUME DS:RESGROUP,ES:NOTHING,SS:NOTHING
|
||
MOV [SINGLECOM],0 ; Turn off singlecom
|
||
MOV ES,[LTPA]
|
||
MOV AH,DEALLOC
|
||
INT int_command ; Free up space used by transient
|
||
MOV BX,[SAVE_PDB]
|
||
MOV AH,SET_CURRENT_PDB
|
||
INT int_command ; Current process is user
|
||
MOV AX,[RETCODE]
|
||
CMP [EXTCOM],0
|
||
JNZ GOTECODE
|
||
XOR AX,AX ; Internals always return 0
|
||
GOTECODE:
|
||
MOV [EXTCOM],1 ; Force external
|
||
JMP [INT_2E_RET] ;"IRET"
|
||
|
||
INT_2E: ; Magic command executer
|
||
ASSUME DS:NOTHING,ES:NOTHING,SS:NOTHING
|
||
POP WORD PTR [INT_2E_RET]
|
||
POP WORD PTR [INT_2E_RET+2] ;Get return address
|
||
POP AX ;Chuck flags
|
||
PUSH CS
|
||
POP ES
|
||
MOV DI,80H
|
||
MOV CX,64
|
||
REP MOVSW
|
||
MOV AH,GET_CURRENT_PDB
|
||
INT int_command ; Get user's header
|
||
MOV [SAVE_PDB],BX
|
||
MOV AH,SET_CURRENT_PDB
|
||
MOV BX,CS
|
||
INT int_command ; Current process is me
|
||
MOV [SINGLECOM],81H
|
||
MOV [EXTCOM],1 ; Make sure this case forced
|
||
|
||
LODCOM: ; Termination handler
|
||
CMP [EXTCOM],0
|
||
JZ LODCOM1 ; If internal, memory already allocated
|
||
MOV BX,0FFFFH
|
||
MOV AH,ALLOC
|
||
INT int_command
|
||
MOV AX,OFFSET TRANGROUP:TRANSPACEEND + 15
|
||
MOV CL,4
|
||
SHR AX,CL
|
||
|
||
IF IBM
|
||
PUSH AX
|
||
MOV AX,OFFSET EGROUP:ZEXECDATAEND + 15
|
||
MOV CL,4
|
||
SHR AX,CL
|
||
POP CX
|
||
ADD AX,CX
|
||
ENDIF
|
||
|
||
ADD AX,20H
|
||
CMP BX,AX ; Is less than 512 byte buffer worth it?
|
||
JNC MEMOK
|
||
BADMEMERRJ:
|
||
JMP BADMEMERR ; Not enough memory
|
||
MEMOK:
|
||
MOV AH,ALLOC
|
||
INT int_command
|
||
JC BADMEMERRJ ; Memory arenas probably trashed
|
||
MOV [EXTCOM],0 ; Flag not to ALLOC again
|
||
MOV [LTPA],AX ; New TPA is base just allocated
|
||
ADD BX,AX
|
||
MOV [MEMSIZ],BX
|
||
|
||
MOV AX,OFFSET TRANGROUP:TRANSPACEEND + 15
|
||
MOV CL,4
|
||
SHR AX,CL
|
||
|
||
IF IBM
|
||
PUSH AX
|
||
MOV AX,OFFSET EGROUP:ZEXECDATAEND + 15
|
||
MOV CL,4
|
||
SHR AX,CL
|
||
POP CX
|
||
ADD AX,CX
|
||
ENDIF
|
||
|
||
SUB BX,AX
|
||
MOV [TRNSEG],BX ; Transient starts here
|
||
LODCOM1:
|
||
MOV AX,CS
|
||
MOV SS,AX
|
||
ASSUME SS:RESGROUP
|
||
MOV SP,OFFSET RESGROUP:RSTACK
|
||
MOV DS,AX
|
||
ASSUME DS:RESGROUP
|
||
CALL HEADFIX ; Make sure files closed stdin and stdout restored
|
||
XOR BP,BP ; Flag command ok
|
||
MOV AX,-1
|
||
XCHG AX,[VERVAL]
|
||
CMP AX,-1
|
||
JZ NOSETVER
|
||
MOV AH,SET_VERIFY_ON_WRITE ; AL has correct value
|
||
INT int_command
|
||
NOSETVER:
|
||
CMP [SINGLECOM],-1
|
||
JNZ NOSNG
|
||
JMP FATALRET2 ; We have finished the single command
|
||
NOSNG:
|
||
CALL SETVECT
|
||
|
||
IF IBMVER
|
||
CALL EXECHK ; Check exe loader
|
||
CMP DX,[EXESUM]
|
||
JNZ BOGUS_COM
|
||
ENDIF
|
||
|
||
CALL CHKSUM ; Check the transient
|
||
CMP DX,[SUM]
|
||
JZ HAVCOM ; Transient OK
|
||
BOGUS_COM:
|
||
MOV [LOADING],1 ; Flag DSKERR routine
|
||
CALL LOADCOM
|
||
CHKSAME:
|
||
|
||
IF IBMVER
|
||
CALL EXECHK
|
||
CMP DX,[EXESUM]
|
||
JNZ ALSO_BOGUS
|
||
ENDIF
|
||
|
||
CALL CHKSUM
|
||
CMP DX,[SUM]
|
||
JZ HAVCOM ; Same COMMAND
|
||
ALSO_BOGUS:
|
||
CALL WRONGCOM
|
||
JMP SHORT CHKSAME
|
||
HAVCOM:
|
||
MOV AX,CHAR_OPER SHL 8
|
||
INT int_command
|
||
MOV [RSWITCHAR],DL
|
||
CMP DL,'/'
|
||
JNZ USESLASH
|
||
MOV [RDIRCHAR],'\' ; Select alt path separator
|
||
USESLASH:
|
||
MOV [LOADING],0 ; Flag to DSKERR
|
||
MOV SI,OFFSET RESGROUP:TRANVARS
|
||
MOV DI,OFFSET TRANGROUP:HEADCALL
|
||
MOV ES,[TRNSEG]
|
||
CLD
|
||
MOV CX,8
|
||
REP MOVSW ; Transfer INFO to transient
|
||
MOV AX,[MEMSIZ]
|
||
MOV WORD PTR DS:[PDB_block_len],AX ; Adjust my own header
|
||
JMP DWORD PTR [TRANS]
|
||
|
||
; Far call to REMCHECK for TRANSIENT
|
||
TREMCHECK PROC FAR
|
||
CALL REMCHECK
|
||
RET
|
||
TREMCHECK ENDP
|
||
|
||
REMCHECK:
|
||
;All registers preserved. Returns zero if media removable, NZ if fixed
|
||
; AL is drive (0=DEF, 1=A,...)
|
||
IF IBM
|
||
PUSH AX
|
||
OR AL,AL
|
||
JNZ GOTDRV2
|
||
MOV AH,GET_DEFAULT_DRIVE
|
||
INT int_command
|
||
INC AL ;A=1
|
||
GOTDRV2:
|
||
PUSH BX
|
||
MOV BL,AL
|
||
INT 11H ;IBM EQUIP CALL
|
||
ROL AL,1
|
||
ROL AL,1
|
||
AND AL,3
|
||
JNZ NOT_SINGLE
|
||
INC AL
|
||
NOT_SINGLE:
|
||
INC AL ; AL is now MAX floppy #
|
||
CMP BL,AL
|
||
POP BX
|
||
JBE SETREM ; Is an IBM floppy and so is removable
|
||
OR AL,AL ; Know AL is non-zero
|
||
JMP SHORT SETNREM
|
||
SETREM:
|
||
ELSE
|
||
PUSH AX
|
||
ENDIF
|
||
|
||
XOR AX,AX ;Zero
|
||
|
||
IF IBM
|
||
SETNREM:
|
||
ENDIF
|
||
|
||
POP AX
|
||
RET
|
||
|
||
; Far call to HEADFIX for TRANSIENT
|
||
THEADFIX PROC FAR
|
||
CALL HEADFIX
|
||
RET
|
||
THEADFIX ENDP
|
||
|
||
HEADFIX:
|
||
XOR BX,BX ; Clean up header
|
||
MOV CX,[IO_SAVE]
|
||
MOV DX,WORD PTR DS:[PDB_JFN_Table]
|
||
CMP CL,DL
|
||
JZ CHK1 ; Stdin matches
|
||
MOV AH,CLOSE
|
||
INT int_command
|
||
MOV DS:[PDB_JFN_Table],CL ; Restore stdin
|
||
CHK1:
|
||
INC BX
|
||
CMP CH,DH ; Stdout matches
|
||
JZ CHKOTHERHAND
|
||
MOV AH,CLOSE
|
||
INT int_command
|
||
MOV DS:[PDB_JFN_Table+1],CH ; Restore stdout
|
||
CHKOTHERHAND:
|
||
ADD BX,4 ; Skip 2,3,4
|
||
MOV CX,FilPerProc - 5 ; Already done 0,1,2,3,4
|
||
CLOSELOOP:
|
||
MOV AH,CLOSE
|
||
INT int_command
|
||
INC BX
|
||
LOOP CLOSELOOP
|
||
RET
|
||
|
||
SAVHAND:
|
||
ASSUME DS:NOTHING,ES:NOTHING,SS:NOTHING
|
||
PUSH DS
|
||
PUSH BX ; Set stdin to sterr, stdout to stderr
|
||
PUSH AX
|
||
MOV AH,GET_CURRENT_PDB
|
||
INT int_command ; Get user's header
|
||
MOV DS,BX
|
||
MOV AX,WORD PTR DS:[PDB_JFN_Table]
|
||
MOV [HANDLE01],AX ; Save user's stdin, stdout
|
||
MOV AL,DS:[PDB_JFN_Table+2]
|
||
MOV AH,AL
|
||
MOV WORD PTR DS:[PDB_JFN_Table],AX ; Dup stderr
|
||
POP AX
|
||
POP BX
|
||
POP DS
|
||
RET
|
||
|
||
ASSUME DS:RESGROUP
|
||
GETCOMDSK2:
|
||
CALL GETCOMDSK
|
||
JMP LODCOM1 ; Memory already allocated
|
||
|
||
RESTHAND:
|
||
PUSH DS
|
||
PUSH BX ; Restore stdin, stdout to user
|
||
PUSH AX
|
||
MOV AH,GET_CURRENT_PDB
|
||
INT int_command ; Point to user's header
|
||
MOV AX,[HANDLE01]
|
||
MOV DS,BX
|
||
ASSUME DS:NOTHING
|
||
MOV WORD PTR DS:[PDB_JFN_Table],AX ; Stuff his old 0 and 1
|
||
POP AX
|
||
POP BX
|
||
POP DS
|
||
RET
|
||
ASSUME DS:RESGROUP,SS:RESGROUP
|
||
|
||
HOPELESS:
|
||
MOV DX,OFFSET RESGROUP:NOCOM
|
||
JMP FATALC
|
||
|
||
GETCOMDSK:
|
||
MOV DX,OFFSET RESGROUP:NEEDCOM
|
||
GETCOMDSK3:
|
||
MOV AL,[COMDRV]
|
||
CALL REMCHECK
|
||
JNZ HOPELESS ;Non-removable media
|
||
CALL RPRINT
|
||
MOV DX,OFFSET RESGROUP:DRVMSG
|
||
CMP [COMDRV],0
|
||
JNZ GETCOM1
|
||
MOV DX,OFFSET RESGROUP:DEFMSG
|
||
GETCOM1:
|
||
CALL RPRINT
|
||
MOV DX,OFFSET RESGROUP:PROMPT
|
||
CALL RPRINT
|
||
CALL GetRawFlushedByte
|
||
RET
|
||
|
||
; flush world and get raw input
|
||
GetRawFlushedByte:
|
||
MOV AX,(STD_CON_INPUT_FLUSH SHL 8) OR RAW_CON_INPUT
|
||
INT int_command ; Get char without testing or echo
|
||
MOV AX,(STD_CON_INPUT_FLUSH SHL 8) + 0
|
||
INT int_command
|
||
return
|
||
|
||
LOADCOM: ; Load in transient
|
||
INC BP ; Flag command read
|
||
MOV DX,OFFSET RESGROUP:COMSPEC
|
||
MOV AX,OPEN SHL 8
|
||
INT int_command ; Open COMMAND.COM
|
||
JNC READCOM
|
||
CMP AX,open_too_many_open_files
|
||
JNZ TRYDOOPEN
|
||
MOV DX,OFFSET RESGROUP:NOHANDMES
|
||
JMP FATALC ; Fatal, will never find a handle
|
||
|
||
TRYDOOPEN:
|
||
CALL GETCOMDSK
|
||
JMP SHORT LOADCOM
|
||
|
||
READCOM:
|
||
MOV BX,AX ; Handle
|
||
MOV DX,OFFSET RESGROUP:TRANSTART
|
||
XOR CX,CX ; Seek loc
|
||
MOV AX,LSEEK SHL 8
|
||
INT int_command
|
||
JC WRONGCOM1
|
||
MOV CX,OFFSET TRANGROUP:TRANSPACEEND - 100H
|
||
|
||
IF IBM
|
||
ADD CX,15
|
||
AND CX,0FFF0H
|
||
ADD CX,OFFSET EGROUP:ZEXECCODEEND
|
||
ENDIF
|
||
|
||
PUSH DS
|
||
MOV DS,[TRNSEG]
|
||
ASSUME DS:NOTHING
|
||
MOV DX,100H
|
||
MOV AH,READ
|
||
INT int_command
|
||
POP DS
|
||
ASSUME DS:RESGROUP
|
||
WRONGCOM1:
|
||
PUSHF
|
||
PUSH AX
|
||
MOV AH,CLOSE
|
||
INT int_command ; Close COMMAND.COM
|
||
POP AX
|
||
POPF
|
||
JC WRONGCOM ; If error on READ
|
||
CMP AX,CX
|
||
JZ RET10 ; Size matched
|
||
WRONGCOM:
|
||
MOV DX,OFFSET RESGROUP:COMBAD
|
||
CALL GETCOMDSK3
|
||
JMP SHORT LOADCOM ; Try again
|
||
|
||
CHKSUM: ; Compute transient checksum
|
||
PUSH DS
|
||
MOV DS,[TRNSEG]
|
||
MOV SI,100H
|
||
MOV CX,OFFSET TRANGROUP:TRANDATAEND - 100H
|
||
|
||
CHECK_SUM:
|
||
CLD
|
||
SHR CX,1
|
||
XOR DX,DX
|
||
CHK:
|
||
LODSW
|
||
ADD DX,AX
|
||
LOOP CHK
|
||
POP DS
|
||
RET10: RET
|
||
|
||
SETVECT: ; Set useful vectors
|
||
MOV DX,OFFSET RESGROUP:LODCOM
|
||
MOV AX,(SET_INTERRUPT_VECTOR SHL 8) OR 22H ; Set Terminate address
|
||
INT int_command
|
||
MOV DX,OFFSET RESGROUP:CONTC
|
||
MOV AX,(SET_INTERRUPT_VECTOR SHL 8) OR 23H ; Set Ctrl-C address
|
||
INT int_command
|
||
MOV DX,OFFSET RESGROUP:DSKERR
|
||
MOV AX,(SET_INTERRUPT_VECTOR SHL 8) OR 24H ; Set Hard Disk Error address
|
||
INT int_command
|
||
RET
|
||
|
||
CODERES ENDS
|
||
|
||
; This TAIL segment is used to produce a PARA aligned label in the resident
|
||
; group which is the location where the transient segments will be loaded
|
||
; initial.
|
||
|
||
TAIL SEGMENT PUBLIC PARA
|
||
ORG 0
|
||
TRANSTART LABEL WORD
|
||
TAIL ENDS
|
||
|
||
; This TAIL segment is used to produce a PARA aligned label in the transient
|
||
; group which is the location where the exec segments will be loaded
|
||
; initial.
|
||
|
||
TRANTAIL SEGMENT PUBLIC PARA
|
||
ORG 0
|
||
EXECSTART LABEL WORD
|
||
TRANTAIL ENDS
|
||
|
||
IF IBMVER
|
||
INCLUDE EXEC.ASM
|
||
ENDIF
|
||
|
||
END PROGSTART
|
||
|