MS-DOS/v2.0/source/DEBCOM2.ASM

1270 lines
35 KiB
NASM

TITLE PART2 DEBUGGER COMMANDS
; Routines to perform debugger commands except ASSEMble and UASSEMble
.xlist
.xcref
INCLUDE DEBEQU.ASM
INCLUDE DOSSYM.ASM
.cref
.list
CODE SEGMENT PUBLIC BYTE 'CODE'
CODE ENDS
CONST SEGMENT PUBLIC BYTE
EXTRN NOTFND:BYTE,NOROOM:BYTE,DRVLET:BYTE,NOSPACE:BYTE,NAMBAD:BYTE
EXTRN TOOBIG:BYTE,ERRMES:BYTE
EXTRN EXEBAD:BYTE,HEXERR:BYTE,EXEWRT:BYTE,HEXWRT:BYTE
EXTRN EXECEMES:BYTE,WRTMES1:BYTE,WRTMES2:BYTE,ACCMES:BYTE
EXTRN FLAGTAB:WORD,EXEC_BLOCK:BYTE,COM_LINE:DWORD,COM_FCB1:DWORD
EXTRN COM_FCB2:DWORD,COM_SSSP:DWORD,COM_CSIP:DWORD,RETSAVE:WORD
EXTRN NEWEXEC:BYTE,HEADSAVE:WORD
EXTRN REGTAB:BYTE,TOTREG:BYTE,NOREGL:BYTE
EXTRN USER_PROC_PDB:WORD,STACK:BYTE,RSTACK:WORD,AXSAVE:WORD
EXTRN BXSAVE:WORD,DSSAVE:WORD,ESSAVE:WORD,CSSAVE:WORD,IPSAVE:WORD
EXTRN SSSAVE:WORD,CXSAVE:WORD,SPSAVE:WORD,FSAVE:WORD
EXTRN SREG:BYTE,SEGTAB:WORD,REGDIF:WORD,RDFLG:BYTE
CONST ENDS
DATA SEGMENT PUBLIC BYTE
EXTRN DEFDUMP:BYTE,TRANSADD:DWORD,INDEX:WORD,BUFFER:BYTE
EXTRN ASMADD:BYTE,DISADD:BYTE,NSEG:WORD,BPTAB:BYTE
EXTRN BRKCNT:WORD,TCOUNT:WORD,SWITCHAR:BYTE,XNXCMD:BYTE,XNXOPT:BYTE
EXTRN AWORD:BYTE,EXTPTR:WORD,HANDLE:WORD,PARSERR:BYTE
DATA ENDS
DG GROUP CODE,CONST,DATA
CODE SEGMENT PUBLIC BYTE 'CODE'
ASSUME CS:DG,DS:DG,ES:DG,SS:DG
PUBLIC DEFIO,SKIP_FILE,PREPNAME,DEBUG_FOUND
PUBLIC REG,COMPARE,GO,INPUT,LOAD
PUBLIC NAME,OUTPUT,TRACE,ZTRACE,DWRITE
if sysver
PUBLIC DISPREG
endif
EXTRN GETHEX:NEAR,GETEOL:NEAR
EXTRN CRLF:NEAR,BLANK:NEAR,OUT:NEAR
EXTRN OUTSI:NEAR,OUTDI:NEAR,INBUF:NEAR,SCANB:NEAR,SCANP:NEAR
EXTRN RPRBUF:NEAR,HEX:NEAR,OUT16:NEAR,DIGIT:NEAR
EXTRN COMMAND:NEAR,DISASLN:NEAR,SET_TERMINATE_VECTOR:NEAR
EXTRN RESTART:NEAR,DABORT:NEAR,TERMINATE:NEAR,DRVERR:NEAR
EXTRN FIND_DEBUG:NEAR,NMIInt:NEAR,NMIIntEnd:NEAR
EXTRN HEXCHK:NEAR,GETHEX1:NEAR,PRINT:NEAR,DSRANGE:NEAR
EXTRN ADDRESS:NEAR,HEXIN:NEAR,PERROR:NEAR
DEBCOM2:
DISPREG:
MOV SI,OFFSET DG:REGTAB
MOV BX,OFFSET DG:AXSAVE
MOV BYTE PTR TOTREG,13
MOV CH,0
MOV CL,NOREGL
REPDISP:
SUB TOTREG,CL
CALL DISPREGLINE
CALL CRLF
MOV CH,0
MOV CL,NOREGL
CMP CL,TOTREG
JL REPDISP
MOV CL,TOTREG
CALL DISPREGLINE
CALL BLANK
CALL DISPFLAGS
CALL CRLF
MOV AX,[IPSAVE]
MOV WORD PTR [DISADD],AX
PUSH AX
MOV AX,[CSSAVE]
MOV WORD PTR [DISADD+2],AX
PUSH AX
MOV [NSEG],-1
CALL DISASLN
POP WORD PTR DISADD+2
POP WORD PTR DISADD
MOV AX,[NSEG]
CMP AL,-1
JZ CRLFJ
CMP AH,-1
JZ NOOVER
XCHG AL,AH
NOOVER:
CBW
MOV BX,AX
SHL BX,1
MOV AX,WORD PTR [BX+SREG]
CALL OUT
XCHG AL,AH
CALL OUT
MOV AL,":"
CALL OUT
MOV DX,[INDEX]
CALL OUT16
MOV AL,"="
CALL OUT
MOV BX,[BX+SEGTAB]
PUSH DS
MOV DS,[BX]
MOV BX,DX
MOV DX,[BX]
POP DS
TEST BYTE PTR [AWORD],-1
JZ OUT8
CALL OUT16
CRLFJ:
JMP CRLF
OUT8:
MOV AL,DL
CALL HEX
JMP CRLF
DISPREGJ:JMP DISPREG
; Perform register dump if no parameters or set register if a
; register designation is a parameter.
REG:
CALL SCANP
JZ DISPREGJ
MOV DL,[SI]
INC SI
MOV DH,[SI]
CMP DH,13
JZ FLAG
INC SI
CALL GETEOL
CMP DH," "
JZ FLAG
MOV DI,OFFSET DG:REGTAB
XCHG AX,DX
PUSH CS
POP ES
MOV CX,REGTABLEN
REPNZ SCASW
JNZ BADREG
OR CX,CX
JNZ NOTPC
DEC DI
DEC DI
MOV AX,CS:[DI-2]
NOTPC:
CALL OUT
MOV AL,AH
CALL OUT
CALL BLANK
PUSH DS
POP ES
LEA BX,[DI+REGDIF-2]
MOV DX,[BX]
CALL OUT16
CALL CRLF
MOV AL,":"
CALL OUT
CALL INBUF
CALL SCANB
JZ RET4
MOV CX,4
CALL GETHEX1
CALL GETEOL
MOV [BX],DX
RET4: RET
BADREG:
MOV AX,5200H+"B" ; BR ERROR
JMP ERR
FLAG:
CMP DL,"F"
JNZ BADREG
CALL DISPFLAGS
MOV AL,"-"
CALL OUT
CALL INBUF
CALL SCANB
XOR BX,BX
MOV DX,[FSAVE]
GETFLG:
LODSW
CMP AL,13
JZ SAVCHG
CMP AH,13
JZ FLGERR
MOV DI,OFFSET DG:FLAGTAB
MOV CX,32
PUSH CS
POP ES
REPNE SCASW
JNZ FLGERR
MOV CH,CL
AND CL,0FH
MOV AX,1
ROL AX,CL
TEST AX,BX
JNZ REPFLG
OR BX,AX
OR DX,AX
TEST CH,16
JNZ NEXFLG
XOR DX,AX
NEXFLG:
CALL SCANP
JMP SHORT GETFLG
DISPREGLINE:
LODS CS:WORD PTR [SI]
CALL OUT
MOV AL,AH
CALL OUT
MOV AL,"="
CALL OUT
MOV DX,[BX]
INC BX
INC BX
CALL OUT16
CALL BLANK
CALL BLANK
LOOP DISPREGLINE
RET
REPFLG:
MOV AX,4600H+"D" ; DF ERROR
FERR:
CALL SAVCHG
ERR:
CALL OUT
MOV AL,AH
CALL OUT
MOV SI,OFFSET DG:ERRMES
JMP PRINT
SAVCHG:
MOV [FSAVE],DX
RET
FLGERR:
MOV AX,4600H+"B" ; BF ERROR
JMP SHORT FERR
DISPFLAGS:
MOV SI,OFFSET DG:FLAGTAB
MOV CX,16
MOV DX,[FSAVE]
DFLAGS:
LODS CS:WORD PTR [SI]
SHL DX,1
JC FLAGSET
MOV AX,CS:[SI+30]
FLAGSET:
OR AX,AX
JZ NEXTFLG
CALL OUT
MOV AL,AH
CALL OUT
CALL BLANK
NEXTFLG:
LOOP DFLAGS
RET
; Input from the specified port and display result
INPUT:
MOV CX,4 ; Port may have 4 digits
CALL GETHEX ; Get port number in DX
CALL GETEOL
IN AL,DX ; Variable port input
CALL HEX ; And display
JMP CRLF
; Output a value to specified port.
OUTPUT:
MOV CX,4 ; Port may have 4 digits
CALL GETHEX ; Get port number
PUSH DX ; Save while we get data
MOV CX,2 ; Byte output only
CALL GETHEX ; Get data to output
CALL GETEOL
XCHG AX,DX ; Output data in AL
POP DX ; Port in DX
OUT DX,AL ; Variable port output
RET5: RET
COMPARE:
CALL DSRANGE
PUSH CX
PUSH AX
PUSH DX
CALL ADDRESS ; Same segment
CALL GETEOL
POP SI
MOV DI,DX
MOV ES,AX
POP DS
POP CX ; Length
DEC CX
CALL COMP ; Do one less than total
INC CX ; CX=1 (do last one)
COMP:
REPE CMPSB
JZ RET5
; Compare error. Print address, value; value, address.
DEC SI
CALL OUTSI
CALL BLANK
CALL BLANK
LODSB
CALL HEX
CALL BLANK
CALL BLANK
DEC DI
MOV AL,ES:[DI]
CALL HEX
CALL BLANK
CALL BLANK
CALL OUTDI
INC DI
CALL CRLF
XOR AL,AL
JMP SHORT COMP
ZTRACE:
IF ZIBO
; just like trace except skips OVER next INT or CALL.
CALL SETADD ; get potential starting point
CALL GETEOL ; check for end of line
MOV [TCOUNT],1 ; only a single go at it
MOV ES,[CSSAVE] ; point to instruction to execute
MOV DI,[IPSAVE] ; include offset in segment
XOR DX,DX ; where to place breakpoint
MOV AL,ES:[DI] ; get the opcode
CMP AL,11101000B ; direct intra call
JZ ZTrace3 ; yes, 3 bytes
CMP AL,10011010B ; direct inter call
JZ ZTrace5 ; yes, 5 bytes
CMP AL,11111111B ; indirect?
JZ ZTraceModRM ; yes, go figure length
CMP AL,11001100B ; short interrupt?
JZ ZTrace1 ; yes, 1 byte
CMP AL,11001101B ; long interrupt?
JZ ZTrace2 ; yes, 2 bytes
CMP AL,11100010B ; loop
JZ ZTrace2 ; 2 byter
CMP AL,11100001B ; loopz/loope
JZ ZTrace2 ; 2 byter
CMP AL,11100000B ; loopnz/loopne
JZ ZTrace2 ; 2 byter
AND AL,11111110B ; check for rep
CMP AL,11110010B ; perhaps?
JNZ Step ; can't do anything special, step
MOV AL,ES:[DI+1] ; next instruction
AND AL,11111110B ; ignore w bit
CMP AL,10100100B ; MOVS
JZ ZTrace2 ; two byte
CMP AL,10100110B ; CMPS
JZ ZTrace2 ; two byte
CMP AL,10101110B ; SCAS
JZ ZTrace2 ; two byte
CMP AL,10101100B ; LODS
JZ ZTrace2 ; two byte
CMP AL,10101010B ; STOS
JZ ZTrace2 ; two byte
JMP Step ; bogus, do single step
ZTraceModRM:
MOV AL,ES:[DI+1] ; get next byte
AND AL,11111000B ; get mod and type
CMP AL,01010000B ; indirect intra 8 bit offset?
JZ ZTrace3 ; yes, three byte whammy
CMP AL,01011000B ; indirect inter 8 bit offset
JZ ZTrace3 ; yes, three byte guy
CMP AL,10010000B ; indirect intra 16 bit offset?
JZ ZTrace4 ; four byte offset
CMP AL,10011000B ; indirect inter 16 bit offset?
JZ ZTrace4 ; four bytes
JMP Step ; can't figger out what this is!
ZTrace5:INC DX
ZTrace4:INC DX
ZTrace3:INC DX
ZTrace2:INC DX
ZTrace1:INC DX
ADD DI,DX ; offset to breakpoint instruction
MOV WORD PTR [BPTab],DI ; save offset
MOV WORD PTR [BPTab+2],ES ; save segment
MOV AL,ES:[DI] ; get next opcode byte
MOV BYTE PTR [BPTab+4],AL ; save it
MOV BYTE PTR ES:[DI],0CCh ; break point it
MOV [BrkCnt],1 ; only this breakpoint
JMP DExit ; start the operation!
ENDIF
; Trace 1 instruction or the number of instruction specified
; by the parameter using 8086 trace mode. Registers are all
; set according to values in save area
TRACE:
CALL SETADD
CALL SCANP
CALL HEXIN
MOV DX,1
JC STOCNT
MOV CX,4
CALL GETHEX
STOCNT:
MOV [TCOUNT],DX
CALL GETEOL
STEP:
MOV [BRKCNT],0
OR BYTE PTR [FSAVE+1],1
DEXIT:
IF NOT SYSVER
MOV BX,[USER_PROC_PDB]
MOV AH,SET_CURRENT_PDB
INT 21H
ENDIF
PUSH DS
XOR AX,AX
MOV DS,AX
MOV WORD PTR DS:[12],OFFSET DG:BREAKFIX ; Set vector 3--breakpoint instruction
MOV WORD PTR DS:[14],CS
MOV WORD PTR DS:[4],OFFSET DG:REENTER ; Set vector 1--Single step
MOV WORD PTR DS:[6],CS
CLI
IF SETCNTC
MOV WORD PTR DS:[8CH],OFFSET DG:CONTC ; Set vector 23H (CTRL-C)
MOV WORD PTR DS:[8EH],CS
ENDIF
POP DS
MOV SP,OFFSET DG:STACK
POP AX
POP BX
POP CX
POP DX
POP BP
POP BP
POP SI
POP DI
POP ES
POP ES
POP SS
MOV SP,[SPSAVE]
PUSH [FSAVE]
PUSH [CSSAVE]
PUSH [IPSAVE]
MOV DS,[DSSAVE]
IRET
STEP1:
CALL CRLF
CALL DISPREG
JMP SHORT STEP
; Re-entry point from CTRL-C. Top of stack has address in 86-DOS for
; continuing, so we must pop that off.
CONTC:
ADD SP,6
JMP SHORT ReEnterReal
; Re-entry point from breakpoint. Need to decrement instruction
; pointer so it points to location where breakpoint actually
; occured.
BREAKFIX:
PUSH BP
MOV BP,SP
DEC WORD PTR [BP].OldIP
POP BP
JMP ReenterReal
; Re-entry point from trace mode or interrupt during
; execution. All registers are saved so they can be
; displayed or modified.
Interrupt_Frame STRUC
OldBP DW ?
OldIP DW ?
OldCS DW ?
OldF DW ?
OlderIP DW ?
OlderCS DW ?
OlderF DW ?
Interrupt_Frame ENDS
REENTER:
PUSH BP
MOV BP,SP ; get a frame to address from
PUSH AX
MOV AX,CS
CMP AX,[BP].OldCS ; Did we interrupt ourselves?
JNZ GoReEnter ; no, go reenter
MOV AX,[BP].OldIP
CMP AX,OFFSET DG:NMIInt ; interrupt below NMI interrupt?
JB GoReEnter ; yes, go reenter
CMP [BP].OLDIP,OFFSET DG:NMIIntEnd
JAE GoReEnter ; interrupt above NMI interrupt?
POP AX ; restore state
POP BP
SUB SP,6 ; switch TRACE and NMI stack frames
PUSH BP
MOV BP,SP ; set up frame
PUSH AX ; get temp variable
MOV AX,[BP].OlderIP ; get NMI Vector
MOV [BP].OldIP,AX ; stuff in new NMI vector
MOV AX,[BP].OlderCS ; get NMI Vector
MOV [BP].OldCS,AX ; stuff in new NMI vector
MOV AX,[BP].OlderF ; get NMI Vector
AND AH,0FEh ; turn off Trace if present
MOV [BP].OldF,AX ; stuff in new NMI vector
MOV [BP].OlderF,AX
MOV [BP].OlderIP,OFFSET DG:ReEnter ; offset of routine
MOV [BP].OlderCS,CS ; and CS
POP AX
POP BP
IRET ; go try again
GoReEnter:
POP AX
POP BP
ReEnterReal:
MOV CS:[SPSAVE+SEGDIF],SP
MOV CS:[SSSAVE+SEGDIF],SS
MOV CS:[FSAVE],CS
MOV SS,CS:[FSAVE]
MOV SP,OFFSET DG:RSTACK
PUSH ES
PUSH DS
PUSH DI
PUSH SI
PUSH BP
DEC SP
DEC SP
PUSH DX
PUSH CX
PUSH BX
PUSH AX
PUSH SS
POP DS
MOV SS,[SSSAVE]
MOV SP,[SPSAVE]
POP [IPSAVE]
POP [CSSAVE]
POP AX
AND AH,0FEH ; turn off trace mode bit
MOV [FSAVE],AX
MOV [SPSAVE],SP
PUSH DS
POP ES
PUSH DS
POP SS
MOV SP,OFFSET DG:STACK
PUSH DS
XOR AX,AX
MOV DS,AX
IF SETCNTC
MOV WORD PTR DS:[8CH],OFFSET DG:DABORT ; Set Ctrl-C vector
MOV WORD PTR DS:[8EH],CS
ENDIF
POP DS
STI
CLD
IF NOT SYSVER
MOV AH,GET_CURRENT_PDB
INT 21H
MOV [USER_PROC_PDB],BX
MOV BX,DS
MOV AH,SET_CURRENT_PDB
INT 21H
ENDIF
DEC [TCOUNT]
JZ CheckDisp
JMP Step1
CheckDisp:
MOV SI,OFFSET DG:BPTAB
MOV CX,[BRKCNT]
JCXZ SHOREG
PUSH ES
CLEARBP:
LES DI,DWORD PTR [SI]
ADD SI,4
MOVSB
LOOP CLEARBP
POP ES
SHOREG:
CALL CRLF
CALL DISPREG
JMP COMMAND
SETADD:
MOV BP,[CSSAVE]
CALL SCANP
CMP BYTE PTR [SI],"="
JNZ RET$5
INC SI
CALL ADDRESS
MOV [CSSAVE],AX
MOV [IPSAVE],DX
RET$5: RET
; Jump to program, setting up registers according to the
; save area. up to 10 breakpoint addresses may be specified.
GO:
CALL SETADD
XOR BX,BX
MOV DI,OFFSET DG:BPTAB
GO1:
CALL SCANP
JZ DEXEC
MOV BP,[CSSAVE]
CALL ADDRESS
MOV [DI],DX ; Save offset
MOV [DI+2],AX ; Save segment
ADD DI,5 ; Leave a little room
INC BX
CMP BX,1+BPMAX
JNZ GO1
MOV AX,5000H+"B" ; BP ERROR
JMP ERR
DEXEC:
MOV [BRKCNT],BX
MOV CX,BX
JCXZ NOBP
MOV DI,OFFSET DG:BPTAB
PUSH DS
SETBP:
LDS SI,ES:DWORD PTR [DI]
ADD DI,4
MOVSB
MOV BYTE PTR [SI-1],0CCH
LOOP SETBP
POP DS
NOBP:
MOV [TCOUNT],1
JMP DEXIT
SKIP_FILE:
MOV AH,CHAR_OPER
INT 21H
MOV [SWITCHAR],DL ; GET THE CURRENT SWITCH CHARACTER
FIND_DELIM:
LODSB
CALL DELIM1
JZ GOTDELIM
CALL DELIM2
JNZ FIND_DELIM
GOTDELIM:
DEC SI
RET
PREPNAME:
MOV ES,DSSAVE
PUSH SI
MOV DI,81H
COMTAIL:
LODSB
STOSB
CMP AL,13
JNZ COMTAIL
SUB DI,82H
XCHG AX,DI
MOV ES:(BYTE PTR [80H]),AL
POP SI
MOV DI,FCB
MOV AX,(PARSE_FILE_DESCRIPTOR SHL 8) OR 01H
INT 21H
MOV BYTE PTR [AXSAVE],AL ; Indicate analysis of first parm
CALL SKIP_FILE
MOV DI,6CH
MOV AX,(PARSE_FILE_DESCRIPTOR SHL 8) OR 01H
INT 21H
MOV BYTE PTR [AXSAVE+1],AL ; Indicate analysis of second parm
RET23: RET
; OPENS A XENIX PATHNAME SPECIFIED IN THE UNFORMATTED PARAMETERS
; VARIABLE [XNXCMD] SPECIFIES WHICH COMMAND TO OPEN IT WITH
;
; VARIABLE [HANDLE] CONTAINS THE HANDLE
; VARIABLE [EXTPTR] POINTS TO THE FILES EXTENSION
DELETE_A_FILE:
MOV BYTE PTR [XNXCMD],UNLINK
JMP SHORT OC_FILE
PARSE_A_FILE:
MOV BYTE PTR [XNXCMD],0
JMP SHORT OC_FILE
EXEC_A_FILE:
MOV BYTE PTR [XNXCMD],EXEC
MOV BYTE PTR [XNXOPT],1
JMP SHORT OC_FILE
OPEN_A_FILE:
MOV BYTE PTR [XNXCMD],OPEN
MOV BYTE PTR [XNXOPT],2 ; Try read write
CALL OC_FILE
JNC RET23
MOV BYTE PTR [XNXCMD],OPEN
MOV BYTE PTR [XNXOPT],0 ; Try read only
JMP SHORT OC_FILE
CREATE_A_FILE:
MOV BYTE PTR [XNXCMD],CREAT
OC_FILE:
PUSH DS
PUSH ES
PUSH AX
PUSH BX
PUSH CX
PUSH DX
PUSH SI
XOR AX,AX
MOV [EXTPTR],AX ; INITIALIZE POINTER TO EXTENSIONS
MOV AH,CHAR_OPER
INT 21H
MOV [SWITCHAR],DL ; GET THE CURRENT SWITCH CHARACTER
MOV SI,81H
OPEN1: CALL GETCHRUP
CALL DELIM2 ; END OF LINE?
JZ OPEN4
CALL DELIM1 ; SKIP LEADING DELIMITERS
JZ OPEN1
MOV DX,SI ; SAVE POINTER TO BEGINNING
DEC DX
OPEN2: CMP AL,"." ; LAST CHAR A "."?
JNZ OPEN3
MOV [EXTPTR],SI ; SAVE POINTER TO THE EXTENSION
OPEN3: CALL GETCHRUP
CALL DELIM1 ; LOOK FOR END OF PATHNAME
JZ OPEN4
CALL DELIM2
JNZ OPEN2
OPEN4: DEC SI ; POINT BACK TO LAST CHAR
PUSH [SI] ; SAVE TERMINATION CHAR
MOV BYTE PTR [SI],0 ; NULL TERMINATE THE STRING
MOV AL,[XNXOPT]
MOV AH,[XNXCMD] ; OPEN OR CREATE FILE
OR AH,AH
JZ OPNRET
MOV BX,OFFSET DG:EXEC_BLOCK
XOR CX,CX
INT 21H
MOV CS:[HANDLE],AX ; SAVE ERROR CODE OR HANDLE
OPNRET: POP [SI]
POP SI
POP DX
POP CX
POP BX
POP AX
POP ES
POP DS
RET
GETCHRUP:
LODSB
CMP AL,"a"
JB GCUR
CMP AL,"z"
JA GCUR
SUB AL,32
MOV [SI-1],AL
GCUR: RET
DELIM0: CMP AL,"["
JZ LIMRET
DELIM1: CMP AL," " ; SKIP THESE GUYS
JZ LIMRET
CMP AL,";"
JZ LIMRET
CMP AL,"="
JZ LIMRET
CMP AL,9
JZ LIMRET
CMP AL,","
JMP SHORT LIMRET
DELIM2: CMP AL,[SWITCHAR] ; STOP ON THESE GUYS
JZ LIMRET
CMP AL,13
LIMRET: RET
NAME:
CALL PREPNAME
MOV AL,BYTE PTR AXSAVE
MOV PARSERR,AL
PUSH ES
POP DS
PUSH CS
POP ES
MOV SI,FCB ; DS:SI points to user FCB
MOV DI,SI ; ES:DI points to DEBUG FCB
MOV CX,82
REP MOVSW
RET6: RET
BADNAM:
MOV DX,OFFSET DG:NAMBAD
JMP RESTART
IFHEX:
CMP BYTE PTR [PARSERR],-1 ; Invalid drive specification?
JZ BADNAM
CALL PARSE_A_FILE
MOV BX,[EXTPTR]
CMP WORD PTR DS:[BX],"EH" ; "HE"
JNZ RET6
CMP BYTE PTR DS:[BX+2],"X"
RET
IFEXE:
PUSH BX
MOV BX,[EXTPTR]
CMP WORD PTR DS:[BX],"XE" ; "EX"
JNZ RETIF
CMP BYTE PTR DS:[BX+2],"E"
RETIF: POP BX
RET
LOAD:
MOV BYTE PTR [RDFLG],READ
JMP SHORT DSKIO
DWRITE:
MOV BYTE PTR [RDFLG],WRITE
DSKIO:
MOV BP,[CSSAVE]
CALL SCANB
JNZ PRIMIO
JMP DEFIO
PRIMIO: CALL ADDRESS
CALL SCANB
JNZ PRMIO
JMP FILEIO
PRMIO: PUSH AX ; Save segment
MOV BX,DX ; Put displacement in proper register
MOV CX,1
CALL GETHEX ; Drive number must be 1 digit
PUSH DX
MOV CX,4
CALL GETHEX ; Logical record number
PUSH DX
MOV CX,3
CALL GETHEX ; Number of records
CALL GETEOL
MOV CX,DX
POP DX ; Logical record number
POP AX ; Drive number
CBW ; Turn off verify after write
MOV BYTE PTR DRVLET,AL ; Save drive in case of error
PUSH AX
PUSH BX
PUSH DX
MOV DL,AL
INC DL
MOV AH,GET_DPB
INT 21H
POP DX
POP BX
OR AL,AL
POP AX
POP DS ; Segment of transfer
JNZ DRVERRJ
CMP CS:BYTE PTR [RDFLG],WRITE
JZ ABSWRT
INT 25H ; Primitive disk read
JMP SHORT ENDABS
ABSWRT:
INT 26H ; Primitive disk write
ENDABS:
JNC RET0
DRVERRJ: JMP DRVERR
RET0:
POPF
RET
DEFIO:
MOV AX,[CSSAVE] ; Default segment
MOV DX,100H ; Default file I/O offset
CALL IFHEX
JNZ EXECHK
XOR DX,DX ; If HEX file, default OFFSET is zero
HEX2BINJ:JMP HEX2BIN
FILEIO:
; AX and DX have segment and offset of transfer, respectively
CALL IFHEX
JZ HEX2BINJ
EXECHK:
CALL IFEXE
JNZ BINFIL
CMP BYTE PTR [RDFLG],READ
JZ EXELJ
MOV DX,OFFSET DG:EXEWRT
JMP RESTART ; Can't write .EXE files
BINFIL:
CMP BYTE PTR [RDFLG],WRITE
JZ BINLOAD
CMP WORD PTR DS:[BX],4F00H + "C" ; "CO"
JNZ BINLOAD
CMP BYTE PTR DS:[BX+2],"M"
JNZ BINLOAD
EXELJ:
DEC SI
CMP DX,100H
JNZ PRER
CMP AX,[CSSAVE]
JZ OAF
PRER: JMP PERROR
OAF: CALL OPEN_A_FILE
JNC GDOPEN
MOV AX,exec_file_not_found
JMP EXECERR
GDOPEN: XOR DX,DX
XOR CX,CX
MOV BX,[HANDLE]
MOV AL,2
MOV AH,LSEEK
INT 21H
CALL IFEXE ; SUBTRACT 512 BYTES FOR EXE
JNZ BIN2 ; FILE LENGTH BECAUSE OF
SUB AX,512 ; THE HEADER
BIN2: MOV [BXSAVE],DX ; SET UP FILE SIZE IN DX:AX
MOV [CXSAVE],AX
MOV AH,CLOSE
INT 21H
JMP EXELOAD
NO_MEM_ERR:
MOV DX,OFFSET DG:TOOBIG
MOV AH,STD_CON_STRING_OUTPUT
INT 21H
JMP COMMAND
WRTFILEJ: JMP WRTFILE
NOFILEJ: JMP NOFILE
BINLOAD:
PUSH AX
PUSH DX
CMP BYTE PTR [RDFLG],WRITE
JZ WRTFILEJ
CALL OPEN_A_FILE
JC NOFILEJ
MOV BX,[HANDLE]
MOV AX,(LSEEK SHL 8) OR 2
XOR DX,DX
MOV CX,DX
INT 21H ; GET SIZE OF FILE
MOV SI,DX
MOV DI,AX ; SIZE TO SI:DI
MOV AX,(LSEEK SHL 8) OR 0
XOR DX,DX
MOV CX,DX
INT 21H ; RESET POINTER BACK TO BEGINNING
POP AX
POP BX
PUSH BX
PUSH AX ; TRANS ADDR TO BX:AX
ADD AX,15
MOV CL,4
SHR AX,CL
ADD BX,AX ; Start of transfer rounded up to seg
MOV DX,SI
MOV AX,DI ; DX:AX is size
MOV CX,16
DIV CX
OR DX,DX
JZ NOREM
INC AX
NOREM: ; AX is number of paras in transfer
ADD AX,BX ; AX is first seg that need not exist
CMP AX,CS:[PDB_block_len]
JA NO_MEM_ERR
MOV CXSAVE,DI
MOV BXSAVE,SI
POP DX
POP AX
RDWR:
; AX:DX is disk transfer address (segment:offset)
; SI:DI is length (32-bit number)
RDWRLOOP:
MOV BX,DX ; Make a copy of the offset
AND DX,000FH ; Establish the offset in 0H-FH range
MOV CL,4
SHR BX,CL ; Shift offset and
ADD AX,BX ; Add to segment register to get new Seg:offset
PUSH AX
PUSH DX ; Save AX,DX register pair
MOV WORD PTR [TRANSADD],DX
MOV WORD PTR [TRANSADD+2],AX
MOV CX,0FFF0H ; Keep request in segment
OR SI,SI ; Need > 64K?
JNZ BIGRDWR
MOV CX,DI ; Limit to amount requested
BIGRDWR:
PUSH DS
PUSH BX
MOV BX,[HANDLE]
MOV AH,[RDFLG]
LDS DX,[TRANSADD]
INT 21H ; Perform read or write
POP BX
POP DS
JC BADWR
CMP BYTE PTR [RDFLG],WRITE
JNZ GOODR
CMP CX,AX
JZ GOODR
BADWR: MOV CX,AX
STC
POP DX ; READ OR WRITE BOMBED OUT
POP AX
RET
GOODR:
MOV CX,AX
SUB DI,CX ; Request minus amount transferred
SBB SI,0 ; Ripple carry
OR CX,CX ; End-of-file?
POP DX ; Restore DMA address
POP AX
JZ RET8
ADD DX,CX ; Bump DMA address by transfer length
MOV BX,SI
OR BX,DI ; Finished with request
JNZ RDWRLOOP
RET8: CLC ; End-of-file not reached
RET
NOFILE:
MOV DX,OFFSET DG:NOTFND
RESTARTJMP:
JMP RESTART
WRTFILE:
CALL CREATE_A_FILE ; Create file we want to write to
MOV DX,OFFSET DG:NOROOM ; Creation error - report error
JC RESTARTJMP
MOV SI,BXSAVE ; Get high order number of bytes to transfer
CMP SI,000FH
JLE WRTSIZE ; Is bx less than or equal to FH
XOR SI,SI ; Ignore BX if greater than FH - set to zero
WRTSIZE:
MOV DX,OFFSET DG:WRTMES1 ; Print number bytes we are writing
CALL RPRBUF
OR SI,SI
JZ NXTBYT
MOV AX,SI
CALL DIGIT
NXTBYT:
MOV DX,CXSAVE
MOV DI,DX
CALL OUT16 ; Amount to write is SI:DI
MOV DX,OFFSET DG:WRTMES2
CALL RPRBUF
POP DX
POP AX
CALL RDWR
JNC CLSFLE
CALL CLSFLE
CALL DELETE_A_FILE
MOV DX,OFFSET DG:NOSPACE
JMP RESTARTJMP
CALL CLSFLE
JMP COMMAND
CLSFLE:
MOV AH,CLOSE
MOV BX,[HANDLE]
INT 21H
RET
EXELOAD:
POP [RETSAVE] ; Suck up return addr
INC BYTE PTR [NEWEXEC]
MOV BX,[USER_PROC_PDB]
MOV AX,DS
CMP AX,BX
JZ DEBUG_CURRENT
JMP FIND_DEBUG
DEBUG_CURRENT:
MOV AX,[DSSAVE]
DEBUG_FOUND:
MOV BYTE PTR [NEWEXEC],0
MOV [HEADSAVE],AX
PUSH [RETSAVE] ; Get the return address back
PUSH AX
MOV BX,CS
SUB AX,BX
PUSH CS
POP ES
MOV BX,AX
ADD BX,10H ; RESERVE HEADER
MOV AH,SETBLOCK
INT 21H
POP AX
MOV WORD PTR [COM_LINE+2],AX
MOV WORD PTR [COM_FCB1+2],AX
MOV WORD PTR [COM_FCB2+2],AX
CALL EXEC_A_FILE
JC EXECERR
CALL SET_TERMINATE_VECTOR ; Reset int 22
MOV AH,GET_CURRENT_PDB
INT 21H
MOV [USER_PROC_PDB],BX
MOV [DSSAVE],BX
MOV [ESSAVE],BX
MOV ES,BX
MOV WORD PTR ES:[PDB_exit],OFFSET DG:TERMINATE
MOV WORD PTR ES:[PDB_exit+2],DS
LES DI,[COM_CSIP]
MOV [CSSAVE],ES
MOV [IPSAVE],DI
MOV WORD PTR [DISADD+2],ES
MOV WORD PTR [DISADD],DI
MOV WORD PTR [ASMADD+2],ES
MOV WORD PTR [ASMADD],DI
MOV WORD PTR [DEFDUMP+2],ES
MOV WORD PTR [DEFDUMP],DI
MOV BX,DS
MOV AH,SET_CURRENT_PDB
INT 21H
LES DI,[COM_SSSP]
MOV AX,ES:[DI]
INC DI
INC DI
MOV [AXSAVE],AX
MOV [SSSAVE],ES
MOV [SPSAVE],DI
RET
EXECERR:
MOV DX,OFFSET DG:NOTFND
CMP AX,exec_file_not_found
JZ GOTEXECEMES
MOV DX,OFFSET DG:ACCMES
CMP AX,error_access_denied
JZ GOTEXECEMES
MOV DX,OFFSET DG:TOOBIG
CMP AX,exec_not_enough_memory
JZ GOTEXECEMES
MOV DX,OFFSET DG:EXEBAD
CMP AX,exec_bad_format
JZ GOTEXECEMES
MOV DX,OFFSET DG:EXECEMES
GOTEXECEMES:
MOV AH,STD_CON_STRING_OUTPUT
INT 21H
JMP COMMAND
HEX2BIN:
MOV [INDEX],DX
MOV DX,OFFSET DG:HEXWRT
CMP BYTE PTR [RDFLG],WRITE
JNZ RDHEX
JMP RESTARTJ2
RDHEX:
MOV ES,AX
CALL OPEN_A_FILE
MOV DX,OFFSET DG:NOTFND
JNC HEXFND
JMP RESTART
HEXFND:
XOR BP,BP
MOV SI,OFFSET DG:(BUFFER+BUFSIZ) ; Flag input buffer as empty
READHEX:
CALL GETCH
CMP AL,":" ; Search for : to start line
JNZ READHEX
CALL GETBYT ; Get byte count
MOV CL,AL
MOV CH,0
JCXZ HEXDONE
CALL GETBYT ; Get high byte of load address
MOV BH,AL
CALL GETBYT ; Get low byte of load address
MOV BL,AL
ADD BX,[INDEX] ; Add in offset
MOV DI,BX
CALL GETBYT ; Throw away type byte
READLN:
CALL GETBYT ; Get data byte
STOSB
CMP DI,BP ; Check if this is the largest address so far
JBE HAVBIG
MOV BP,DI ; Save new largest
HAVBIG:
LOOP READLN
JMP SHORT READHEX
GETCH:
CMP SI,OFFSET DG:(BUFFER+BUFSIZ)
JNZ NOREAD
MOV DX,OFFSET DG:BUFFER
MOV SI,DX
MOV AH,READ
PUSH BX
PUSH CX
MOV CX,BUFSIZ
MOV BX,[HANDLE]
INT 21H
POP CX
POP BX
OR AX,AX
JZ HEXDONE
NOREAD:
LODSB
CMP AL,1AH
JZ HEXDONE
OR AL,AL
JNZ RET7
HEXDONE:
MOV [CXSAVE],BP
MOV BXSAVE,0
RET
HEXDIG:
CALL GETCH
CALL HEXCHK
JNC RET7
MOV DX,OFFSET DG:HEXERR
RESTARTJ2:
JMP RESTART
GETBYT:
CALL HEXDIG
MOV BL,AL
CALL HEXDIG
SHL BL,1
SHL BL,1
SHL BL,1
SHL BL,1
OR AL,BL
RET7: RET
CODE ENDS
END DEBCOM2