mirror of https://github.com/microsoft/MS-DOS.git
838 lines
22 KiB
NASM
838 lines
22 KiB
NASM
|
TITLE DEBUGger for MS-DOS
|
|||
|
; DEBUG-86 8086 debugger runs under 86-DOS version 2.30
|
|||
|
;
|
|||
|
; Modified 5/4/82 by AaronR to do all I/O direct to devices
|
|||
|
; Runs on MS-DOS 1.28 and above
|
|||
|
; REV 1.20
|
|||
|
; Tab expansion
|
|||
|
; New device interface (1.29 and above)
|
|||
|
; REV 2.0
|
|||
|
; line by line assembler added by C. Peters
|
|||
|
; REV 2.1
|
|||
|
; Uses EXEC system call
|
|||
|
; REV 2.2
|
|||
|
; Ztrace mode by zibo.
|
|||
|
; Fix dump display to indent properly
|
|||
|
; Parity nonsense by zibo
|
|||
|
;
|
|||
|
; REV 2.3
|
|||
|
; Split into seperate modules to allow for
|
|||
|
; assembly on an IBM PC
|
|||
|
;
|
|||
|
|
|||
|
|
|||
|
|
|||
|
.xlist
|
|||
|
.xcref
|
|||
|
INCLUDE DEBEQU.ASM
|
|||
|
INCLUDE DOSSYM.ASM
|
|||
|
.cref
|
|||
|
.list
|
|||
|
|
|||
|
IF SYSVER
|
|||
|
|
|||
|
; Structure for system call 72
|
|||
|
|
|||
|
SYSINITVAR STRUC
|
|||
|
DPBHEAD DD ? ; Pointer to head of DPB-FAT list
|
|||
|
sft_addr DD ? ; Pointer to first FCB table
|
|||
|
; The following address points to the CLOCK device
|
|||
|
BCLOCK DD ?
|
|||
|
; The following address is used by DISKSTATCHK it is always
|
|||
|
; points to the console input device header
|
|||
|
BCON DD ? ; Console device entry points
|
|||
|
NUMIO DB 0 ; Number of disk tables
|
|||
|
MAXSEC DW 0 ; Maximum allowed sector size
|
|||
|
BUFFHEAD DD ?
|
|||
|
DEVHEAD DD ?
|
|||
|
SYSINITVAR ENDS
|
|||
|
|
|||
|
ENDIF
|
|||
|
|
|||
|
|
|||
|
CODE SEGMENT PUBLIC 'CODE'
|
|||
|
CODE ENDS
|
|||
|
|
|||
|
CONST SEGMENT PUBLIC BYTE
|
|||
|
|
|||
|
EXTRN USER_PROC_PDB:WORD,STACK:BYTE,CSSAVE:WORD,DSSAVE:WORD
|
|||
|
EXTRN SPSAVE:WORD,IPSAVE:WORD,LINEBUF:BYTE,QFLAG:BYTE
|
|||
|
EXTRN NEWEXEC:BYTE,HEADSAVE:WORD,LBUFSIZ:BYTE,BACMES:BYTE
|
|||
|
EXTRN BADVER:BYTE,ENDMES:BYTE,CARRET:BYTE,ParityMes:BYTE
|
|||
|
|
|||
|
IF IBMVER
|
|||
|
EXTRN DSIZ:BYTE,NOREGL:BYTE,DISPB:WORD
|
|||
|
ENDIF
|
|||
|
|
|||
|
IF SYSVER
|
|||
|
EXTRN CONFCB:BYTE,POUT:DWORD,COUT:DWORD,CIN:DWORD,IOBUFF:BYTE
|
|||
|
EXTRN IOADDR:DWORD,IOCALL:BYTE,IOCOM:BYTE,IOSTAT:WORD,IOCNT:WORD
|
|||
|
EXTRN IOSEG:WORD,COLPOS:BYTE,BADDEV:BYTE,BADLSTMES:BYTE
|
|||
|
EXTRN LBUFFCNT:BYTE,PFLAG:BYTE
|
|||
|
ENDIF
|
|||
|
|
|||
|
CONST ENDS
|
|||
|
|
|||
|
DATA SEGMENT PUBLIC BYTE
|
|||
|
|
|||
|
EXTRN PARSERR:BYTE,DATAEND:WORD,ParityFlag:BYTE,DISADD:BYTE
|
|||
|
EXTRN ASMADD:BYTE,DEFDUMP:BYTE,BYTEBUF:BYTE
|
|||
|
|
|||
|
DATA ENDS
|
|||
|
|
|||
|
DG GROUP CODE,CONST,DATA
|
|||
|
|
|||
|
|
|||
|
CODE SEGMENT PUBLIC 'CODE'
|
|||
|
ASSUME CS:DG,DS:DG,ES:DG,SS:DG
|
|||
|
|
|||
|
PUBLIC RESTART,SET_TERMINATE_VECTOR,DABORT,TERMINATE,COMMAND
|
|||
|
PUBLIC FIND_DEBUG,CRLF,BLANK,TAB,OUT,INBUF,SCANB,SCANP
|
|||
|
PUBLIC PRINTMES,RPRBUF,HEX,OUTSI,OUTDI,OUT16,DIGIT,BACKUP,RBUFIN
|
|||
|
|
|||
|
IF SYSVER
|
|||
|
PUBLIC SETUDEV,DEVIOCALL
|
|||
|
EXTRN DISPREG:NEAR,IN:NEAR
|
|||
|
ENDIF
|
|||
|
|
|||
|
EXTRN PERR:NEAR,COMPARE:NEAR,DUMP:NEAR,ENTER:NEAR,FILL:NEAR
|
|||
|
EXTRN GO:NEAR,INPUT:NEAR,LOAD:NEAR,MOVE:NEAR,NAME:NEAR
|
|||
|
EXTRN REG:NEAR,SEARCH:NEAR,DWRITE:NEAR,UNASSEM:NEAR,ASSEM:NEAR
|
|||
|
EXTRN OUTPUT:NEAR,ZTRACE:NEAR,TRACE:NEAR,GETHEX:NEAR,GETEOL:NEAR
|
|||
|
|
|||
|
EXTRN PREPNAME:NEAR,DEFIO:NEAR,SKIP_FILE:NEAR,DEBUG_FOUND:NEAR
|
|||
|
EXTRN TrapParity:NEAR,ReleaseParity:NEAR
|
|||
|
|
|||
|
ORG 100H
|
|||
|
|
|||
|
START:
|
|||
|
DEBUG:
|
|||
|
JMP SHORT DSTRT
|
|||
|
|
|||
|
HEADER DB "Vers 2.30"
|
|||
|
|
|||
|
DSTRT:
|
|||
|
DOSVER_HIGH EQU 0200H ; 2.00 in hex
|
|||
|
MOV AH,GET_VERSION
|
|||
|
INT 21H
|
|||
|
XCHG AH,AL ; Turn it around to AH.AL
|
|||
|
CMP AX,DOSVER_HIGH
|
|||
|
JAE OKDOS
|
|||
|
GOTBADDOS:
|
|||
|
MOV DX,OFFSET DG:BADVER
|
|||
|
MOV AH,STD_CON_STRING_OUTPUT
|
|||
|
INT 21H
|
|||
|
INT 20H
|
|||
|
|
|||
|
OKDOS:
|
|||
|
CALL TrapParity ; scarf up those parity guys
|
|||
|
MOV AH,GET_CURRENT_PDB
|
|||
|
INT 21H
|
|||
|
MOV [USER_PROC_PDB],BX ; Initially set to DEBUG
|
|||
|
|
|||
|
IF SYSVER
|
|||
|
MOV [IOSEG],CS
|
|||
|
ENDIF
|
|||
|
|
|||
|
MOV SP,OFFSET DG:STACK
|
|||
|
MOV [PARSERR],AL
|
|||
|
MOV AH,GET_IN_VARS
|
|||
|
INT 21H
|
|||
|
|
|||
|
|
|||
|
IF SYSVER
|
|||
|
LDS SI,ES:[BX.BCON]
|
|||
|
MOV WORD PTR CS:[CIN+2],DS
|
|||
|
MOV WORD PTR CS:[CIN],SI
|
|||
|
MOV WORD PTR CS:[COUT+2],DS
|
|||
|
MOV WORD PTR CS:[COUT],SI
|
|||
|
PUSH CS
|
|||
|
POP DS
|
|||
|
MOV DX,OFFSET DG:CONFCB
|
|||
|
MOV AH,FCB_OPEN
|
|||
|
INT 21H
|
|||
|
OR AL,AL
|
|||
|
JZ GOTLIST
|
|||
|
MOV DX,OFFSET DG:BADLSTMES
|
|||
|
CALL RPRBUF
|
|||
|
CALL RBUFIN
|
|||
|
CALL CRLF
|
|||
|
MOV CL,[LBUFFCNT]
|
|||
|
OR CL,CL
|
|||
|
JZ NOLIST1 ; User didn't specify one
|
|||
|
XOR CH,CH
|
|||
|
MOV DI,OFFSET DG:(CONFCB + 1)
|
|||
|
MOV SI,OFFSET DG:LINEBUF
|
|||
|
REP MOVSB
|
|||
|
MOV DX,OFFSET DG:CONFCB
|
|||
|
MOV AH,FCB_OPEN
|
|||
|
INT 21H
|
|||
|
OR AL,AL
|
|||
|
JZ GOTLIST ; GOOD
|
|||
|
MOV DX,OFFSET DG:BADDEV
|
|||
|
CALL RPRBUF
|
|||
|
NOLIST1:
|
|||
|
MOV WORD PTR [POUT+2],CS
|
|||
|
MOV WORD PTR [POUT],OFFSET DG:LONGRET
|
|||
|
JMP NOLIST
|
|||
|
|
|||
|
XXX PROC FAR
|
|||
|
LONGRET:RET
|
|||
|
XXX ENDP
|
|||
|
ENDIF
|
|||
|
|
|||
|
GOTLIST:
|
|||
|
IF SYSVER
|
|||
|
MOV SI,DX
|
|||
|
LDS SI,DWORD PTR DS:[SI.fcb_FIRCLUS]
|
|||
|
MOV WORD PTR CS:[POUT+2],DS
|
|||
|
MOV WORD PTR CS:[POUT],SI
|
|||
|
ENDIF
|
|||
|
NOLIST:
|
|||
|
MOV AX,CS
|
|||
|
MOV DS,AX
|
|||
|
MOV ES,AX
|
|||
|
|
|||
|
; Code to print header
|
|||
|
; MOV DX,OFFSET DG:HEADER
|
|||
|
; CALL RPRBUF
|
|||
|
|
|||
|
CALL SET_TERMINATE_VECTOR
|
|||
|
|
|||
|
IF SETCNTC
|
|||
|
MOV AL,23H ; Set vector 23H
|
|||
|
MOV DX,OFFSET DG:DABORT
|
|||
|
INT 21H
|
|||
|
ENDIF
|
|||
|
|
|||
|
MOV DX,CS ; Get DEBUG's segment
|
|||
|
MOV AX,OFFSET DG:DATAEND + 15 ; End of debug
|
|||
|
SHR AX,1 ; Convert to segments
|
|||
|
SHR AX,1
|
|||
|
SHR AX,1
|
|||
|
SHR AX,1
|
|||
|
ADD DX,AX ; Add siz of debug in paragraphs
|
|||
|
MOV AH,CREATE_PROCESS_DATA_BLOCK ; create program segment just after DEBUG
|
|||
|
INT 21H
|
|||
|
MOV AX,DX
|
|||
|
MOV DI,OFFSET DG:DSSAVE
|
|||
|
CLD
|
|||
|
STOSW
|
|||
|
STOSW
|
|||
|
STOSW
|
|||
|
STOSW
|
|||
|
MOV WORD PTR [DISADD+2],AX
|
|||
|
MOV WORD PTR [ASMADD+2],AX
|
|||
|
MOV WORD PTR [DEFDUMP+2],AX
|
|||
|
MOV AX,100H
|
|||
|
MOV WORD PTR[DISADD],AX
|
|||
|
MOV WORD PTR[ASMADD],AX
|
|||
|
MOV WORD PTR [DEFDUMP],AX
|
|||
|
MOV DS,DX
|
|||
|
MOV ES,DX
|
|||
|
MOV DX,80H
|
|||
|
MOV AH,SET_DMA
|
|||
|
INT 21H ; Set default DMA address to 80H
|
|||
|
MOV AX,WORD PTR DS:[6]
|
|||
|
MOV BX,AX
|
|||
|
CMP AX,0FFF0H
|
|||
|
PUSH CS
|
|||
|
POP DS
|
|||
|
JAE SAVSTK
|
|||
|
MOV AX,WORD PTR DS:[6]
|
|||
|
PUSH BX
|
|||
|
MOV BX,OFFSET DG:DATAEND + 15
|
|||
|
AND BX,0FFF0H ; Size of DEBUG in bytes (rounded up to PARA)
|
|||
|
SUB AX,BX
|
|||
|
POP BX
|
|||
|
SAVSTK:
|
|||
|
PUSH BX
|
|||
|
DEC AX
|
|||
|
DEC AX
|
|||
|
MOV BX,AX
|
|||
|
MOV WORD PTR [BX],0
|
|||
|
POP BX
|
|||
|
MOV SPSAVE,AX
|
|||
|
DEC AH
|
|||
|
MOV ES:WORD PTR [6],AX
|
|||
|
SUB BX,AX
|
|||
|
MOV CL,4
|
|||
|
SHR BX,CL
|
|||
|
ADD ES:WORD PTR [8],BX
|
|||
|
|
|||
|
IF IBMVER
|
|||
|
; Get screen size and initialize display related variables
|
|||
|
MOV AH,15
|
|||
|
INT 10H
|
|||
|
CMP AH,40
|
|||
|
JNZ PARSCHK
|
|||
|
MOV BYTE PTR DSIZ,7
|
|||
|
MOV BYTE PTR NOREGL,4
|
|||
|
MOV DISPB,64
|
|||
|
ENDIF
|
|||
|
|
|||
|
PARSCHK:
|
|||
|
; Copy rest of command line to test program's parameter area
|
|||
|
MOV DI,FCB
|
|||
|
MOV SI,81H
|
|||
|
MOV AX,(PARSE_FILE_DESCRIPTOR SHL 8) OR 01H
|
|||
|
INT 21H
|
|||
|
CALL SKIP_FILE ; Make sure si points to delimiter
|
|||
|
CALL PREPNAME
|
|||
|
PUSH CS
|
|||
|
POP ES
|
|||
|
FILECHK:
|
|||
|
MOV DI,80H
|
|||
|
CMP BYTE PTR ES:[DI],0 ; ANY STUFF FOUND?
|
|||
|
JZ COMMAND ; NOPE
|
|||
|
FILOOP: INC DI
|
|||
|
CMP BYTE PTR ES:[DI],13 ; COMMAND LINE JUST SPACES?
|
|||
|
JZ COMMAND
|
|||
|
CMP BYTE PTR ES:[DI]," "
|
|||
|
JZ FILOOP
|
|||
|
CMP BYTE PTR ES:[DI],9
|
|||
|
JZ FILOOP
|
|||
|
|
|||
|
CALL DEFIO ; WELL READ IT IN
|
|||
|
MOV AX,CSSAVE
|
|||
|
MOV WORD PTR DISADD+2,AX
|
|||
|
MOV WORD PTR ASMADD+2,AX
|
|||
|
MOV AX,IPSAVE
|
|||
|
MOV WORD PTR DISADD,AX
|
|||
|
MOV WORD PTR ASMADD,AX
|
|||
|
COMMAND:
|
|||
|
CLD
|
|||
|
MOV AX,CS
|
|||
|
MOV DS,AX
|
|||
|
MOV ES,AX
|
|||
|
MOV SS,AX
|
|||
|
MOV SP,OFFSET DG:STACK
|
|||
|
STI
|
|||
|
CMP [ParityFlag],0 ; did we detect a parity error?
|
|||
|
JZ GoPrompt ; nope, go prompt
|
|||
|
MOV [ParityFlag],0 ; reset flag
|
|||
|
MOV DX,OFFSET DG:ParityMes ; message to print
|
|||
|
MOV AH,STD_CON_STRING_OUTPUT; easy way out
|
|||
|
INT 21h ; blam
|
|||
|
GoPrompt:
|
|||
|
MOV AL,PROMPT
|
|||
|
CALL OUT
|
|||
|
CALL INBUF ; Get command line
|
|||
|
; From now and throughout command line processing, DI points
|
|||
|
; to next character in command line to be processed.
|
|||
|
CALL SCANB ; Scan off leading blanks
|
|||
|
JZ COMMAND ; Null command?
|
|||
|
LODSB ; AL=first non-blank character
|
|||
|
; Prepare command letter for table lookup
|
|||
|
SUB AL,"A" ; Low end range check
|
|||
|
JB ERR1
|
|||
|
CMP AL,"Z"-"A" ; Upper end range check
|
|||
|
JA ERR1
|
|||
|
SHL AL,1 ; Times two
|
|||
|
CBW ; Now a 16-bit quantity
|
|||
|
XCHG BX,AX ; In BX we can address with it
|
|||
|
CALL CS:[BX+COMTAB] ; Execute command
|
|||
|
JMP SHORT COMMAND ; Get next command
|
|||
|
ERR1: JMP PERR
|
|||
|
|
|||
|
SET_TERMINATE_VECTOR:
|
|||
|
MOV AX,(SET_INTERRUPT_VECTOR SHL 8) OR 22H ; Set vector 22H
|
|||
|
MOV DX,OFFSET DG:TERMINATE
|
|||
|
INT 21H
|
|||
|
RET
|
|||
|
|
|||
|
TERMINATE:
|
|||
|
CMP BYTE PTR CS:[QFLAG],0
|
|||
|
JNZ QUITING
|
|||
|
MOV CS:[USER_PROC_PDB],CS
|
|||
|
CMP BYTE PTR CS:[NEWEXEC],0
|
|||
|
JZ NORMTERM
|
|||
|
MOV AX,CS
|
|||
|
MOV DS,AX
|
|||
|
MOV SS,AX
|
|||
|
MOV SP,OFFSET DG:STACK
|
|||
|
MOV AX,[HEADSAVE]
|
|||
|
JMP DEBUG_FOUND
|
|||
|
|
|||
|
NORMTERM:
|
|||
|
MOV DX,OFFSET DG:ENDMES
|
|||
|
JMP SHORT RESTART
|
|||
|
|
|||
|
QUITING:
|
|||
|
MOV AX,(EXIT SHL 8)
|
|||
|
INT 21H
|
|||
|
|
|||
|
DABORT:
|
|||
|
MOV DX,OFFSET DG:CARRET
|
|||
|
RESTART:
|
|||
|
MOV AX,CS
|
|||
|
MOV DS,AX
|
|||
|
MOV SS,AX
|
|||
|
MOV SP,OFFSET DG:STACK
|
|||
|
CALL RPRBUF
|
|||
|
JMP COMMAND
|
|||
|
|
|||
|
IF SYSVER
|
|||
|
SETUDEV:
|
|||
|
MOV DI,OFFSET DG:CONFCB
|
|||
|
MOV AX,(PARSE_FILE_DESCRIPTOR SHL 8) OR 01H
|
|||
|
INT 21H
|
|||
|
CALL USERDEV
|
|||
|
JMP DISPREG
|
|||
|
|
|||
|
USERDEV:
|
|||
|
MOV DX,OFFSET DG:CONFCB
|
|||
|
MOV AH,FCB_OPEN
|
|||
|
INT 21H
|
|||
|
OR AL,AL
|
|||
|
JNZ OPENERR
|
|||
|
MOV SI,DX
|
|||
|
TEST BYTE PTR [SI.fcb_DEVID],080H ; Device?
|
|||
|
JZ OPENERR ; NO
|
|||
|
LDS SI,DWORD PTR [CONFCB.fcb_FIRCLUS]
|
|||
|
MOV WORD PTR CS:[CIN],SI
|
|||
|
MOV WORD PTR CS:[CIN+2],DS
|
|||
|
MOV WORD PTR CS:[COUT],SI
|
|||
|
MOV WORD PTR CS:[COUT+2],DS
|
|||
|
PUSH CS
|
|||
|
POP DS
|
|||
|
RET
|
|||
|
|
|||
|
|
|||
|
OPENERR:
|
|||
|
MOV DX,OFFSET DG:BADDEV
|
|||
|
CALL RPRBUF
|
|||
|
RET
|
|||
|
ENDIF
|
|||
|
|
|||
|
; Get input line. Convert all characters NOT in quotes to upper case.
|
|||
|
|
|||
|
INBUF:
|
|||
|
CALL RBUFIN
|
|||
|
MOV SI,OFFSET DG:LINEBUF
|
|||
|
MOV DI,OFFSET DG:BYTEBUF
|
|||
|
CASECHK:
|
|||
|
LODSB
|
|||
|
CMP AL,'a'
|
|||
|
JB NOCONV
|
|||
|
CMP AL,'z'
|
|||
|
JA NOCONV
|
|||
|
ADD AL,"A"-"a" ; Convert to upper case
|
|||
|
NOCONV:
|
|||
|
STOSB
|
|||
|
CMP AL,13
|
|||
|
JZ INDONE
|
|||
|
CMP AL,'"'
|
|||
|
JZ QUOTSCAN
|
|||
|
CMP AL,"'"
|
|||
|
JNZ CASECHK
|
|||
|
QUOTSCAN:
|
|||
|
MOV AH,AL
|
|||
|
KILLSTR:
|
|||
|
LODSB
|
|||
|
STOSB
|
|||
|
CMP AL,13
|
|||
|
JZ INDONE
|
|||
|
CMP AL,AH
|
|||
|
JNZ KILLSTR
|
|||
|
JMP SHORT CASECHK
|
|||
|
|
|||
|
INDONE:
|
|||
|
MOV SI,OFFSET DG:BYTEBUF
|
|||
|
|
|||
|
; Output CR/LF sequence
|
|||
|
|
|||
|
CRLF:
|
|||
|
MOV AL,13
|
|||
|
CALL OUT
|
|||
|
MOV AL,10
|
|||
|
JMP OUT
|
|||
|
|
|||
|
; Physical backspace - blank, backspace, blank
|
|||
|
|
|||
|
BACKUP:
|
|||
|
MOV SI,OFFSET DG:BACMES
|
|||
|
|
|||
|
; Print ASCII message. Last char has bit 7 set
|
|||
|
|
|||
|
PRINTMES:
|
|||
|
LODS CS:BYTE PTR [SI] ; Get char to print
|
|||
|
CALL OUT
|
|||
|
SHL AL,1 ; High bit set?
|
|||
|
JNC PRINTMES
|
|||
|
RET
|
|||
|
|
|||
|
; Scan for parameters of a command
|
|||
|
|
|||
|
SCANP:
|
|||
|
CALL SCANB ; Get first non-blank
|
|||
|
CMP BYTE PTR [SI],"," ; One comma between params OK
|
|||
|
JNE EOLCHK ; If not comma, we found param
|
|||
|
INC SI ; Skip over comma
|
|||
|
|
|||
|
; Scan command line for next non-blank character
|
|||
|
|
|||
|
SCANB:
|
|||
|
PUSH AX
|
|||
|
SCANNEXT:
|
|||
|
LODSB
|
|||
|
CMP AL," "
|
|||
|
JZ SCANNEXT
|
|||
|
CMP AL,9
|
|||
|
JZ SCANNEXT
|
|||
|
DEC SI ; Back to first non-blank
|
|||
|
POP AX
|
|||
|
EOLCHK:
|
|||
|
CMP BYTE PTR [SI],13
|
|||
|
RET
|
|||
|
|
|||
|
; Hex addition and subtraction
|
|||
|
|
|||
|
HEXADD:
|
|||
|
MOV CX,4
|
|||
|
CALL GETHEX
|
|||
|
MOV DI,DX
|
|||
|
MOV CX,4
|
|||
|
CALL GETHEX
|
|||
|
CALL GETEOL
|
|||
|
PUSH DX
|
|||
|
ADD DX,DI
|
|||
|
CALL OUT16
|
|||
|
CALL BLANK
|
|||
|
CALL BLANK
|
|||
|
POP DX
|
|||
|
SUB DI,DX
|
|||
|
MOV DX,DI
|
|||
|
CALL OUT16
|
|||
|
JMP SHORT CRLF
|
|||
|
|
|||
|
; Print the hex address of DS:SI
|
|||
|
|
|||
|
OUTSI:
|
|||
|
MOV DX,DS ; Put DS where we can work with it
|
|||
|
CALL OUT16 ; Display segment
|
|||
|
MOV AL,":"
|
|||
|
CALL OUT
|
|||
|
MOV DX,SI
|
|||
|
JMP SHORT OUT16 ; Output displacement
|
|||
|
|
|||
|
; Print hex address of ES:DI
|
|||
|
; Same as OUTSI above
|
|||
|
|
|||
|
OUTDI:
|
|||
|
MOV DX,ES
|
|||
|
CALL OUT16
|
|||
|
MOV AL,":"
|
|||
|
CALL OUT
|
|||
|
MOV DX,DI
|
|||
|
|
|||
|
; Print out 16-bit value in DX in hex
|
|||
|
|
|||
|
OUT16:
|
|||
|
MOV AL,DH ; High-order byte first
|
|||
|
CALL HEX
|
|||
|
MOV AL,DL ; Then low-order byte
|
|||
|
|
|||
|
; Output byte in AL as two hex digits
|
|||
|
|
|||
|
HEX:
|
|||
|
MOV AH,AL ; Save for second digit
|
|||
|
; Shift high digit into low 4 bits
|
|||
|
PUSH CX
|
|||
|
MOV CL,4
|
|||
|
SHR AL,CL
|
|||
|
POP CX
|
|||
|
|
|||
|
CALL DIGIT ; Output first digit
|
|||
|
MOV AL,AH ; Now do digit saved in AH
|
|||
|
DIGIT:
|
|||
|
AND AL,0FH ; Mask to 4 bits
|
|||
|
; Trick 6-byte hex conversion works on 8086 too.
|
|||
|
ADD AL,90H
|
|||
|
DAA
|
|||
|
ADC AL,40H
|
|||
|
DAA
|
|||
|
|
|||
|
; Console output of character in AL. No registers affected but bit 7
|
|||
|
; is reset before output.
|
|||
|
|
|||
|
IF SYSVER
|
|||
|
OUT:
|
|||
|
PUSH AX
|
|||
|
AND AL,7FH
|
|||
|
CMP AL,7FH
|
|||
|
JNZ NOTDEL
|
|||
|
MOV AL,8 ; DELETE same as backspace
|
|||
|
NOTDEL:
|
|||
|
CMP AL,9
|
|||
|
JZ TABDO
|
|||
|
CALL DOCONOUT
|
|||
|
CMP AL,0DH
|
|||
|
JZ ZEROPOS
|
|||
|
CMP AL,0AH
|
|||
|
JZ ZEROPOS
|
|||
|
CMP AL,8
|
|||
|
JNZ OOKRET
|
|||
|
MOV AL," "
|
|||
|
CALL DOCONOUT
|
|||
|
MOV AL,8
|
|||
|
CALL DOCONOUT
|
|||
|
CMP BYTE PTR CS:[COLPOS],0
|
|||
|
JZ NOTINC
|
|||
|
DEC BYTE PTR CS:[COLPOS]
|
|||
|
JMP NOTINC
|
|||
|
ZEROPOS:
|
|||
|
MOV BYTE PTR CS:[COLPOS],0FFH
|
|||
|
OOKRET:
|
|||
|
INC BYTE PTR CS:[COLPOS]
|
|||
|
NOTINC:
|
|||
|
TEST BYTE PTR CS:[PFLAG],1
|
|||
|
JZ POPRET
|
|||
|
CALL LISTOUT
|
|||
|
POPRET:
|
|||
|
POP AX
|
|||
|
RET
|
|||
|
|
|||
|
TABDO:
|
|||
|
MOV AL,CS:[COLPOS]
|
|||
|
OR AL,0F8H
|
|||
|
NEG AL
|
|||
|
PUSH CX
|
|||
|
MOV CL,AL
|
|||
|
XOR CH,CH
|
|||
|
JCXZ POPTAB
|
|||
|
TABLP:
|
|||
|
MOV AL," "
|
|||
|
CALL OUT
|
|||
|
LOOP TABLP
|
|||
|
POPTAB:
|
|||
|
POP CX
|
|||
|
POP AX
|
|||
|
RET
|
|||
|
|
|||
|
|
|||
|
DOCONOUT:
|
|||
|
PUSH DS
|
|||
|
PUSH SI
|
|||
|
PUSH AX
|
|||
|
CONOWAIT:
|
|||
|
LDS SI,CS:[COUT]
|
|||
|
MOV AH,10
|
|||
|
CALL DEVIOCALL
|
|||
|
MOV AX,CS:[IOSTAT]
|
|||
|
AND AX,200H
|
|||
|
JNZ CONOWAIT
|
|||
|
POP AX
|
|||
|
PUSH AX
|
|||
|
MOV AH,8
|
|||
|
CALL DEVIOCALL
|
|||
|
POP AX
|
|||
|
POP SI
|
|||
|
POP DS
|
|||
|
RET
|
|||
|
|
|||
|
|
|||
|
LISTOUT:
|
|||
|
PUSH DS
|
|||
|
PUSH SI
|
|||
|
PUSH AX
|
|||
|
LISTWAIT:
|
|||
|
LDS SI,CS:[POUT]
|
|||
|
MOV AH,10
|
|||
|
CALL DEVIOCALL
|
|||
|
MOV AX,CS:[IOSTAT]
|
|||
|
AND AX,200H
|
|||
|
JNZ LISTWAIT
|
|||
|
POP AX
|
|||
|
PUSH AX
|
|||
|
MOV AH,8
|
|||
|
CALL DEVIOCALL
|
|||
|
POP AX
|
|||
|
POP SI
|
|||
|
POP DS
|
|||
|
RET
|
|||
|
|
|||
|
DEVIOCALL:
|
|||
|
PUSH ES
|
|||
|
PUSH BX
|
|||
|
PUSH CS
|
|||
|
POP ES
|
|||
|
MOV BX,OFFSET DG:IOCALL
|
|||
|
MOV CS:[IOCOM],AH
|
|||
|
MOV WORD PTR CS:[IOSTAT],0
|
|||
|
MOV WORD PTR CS:[IOCNT],1
|
|||
|
MOV CS:[IOBUFF],AL
|
|||
|
MOV WORD PTR CS:[IOADDR+2],DS
|
|||
|
MOV AX,[SI+6]
|
|||
|
MOV WORD PTR CS:[IOADDR],AX
|
|||
|
CALL DWORD PTR CS:[IOADDR]
|
|||
|
MOV AX,[SI+8]
|
|||
|
MOV WORD PTR CS:[IOADDR],AX
|
|||
|
CALL DWORD PTR CS:[IOADDR]
|
|||
|
MOV AL,CS:[IOBUFF]
|
|||
|
POP BX
|
|||
|
POP ES
|
|||
|
RET
|
|||
|
ELSE
|
|||
|
|
|||
|
OUT:
|
|||
|
PUSH DX
|
|||
|
PUSH AX
|
|||
|
AND AL,7FH
|
|||
|
MOV DL,AL
|
|||
|
MOV AH,2
|
|||
|
INT 21H
|
|||
|
POP AX
|
|||
|
POP DX
|
|||
|
RET
|
|||
|
ENDIF
|
|||
|
|
|||
|
|
|||
|
IF SYSVER
|
|||
|
RBUFIN:
|
|||
|
PUSH AX
|
|||
|
PUSH ES
|
|||
|
PUSH DI
|
|||
|
PUSH CS
|
|||
|
POP ES
|
|||
|
MOV BYTE PTR [LBUFFCNT],0
|
|||
|
MOV DI,OFFSET DG:LINEBUF
|
|||
|
FILLBUF:
|
|||
|
CALL IN
|
|||
|
CMP AL,0DH
|
|||
|
JZ BDONE
|
|||
|
CMP AL,8
|
|||
|
JZ ECHR
|
|||
|
CMP AL,7FH
|
|||
|
JZ ECHR
|
|||
|
CMP BYTE PTR [LBUFFCNT],BUFLEN
|
|||
|
JAE BFULL
|
|||
|
STOSB
|
|||
|
INC BYTE PTR [LBUFFCNT]
|
|||
|
JMP SHORT FILLBUF
|
|||
|
|
|||
|
BDONE:
|
|||
|
STOSB
|
|||
|
POP DI
|
|||
|
POP ES
|
|||
|
POP AX
|
|||
|
RET
|
|||
|
|
|||
|
BFULL:
|
|||
|
MOV AL,8
|
|||
|
CALL OUT
|
|||
|
MOV AL,7
|
|||
|
CALL OUT
|
|||
|
JMP SHORT FILLBUF
|
|||
|
|
|||
|
ECHR:
|
|||
|
CMP DI,OFFSET DG:LINEBUF
|
|||
|
JZ FILLBUF
|
|||
|
DEC DI
|
|||
|
DEC BYTE PTR [LBUFFCNT]
|
|||
|
JMP SHORT FILLBUF
|
|||
|
ELSE
|
|||
|
|
|||
|
RBUFIN:
|
|||
|
PUSH AX
|
|||
|
PUSH DX
|
|||
|
MOV AH,10
|
|||
|
MOV DX,OFFSET DG:LBUFSIZ
|
|||
|
INT 21H
|
|||
|
POP DX
|
|||
|
POP AX
|
|||
|
RET
|
|||
|
ENDIF
|
|||
|
|
|||
|
|
|||
|
IF SYSVER
|
|||
|
RPRBUF:
|
|||
|
PUSHF
|
|||
|
PUSH AX
|
|||
|
PUSH SI
|
|||
|
MOV SI,DX
|
|||
|
PLOOP:
|
|||
|
LODSB
|
|||
|
CMP AL,"$"
|
|||
|
JZ PRTDONE
|
|||
|
CALL OUT
|
|||
|
JMP SHORT PLOOP
|
|||
|
PRTDONE:
|
|||
|
POP SI
|
|||
|
POP AX
|
|||
|
POPF
|
|||
|
RET
|
|||
|
ELSE
|
|||
|
|
|||
|
RPRBUF:
|
|||
|
MOV AH,9
|
|||
|
INT 21H
|
|||
|
RET
|
|||
|
ENDIF
|
|||
|
|
|||
|
; Output one space
|
|||
|
|
|||
|
BLANK:
|
|||
|
MOV AL," "
|
|||
|
JMP OUT
|
|||
|
|
|||
|
; Output the number of blanks in CX
|
|||
|
|
|||
|
TAB:
|
|||
|
CALL BLANK
|
|||
|
LOOP TAB
|
|||
|
RET
|
|||
|
|
|||
|
; Command Table. Command letter indexes into table to get
|
|||
|
; address of command. PERR prints error for no such command.
|
|||
|
|
|||
|
COMTAB DW ASSEM ; A
|
|||
|
DW PERR ; B
|
|||
|
DW COMPARE ; C
|
|||
|
DW DUMP ; D
|
|||
|
DW ENTER ; E
|
|||
|
DW FILL ; F
|
|||
|
DW GO ; G
|
|||
|
DW HEXADD ; H
|
|||
|
DW INPUT ; I
|
|||
|
DW PERR ; J
|
|||
|
DW PERR ; K
|
|||
|
DW LOAD ; L
|
|||
|
DW MOVE ; M
|
|||
|
DW NAME ; N
|
|||
|
DW OUTPUT ; O
|
|||
|
IF ZIBO
|
|||
|
DW ZTRACE
|
|||
|
ELSE
|
|||
|
DW PERR ; P
|
|||
|
ENDIF
|
|||
|
DW QUIT ; Q (QUIT)
|
|||
|
DW REG ; R
|
|||
|
DW SEARCH ; S
|
|||
|
DW TRACE ; T
|
|||
|
DW UNASSEM ; U
|
|||
|
DW PERR ; V
|
|||
|
DW DWRITE ; W
|
|||
|
IF SYSVER
|
|||
|
DW SETUDEV ; X
|
|||
|
ELSE
|
|||
|
DW PERR
|
|||
|
ENDIF
|
|||
|
DW PERR ; Y
|
|||
|
DW PERR ; Z
|
|||
|
|
|||
|
QUIT:
|
|||
|
INC BYTE PTR [QFLAG]
|
|||
|
MOV BX,[USER_PROC_PDB]
|
|||
|
FIND_DEBUG:
|
|||
|
IF NOT SYSVER
|
|||
|
MOV AH,SET_CURRENT_PDB
|
|||
|
INT 21H
|
|||
|
ENDIF
|
|||
|
CALL ReleaseParity ; let system do normal parity stuff
|
|||
|
MOV AX,(EXIT SHL 8)
|
|||
|
INT 21H
|
|||
|
|
|||
|
CODE ENDS
|
|||
|
END START
|
|||
|
|