mirror of https://github.com/microsoft/MS-DOS.git
439 lines
12 KiB
NASM
439 lines
12 KiB
NASM
;
|
||
; Device call routines for MSDOS
|
||
;
|
||
|
||
INCLUDE DOSSEG.ASM
|
||
|
||
IFNDEF KANJI
|
||
KANJI EQU 0 ;FALSE
|
||
ENDIF
|
||
|
||
CODE SEGMENT BYTE PUBLIC 'CODE'
|
||
ASSUME SS:DOSGROUP,CS:DOSGROUP
|
||
|
||
.xlist
|
||
.xcref
|
||
INCLUDE DOSSYM.ASM
|
||
INCLUDE DEVSYM.ASM
|
||
.cref
|
||
.list
|
||
|
||
TITLE DEV - Device call routines
|
||
NAME Dev
|
||
|
||
i_need IOXAD,DWORD
|
||
i_need IOSCNT,WORD
|
||
i_need DEVIOBUF,4
|
||
i_need IOCALL,BYTE
|
||
i_need IOMED,BYTE
|
||
i_need IORCHR,BYTE
|
||
i_need CALLSCNT,WORD
|
||
i_need DMAAdd,DWORD
|
||
i_need NullDevPt,DWORD
|
||
i_need CallDevAd,DWORD
|
||
i_need Attrib,BYTE
|
||
i_need NULDEV,DWORD
|
||
i_need Name1,BYTE
|
||
i_need DevPt,DWORD
|
||
i_need DPBHead,DWORD
|
||
i_need NumIO,BYTE
|
||
i_need ThisDPB,DWORD
|
||
i_need DevCall,DWORD
|
||
i_need VerFlg,BYTE
|
||
|
||
SUBTTL IOFUNC -- DO FUNCTION 1-12 I/O
|
||
PAGE
|
||
IOFUNC_RETRY:
|
||
ASSUME DS:NOTHING,ES:NOTHING
|
||
invoke restore_world
|
||
|
||
procedure IOFUNC,NEAR
|
||
ASSUME DS:NOTHING,ES:NOTHING
|
||
|
||
; Inputs:
|
||
; DS:SI Points to FCB
|
||
; AH is function code
|
||
; = 0 Input
|
||
; = 1 Input Status
|
||
; = 2 Output
|
||
; = 3 Output Status
|
||
; = 4 Flush
|
||
; AL = character if output
|
||
; Function:
|
||
; Perform indicated I/O to device or file
|
||
; Outputs:
|
||
; AL is character if input
|
||
; If a status call
|
||
; zero set if not ready
|
||
; zero reset if ready (character in AL for input status)
|
||
; For regular files:
|
||
; Input Status
|
||
; Gets character but restores fcb_RR field
|
||
; Zero set on EOF
|
||
; Input
|
||
; Gets character advances fcb_RR field
|
||
; Returns ^Z on EOF
|
||
; Output Status
|
||
; Always ready
|
||
; AX altered, all other registers preserved
|
||
|
||
MOV WORD PTR [IOXAD+2],SS
|
||
MOV WORD PTR [IOXAD],OFFSET DOSGROUP:DEVIOBUF
|
||
MOV WORD PTR [IOSCNT],1
|
||
MOV WORD PTR [DEVIOBUF],AX
|
||
|
||
IOFUNC2:
|
||
TEST [SI.fcb_DEVID],080H
|
||
JNZ IOTODEV
|
||
JMP IOTOFILE
|
||
|
||
IOTODEV:
|
||
invoke save_world
|
||
PUSH DS
|
||
PUSH SS
|
||
POP ES
|
||
PUSH SS
|
||
POP DS
|
||
ASSUME DS:DOSGROUP
|
||
XOR BX,BX
|
||
MOV [IOCALL.REQSTAT],BX
|
||
MOV BYTE PTR [IOMED],BL
|
||
|
||
MOV BX,OFFSET DOSGROUP:IOCALL
|
||
|
||
MOV CX,(DEVRD SHL 8) OR DRDWRHL
|
||
OR AH,AH
|
||
JZ DCALLR
|
||
MOV CX,(DEVRDND SHL 8) OR DRDNDHL
|
||
DEC AH
|
||
JZ DCALLR
|
||
MOV CX,(DEVWRT SHL 8) OR DRDWRHL
|
||
DEC AH
|
||
JZ DCALLO
|
||
MOV CX,(DEVOST SHL 8) OR DSTATHL
|
||
DEC AH
|
||
JZ DCALLO
|
||
DFLUSH:
|
||
MOV CX,(DEVIFL SHL 8) OR DFLSHL
|
||
DCALLR:
|
||
MOV AH,86H
|
||
DCALL:
|
||
MOV [IOCALL.REQLEN],CL
|
||
MOV [IOCALL.REQFUNC],CH
|
||
MOV CL,AH
|
||
POP DS
|
||
ASSUME DS:NOTHING
|
||
CALL DEVIOCALL
|
||
MOV DI,[IOCALL.REQSTAT]
|
||
TEST DI,STERR
|
||
JZ OKDEVIO
|
||
MOV AH,CL
|
||
invoke CHARHARD
|
||
CMP AL,1
|
||
JZ IOFUNC_RETRY
|
||
;Know user must have wanted ignore. Make sure device shows ready so
|
||
;that DOS doesn't get caught in a status loop when user simply wants
|
||
;to ignore the error.
|
||
AND BYTE PTR [IOCALL.REQSTAT+1], NOT (STBUI SHR 8)
|
||
OKDEVIO:
|
||
PUSH SS
|
||
POP DS
|
||
ASSUME DS:DOSGROUP
|
||
CMP CH,DEVRDND
|
||
JNZ DNODRD
|
||
MOV AL,BYTE PTR [IORCHR]
|
||
MOV [DEVIOBUF],AL
|
||
|
||
DNODRD: MOV AH,BYTE PTR [IOCALL.REQSTAT+1]
|
||
NOT AH ; Zero = busy, not zero = ready
|
||
AND AH,STBUI SHR 8
|
||
|
||
invoke restore_world
|
||
ASSUME DS:NOTHING
|
||
MOV AX,WORD PTR [DEVIOBUF]
|
||
return
|
||
|
||
DCALLO:
|
||
MOV AH,87H
|
||
JMP SHORT DCALL
|
||
|
||
IOTOFILE:
|
||
ASSUME DS:NOTHING
|
||
OR AH,AH
|
||
JZ IOIN
|
||
DEC AH
|
||
JZ IOIST
|
||
DEC AH
|
||
JZ IOUT
|
||
return ; NON ZERO FLAG FOR OUTPUT STATUS
|
||
|
||
IOIST:
|
||
PUSH WORD PTR [SI.fcb_RR] ; Save position
|
||
PUSH WORD PTR [SI.fcb_RR+2]
|
||
CALL IOIN
|
||
POP WORD PTR [SI.fcb_RR+2] ; Restore position
|
||
POP WORD PTR [SI.fcb_RR]
|
||
return
|
||
|
||
IOUT:
|
||
CALL SETXADDR
|
||
invoke STORE
|
||
invoke FINNOSAV
|
||
CALL RESTXADDR ; If you change this into a jmp don't come
|
||
return ; crying to me when things don't work ARR
|
||
|
||
IOIN:
|
||
CALL SETXADDR
|
||
invoke LOAD
|
||
PUSH CX
|
||
invoke FINNOSAV
|
||
POP CX
|
||
OR CX,CX ; Check EOF
|
||
CALL RESTXADDR
|
||
MOV AL,[DEVIOBUF] ; Get byte from trans addr
|
||
retnz
|
||
MOV AL,1AH ; ^Z if EOF
|
||
return
|
||
|
||
SETXADDR:
|
||
POP WORD PTR [CALLSCNT] ; Return address
|
||
invoke save_world
|
||
PUSH WORD PTR [DMAADD] ; Save Disk trans addr
|
||
PUSH WORD PTR [DMAADD+2]
|
||
PUSH DS
|
||
PUSH SS
|
||
POP DS
|
||
ASSUME DS:DOSGROUP
|
||
MOV CX,WORD PTR [IOXAD+2]
|
||
MOV WORD PTR [DMAADD+2],CX
|
||
MOV CX,WORD PTR [IOXAD]
|
||
MOV WORD PTR [DMAADD],CX ; Set byte trans addr
|
||
MOV CX,[IOSCNT] ; ioscnt specifies length of buffer
|
||
POP DS
|
||
ASSUME DS:NOTHING
|
||
MOV [SI.fcb_RECSIZ],1 ; One byte per record
|
||
MOV DX,SI ; FCB to DS:DX
|
||
invoke GETRRPOS
|
||
JMP SHORT RESTRET ; RETURN ADDRESS
|
||
|
||
RESTXADDR:
|
||
POP WORD PTR [CALLSCNT] ; Return address
|
||
POP WORD PTR [DMAADD+2] ; Restore Disk trans addr
|
||
POP WORD PTR [DMAADD]
|
||
invoke restore_world
|
||
RESTRET:JMP WORD PTR [CALLSCNT] ; Return address
|
||
IOFUNC ENDP
|
||
|
||
SUBTTL DEVIOCALL, DEVIOCALL2 - CALL A DEVICE
|
||
PAGE
|
||
procedure DEVIOCALL,NEAR
|
||
ASSUME DS:NOTHING,ES:NOTHING
|
||
|
||
; Inputs:
|
||
; DS:SI Points to device FCB
|
||
; ES:BX Points to request data
|
||
; Function:
|
||
; Call the device
|
||
; Outputs:
|
||
; None
|
||
; DS:SI,AX destroyed, others preserved
|
||
|
||
LDS SI,DWORD PTR [SI.fcb_FIRCLUS]
|
||
|
||
entry DEVIOCALL2
|
||
; As above only DS:SI points to device header on entry, and DS:SI is preserved
|
||
MOV AX,[SI.SDEVSTRAT]
|
||
MOV WORD PTR [CALLDEVAD],AX
|
||
MOV WORD PTR [CALLDEVAD+2],DS
|
||
CALL DWORD PTR [CALLDEVAD]
|
||
MOV AX,[SI.SDEVINT]
|
||
MOV WORD PTR [CALLDEVAD],AX
|
||
CALL DWORD PTR [CALLDEVAD]
|
||
return
|
||
DEVIOCALL ENDP
|
||
|
||
SUBTTL DEVNAME - LOOK FOR NAME OF DEVICE
|
||
PAGE
|
||
procedure DEVNAME,NEAR
|
||
ASSUME DS:DOSGROUP,ES:DOSGROUP
|
||
|
||
; Inputs:
|
||
; DS,ES:DOSGROUP
|
||
; Filename in NAME1
|
||
; Function:
|
||
; Determine if file is in list of I/O drivers
|
||
; Outputs:
|
||
; Carry set if name not found
|
||
; ELSE
|
||
; Zero flag set
|
||
; BH = Bit 7,6 = 1, bit 5 = 0 (cooked mode)
|
||
; bits 0-4 set from low byte of attribute word
|
||
; DEVPT = DWORD pointer to Device header of device
|
||
; Registers BX destroyed
|
||
|
||
PUSH SI
|
||
PUSH DI
|
||
PUSH CX
|
||
|
||
IF KANJI
|
||
PUSH WORD PTR [NAME1]
|
||
CMP [NAME1],5
|
||
JNZ NOKTR
|
||
MOV [NAME1],0E5H
|
||
NOKTR:
|
||
ENDIF
|
||
|
||
TEST BYTE PTR [ATTRIB],attr_volume_id ; If looking for VOL id don't find devs
|
||
JNZ RET31
|
||
MOV SI,OFFSET DOSGROUP:NULDEV
|
||
LOOKIO:
|
||
ASSUME DS:NOTHING
|
||
TEST [SI.SDEVATT],DEVTYP
|
||
JZ SKIPDEV ; Skip block devices
|
||
PUSH SI
|
||
ADD SI,SDEVNAME
|
||
MOV DI,OFFSET DOSGROUP:NAME1
|
||
MOV CX,4 ; All devices are 8 letters
|
||
REPE CMPSW ; Check for name in list
|
||
POP SI
|
||
JZ IOCHK ; Found it?
|
||
SKIPDEV:
|
||
LDS SI,DWORD PTR [SI] ; Get address of next device
|
||
CMP SI,-1 ; At end of list?
|
||
JNZ LOOKIO
|
||
RET31: STC ; Not found
|
||
RETNV: PUSH SS
|
||
POP DS
|
||
ASSUME DS:DOSGROUP
|
||
|
||
IF KANJI
|
||
POP WORD PTR [NAME1]
|
||
ENDIF
|
||
|
||
POP CX
|
||
POP DI
|
||
POP SI
|
||
RET
|
||
|
||
IOCHK:
|
||
ASSUME DS:NOTHING
|
||
MOV WORD PTR [DEVPT+2],DS ; Save pointer to device
|
||
MOV BH,BYTE PTR [SI.SDEVATT]
|
||
OR BH,0C0H
|
||
AND BH,NOT 020H ;Clears Carry
|
||
MOV WORD PTR [DEVPT],SI
|
||
JMP RETNV
|
||
DevName ENDP
|
||
|
||
procedure GetBP,NEAR
|
||
ASSUME DS:DOSGROUP,ES:NOTHING
|
||
|
||
; Inputs:
|
||
; AL = Logical unit number (A = 0)
|
||
; Function:
|
||
; Find Drive Parameter Block
|
||
; Outputs:
|
||
; ES:BP points to DPB
|
||
; [THISDPB] = ES:BP
|
||
; Carry set if unit number bad
|
||
; No other registers altered
|
||
|
||
LES BP,[DPBHEAD] ; Just in case drive isn't valid
|
||
AND AL,3FH ; Mask out dirty and device bits
|
||
CMP AL,BYTE PTR [NUMIO]
|
||
CMC
|
||
JC GOTDPB ; Get drive A
|
||
FNDDPB:
|
||
CMP AL,ES:[BP.dpb_drive]
|
||
JZ GOTDPB ; Carry is clear if jump executed
|
||
LES BP,ES:[BP.dpb_next_dpb]
|
||
JMP SHORT FNDDPB
|
||
GOTDPB:
|
||
MOV WORD PTR [THISDPB],BP
|
||
MOV WORD PTR [THISDPB+2],ES
|
||
RET
|
||
GetBP ENDP
|
||
|
||
SUBTTL SETREAD, SETWRITE -- SET UP HEADER BLOCK
|
||
PAGE
|
||
procedure SETREAD,NEAR
|
||
ASSUME DS:NOTHING,ES:NOTHING
|
||
|
||
; Inputs:
|
||
; DS:BX = Transfer Address
|
||
; CX = Record Count
|
||
; DX = Starting Record
|
||
; AH = Media Byte
|
||
; AL = Unit Code
|
||
; Function:
|
||
; Set up the device call header at DEVCALL
|
||
; Output:
|
||
; ES:BX Points to DEVCALL
|
||
; No other registers effected
|
||
|
||
PUSH DI
|
||
PUSH CX
|
||
PUSH AX
|
||
MOV CL,DEVRD
|
||
SETCALLHEAD:
|
||
MOV AL,DRDWRHL
|
||
PUSH SS
|
||
POP ES
|
||
MOV DI,OFFSET DOSGROUP:DEVCALL
|
||
STOSB ; length
|
||
POP AX
|
||
STOSB ; Unit
|
||
PUSH AX
|
||
MOV AL,CL
|
||
STOSB ; Command code
|
||
XOR AX,AX
|
||
STOSW ; Status
|
||
ADD DI,8 ; Skip link fields
|
||
POP AX
|
||
XCHG AH,AL
|
||
STOSB ; Media byte
|
||
XCHG AL,AH
|
||
PUSH AX
|
||
MOV AX,BX
|
||
STOSW
|
||
MOV AX,DS
|
||
STOSW ; Transfer addr
|
||
POP CX ; Real AX
|
||
POP AX ; Real CX
|
||
STOSW ; Count
|
||
XCHG AX,DX ; AX=Real DX, DX=real CX, CX=real AX
|
||
STOSW ; Start
|
||
XCHG AX,CX
|
||
XCHG DX,CX
|
||
POP DI
|
||
MOV BX,OFFSET DOSGROUP:DEVCALL
|
||
RET
|
||
|
||
entry SETWRITE
|
||
ASSUME DS:NOTHING,ES:NOTHING
|
||
|
||
; Inputs:
|
||
; DS:BX = Transfer Address
|
||
; CX = Record Count
|
||
; DX = Starting Record
|
||
; AH = Media Byte
|
||
; AL = Unit Code
|
||
; Function:
|
||
; Set up the device call header at DEVCALL
|
||
; Output:
|
||
; ES:BX Points to DEVCALL
|
||
; No other registers effected
|
||
|
||
PUSH DI
|
||
PUSH CX
|
||
PUSH AX
|
||
MOV CL,DEVWRT
|
||
ADD CL,[VERFLG]
|
||
JMP SHORT SETCALLHEAD
|
||
SETREAD ENDP
|
||
|
||
do_ext
|
||
|
||
CODE ENDS
|
||
END
|
||
|