mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
4155 lines
151 KiB
4155 lines
151 KiB
page,132
|
|
;---------------------------Module-Header-------------------------------;
|
|
; Module Name: IBMSETUP.ASM
|
|
;
|
|
; Copyright (c) Microsoft Corporation 1985-1990. All Rights Reserved.
|
|
;
|
|
; General Description:
|
|
;
|
|
; History:
|
|
; sudeepb 10-Jan-1993 changed the costly cli/sti with non-trapping
|
|
; FCLI/FSTI macros
|
|
;
|
|
;-----------------------------------------------------------------------;
|
|
|
|
title IBMSetup - IBM PC, PC-XT, PC-AT, PS/2 Communications Interface
|
|
|
|
.xlist
|
|
include cmacros.inc
|
|
include comdev.inc
|
|
include ins8250.inc
|
|
include ibmcom.inc
|
|
include BIMODINT.INC
|
|
include vint.inc
|
|
.list
|
|
|
|
|
|
EBIS_Sel1 equ SIZE Bimodal_Int_Struc
|
|
EBIS_Sel2 equ EBIS_Sel1 + (SIZE EBIS_Sel_Struc)
|
|
|
|
externA __WinFlags
|
|
|
|
externFP GetSystemMsecCount
|
|
externFP CreateSystemTimer
|
|
externFP AllocCStoDSAlias
|
|
externFP LockSegment
|
|
externFP UnlockSegment
|
|
externFP FreeSelector
|
|
externFP GetSelectorBase
|
|
externFP GetModuleHandle
|
|
externFP GetProcAddress
|
|
externFP GetPrivateProfileInt
|
|
externFP GetPrivateProfileString
|
|
externFP GetAppCompatFlags
|
|
ifdef NEC_98
|
|
externFP KillSystemTimer ; (ins 92.xx.xx) |
|
|
externFP TickEntry1 ; (ins 92.09.25) |
|
|
externFP TickEntry2 ; (ins 92.09.25) |
|
|
externFP TickEntry3 ; (ins 92.09.25) |
|
|
externNP INT_1AH_call ; (ins 92.xx.xx) |
|
|
externNP INT_1AH_Close ; (ins 93.10.13) |
|
|
externFP MSR_READ_Call ; (ins 92.xx.xx) |
|
|
else ; NEC_98
|
|
externFP WowCloseComPort
|
|
endif ; NEC_98
|
|
|
|
externNP $RECCOM
|
|
|
|
externA __0040H
|
|
externA __F000h
|
|
|
|
externB IRQhooks
|
|
|
|
IF 0
|
|
externD OldIntVecIntB
|
|
externD OldIntVecIntC
|
|
externD OurIntVecIntB
|
|
externD OurIntVecIntC
|
|
ENDIF
|
|
|
|
externB szMessage
|
|
externB pLPTByte
|
|
externB szCOMMessage
|
|
externB pCOMByte
|
|
externB _szTitle
|
|
|
|
|
|
MULTIPLEX equ 2Fh ; multiplex interrupt number
|
|
GET386API equ 1684h ; Get API entry point from VxD
|
|
VPD equ 000Fh ; device ID of VPD device
|
|
VPD_GETPORT equ 0004h ; function: assign port to current VM
|
|
VPD_RELPORT equ 0005h ; function: release port
|
|
ifdef NEC_98
|
|
VPD_GETVER equ 0000h ; get version API (ins 94.02.24) |
|
|
endif ; NEC_98
|
|
VCD equ 000Eh ; device ID of VCD device
|
|
VCD_GETVER equ 0000h ; get version API
|
|
VCD_GETPORT equ 0004h ; function: assign port to current VM
|
|
VCD_RELPORT equ 0005h ; function: release port
|
|
VCD_STEALPORT equ 0006h
|
|
VPICD equ 0003h ; device ID of VPICD device
|
|
|
|
POSTMESSAGE equ 110 ; export ordinal of PostMessage()
|
|
MESSAGEBOX equ 1 ; export ordinal of MessageBox()
|
|
MB_TASKMODAL equ 2000h
|
|
MB_YESNO equ 0004h ; messagebox flags
|
|
MB_ICONEXCLAMATION equ 0030h
|
|
IDYES equ 6
|
|
|
|
ifdef NEC_98
|
|
NULL equ 00h ; [QN] (INS 92.08.05) |
|
|
BIOS_FLAG5 equ 58h ; [QN] (ins 92.11.06) |
|
|
BIOS_FLAG7 equ 5bh ; [QN] (ins 92.11.06) |
|
|
endif ; NEC_98
|
|
|
|
createSeg _INTERRUPT,IntCode,word,public,CODE
|
|
sBegin IntCode
|
|
assumes cs,IntCode
|
|
|
|
externFP FakeCOMIntFar
|
|
externFP TimerProc
|
|
externFP Control
|
|
externFP COMHandler
|
|
externFP APIHandler
|
|
IFDEF No_DOSX_Bimodal_Services
|
|
externW RM_IntDataSeg
|
|
externFP RM_APIHandler
|
|
externFP Entry_From_RM
|
|
externD RM_CallBack
|
|
ENDIF
|
|
|
|
sEnd IntCode
|
|
|
|
page
|
|
sBegin Data
|
|
|
|
externB lpCommBase
|
|
externB CommBaseX
|
|
externB lpCommIrq
|
|
externB CommIrqX
|
|
externB lpCommFifo
|
|
externB CommFifoX
|
|
externB lpCommDSR
|
|
externB CommDSRx
|
|
|
|
externB lpCommSection
|
|
externB lpSYSTEMINI
|
|
|
|
|
|
;------------------------------------------------------------------------------
|
|
;
|
|
; Reserve data space for COM ports
|
|
;
|
|
DefineCommX MACRO num
|
|
public Comm&num
|
|
Comm&num label byte
|
|
db num-1
|
|
.errnz DCB_Id
|
|
db ((DCBSize+1) AND 0FFFEh)-1 DUP (0) ; ComDCB
|
|
dw 0 ; ComErr
|
|
dw 0 ; Port
|
|
dw 0 ; NotifyHandle
|
|
dw 0 ; NotifyFlags
|
|
dw -1 ; RecvTrigger
|
|
dw 0 ; SendTrigger
|
|
.errnz IRQhook - SendTrigger - 2
|
|
db (SIZE ComDEB) - IRQhook DUP(0)
|
|
.errnz $ - Comm&num - (SIZE ComDEB)
|
|
Declare_PM_BIS 0,Control,COMHandler,APIHandler,_INTERRUPT,_DATA
|
|
db (SIZE EBIS_Sel_Struc) * 2 DUP(0) ; res space for 2 selectors
|
|
ENDM
|
|
DW_OFFSET_CommX MACRO num
|
|
dw DataOFFSET Comm&num
|
|
ENDM
|
|
|
|
|
|
??portnum = 1
|
|
REPT MAXCOM+1
|
|
DefineCommX %??portnum
|
|
??portnum = ??portnum+1
|
|
ENDM
|
|
|
|
PUBLIC COMptrs ; table of offsets to CommX's declared above
|
|
COMptrs label word
|
|
??portnum = 1
|
|
REPT MAXCOM+1
|
|
DW_OFFSET_CommX %??portnum
|
|
??portnum = ??portnum+1
|
|
ENDM
|
|
|
|
PURGE DefineCommX
|
|
PURGE DW_OFFSET_CommX
|
|
|
|
;------------------------------------------------------------------------------
|
|
;
|
|
; Reserve data space for LPT ports
|
|
;
|
|
DefineLPTx MACRO num
|
|
public LPT&num
|
|
LPT&num label byte
|
|
db num-1+LPTx
|
|
.errnz DCB_Id
|
|
db ((DCBSize+1) AND 0FFFEh)-1 DUP (0) ; xComDCB
|
|
dw 0 ; xComErr
|
|
dw 0 ; xPort
|
|
dw 0 ; xNotifyHandle
|
|
dw 0 ; xNotifyFlags
|
|
dw -1 ; xRecvTrigger
|
|
dw 0 ; xSendTrigger
|
|
IF num LE 3
|
|
dw LPTB + (num-1)*2
|
|
ELSE
|
|
dw 0 ; BIOSPortLoc
|
|
ENDIF
|
|
.errnz $-LPT&num - SIZE LptDEB
|
|
ENDM
|
|
|
|
??portnum = 1
|
|
REPT MAXLPT+1
|
|
DefineLPTx %??portnum
|
|
??portnum = ??portnum+1
|
|
ENDM
|
|
|
|
PURGE DefineLPTx
|
|
|
|
page
|
|
|
|
PUBLIC $MachineID, Using_DPMI
|
|
$MachineID db 0 ;IBM Machine ID
|
|
Using_DPMI db 0 ; 0FFh, if TRUE
|
|
|
|
ALIGN 2
|
|
|
|
PUBLIC activeCOMs
|
|
activeCOMs dw 0
|
|
|
|
PUBLIC lpPostMessage, lpfnMessageBox, lpfnVPD, fVPD
|
|
|
|
lpPostMessage dd 0
|
|
lpfnMessageBox dd 0
|
|
|
|
lpfnVPD dd 0 ; far pointer to win 386 VPD entry point
|
|
lpfnVCD dd 0 ; far pointer to win 386 VCD entry point
|
|
lpfnVPICD dd 0 ; far pointer to win 386 VPICD entry point
|
|
ifdef NEC_98
|
|
lpfnWPSVPD dd 0 ; far pointer to win 386 VPICD entry point(ins 94.02.24)|
|
|
endif ; NEC_98
|
|
PUBLIC VCD_int_callback
|
|
VCD_int_callback df 0 ; VCD returns the address for this callback
|
|
; on every call to acquire a COM port, but
|
|
; it is always the same address, so we will
|
|
; just maintain it globally.
|
|
fVPD db 0 ; 0-not checked, 1 vpd present, -1 no vpd
|
|
fVCD db 0 ; 0-not checked, 1 vcd present, -1 no vcd
|
|
fVPICD db 0 ; 0-not checked, 1 vpicd present, -1 no vpicd
|
|
ifdef NEC_98
|
|
fWPSVPD db 0 ; 0-not checked, 1 WPS present, -1 no WPS (ins 94.02.24) |
|
|
endif
|
|
|
|
szUser db 'USER',0
|
|
|
|
|
|
default_table db 4, 3, 4, 3, 0 ; Default IRQ's (COM3 default is changed to
|
|
; 3 for PS/2's during LoadLib)
|
|
|
|
|
|
IFDEF No_DOSX_Bimodal_Services
|
|
RM_Call_Struc Real_Mode_Call_Struc <>
|
|
ENDIF
|
|
|
|
IFDEF DEBUG_TimeOut
|
|
%OUT including code to display MsgBox, if closing comm with data in buffer
|
|
szSendTO db 'TimedOut CloseComm with data in buffer. Retry?', 0
|
|
ENDIF
|
|
|
|
ifdef NEC_98
|
|
;;;;;;;************ DEB Value of Each Port ************ |
|
|
;;;;;;; |
|
|
;;;;;;; +---------------------------------------ComDCB |
|
|
;;;;;;; |+--------------------------------------ComErr |
|
|
;;;;;;; ||+-------------------------------------Port |
|
|
;;;;;;; |||+------------------------------------NotifyHandle |
|
|
;;;;;;; ||||+-----------------------------------NotifyFlag |
|
|
;;;;;;; |||||+----------------------------------RecvTrigger |
|
|
;;;;;;; ||||||+---------------------------------SendTrigger |
|
|
;;;;;;; ||||||| +----------------------------TimerProcAdr |
|
|
;;;;;;; ||||||| | +-------------------MaskFFPort |
|
|
;;;;;;; ||||||| | | +--------------ReadSigPort |
|
|
;;;;;;; ||||||| | | | +---------DataPort |
|
|
;;;;;;; ||||||| | | | | +----CommandPort |
|
|
;;;;;;; ||||||| | | | | | +--StatusPort |
|
|
;;;;;;; ||||||| | | | | | | |
|
|
;;;;;;Comm1 ComDEB <,,,,,,,TickEntry1,035h,033h,030h,032h,032h,> ;Com1 |
|
|
;;;;;;Comm2 ComDEB <,,,,,,,TickEntry2,0B0h,0B0h,0B1h,0B3h,0B3h,> ;Com2 |
|
|
;;;;;;Comm3 ComDEB <,,,,,,,TickEntry3,0B2h,0B2h,0B9h,0BBh,0BBh,> ;Com3 |
|
|
; |
|
|
public Port_TBL,Port_EOI,Vect_TBL,Mask_TBL,PRN_CHAR ; |
|
|
public Board_66L,Board_59,FindCOMPort ; |
|
|
; |
|
|
;************************** Data Table ************************** |
|
|
; using IRQ as 'index'.
|
|
Port_TBL dw 00h,00h,00h,02h,00h,02h,02h,00h
|
|
dw 00h,0ah,0ah,00h,0ah,0ah,00h,00h
|
|
Port_EOI dw 00h,00h,00h,0bh,00h,0dh,0eh,00h
|
|
dw 00h,11h,13h,00h,14h,15h,00h,00h
|
|
Vect_TBL db 00h,00h,00h,00h,00h,00h,00h,00h
|
|
db 00h,08h,08h,00h,08h,08h,00h,00h
|
|
Mask_TBL db 00h,00h,00h,08h,00h,20h,40h,00h
|
|
db 00h,02h,08h,00h,10h,20h,00h,00h
|
|
PRN_CHAR db 0 ;Lpt Output Char Buffer |
|
|
; |
|
|
Board_66L dw 0BAh ;PC-980166/L board |
|
|
Board_59 dw 0BFh ;PC-980159 board |
|
|
; |
|
|
out5f_flag db 0 ; ins 93.03.24 (IOrecovery) |
|
|
public RECLoopCounter ; ins 93.03.31 |
|
|
RECLoopCounter dw 0 ; ins 93.03.31 |
|
|
endif ; NEC_98
|
|
|
|
sEnd Data
|
|
|
|
ROMBios segment at 0F000h
|
|
org 0FFFEh
|
|
|
|
MachineID label byte
|
|
RomBios Ends
|
|
|
|
|
|
sBegin Code
|
|
assumes cs,Code
|
|
assumes ds,Data
|
|
|
|
page
|
|
|
|
IFDEF No_DOSX_Bimodal_Services
|
|
;----------------------------Private-Routine----------------------------;
|
|
; SegmentFromSelector
|
|
;
|
|
; Converts a selector to a segment...note that this routine assumes
|
|
; the memory pointed to by the selector is below the 1Meg line!
|
|
;
|
|
; Params:
|
|
; AX = selector to convert to segment
|
|
;
|
|
; Returns:
|
|
; AX = segment of selector given
|
|
;
|
|
; Error Returns:
|
|
; None
|
|
;
|
|
; Registers Destroyed:
|
|
; CX
|
|
;
|
|
;-----------------------------------------------------------------------;
|
|
|
|
assumes ds,Data
|
|
assumes es,nothing
|
|
|
|
public SegmentFromSelector
|
|
SegmentFromSelector proc far
|
|
|
|
.286
|
|
push dx
|
|
cCall GetSelectorBase,<ax> ;DX:AX = segment of selector
|
|
shr ax, 4
|
|
shl dl, 4
|
|
or ah, dl ;AX now points to interrupt *segment*
|
|
pop dx
|
|
ret
|
|
.8086
|
|
|
|
SegmentFromSelector endp
|
|
ENDIF
|
|
page
|
|
|
|
;------------------------------------------------------------------------------
|
|
;
|
|
; Get_API_Entry
|
|
;
|
|
; entry - BX = device id
|
|
; DS:DI -> DWORD for proc address
|
|
; exit - Z flag set, if failed
|
|
;
|
|
Get_API_Entry proc near
|
|
|
|
push di
|
|
xor di, di
|
|
mov es, di
|
|
mov ax, GET386API
|
|
int MULTIPLEX
|
|
mov ax, di
|
|
pop di
|
|
mov [di], ax
|
|
mov [di+2], es
|
|
or ax, [di+2]
|
|
ret
|
|
|
|
Get_API_Entry endp
|
|
|
|
;----------------------------Private-Routine----------------------------;
|
|
;
|
|
; Contention_Dlg
|
|
;
|
|
; If running under Win386, this routine can be called to ask the user to
|
|
; resolve contention for a COM or LPT port.
|
|
;
|
|
; entry - CX is offset of message string for dialog box
|
|
;
|
|
; exit - Z flag set, if user specified that Windows should steal the port
|
|
|
|
Contention_Dlg proc near
|
|
PUBLIC Contention_Dlg
|
|
|
|
xor ax,ax
|
|
push ax ; hwndOwner
|
|
push ds
|
|
push cx ; message ptr
|
|
|
|
cmp wo lpfnMessageBox[2], 0 ;Q: ptr to MessageBox proc valid?
|
|
jne short gmbp_done ; Y: we can call it
|
|
push ds ; N: get module handle of USER
|
|
lea ax, szUser
|
|
push ax
|
|
cCall GetModuleHandle
|
|
|
|
push ax ; module handle
|
|
mov ax, MESSAGEBOX
|
|
cwd
|
|
push dx
|
|
push ax
|
|
cCall GetProcAddress
|
|
mov wo lpfnMessageBox[0], ax ; save received proc address
|
|
mov wo lpfnMessageBox[2], dx
|
|
gmbp_done:
|
|
|
|
push ds
|
|
lea ax, _szTitle
|
|
push ax
|
|
mov ax, MB_ICONEXCLAMATION or MB_YESNO or MB_TASKMODAL
|
|
push ax
|
|
cCall lpfnMessageBox
|
|
cmp ax, IDYES ; user allows us to take the port?
|
|
ret
|
|
Contention_Dlg endp
|
|
|
|
ifdef NEC_98
|
|
;---------------------------------------------------------------(ins 94.02.24) |
|
|
; WPS386 (ins 94.02.24) |
|
|
; Request ON/OFF of port trap to WPSVPD (ins 94.02.24) |
|
|
; (The function is toggled by calling WPSVPD) (ins 94.02.24) |
|
|
; (ins 94.02.24) |
|
|
; entry - DI contains offset in ROM area of port... (ins 94.02.24) |
|
|
; 8 - LPT1, A - LPT2, etc (ins 94.02.24) |
|
|
;---------------------------------------------------------------(ins 94.02.24) |
|
|
WPS386 proc near ; (ins 94.02.24) |
|
|
.286 ; (ins 94.02.24) |
|
|
pusha ; (ins 94.02.24) |
|
|
.8086 ; (ins 94.02.24) |
|
|
mov di, [si.BIOSPortLoc] ; (ins 94.02.24) |
|
|
sub di, LPTB ; (ins 94.02.24) |
|
|
shr di, 1 ; DI into port number (ins 94.02.24) |
|
|
cmp fWPSVPD, 1 ; Is there WPSVPD ? (ins 94.02.24) |
|
|
je short beWPSVPD ; Y: Go Next (ins 94.02.24) |
|
|
; (ins 94.02.24) |
|
|
push di ; (ins 94.02.24) |
|
|
mov bx, VPD ; (ins 94.02.24) |
|
|
mov di, DataOFFSET lpfnWPSVPD ; (ins 94.02.24) |
|
|
call Get_API_Entry ;Is there VPD ? (ins 94.02.24) |
|
|
pop di ; (ins 94.02.24) |
|
|
jnz short chkWPSVPD ; Y: Go Next (ins 94.02.24) |
|
|
mov fWPSVPD, -1 ; (ins 94.02.24) |
|
|
jmp short WPSVPD_exit ; (ins 94.02.24) |
|
|
chkWPSVPD: ; (ins 94.02.24) |
|
|
push di ; (ins 94.02.24) |
|
|
mov dx, VPD_GETVER ; Get VPD version (ins 94.02.24) |
|
|
call [lpfnWPSVPD] ; (ins 94.02.24) |
|
|
pop di ; (ins 94.02.24) |
|
|
cmp ax, 030bh ; WPSVPD(=30bh)? (ins 94.02.24) |
|
|
je short insWPSVPD ; Y: Next(IS WPSVPD) (ins 94.02.24) |
|
|
mov fWPSVPD, -1 ; ) (ins 94.02.24) |
|
|
jne short WPSVPD_exit ; (ins 94.02.24) |
|
|
insWPSVPD: ; (ins 94.02.24) |
|
|
mov fWPSVPD, 1 ; There is WPSVPD (ins 94.02.24) |
|
|
beWPSVPD: ; (ins 94.02.24) |
|
|
xor ax, ax ; (ins 94.02.24) |
|
|
mov cx, di ; Set LPT ID (ins 94.02.24) |
|
|
mov dx, VPD_GETPORT ; (ins 94.02.24) |
|
|
call [lpfnWPSVPD] ; Call VPD_PM_Aquire_Port(ins 94.02.24) |
|
|
WPSVPD_exit: ; (ins 94.02.24) |
|
|
.286 ; (ins 94.02.24) |
|
|
popa ; (ins 94.02.24) |
|
|
ret ; (ins 94.02.24) |
|
|
.8086 ; (ins 94.02.24) |
|
|
WPS386 endp ; (ins 94.02.24) |
|
|
endif ; NEC_98
|
|
|
|
;----------------------------Private-Routine----------------------------;
|
|
;
|
|
; GetPort386
|
|
;
|
|
; If running under Win386, tell the VPD to assign an LPT port to us.
|
|
; The comm driver will handle contention.
|
|
;
|
|
; entry - DI contains offset in ROM area of port...
|
|
; 8 - LPT1, A - LPT2, etc
|
|
;
|
|
; exit - registers saved, carry = clear if OK to proceed, set if
|
|
; user won't allow assignment of port or Win386 error
|
|
;
|
|
|
|
GetPort386 proc near
|
|
public GetPort386
|
|
|
|
cmp fVPD, 0
|
|
jl getport_VPDNotInstalled
|
|
jnz short getport_CallVPD
|
|
|
|
push di
|
|
mov bx, VPD
|
|
mov di, DataOFFSET lpfnVPD
|
|
call Get_API_Entry
|
|
pop di
|
|
jnz short getport_CallVPD
|
|
mov fVPD, -1
|
|
|
|
getport_VPDNotInstalled:
|
|
clc
|
|
jmp short getport_exit
|
|
|
|
getport_CallVPD:
|
|
mov fVPD, 1
|
|
push di
|
|
sub di, LPTB
|
|
shr di, 1 ; turn DI into port number
|
|
|
|
xor ax, ax
|
|
mov dx, VPD_GETPORT
|
|
mov cx, di
|
|
call [lpfnVPD]
|
|
jnc getport_gotit
|
|
|
|
; port owned by another VM... ask the user for it
|
|
|
|
add cl, '1' ; fix up the port name...
|
|
mov pLPTByte, cl ; HACK HACK HACK
|
|
lea cx, szMessage
|
|
call Contention_Dlg
|
|
jnz getport_userwontallow
|
|
|
|
mov ax, 1 ; tell win386 we really do want it
|
|
mov cx, di ;
|
|
mov dx, VPD_GETPORT ;
|
|
call [lpfnVPD] ; return with C set or clear...
|
|
jmp short getport_gotit
|
|
|
|
getport_userwontallow:
|
|
stc
|
|
|
|
getport_gotit:
|
|
pop di
|
|
|
|
getport_exit:
|
|
ret
|
|
|
|
GetPort386 endp
|
|
|
|
;----------------------------Private-Routine----------------------------;
|
|
;
|
|
; ReleasePort386
|
|
;
|
|
; If running under Win386, tell the VPD to deassign an LPT port.
|
|
;
|
|
; entry - DS:SI -> COMDEB
|
|
;
|
|
|
|
ReleasePort386 proc near
|
|
|
|
cmp fVPD, 1
|
|
jne release_noVPD
|
|
|
|
xor cx, cx
|
|
mov cl, [si.DCB_id]
|
|
and cl, NOT LPTx ; clear high bit
|
|
mov dx, VPD_RELPORT
|
|
call [lpfnVPD]
|
|
|
|
release_noVPD:
|
|
ret
|
|
|
|
ReleasePort386 endp
|
|
|
|
|
|
;----------------------------Private-Routine----------------------------;
|
|
;
|
|
; GetCOMport386
|
|
;
|
|
; If running under Win386, tell the VCD to assign a COM port to us.
|
|
; The comm driver will handle contention.
|
|
;
|
|
; entry - DS:SI -> COMDEB
|
|
;
|
|
; exit - registers saved, carry = clear if OK to proceed, set if
|
|
; user won't allow assignment of port or Win386 error
|
|
;
|
|
.386
|
|
GetCOMport386 proc near
|
|
public GetCOMport386
|
|
|
|
push es
|
|
pushad
|
|
cmp fVCD, 0
|
|
jl short getcomport_VCDNotInstalled
|
|
jnz short getcomport_CallVCD
|
|
|
|
mov bx, VCD
|
|
mov di, DataOFFSET lpfnVCD
|
|
call Get_API_Entry
|
|
jz short getcomport_checknoVCD
|
|
|
|
mov dx, VCD_GETVER
|
|
call [lpfnVCD]
|
|
cmp ax, 30Ah ;Q: 3.1 or greater?
|
|
jae short getcomport_CallVCD ; Y:
|
|
|
|
getcomport_checknoVCD:
|
|
mov fVCD, -1
|
|
|
|
getcomport_VCDNotInstalled:
|
|
clc
|
|
jmp short getcomport_exit
|
|
|
|
getcomport_CallVCD:
|
|
mov fVCD, 1
|
|
|
|
mov ax, 10b ; flag ring0 int handler
|
|
call VCD_GetPort_API
|
|
jnc short getcomport_success ; jump if acquire worked
|
|
jnz short getcomport_noport ; jump if port doesn't exist
|
|
|
|
; port owned by another VM... ask the user for it
|
|
|
|
mov cl, [si.DCB_id]
|
|
add cl, '1' ; fix up the port name...
|
|
mov pCOMByte, cl
|
|
lea cx, szCOMMessage
|
|
call Contention_Dlg
|
|
stc
|
|
jnz short getcomport_exit
|
|
|
|
mov ax, 11b ; tell win386 we really do want it
|
|
call VCD_GetPort_API
|
|
jc short getcomport_exit
|
|
|
|
getcomport_success:
|
|
mov dword ptr [VCD_int_callback], edi
|
|
mov word ptr [VCD_int_callback+4], cx
|
|
mov [si.VCD_data], ebx
|
|
xchg ax, [si.Port]
|
|
or ax, ax ;Q: already had port base?
|
|
jnz short getcomport_exit ; Y: don't update vector #, or FIFO
|
|
mov [si.IntVecNum], dl
|
|
call GetPortFlags
|
|
clc
|
|
|
|
getcomport_exit:
|
|
popad
|
|
pop es
|
|
ret
|
|
|
|
getcomport_noport:
|
|
mov [si.Port], -1
|
|
jmp getcomport_exit
|
|
|
|
GetCOMport386 endp
|
|
|
|
VCD_GetPort_API proc near
|
|
mov dx, VCD_GETPORT
|
|
xor cx, cx
|
|
mov cl, [si.DCB_Id] ; cx = port #
|
|
mov di, VCDflags ; offset from start of DEB
|
|
call [lpfnVCD]
|
|
ret
|
|
VCD_GetPort_API endp
|
|
.8086
|
|
|
|
;----------------------------Private-Routine----------------------------;
|
|
;
|
|
; ReleaseCOMport386
|
|
;
|
|
; If running under Win386, tell the VCD to deassign a COM port.
|
|
;
|
|
; entry - DS:SI -> COMDEB
|
|
;
|
|
|
|
ReleaseCOMport386 proc near
|
|
|
|
ifdef NEC_98
|
|
cmp fVCD, 1
|
|
jne release_noVCD
|
|
|
|
xor cx, cx
|
|
mov cl, [si.DCB_id]
|
|
mov dx, VCD_RELPORT
|
|
call [lpfnVCD]
|
|
else ; NEC_98
|
|
ifndef WOW
|
|
cmp fVCD, 1
|
|
jne release_noVCD
|
|
|
|
xor cx, cx
|
|
mov cl, [si.DCB_id]
|
|
mov dx, VCD_RELPORT
|
|
call [lpfnVCD]
|
|
else
|
|
xor cx, cx
|
|
mov cl, [si.DCB_id]
|
|
push cx
|
|
call WowCloseComPort
|
|
endif
|
|
endif ; NEC_98
|
|
|
|
release_noVCD:
|
|
ret
|
|
|
|
ReleaseCOMport386 endp
|
|
|
|
PUBLIC StealPort
|
|
StealPort proc near
|
|
|
|
cmp fVCD, 1
|
|
jne sp_yes
|
|
mov dx, VCD_STEALPORT
|
|
xor cx, cx
|
|
mov cl, [si.DCB_id]
|
|
call [lpfnVCD]
|
|
or al, al
|
|
jnz sp_yes
|
|
|
|
sp_no:
|
|
stc
|
|
ret
|
|
|
|
sp_yes:
|
|
clc
|
|
mov [si.VCDflags], 0
|
|
ret
|
|
|
|
StealPort endp
|
|
|
|
page
|
|
|
|
;------------------------------------------------------------------------------
|
|
;
|
|
; ENTER: DS:SI -> ComDEB
|
|
; EXIT: AL = 0, if IRQ was unmasked, else -1, if IRQ was already masked
|
|
;
|
|
MaskIRQ proc near
|
|
push es
|
|
push di
|
|
mov di, ds
|
|
mov es, di
|
|
lea di, [si+SIZE ComDEB]
|
|
mov ax, BIH_API_Get_Mask
|
|
call APIHandler ; returns Carry Set, if masked
|
|
jc short already_masked
|
|
pushf
|
|
mov ax, BIH_API_Mask
|
|
call APIHandler ; mask IRQ
|
|
xor ax, ax
|
|
popf
|
|
jnc short mask_exit
|
|
already_masked:
|
|
or al, -1
|
|
mask_exit:
|
|
pop di
|
|
pop es
|
|
ret
|
|
MaskIRQ endp
|
|
|
|
;------------------------------------------------------------------------------
|
|
;
|
|
; ENTER: DS:SI -> ComDEB
|
|
;
|
|
UnmaskIRQ proc near
|
|
push es
|
|
push di
|
|
mov di, ds
|
|
mov es, di
|
|
lea di, [si+SIZE ComDEB]
|
|
mov ax, BIH_API_Unmask
|
|
call APIHandler
|
|
pop di
|
|
pop es
|
|
ret
|
|
UnmaskIRQ endp
|
|
page
|
|
|
|
;----------------------------Public Routine-----------------------------;
|
|
;
|
|
; $INICOM - Initialize A Port
|
|
;
|
|
; Initalizes the requested port if present, and sets
|
|
; up the port with the given attributes when they are valid.
|
|
; This routine also initializes communications buffer control
|
|
; variables. This routine is passed the address of a device
|
|
; control block.
|
|
;
|
|
; The RLSD, CTS, and DSR signals should be ignored by all COM
|
|
; routines if the corresponding timeout values are 0.
|
|
;
|
|
; For the LPT ports, a check is performed to see if the hardware
|
|
; is present (via the LPT port addresses based at 40:8h. If the
|
|
; port is unavailable, an error is returned. If the port is
|
|
; available, then the DEB is set up for the port. $SETCOM will
|
|
; be called to set up the DEB so that there will be something
|
|
; valid to pass back to the caller when he inquires the DEB.
|
|
;
|
|
; No hardware initialization will be performed to prevent the
|
|
; RESET line from being asserted and resetting the printer every
|
|
; time this routine is called.
|
|
;
|
|
; Entry:
|
|
; EX:BX --> Device Control Block with all fields set.
|
|
; Returns:
|
|
; AX = 0 if no errors occured
|
|
; Error Returns:
|
|
; AX = initialization error code otherwise
|
|
; Registers Preserved:
|
|
; None
|
|
; Registers Destroyed:
|
|
; AX,BX,CX,DX,ES,FLAGS
|
|
; History:
|
|
;-----------------------------------------------------------------------;
|
|
|
|
;------------------------------Pseudo-Code------------------------------;
|
|
; {
|
|
; }
|
|
;-----------------------------------------------------------------------;
|
|
|
|
assumes ds,Data
|
|
assumes es,nothing
|
|
|
|
public $INICOM
|
|
$INICOM proc near
|
|
push si ;As usual, save register variables
|
|
push di
|
|
mov ah,es:[bx.DCB_Id] ;Get device i.d.
|
|
call GetDEB ;--> DEB for this device
|
|
mov ax, IE_BADID ; call it a bad id (spooler uses DOS)
|
|
jc InitCom15 ;Invalid device
|
|
jns InitCom20 ; jmp if COM port
|
|
|
|
ifndef NEC_98
|
|
push ds
|
|
|
|
mov di, [si.BIOSPortLoc]
|
|
cmp di, LPTB
|
|
jb short InitLPT_Installed
|
|
|
|
mov cx,__0040H ;[rkh] ...
|
|
mov ds,cx ;Point DS: at ROM Save Area.
|
|
assumes ds,nothing
|
|
|
|
mov ax, IE_HARDWARE
|
|
mov cx, wo [di]
|
|
jcxz InitCom10 ; if zero, no hardware
|
|
|
|
mov ax,IE_BadID ;Show bad device
|
|
cmp ch, 0 ; zero hibyte -> not valid (redir)
|
|
jz InitCom10 ; call it a bad id (spooler uses DOS)
|
|
|
|
cmp di, LPTB ; LPT1?
|
|
jz InitLPT_Installed ; yes, must be installed
|
|
|
|
cmp cx, wo [di-2] ;Q: duplicate of previous port
|
|
je InitCom10 ; Y: (redirected port)
|
|
|
|
InitLPT_Installed:
|
|
pop ds
|
|
mov [si.Port], cx
|
|
endif ; NEC_98
|
|
call $SETCOM
|
|
ifdef NEC_98
|
|
mov di, [si.BIOSPortLoc] ; (ins 93.12.19) |
|
|
endif ; NEC_98
|
|
call GetPort386 ; tell win386 we're using the port
|
|
mov ax, IE_OPEN ; port already open (by another VM)
|
|
jc InitCom15 ; error
|
|
ifdef NEC_98
|
|
call WPS386 ; WPS386 I/O trap ON (ins 94.02.24) |
|
|
mov ah, 1 ; Check H/W environment (ins 93.10.29) |
|
|
call int_1ah_call ; and set to [si.Port](ins 93.10.29) |
|
|
call WPS386 ; WPS386 I/O trap OFF (ins 94.02.24) |
|
|
cmp wo [si.Port], -1 ; Port = -1 ? (ins 93.12.19) |
|
|
mov ax, IE_HARDWARE ; (no H/W?) (ins 93.12.19) |
|
|
jne @f ; N: success (ins 93.12.19) |
|
|
call ReleasePort386 ; Y: release port (ins 93.12.19) |
|
|
jmp InitCom100 ; (ins 93.12.19) |
|
|
@@: ; (ins 93.12.19) |
|
|
endif ; NEC_98
|
|
jmp InitCom90 ;That's all
|
|
|
|
InitCom10:
|
|
pop ds ; get DS back
|
|
InitCom15:
|
|
jmp InitCom100
|
|
|
|
|
|
assumes ds,Data
|
|
InitCom17:
|
|
ifdef NEC_98
|
|
mov ax, IE_OPEN ; (ins 92.08.xx) |
|
|
mov dx,StatusPort[si] ;dx = StatusPort[si] (ins 92.08.xx) |
|
|
NEWIODELAY 5 ;<OUT 5F,AL> (ins 94.05.11) |
|
|
in al,dx ; (ins 92.08.xx) |
|
|
cmp al,-1 ; (ins 92.08.xx) |
|
|
jne InitCom15 ; N:return IE_OPEN (ins 92.08.xx) |
|
|
jmp InitCom27 ; Y:return IE_HARDWARE(ins 92.08.xx) |
|
|
else ; NEC_98
|
|
mov ax, IE_OPEN
|
|
cmp [si.Port], -1 ;Q: determined that port didn't exist?
|
|
jne InitCom15 ; N: return IE_OPEN
|
|
jmp short InitCom27 ; Y: return IE_HARDWARE
|
|
endif ; NEC_98
|
|
|
|
; *** Set up serial port ***
|
|
;
|
|
InitCom20:
|
|
ifdef NEC_98
|
|
push cx ; |
|
|
push ax ; |
|
|
; |
|
|
call OUT5FCheck ; (ins 93.03.24) |
|
|
; |
|
|
mov cx,500 ; |
|
|
mov ax, __WinFlags ; [QN][BA](ins 93.03.30) |
|
|
and ax, WF_ENHANCED ; [QN][BA](ins 93.03.30) |
|
|
jnz Go_EnhSET ; [QN][BA](ins 93.03.30) |
|
|
; |
|
|
cmp [out5f_flag],1 ;out 5f ok ?? (ins 93.03.24) |
|
|
je @f ;yes (ins 93.03.24) |
|
|
mov cx,5000 ; |
|
|
@@: ; |
|
|
jmp short OUT5FSetEND ; |
|
|
; |
|
|
Go_EnhSET: ; |
|
|
cmp [out5f_flag],1 ;out 5f ok ?? (ins 93.03.24) |
|
|
jne @f ;No (ins 93.03.24) |
|
|
mov cx,1 ; |
|
|
@@: ; |
|
|
; |
|
|
OUT5FSetEND: ; |
|
|
mov [RECLoopCounter] , cx ; |
|
|
pop ax ; |
|
|
pop cx ; |
|
|
; |
|
|
mov ah,es:[bx.DCB_Id] ;Get device i.d. (ins 92.08.xx) |
|
|
cmp ah,ID_COM1 ;COM1? (ins 92.08.xx) |
|
|
je InitCom22 ;Yes, InitCom22 に飛ぶ。(ins 92.08.xx) |
|
|
InitCom21: ; (ins 94.04.15) |
|
|
or EFlags[si], fNoFIFO ; (ins 94.04.21) |
|
|
mov dx,[Board_66L] ; (ins 92.08.xx) |
|
|
in al,dx ;PC-980166/Lボード (ins 92.08.xx) |
|
|
cmp al,0FFH ;通信制御アダプタが (ins 92.08.xx) |
|
|
je @f ; (ins 94.05.19) |
|
|
jmp InitCom27 ; (ins 94.05.19) |
|
|
@@: ; (ins 94.05.19) |
|
|
mov dx,[Board_59] ; (ins 92.08.xx) |
|
|
in al,dx ;PC-980159ボード (ins 92.08.xx) |
|
|
cmp al,0FFH ;高速回線アダプタが (ins 92.08.xx) |
|
|
je @f
|
|
jmp InitCom27
|
|
@@:
|
|
mov dx,StatusPort[si] ;dx = StatusPort[si] (ins 92.08.xx) |
|
|
in al,dx ; (ins 92.08.xx) |
|
|
cmp al,-1 ; (ins 92.08.xx) |
|
|
jne @f
|
|
jmp InitCom27
|
|
@@:
|
|
; |
|
|
InitCom22: ; (ins 92.08.xx) |
|
|
else ; NEC_98
|
|
cmp [si.Port], -1 ;Q: port exists?
|
|
je InitCom27 ; N: report not found
|
|
endif ; NEC_98
|
|
|
|
mov ax, __WinFlags
|
|
test ax, WF_ENHANCED
|
|
jz short @F
|
|
call GetCOMport386
|
|
ifdef NEC_98
|
|
jc InitCom23 ; (ins 94.04.18) |
|
|
jmp short @F ; (ins 94.04.18) |
|
|
InitCom23: ; (ins 94.04.18) |
|
|
jmp InitCom17 ; (ins 94.04.18) |
|
|
@@: ; (ins 94.04.18) |
|
|
mov ah,es:[bx.DCB_Id] ;Get device i.d. (mov 94.05.19) |
|
|
or EFlags[si], fNoFIFO ;no AOBA on board (ins 94.05.31) |
|
|
cmp ah,ID_COM1 ;COM1? (mov 94.05.19) |
|
|
jne InitCom24_1 ;No, (mov 94.05.19) |
|
|
mov Port[si],0 ; (mov 94.05.19) |
|
|
call AOBACheck ;Check AOBA (mov 94.05.19) |
|
|
FCLI ; (mov 94.05.19) |
|
|
test EFlags[si], fNoFIFO ;AOBA on board ? (mov 94.05.19) |
|
|
jnz InitCom24_2 ; (mov 94.05.19) |
|
|
mov Port[si],130H ; (mov 94.05.19) |
|
|
mov dx,Port[si] ;AOBA BASE PORT (mov 94.05.19) |
|
|
add dx,ACE_FCR ; (mov 94.05.19) |
|
|
in al,dx ; (mov 94.05.19) |
|
|
mov preFCRShadow[si],al ;Save FCR when WIN start(ins 94.06.06) |
|
|
test al,ACE_EFIFO ;FIFO mode ? (mov 94.05.19) |
|
|
jz short @F ;No, 8251 mode (mov 94.05.19) |
|
|
call Set8251mode ;Set 8251 mode (mov 94.05.19) |
|
|
@@: ; (mov 94.05.19) |
|
|
jmp short InitCom24_2 ; (mov 94.05.19) |
|
|
InitCom24_1: ; (mov 94.05.19) |
|
|
InitCom24_2: ; (mov 94.05.19) |
|
|
|
|
; Entry: ;(ins 92.08.xx) |
|
|
; DS:SI --> DEB ;(ins 92.08.xx) |
|
|
; CX = Port base address ;(ins 92.08.xx) |
|
|
; DS = Data ;(ins 92.08.xx) |
|
|
;(ins 92.08.xx) |
|
|
call Get_PortAddr_and_INTVect ;割り込みベクタを得る ;(ins 92.08.xx) |
|
|
;(ins 92.08.xx) |
|
|
; Returns: ;(ins 92.08.xx) |
|
|
; AH = IRQ number ;(ins 92.08.xx) |
|
|
; AL = 8259 Mask ;(ins 92.08.xx) |
|
|
; DI:DX --> interrupt handler ;(ins 92.08.xx) |
|
|
;(ins 92.08.xx) |
|
|
inc IntVecIntCount[si] ; USE Rotate_Pic() (ins 92.08.xx) |
|
|
call FindCOMPort ; (MOVE 93.03.20) |
|
|
else ; NEC_98
|
|
jc InitCom17
|
|
|
|
@@:
|
|
cmp [si.Port], 0 ;Q: already got info?
|
|
jnz @F
|
|
call FindCOMPort
|
|
jc InitCom27 ; report not found, if error
|
|
mov [si.Port], ax
|
|
mov [si.IntVecNum], dl
|
|
@@:
|
|
endif ; NEC_98
|
|
|
|
push es ;Save these registers
|
|
push di
|
|
push cx ;needed later for $SETCOM etc
|
|
push bx
|
|
|
|
mov al, [si.IntVecNum]
|
|
xor ah, ah
|
|
lea di, [si+SIZE ComDEB]
|
|
mov [di.BIS_IRQ_Number], ax
|
|
|
|
mov di, DataOFFSET IRQhooks
|
|
ifdef NEC_98
|
|
xor cx,cx ; [QN] (ins 92.09.xx) |
|
|
mov cl, [si.DCB_Id] ; [QN] (ins 92.09.xx) |
|
|
jcxz InitCom30 ; [QN] (ins 92.09.xx) |
|
|
else ; NEC_98
|
|
mov cx, MAXCOM+1
|
|
endif ; NEC_98
|
|
InitCom25:
|
|
ifndef NEC_98
|
|
cmp al, [di.IRQn] ;Q: hooked IRQ matches ours?
|
|
je short InitCom30 ; Y:
|
|
cmp [di.IRQn], 0 ;Q: end of hooked IRQ list?
|
|
je short InitCom35 ; Y:
|
|
endif ; NEC_98
|
|
add di, SIZE IRQ_Hook_Struc ; N: check next hook
|
|
loop InitCom25
|
|
ifdef NEC_98
|
|
jmp short InitCom30 ; [QN] (ins 92.09.xx) |
|
|
endif ; NEC_98
|
|
int 3 ; data structures corrupt if we
|
|
; get here, because no hook table
|
|
; entries exist and there is suppose
|
|
; to be at least 1 for each DEB
|
|
InitCom26:
|
|
call ReleaseCOMport386 ; give port back to 386...
|
|
pop bx
|
|
pop cx
|
|
pop di
|
|
pop es
|
|
|
|
InitCom27:
|
|
mov ax, IE_HARDWARE ; jump if port not available
|
|
jmp InitCom100
|
|
|
|
InitCom30:
|
|
cmp [di.HookCnt], 0 ;Q: IRQ still hooked?
|
|
je short InitCom35 ; N: rehook
|
|
inc [di.HookCnt] ; Y: inc hook count
|
|
mov [si.IRQhook], di ; & link DEB into list
|
|
mov ax, [di.First_DEB]
|
|
mov [si.NextDEB], ax
|
|
mov [di.First_DEB], si
|
|
jmp short InitCom40
|
|
|
|
InitCom35:
|
|
mov [di.IRQn], al ; hook IRQ for first time, or rehook
|
|
mov [si.IRQhook], di
|
|
mov [di.First_DEB], si
|
|
mov [di.HookCnt], 1
|
|
call MaskIRQ
|
|
mov [di.OldMask], al
|
|
|
|
InitCom40: ; di -> IRQ_Hook_Struc
|
|
|
|
cmp [fVPICD], 0 ;Q: VPICD bimodel services available?
|
|
jl short InitCom415 ; N:
|
|
mov ax, ds ; Y: use them
|
|
mov es, ax
|
|
lea di, [si+SIZE ComDEB]
|
|
|
|
mov [di.BIS_Descriptor_Count], 2
|
|
mov ax, word ptr [si.QInAddr+2] ; get selector of in queue
|
|
mov [di.EBIS_Sel1.EBIS_User_Mode_Sel], ax
|
|
mov ax, word ptr [si.QOutAddr+2] ; get selector of out queue
|
|
mov [di.EBIS_Sel2.EBIS_User_Mode_Sel], ax
|
|
|
|
mov ax, VPICD_Install_Handler
|
|
call [lpfnVPICD]
|
|
jnc InitCom42
|
|
cmp [di.OldMask], 0
|
|
jne InitCom26
|
|
call UnmaskIRQ
|
|
jmp InitCom26
|
|
|
|
InitCom42:
|
|
;
|
|
; save newly allocated selectors/segments into "Alt" queue pointers
|
|
;
|
|
mov ax, [di.EBIS_Sel1.EBIS_Super_Mode_Sel]
|
|
mov word ptr [si.AltQInAddr+2], ax
|
|
mov ax, [di.EBIS_Sel2.EBIS_Super_Mode_Sel]
|
|
mov word ptr [si.AltQOutAddr+2], ax
|
|
|
|
InitCom414:
|
|
jmp InitCom59
|
|
|
|
InitCom415:
|
|
cmp [di.VecN], 0FFh ;Q: int already hooked?
|
|
IFDEF No_DOSX_Bimodal_Services
|
|
jnz short InitCom52 ; Y: init RMode ptrs in BIS
|
|
ELSE
|
|
jnz InitCom414 ; Y:
|
|
ENDIF
|
|
mov al, [si.IntVecNum]
|
|
add al, 8 ; 1st PIC starts at vector 8h
|
|
ifndef NEC_98
|
|
cmp al, 16 ;Q: 2nd PIC?
|
|
jb short InitCom418 ; N:
|
|
add al, 70h-16 ; Y: 2nd PIC starts at vector 70h
|
|
endif ; NEC_98
|
|
InitCom418:
|
|
mov [di.VecN], al
|
|
|
|
; *** Set interrupt vectors ***
|
|
;
|
|
mov ah,35h ;Get the DOS vector
|
|
int 21h ;DOS Get Vector Function
|
|
mov wo [di.OldIntVec][0], bx
|
|
mov wo [di.OldIntVec][2], es
|
|
|
|
InitCom50:
|
|
push ds ;Save original DS
|
|
mov dx, [di.HandlerOff]
|
|
mov bx, _INTERRUPT
|
|
mov ds, bx ;Interrupt handler address in ds:dx
|
|
assumes ds,nothing
|
|
mov ah, 25h ;DOS Set Vector Function
|
|
int 21h ;Set the DOS vector
|
|
pop ds ;Original DS
|
|
assumes ds,Data
|
|
|
|
IFDEF No_DOSX_Bimodal_Services
|
|
InitCom52:
|
|
cmp [Using_DPMI], 0
|
|
jz short InitCom57
|
|
|
|
mov ax, Int31_Get_Version SHL 8
|
|
int 31h
|
|
mov bl, [si.IntVecNum]
|
|
mov bh, bl
|
|
add bl, dh ; assume master PIC
|
|
sub bh, 8 ;Q: IRQ in master?
|
|
jb @f ; Y: add master's base vec
|
|
add bh, dl ; N: add slave's base vec
|
|
mov bl, bh
|
|
@@:
|
|
mov ax, Get_RM_IntVector
|
|
int 31h
|
|
mov wo [di.RM_OldIntVec][0], dx
|
|
mov wo [di.RM_OldIntVec][2], cx
|
|
|
|
mov dx, [di.RM_HandlerOff]
|
|
mov ax, _INTERRUPT
|
|
call SegmentFromSelector
|
|
mov cx, ax
|
|
push cx
|
|
mov ax, Set_RM_IntVector
|
|
int 31h
|
|
|
|
lea di, [si+SIZE ComDEB]
|
|
mov wo [di.BIS_Super_Mode_API], IntCodeOFFSET RM_APIHandler
|
|
pop cx
|
|
mov wo [di.BIS_Super_Mode_API+2], cx
|
|
|
|
;
|
|
; Get segment addresses for the Q's and set AltQInAddr and AltQOutAddr
|
|
;
|
|
mov ax, wo [si.AltQInAddr+2]
|
|
call SegmentFromSelector
|
|
mov wo [si.AltQInAddr+2], ax
|
|
mov ax, wo [si.AltQOutAddr+2]
|
|
call SegmentFromSelector
|
|
mov wo [si.AltQOutAddr+2], ax
|
|
InitCom57:
|
|
ENDIF
|
|
mov ax, __WinFlags ;In Standard mode, the PIC IRQ
|
|
test al, WF_STANDARD ; priorities get rotated to favor
|
|
jz InitCom59 ; the comm ports.
|
|
|
|
call Rotate_PIC
|
|
|
|
; *** Interrupt handler set : jump here if handler is already installed ***
|
|
;
|
|
InitCom59:
|
|
pop bx
|
|
pop cx
|
|
pop di
|
|
pop es
|
|
|
|
ifdef NEC_98
|
|
cmp es:[bx.DCB_Id],ID_Com1 ;Is this ID Com1 ? (ins 92.08.xx)|
|
|
jne InitCom60 ;No, jump InitCom60 (ins 92.08.xx)|
|
|
mov dx,MaskFFPort[si] ;F/F Port Mask Bit (ins 92.08.xx)|
|
|
in al,dx ; (ins 92.08.xx)|
|
|
mov MaskFFShadow[si],al ;Save Old Mask bit (ins 92.08.xx)|
|
|
; |
|
|
InitCom60: ; |
|
|
mov al,MaskFFShadow[si] ;Get 8251 INT Mask bit (ins 92.08.xx)|
|
|
and al,not(Enable_TxR or Enable_TxE or Enable_RxR) ;(ins 92.08.xx)|
|
|
mov dx,MaskFFPort[si] ;(ins 92.08.xx)|
|
|
out dx,al ;(ins 92.08.xx)|
|
|
mov MaskFFShadow[si],al ;Sasve Mask bit ;(ins 92.08.xx)|
|
|
call FlagNotActive ;(ins 92.08.xx)|
|
|
else ; NEC_98
|
|
InitCom60:
|
|
mov dx,[si.Port] ;Set comm card address
|
|
xor ax,ax ;Need a zero
|
|
inc dx ;--> Interrupt Enable Register
|
|
.errnz ACE_IER-ACE_RBR-1
|
|
out dx,al ;Turn off interrupts
|
|
call FlagNotActive
|
|
add dl,ACE_MCR-ACE_IER ;--> Modem Control Register
|
|
in al,dx
|
|
and al,ACE_DTR+ACE_RTS ;Leave DTR, RTS high if already so
|
|
iodelay ; but tri-state IRQ line
|
|
out dx,al
|
|
endif ; NEC_98
|
|
|
|
InitCom70:
|
|
push es ;Zero queue counts and indexes
|
|
|
|
push ds
|
|
pop es
|
|
assumes es,Data
|
|
|
|
lea di,QInCount[si]
|
|
mov cx,(EFlags-QInCount)/2
|
|
.errnz (EFlags-QInCount) AND 1
|
|
xor ax,ax
|
|
cld
|
|
rep stosw
|
|
|
|
.errnz QInGet-QInCount-2
|
|
.errnz QInPut-QInGet-2
|
|
.errnz QOutCount-QInPut-2
|
|
.errnz QOutGet-QOutCount-2
|
|
.errnz QOutPut-QOutGet-2
|
|
.errnz EFlags-QOutPut-2 ;First non-queue item
|
|
|
|
pop es
|
|
assumes es,nothing
|
|
|
|
mov HSFlag[si],al ;Show no handshakes yet
|
|
mov MiscFlags[si],al ;Show not discarding
|
|
mov EvtWord[si],ax ;Show no events
|
|
mov [si.VCDflags], al
|
|
|
|
mov [si.SendTrigger], ax
|
|
dec ax
|
|
mov [si.RecvTrigger], ax
|
|
|
|
;Call $SETCOM to perform further hardware initialization.
|
|
|
|
InitCom80:
|
|
ifndef NEC_98
|
|
sub dl,ACE_MCR-ACE_FCR ; dx -> FCR
|
|
in al, dx
|
|
iodelay
|
|
test al, ACE_FIFO_E2 ;Q: FIFO already on?
|
|
jz short @F ; N:
|
|
or EFlags[si], fFIFOpre ; Y: flag it
|
|
@@:
|
|
endif ; NEC_98
|
|
|
|
; needs si, di, and es to be saved from the beginning of inicom
|
|
call $SETCOM ;Set up Comm Device
|
|
jnz short InitCom110 ;jump if failed
|
|
|
|
call UnmaskIRQ
|
|
and EFlags[si], fEFlagsMask ;Clear internal state
|
|
|
|
InitCom90:
|
|
xor ax,ax ;Return AX = 0 to show success
|
|
mov ComErr[si],ax ;Get rid of any bogus init error
|
|
|
|
InitCom100:
|
|
ifdef NEC_98
|
|
FSTI ; (ins 94.05.17)
|
|
endif ; NEC_98
|
|
pop di
|
|
pop si
|
|
ret
|
|
|
|
;
|
|
; jump to here, if call to $SETCOM failed
|
|
;
|
|
; DANGER! *** Call into middle of Terminate to clean things up *** DANGER!
|
|
;
|
|
InitCom110:
|
|
push ax ;Failure, save error code
|
|
call Terminate45 ;Restore port address, int vec
|
|
pop ax ;Restore error code and exit
|
|
jmp InitCom100
|
|
|
|
$INICOM endp
|
|
page
|
|
|
|
;----------------------------Public Routine-----------------------------;
|
|
;
|
|
; $TRMCOM - Terminate Communications Channel
|
|
;
|
|
; Wait for any outbound data to be transmitted, drop the hardware
|
|
; handshaking lines, and disable interrupts. If the output queue
|
|
; contained data when it was closed, an error will be returned
|
|
;
|
|
; LPT devices have it easy. They just need to restore the I/O port
|
|
; address.
|
|
;
|
|
; Entry:
|
|
; AH = Device ID
|
|
; Returns:
|
|
; AX = 0
|
|
; Error Returns:
|
|
; AX = 8000h if invalid device ID
|
|
; AX = -2 if output queue timeout occured
|
|
; Registers Destroyed:
|
|
; AX,BX,CX,DX,ES,FLAGS
|
|
; History:
|
|
;-----------------------------------------------------------------------;
|
|
|
|
;------------------------------Pseudo-Code------------------------------;
|
|
; {
|
|
; }
|
|
;-----------------------------------------------------------------------;
|
|
|
|
assumes ds,Data
|
|
assumes es,nothing
|
|
|
|
public $TRMCOM
|
|
$TRMCOM proc near
|
|
|
|
push si
|
|
push di
|
|
xor cx,cx ;Show no error if LPT port
|
|
call GetDEB
|
|
jc TermCom60 ;ID is invalid, return error
|
|
js TermCom30 ;Port is a LPT port
|
|
|
|
push ax ;Save port id
|
|
or MiscFlags[si],Discard ;Show discarding serial data
|
|
mov ComErr[si],cx ;Clear error flags
|
|
mov QInCount[si],cx ;Show no chars in input queue
|
|
call $RECCOM ;Send XON if needed
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; We have to wait for the output queue to empty. To do this,
|
|
; a timer will be created. If no character has been transmitted
|
|
; when the timeout occurs, then an error will be indicated and
|
|
; the port closed anyway. If the timer cannot be created, then
|
|
; just loop until the queue empties, which will be better than
|
|
; discarding charatcers if there are any
|
|
;-----------------------------------------------------------------------;
|
|
|
|
test [si.HSFlag], HHSAlwaysDown ; Q: handshaking ever up?
|
|
jnz TermCom17 ; N: skip wait loop
|
|
|
|
TermCom10:
|
|
mov cx,QOutCount[si] ;Get current queue count
|
|
jcxz TermCom20 ;No characters in queue
|
|
|
|
cCall GetSystemMsecCount
|
|
mov di, ax
|
|
|
|
TermCom15:
|
|
cmp QOutCount[si],cx ;Queue count change?
|
|
jne TermCom10 ; Yes, restart timeout
|
|
|
|
cCall GetSystemMsecCount
|
|
sub ax, di
|
|
ifdef NEC_98
|
|
cmp ax, 20000 ;Q: Timeout reached? (ins 94.06.08)
|
|
else ; NEC_98
|
|
cmp ax, Timeout * 1000 ;Q: Timeout reached?
|
|
endif ; NEC_98
|
|
jb TermCom15 ; No, keep waiting
|
|
|
|
IFDEF DEBUG_TimeOut
|
|
.286
|
|
pusha
|
|
lea cx, szSendTO
|
|
call Contention_Dlg
|
|
popa
|
|
jz TermCom10
|
|
.8086
|
|
ENDIF
|
|
|
|
TermCom17:
|
|
mov cx, TimeoutError ; Yes, show timeout error
|
|
|
|
TermCom20:
|
|
pop ax ;Restore cid
|
|
ifdef NEC_98
|
|
test [si.AOBA_flag],fFIFO_Mode ; (ins 94.04.11)|
|
|
jnz @F ;Now, FIFO mode (ins 94.04.11)|
|
|
TermCom25: ; (ins 92.08.xx)|
|
|
mov dx,ReadSigPort[si] ;CI/CS/CD Signal Port (ins 94.06.08)|
|
|
in al,dx ;in al,33h (ins 94.06.08)|
|
|
test al,40h ;CS Off ? (ins 94.06.08)|
|
|
jnz @f ;y: jump (ins 94.06.08)|
|
|
; (ins 94.06.08)|
|
|
push ax ; (ins 92.08.xx)|
|
|
push dx ; (ins 92.08.xx)|
|
|
mov dx,StatusPort[si] ;8251 status port read (ins 92.08.xx)|
|
|
in al,dx ; (ins 92.08.xx)|
|
|
; (ins 92.08.xx)|
|
|
test al,(TX_EMP) ;tx_empty check (ins 92.08.xx)|
|
|
pop dx ; (ins 92.08.xx)|
|
|
pop ax ; (ins 92.08.xx)|
|
|
jz TermCom25 ; (ins 92.08.xx)|
|
|
@@: ; (ins 94.04.11)|
|
|
;-------------------------------------------------------------- (ins 92.08.xx)|
|
|
; Close process of system timer (ins 92.08.xx)|
|
|
;-------------------------------------------------------------- (ins 92.08.xx)|
|
|
push ax ; (ins 92.08.xx)|
|
|
mov ax,hTimer[si] ; Check hTimer (ins 94.04.12)|
|
|
or ax,ax ; Is there TimerProcess?(ins 92.08.xx)|
|
|
jz NoKillTimer ; Y:(Nothing) (ins 92.08.xx)|
|
|
cCall KillSystemTimer,<ax> ; Kill system timer (ins 92.08.xx)|
|
|
mov hTimer[si] ,NULL ; Clear hTimer (ins 94.04.12)|
|
|
NoKillTimer: ; (ins 92.08.xx)|
|
|
pop ax ; (ins 92.08.xx)|
|
|
endif ; NEC_98
|
|
|
|
TermCom30:
|
|
mov dx,Port[si] ;Get port base address
|
|
call Terminate ;The real work is done here
|
|
ifndef NEC_98
|
|
mov ax,cx ;Set return code
|
|
endif ; NEC_98
|
|
|
|
TermCom60:
|
|
pop di
|
|
pop si
|
|
ret
|
|
|
|
$TRMCOM endp
|
|
page
|
|
|
|
;----------------------------Private-Routine----------------------------;
|
|
;
|
|
; Terminate - Terminate Device
|
|
;
|
|
; Restore the port I/O address and make sure that interrupts are off
|
|
;
|
|
; Entry:
|
|
; AH = Device Id.
|
|
; DX = Device I/O port address.
|
|
; SI --> DEB
|
|
; Returns:
|
|
; AX = 0
|
|
; Error Returns:
|
|
; AX = -1
|
|
; Registers Destroyed:
|
|
; AX,BX,DX,FLAGS
|
|
; History:
|
|
;-----------------------------------------------------------------------;
|
|
|
|
assumes ds,Data
|
|
assumes es,nothing
|
|
|
|
public Terminate ;Public for debugging
|
|
Terminate proc near
|
|
|
|
or ah,ah ;LPT port?
|
|
jns Terminate10 ; No, process COM port
|
|
.errnz LPTx-10000000b
|
|
|
|
Terminate5:
|
|
ifdef NEC_98
|
|
call WPS386 ; WPS386 I/O trap ON (ins 940224)
|
|
call INT_1AH_Close ; (ins 931027)
|
|
call WPS386 ; WPS386 I/O trap OFF (ins 940224)
|
|
endif ; NEC_98
|
|
call ReleasePort386 ; give port back to 386...
|
|
jmp Terminate50 ;That's all
|
|
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; It is a com port!
|
|
;
|
|
; We delay for a bit while the last character finishes transmitting
|
|
; Then we drop DTR and RTS, and disable the interrupt generation at
|
|
; the 8250. Even if fRTSDisable or fDTRDisable is set, those lines
|
|
; will be dropped when the port is closed.
|
|
;-----------------------------------------------------------------------;
|
|
;
|
|
; When the OUT2 bit is reset to 0 to disable interrupts, many ports
|
|
; generate an interrupt which can not be identified, because the the
|
|
; interrupt ID register will not be set. To work around this hardware
|
|
; problem we first mask the IRQ, then set the port into loopback mode
|
|
; and output a NULL to generate a receive interrupt request. Then we
|
|
; reset OUT2 and unmask the IRQ. This will cause the interrupt to occur
|
|
; and the interrupt handler will be able to correctly identify the
|
|
; interrupt as coming from the com port.
|
|
|
|
Terminate10:
|
|
ifdef NEC_98
|
|
push ax ; (ins 94.04.18) |
|
|
call FlagNotActive ; don't need to check (ins 94.04.11) |
|
|
; on timer ticks |
|
|
pop ax ; (ins 94.04.18) |
|
|
FCLI ; (ins 94.06.06) |
|
|
test [si.AOBA_flag],fFIFO_Mode ; (ins 94.04.11) |
|
|
jz AOBA_Term35 ;Now, 8251 mode (ins 94.04.11) |
|
|
; |
|
|
;;;;;;; begin code : process for AOBA---------------------------(ins 94.04.11) |
|
|
push ax ; |
|
|
push dx ; |
|
|
; |
|
|
mov dx,Port[si] ; (ins 94.06.08)|
|
|
add dx,ACE_MSR ;Modem Status = 134h (ins 94.06.08)|
|
|
in al,dx ;Wait until xmit is empty (ins 94.06.08)|
|
|
out 05fh ,al ; (ins 94.06.08)|
|
|
test al,ACE_CTS ; CS = Off ? (ins 94.06.08)|
|
|
jz AOBA_Term30 ; Y; jump (ins 94.06.08)|
|
|
; |
|
|
mov dx,Port[si] ; |
|
|
add dx,ACE_LSR ;Line Status register |
|
|
public AOBA_Term20 ; |
|
|
AOBA_Term20: ; |
|
|
in al,dx ;Wait until xmit is empty |
|
|
out 05fh ,al ; (ins 94.06.07) |
|
|
test al,ACE_THRE ; Send FIFO and CCU Send Reg All Empty |
|
|
jz AOBA_Term20 ;Not empty yet |
|
|
; |
|
|
public AOBA_Term30 ; (ins 94.06.08)|
|
|
AOBA_Term30: ; (ins 94.06.08)|
|
|
call Set8251mode ;Change to 8251 mode |
|
|
pop dx ; |
|
|
pop ax ; |
|
|
;;;;;;; end code : process for AOBA-----------------------------(ins 94.04.11) |
|
|
; |
|
|
public AOBA_Term35 ; |
|
|
AOBA_Term35: ; (ins 94.04.11) |
|
|
; |
|
|
;;;;;;;;;;;;;;;;8251 F/F interrupt mask ; (ins 92.08.xx) |
|
|
; (ins 92.08.xx) |
|
|
Terminate11: ; (ins 92.08.xx) |
|
|
cmp ah,ID_Com1 ; com1 ? (ins 92.08.xx) |
|
|
jne Terminate12 ; No: (ins 92.08.xx) |
|
|
mov dx,MaskFFPort[si] ; (ins 92.08.xx) |
|
|
in al,dx ; Read Port C data (ins 92.08.xx) |
|
|
mov MaskFFShadow[si],al ;Int mask data (ins 92.08.xx) |
|
|
; Port C save (ins 92.08.xx) |
|
|
Terminate12: ; (ins 92.08.xx) |
|
|
mov dx,MaskFFPort[si] ; (ins 92.08.xx) |
|
|
mov al,MaskFFShadow[si] ; (ins 92.08.xx) |
|
|
and al,not(MSK_TXE+MSK_TXR+MSK_RXR);8251 F/F INT mask (ins 92.08.xx) |
|
|
mov MaskFFShadow[si],al ;8251 Mask data save (ins 92.08.xx) |
|
|
out dx,al ;Send mask data (ins 92.08.xx) |
|
|
; (ins 92.08.xx) |
|
|
;;;;;;;;;;;;;;;;;;;;8259 interrupt mask on IMR ; (ins 92.08.xx) |
|
|
; (ins 92.08.xx) |
|
|
push bx ;Save bx (ins 92.08.xx) |
|
|
push dx ;Save dx (ins 92.08.xx) |
|
|
; (ins 92.08.xx) |
|
|
mov dx,IMRPort[si] ;INT Mask Register Port (ins 92.08.xx) |
|
|
mov bl,Mask8259[si] ;8259 Mask bit (ins 92.08.xx) |
|
|
in al,dx ;AL=IMRPort (ins 92.08.xx) |
|
|
NEWIODELAY 1 ;<OUT 5F,AL> (ins 92.11.14) |
|
|
mov cx,ax ; (ins 92.08.xx) |
|
|
or al,bl ;Mask RS232c (ins 92.08.xx) |
|
|
out dx,al ; (ins 92.08.xx) |
|
|
NEWIODELAY 1 ;<OUT 5F,AL> (ins 92.11.14) |
|
|
pop dx ; (ins 92.08.xx) |
|
|
pop bx ; (ins 92.08.xx) |
|
|
; (ins 92.08.xx) |
|
|
;--------- internal reset ------------------------------- (ins 92.08.xx) |
|
|
; (ins 92.08.xx) |
|
|
Terminate15: ; (ins 92.08.xx) |
|
|
push cx ; (ins 92.08.xx) |
|
|
push ax ; (ins 92.08.xx) |
|
|
push dx ; (ins 92.08.xx) |
|
|
mov cx,3 ; (ins 92.08.xx) |
|
|
xor ax,ax ; (ins 92.08.xx) |
|
|
; (ins 92.08.xx) |
|
|
Terminate20: ; (ins 92.08.xx) |
|
|
mov dx,CommandPort[si] ; (ins 92.08.xx) |
|
|
out dx,al ; (ins 92.08.xx) |
|
|
NEWIODELAY 20 ;<OUT 5F,AL> (ins 92.11.14) |
|
|
loop Terminate20 ; (ins 92.08.xx) |
|
|
mov al,40h ; (ins 92.08.xx) |
|
|
out dx,al ; (ins 92.08.xx) |
|
|
pop dx ; (ins 92.08.xx) |
|
|
pop ax ; (ins 92.08.xx) |
|
|
pop cx ; (ins 92.08.xx) |
|
|
; (ins 92.08.xx) |
|
|
Terminate30: ; (ins 94.04.15) |
|
|
else ; NEC_98
|
|
inc dx ;Disable chip interrupts
|
|
.errnz ACE_IER-ACE_RBR-1
|
|
mov al, ACE_ERBFI ; except receive
|
|
out dx,al
|
|
call FlagNotActive ; don't need to check for postmessage
|
|
; on timer ticks
|
|
add dl,ACE_LSR-ACE_IER ;--> line status register
|
|
iodelay
|
|
|
|
Terminate20:
|
|
in al,dx ;Wait until xmit is empty
|
|
and al,ACE_THRE+ACE_TSRE
|
|
cmp al,ACE_THRE+ACE_TSRE
|
|
jne Terminate20 ;Not empty yet
|
|
|
|
Terminate30:
|
|
xor al, al
|
|
test EFlags[si], fFIFOpre ;Q: leave FIFO enabled?
|
|
jz short @F ; N:
|
|
mov al, ACE_TRIG14 OR ACE_EFIFO OR ACE_CRFIFO OR ACE_CTFIFO
|
|
@@:
|
|
sub dl, ACE_LSR-ACE_FCR
|
|
out dx, al
|
|
iodelay
|
|
call MaskIRQ
|
|
add dl, ACE_MCR-ACE_FCR ;--> Modem Control Register
|
|
in al,dx
|
|
iodelay
|
|
mov ah, al
|
|
or al,ACE_LOOP ; turn on loopback
|
|
out dx, al
|
|
iodelay
|
|
sub dl, ACE_MCR-ACE_THR
|
|
xor al, al
|
|
out dx, al ; output a NULL to generate an int
|
|
iodelay
|
|
add dl, ACE_LSR-ACE_THR
|
|
Terminate35:
|
|
in al,dx ;Wait until xmit is empty
|
|
and al,ACE_THRE+ACE_TSRE
|
|
cmp al,ACE_THRE+ACE_TSRE
|
|
jne Terminate35 ;Not empty yet
|
|
mov al, ah
|
|
dec dl ; now clear OUT2 and loopback
|
|
.errnz ACE_LSR-ACE_MCR-1
|
|
and al,ACE_DTR+ACE_RTS ;Leave DTR, RTS high if already so
|
|
out dx,al ; but tri-state IRQ line
|
|
|
|
call UnmaskIRQ ; this will cause the receive int
|
|
; to occur and be processed
|
|
sub dl, ACE_MCR-ACE_IER ; clear the receive int enable
|
|
xor al, al
|
|
out dx, al
|
|
dec dx
|
|
.errnz ACE_IER-ACE_RBR-1
|
|
call MaskIRQ
|
|
endif ; NEC_98
|
|
|
|
;******* DANGER! ***** NOTICE! ***** DANGER! ***** WARNING! ***** NOTICE!
|
|
;
|
|
; Terminate45 is a secondary entrypoint into this routine--it's called
|
|
; by the initialization code when that code is unable to properly init
|
|
; a com port and needs to clean-up the mess it's made.
|
|
;
|
|
;******* DANGER! ***** NOTICE! ***** DANGER! ***** WARNING! ***** NOTICE!
|
|
|
|
Terminate45:
|
|
push cx ;Save original cx
|
|
push bx ;Save original bx
|
|
|
|
ifdef NEC_98
|
|
dec IntVecIntCount[si] ;USE Rotate_Pic() (ins 92.08.xx)
|
|
endif ; NEC_98
|
|
cmp [fVPICD], 0 ;Q: VPICD bimodel services available?
|
|
jl short @F ; N:
|
|
mov ax, ds ; Y: use them
|
|
mov es, ax
|
|
lea di, [si+SIZE ComDEB]
|
|
mov ax, VPICD_Remove_Handler
|
|
call [lpfnVPICD]
|
|
@@:
|
|
|
|
mov di, [si.IRQhook]
|
|
dec [di.HookCnt] ;Q: last port using IRQ?
|
|
jne short Terminate495 ; N: unmask IRQ again
|
|
mov al, 0FFh
|
|
xchg al, [di.VecN] ;Interrupt vector number
|
|
cmp al, 0FFh ;Q: IRQ vector hooked?
|
|
je short Terminate49 ; no...
|
|
|
|
IFDEF No_DOSX_Bimodal_Services
|
|
cmp [Using_DPMI], 0
|
|
jz short term_no_dpmi
|
|
|
|
;
|
|
; unhook RM vector thru DPMI for standard mode
|
|
;
|
|
push ax
|
|
mov ax, Int31_Get_Version SHL 8
|
|
int 31h
|
|
mov bl, [si.IntVecNum]
|
|
mov bh, bl
|
|
add bl, dh ; assume master PIC
|
|
ifndef NEC_98
|
|
sub bh, 8 ;Q: IRQ in master?
|
|
jb @f ; Y: add master's base vec
|
|
add bh, dl ; N: add slave's base vec
|
|
mov bl, bh
|
|
endif ; NEC_98
|
|
@@:
|
|
mov dx, wo [di.RM_OldIntVec][0]
|
|
mov cx, wo [di.RM_OldIntVec][2]
|
|
mov ax, Set_RM_IntVector
|
|
int 31h
|
|
pop ax
|
|
term_no_dpmi:
|
|
ENDIF
|
|
mov dx, __WinFlags ;In Standard mode the PIC interrupt
|
|
test dl, WF_STANDARD ; priorities are changed to favor
|
|
jz Terminate48 ; the comm ports.
|
|
|
|
call Rotate_PIC ;This port no longer needs priority
|
|
|
|
Terminate48:
|
|
; *** reset int vector to it's previous state
|
|
assumes ds,nothing
|
|
push ds ;Save original DS [rkh] ...
|
|
lds dx, [di.OldIntVec]
|
|
mov ah, 25h ;DOS Set Vector Function
|
|
int 21h ;Set the DOS vector
|
|
pop ds ;Original DS
|
|
assumes ds,data
|
|
|
|
; *** interrupt vectors have been reset if needed at this point ***
|
|
;
|
|
Terminate49:
|
|
mov cl, [di.OldMask]
|
|
|
|
; Set the 8259 interrupt mask bit for this IRQ. Leave interrupts enabled
|
|
; if they were already enabled when the comm port was initialized by us.
|
|
|
|
or cl, cl
|
|
jnz @f
|
|
Terminate495:
|
|
call UnmaskIRQ
|
|
@@:
|
|
|
|
xor ax, ax
|
|
xchg ax, [si.NextDEB]
|
|
cmp [di.First_DEB], si ;Q: DEB first for IRQ hook?
|
|
je short Terminate46 ; Y:
|
|
mov bx, [di.First_DEB] ; N: get first
|
|
Terminate453:
|
|
cmp [bx.NextDEB], si ;Q: does this DEB point to one terminating?
|
|
je Terminate455 ; Y:
|
|
mov bx, [bx.NextDEB] ; N: get next DEB
|
|
jmp Terminate453
|
|
Terminate455:
|
|
mov [bx.NextDEB], ax ; link previous DEB to NextDEB
|
|
jmp short Terminate47
|
|
Terminate46:
|
|
mov [di.First_DEB], ax ; point IRQ hook at NextDEB
|
|
Terminate47:
|
|
ifdef NEC_98
|
|
test EFlags[si], fNoFIFO ; (ins 94.06.06) |
|
|
jnz @F ; (ins 94.06.06) |
|
|
mov al,preFCRShadow ; (ins 94.06.06) |
|
|
and al,00000110b ;Clear FIFO buffer (ins 94.06.06) |
|
|
mov dx,Port[si] ; (ins 94.06.06) |
|
|
add dx,ACE_FCR ;FIFO control register (ins 94.06.06) |
|
|
out dx, al ; (ins 94.06.06) |
|
|
@@: ; (ins 94.04.13) |
|
|
FSTI ; (ins 94.06.06) |
|
|
endif ; NEC_98
|
|
pop bx ;Original BX
|
|
|
|
call ReleaseCOMport386 ; give port back to 386...
|
|
|
|
pop cx ;Original CX
|
|
|
|
Terminate50: ;Also called from $INICOM !
|
|
xor ax,ax ;Indicate no error
|
|
ret ;Port is closed and deallocated
|
|
|
|
Terminate endp
|
|
page
|
|
|
|
;----------------------------Public Routine-----------------------------;
|
|
;
|
|
; $ENANOTIFY - Enable Event Notification
|
|
;
|
|
; Entry:
|
|
; AH = Device ID
|
|
; BX = Window handle for PostMessage
|
|
; CX = Receive threshold
|
|
; DX = Transmit threshold
|
|
; Returns:
|
|
; AX = 1, if no errors occured
|
|
; Error Returns:
|
|
; AX = 0
|
|
; Registers Preserved:
|
|
; BX,SI,DI,DS
|
|
; Registers Destroyed:
|
|
; AX,CX,DX,ES,FLAGS
|
|
; History:
|
|
;-----------------------------------------------------------------------;
|
|
|
|
;------------------------------Pseudo-Code------------------------------;
|
|
; {
|
|
; }
|
|
;-----------------------------------------------------------------------;
|
|
|
|
assumes ds,Data
|
|
assumes es,nothing
|
|
|
|
public $ENANOTIFY
|
|
$ENANOTIFY proc near
|
|
push si
|
|
call GetDEB
|
|
mov ax, 0
|
|
jc scb_exit
|
|
|
|
mov ax, cx
|
|
inc ax
|
|
jz short scb_recv_ok
|
|
cmp cx, [si.QInSize] ;Q: receive threshold reasonable?
|
|
jb short scb_recv_ok ; Y:
|
|
%OUT should we return an error, if thresholds invalid?
|
|
mov cx, [si.QInSize] ; N:
|
|
sub cx, 10
|
|
scb_recv_ok:
|
|
inc dx
|
|
jz short scb_send_ok
|
|
dec dx
|
|
cmp dx, [si.QOutSize] ;Q: receive threshold reasonable?
|
|
jb short scb_send_ok ; Y:
|
|
mov dx, [si.QOutSize] ; N:
|
|
sub dx, 10
|
|
scb_send_ok:
|
|
mov [si.NotifyHandle], bx
|
|
mov [si.NotifyFlagsHI], CN_Notify
|
|
or bx, bx ;Q: null callback?
|
|
jnz scb_save_thresholds ; N: save thresholds
|
|
or cx, -1 ; Y: zero thresholds
|
|
xor dx, dx
|
|
mov [si.NotifyFlagsHI], 0
|
|
scb_save_thresholds:
|
|
mov [si.RecvTrigger], cx
|
|
mov [si.SendTrigger], dx
|
|
or [si.NotifyFlagsHI], CN_TRANSMIT ; we don't want to send
|
|
; a transmit trigger notification until
|
|
; the transmit buffer has been filled
|
|
; above the trigger level and then
|
|
; emptied below it again!
|
|
|
|
cmp wo lpPostMessage[2], 0 ;Q: gotten addr of PostMessage yet?
|
|
jne short scb_good ; Y:
|
|
push ds ; N: get module handle of USER
|
|
lea ax, szUser
|
|
push ax
|
|
cCall GetModuleHandle
|
|
|
|
push ax ; module handle
|
|
mov ax, POSTMESSAGE
|
|
cwd
|
|
push dx
|
|
push ax
|
|
cCall GetProcAddress
|
|
mov wo lpPostMessage[0], ax ; save received proc address
|
|
mov wo lpPostMessage[2], dx
|
|
|
|
scb_good:
|
|
mov ax, 1
|
|
|
|
scb_exit:
|
|
pop si
|
|
ret
|
|
$ENANOTIFY endp
|
|
page
|
|
|
|
;----------------------------Public Routine-----------------------------;
|
|
;
|
|
; $SETQUE - Set up Queue Pointers
|
|
;
|
|
; Sets pointers to Receive and Transmit Queues, as provided by the
|
|
; caller, and initializes those queues to be empty.
|
|
;
|
|
; Queues must be set before $INICOM is called!
|
|
;
|
|
; Entry:
|
|
; AH = Device ID
|
|
; ES:BX --> Queue Definition Block
|
|
; Returns:
|
|
; AX = 0 if no errors occured
|
|
; Error Returns:
|
|
; AX = error code
|
|
; Registers Preserved:
|
|
; BX,DX,SI,DI,DS
|
|
; Registers Destroyed:
|
|
; AX,CX,ES,FLAGS
|
|
; History:
|
|
;-----------------------------------------------------------------------;
|
|
|
|
;------------------------------Pseudo-Code------------------------------;
|
|
; {
|
|
; }
|
|
;-----------------------------------------------------------------------;
|
|
|
|
assumes ds,Data
|
|
assumes es,nothing
|
|
|
|
public $SETQUE
|
|
$SETQUE proc near
|
|
|
|
push si ;These will be used
|
|
push di
|
|
call GetDEB ;Get DEB
|
|
jc SetQue10 ;Invalid, ignore the call
|
|
js SetQue10 ;Ignore call for LPT ports
|
|
push ds ;Set ds:si --> QDB
|
|
push es ;Set es:di --> to ComDCB.QInAddr
|
|
pop ds
|
|
assumes ds,nothing
|
|
pop es
|
|
assumes es,Data
|
|
lea di,QInAddr[si]
|
|
mov si,bx
|
|
cld
|
|
FCLI ;No one else can play with queues
|
|
movsw ; QInAddr = QueueRxAddr
|
|
movsw
|
|
.errnz QueueRxAddr
|
|
sub si, 4 ; AltQInAddr = QueueRxAddr
|
|
mov cx, 5 ; QInSize = QueueRxSize
|
|
rep movsw ; QOutAddr = QueueTxAddr
|
|
sub si, 4
|
|
mov cx, 3 ; AltQOutAddr = QueueTxAddr
|
|
rep movsw ; QOutSize = QueueTxSize
|
|
|
|
xor ax,ax ;Will do some zero filling
|
|
mov cl,(EFlags-QInCount)/2
|
|
.errnz (EFlags-QInCount) AND 0FE01h
|
|
rep stosw
|
|
FSTI
|
|
push es ;Restore the data segment
|
|
pop ds
|
|
assumes ds,Data
|
|
assumes es,nothing
|
|
|
|
SetQue10:
|
|
pop di ;Restore saved registers
|
|
pop si
|
|
ret
|
|
|
|
; The above code made a few assumptions about how memory
|
|
; was allocated within the structures:
|
|
|
|
.errnz AltQInAddr-QInAddr-4
|
|
.errnz (QueueRxSize-QueueRxAddr)-(QInSize-AltQInAddr)
|
|
.errnz (QueueTxAddr-QueueRxSize)-(QOutAddr-QInSize)
|
|
.errnz AltQOutAddr-QOutAddr-4
|
|
.errnz (QueueTxSize-QueueTxAddr)-(QOutSize-AltQOutAddr)
|
|
|
|
.errnz QueueRxSize-QueueRxAddr-4
|
|
.errnz QueueTxAddr-QueueRxSize-2
|
|
.errnz QueueTxSize-QueueTxAddr-4
|
|
|
|
.errnz QInSize-AltQInAddr-4
|
|
.errnz QOutAddr-QInSize-2
|
|
.errnz QOutSize-AltQOutAddr-4
|
|
|
|
.errnz QInCount-QOutSize-2
|
|
.errnz QInGet-QInCount-2
|
|
.errnz QInPut-QInGet-2
|
|
.errnz QOutCount-QInPut-2
|
|
.errnz QOutGet-QOutCount-2
|
|
.errnz QOutPut-QOutGet-2
|
|
.errnz EFlags-QOutPut-2 ;First non-queue item
|
|
|
|
$SETQUE endp
|
|
page
|
|
|
|
;----------------------------Public Routine-----------------------------;
|
|
;
|
|
; $SETCOM - Set Communications parameters
|
|
;
|
|
; Re-initalizes the requested port if present, and sets up the
|
|
; port with the given attributes when they are valid.
|
|
;
|
|
; For LPT ports, just copies whatever is given since it's ignored
|
|
; anyway.
|
|
;
|
|
; Entry:
|
|
; ES:BX --> DCB with all fields set.
|
|
; Returns:
|
|
; 'Z' Set if no errors occured
|
|
; AX = 0
|
|
; Error Returns:
|
|
; 'Z' clear if errors occured
|
|
; AX = initialization error code.
|
|
; Registers Destroyed:
|
|
; AX,BX,CX,DX,ES,FLAGS
|
|
; History:
|
|
;-----------------------------------------------------------------------;
|
|
|
|
;------------------------------Pseudo-Code------------------------------;
|
|
; {
|
|
; }
|
|
;-----------------------------------------------------------------------;
|
|
|
|
assumes ds,Data
|
|
assumes es,nothing
|
|
|
|
public $SETCOM
|
|
$SETCOM proc near
|
|
|
|
cld
|
|
push si
|
|
push di
|
|
mov ah,es:[bx.DCB_Id] ;Get device i.d.
|
|
call GetDEB ;Get DEB pointer in SI
|
|
mov ax,IE_BadID ;Assume unknown device
|
|
jc SetCom10 ;Invalid device, return error
|
|
jns SetCom20 ;COM port
|
|
call SetCom100 ;Copy the DCB
|
|
|
|
SetCom5:
|
|
xor ax,ax ;Show no error
|
|
|
|
SetCom10:
|
|
or ax,ax ;Set/clear 'Z'
|
|
pop di ; and exit
|
|
pop si
|
|
ret
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; Have a comm device, check all the serial parameters to make
|
|
; sure they are correct before moving the new DCB into our space
|
|
; and changing the ACE parameters.
|
|
;-----------------------------------------------------------------------;
|
|
|
|
SetCom20:
|
|
ifdef NEC_98
|
|
FCLI ; (ins 94.05.17) |
|
|
test EFlags[si], fNoFIFO ; (ins 94.04.13) |
|
|
jnz @F ;no AOBA on board (ins 94.04.13) |
|
|
call Set8251mode ;Change to 8251 mode (ins 94.04.12) |
|
|
@@: ; (ins 94.04.13) |
|
|
FSTI ; (ins 94.06.11)
|
|
endif ; NEC_98
|
|
call SetCom300 ;Baud rate valid?
|
|
jcxz SetCom10 ; No, return error
|
|
call SetCom400 ;Byte size/parity/stop bits correct?
|
|
jc SetCom10 ; No, return error
|
|
|
|
; The parameters seem correct. Copy the DCB into our space and
|
|
; initialize the ACE with the new parameters
|
|
|
|
ifdef NEC_98
|
|
mov dx,MaskFFPort[si] ;Disable INT from 8251 (ins 92.08.xx)|
|
|
cmp es:[bx.DCB_Id],ID_COM1 ; com1 ? (ins 92.08.xx)|
|
|
jne SetCom2001 ; no, Jump SetCom2001 (ins 92.08.xx)|
|
|
in al,dx ; Read port_c (ins 92.08.xx)|
|
|
jmp short SetCom2002 ; (ins 92.08.xx)|
|
|
; (ins 92.08.xx)|
|
|
SetCom2001: ; (ins 92.08.xx)|
|
|
mov al,MaskFFShadow[si] ;read 8251 mask data (ins 92.08.xx)|
|
|
; (ins 92.08.xx)|
|
|
SetCom2002: ; (ins 92.08.xx)|
|
|
and al,not(MSK_TXE+MSK_TXR+MSK_RXR) ; (ins 92.08.xx)|
|
|
mov MaskFFShadow[si],al ;write 8251 mask data (ins 92.08.xx)|
|
|
out dx,al ; (ins 92.08.xx)|
|
|
else ; NEC_98
|
|
mov dx,Port[si] ;Disable interrupts from the 8250
|
|
inc dx
|
|
.errnz ACE_IER-1
|
|
xor ax,ax
|
|
out dx,al
|
|
endif ; NEC_98
|
|
call FlagNotActive
|
|
|
|
call SetCom100 ;Copy the DCB
|
|
mov bx,si ;Set ES:BX --> DCB
|
|
ifdef NEC_98
|
|
FCLI ; (ins 94.06.11) |
|
|
endif ; NEC_98
|
|
call SetCom200 ;Get timeout masks
|
|
ifdef NEC_98
|
|
FSTI ; (ins 94.06.11)
|
|
endif ; NEC_98
|
|
xchg al,ah ;Want them in the correct registers
|
|
mov wo MSRMask[si],ax
|
|
.errnz MSRInfinite-MSRMask-1
|
|
|
|
call SetCom400 ;Get line control byte
|
|
push ax ; and save LCR value
|
|
ifndef NEC_98
|
|
inc dx ;--> LCR
|
|
inc dx
|
|
.errnz ACE_LCR-ACE_IER-2
|
|
or al,ACE_DLAB ;Want access to divisor latch
|
|
out dx,al
|
|
endif ; NEC_98
|
|
mov RxMask[si],ah ;Save Receive character mask
|
|
mov ax,di ;Get flags mask, error mask
|
|
and [si.DCB_Flags],ah ;Disable parity checking if no parity
|
|
mov ErrorMask[si],al ;Save line status error mask
|
|
|
|
call SetCom300 ;Get baud rate
|
|
ifdef NEC_98
|
|
;-----------------------------------------------------------------------------+
|
|
; timer set !! (ins 92.08.xx)|
|
|
;---------------------------------------------------------------(ins 92.08.xx)|
|
|
cmp es:[bx.DCB_Id],00h ;Comm1 ? (ins 92.08.xx)|
|
|
jne SetCom21 ; No (ins 92.08.xx)|
|
|
mov al,RS_TIME_MOD ;Mode = counter2,LBS->MSB,mode2 (ins 92.08.xx)|
|
|
out TIME_CTL,al ; (8253) (priscaler) (ins 92.08.xx)|
|
|
NEWIODELAY 2 ; <OUT 5F,AL> (ins 92.11.14)|
|
|
mov ax,cx ;divisor data set (LSB) (ins 92.08.xx)|
|
|
out TIMER2,al ; '=075H' (ins 92.08.xx)|
|
|
NEWIODELAY 2 ; <OUT 5F,AL> (ins 92.11.14)|
|
|
mov al,ah ;divisor data set (MSB) (ins 92.08.xx)|
|
|
out TIMER2,al ; (ins 92.08.xx)|
|
|
; (ins 92.08.xx)|
|
|
SetCom21: ; (ins 92.08.xx)|
|
|
mov dx,CommandPort[si] ;8251 mode Inst Addr (ins 92.08.xx)|
|
|
xor ax,ax ; (ins 92.08.xx)|
|
|
FCLI ; (ins 94.06.11)|
|
|
out dx,al ;null out (ins 92.08.xx)|
|
|
NEWIODELAY 20 ; <OUT 5F,AL> (ins 92.11.14)|
|
|
out dx,al ;null out (ins 92.08.xx)|
|
|
NEWIODELAY 20 ; <OUT 5F,AL> (ins 92.11.14)|
|
|
out dx,al ;null out (ins 92.08.xx)|
|
|
NEWIODELAY 20 ; <OUT 5F,AL> (ins 92.11.14)|
|
|
mov al,INTER_RESET ;reset (=40h) (ins 92.08.xx)|
|
|
out dx,al ; (ins 92.08.xx)|
|
|
pop ax ;Get Mode Inst (8251) (ins 92.08.xx)|
|
|
NEWIODELAY 20 ; <OUT 5F,AL> (ins 92.11.14)|
|
|
out dx,al ;mode set (ins 92.08.xx)|
|
|
FSTI ; (ins 94.06.11)|
|
|
;-------------------------------------------------------------------------- |
|
|
; relative DTR, RTS to FRTSDisable & FDTRDisable |
|
|
; for hardware handshake !? |
|
|
;-------------------------------------------------------------------------- |
|
|
;NOW,DCB_Flags is 8250's format !! |
|
|
mov al,[si.DCB_Flags] ;Align DTR/RTS disable flags for 8251 |
|
|
mov ah,al ; |
|
|
and al,FRTSDisable ;= 00000010b ______. |
|
|
and ah,FDTRDisable ;= 10000000b ______I___________. |
|
|
shl al,1 ; I I |
|
|
shl al,1 ; 8251 mode (7)(6)(5)(4)(3)(2)(1)(0) |
|
|
shl al,1 ; I I I I |
|
|
shl al,1 ; RTS ____I RX_E __I I I |
|
|
rol ah,1 ; DTR ______I TX_E |
|
|
rol ah,1 ; |
|
|
or al,ah ; |
|
|
mov cl,al ; |
|
|
or al,RX_E+TX_E ;add the tx.disable and rx.disable |
|
|
xor al,DTR+RTS ; Enable = "On" |
|
|
FCLI ; (ins 94.06.11)|
|
|
out dx,al ;NOW,DCB_Flags is 8251's format !! |
|
|
FSTI ; (ins 94.06.11)|
|
|
mov CommandShadow[si],al ;Save Command |
|
|
else ; NEC_98
|
|
sub dl,ACE_LCR-ACE_DLL ;--> LSB of divisor latch
|
|
mov al,cl
|
|
out dx,al
|
|
mov al,ch
|
|
inc dx ;--> MSB of divisor latch
|
|
.errnz ACE_DLM-ACE_DLL-1
|
|
iodelay
|
|
out dx,al
|
|
inc dx ;--> LCR and clear divisor access bit
|
|
inc dx
|
|
.errnz ACE_LCR-ACE_DLM-2
|
|
pop ax
|
|
out dx,al
|
|
|
|
inc dx ;--> Modem Control Register
|
|
.errnz ACE_MCR-ACE_LCR-1
|
|
endif ; NEC_98
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; Compute initial state of DTR and RTS. If they have been disabled,
|
|
; then do not raise them, and disallow being used as a handshaking
|
|
; line. Also compute the bits to use as hardware handshake bits
|
|
; (DTR and/or RTS as indicated, qualified with the disabled flags).
|
|
;-----------------------------------------------------------------------;
|
|
|
|
mov al,[si.DCB_Flags] ;Align DTR/RTS disable flags for 8250
|
|
and al,fRTSDisable+fDTRDisable
|
|
rol al,1 ;d0 = DTR, d2 = RTS (1 = disabled)
|
|
shr al,1 ;'C'= DTR, d1 = RTS
|
|
adc al,0 ;d0 = DTR, d1 = RTS
|
|
.errnz fRTSDisable-00000010b
|
|
.errnz fDTRDisable-10000000b
|
|
.errnz ACE_DTR-00000001b
|
|
.errnz ACE_RTS-00000010b
|
|
|
|
mov ah,al ;Save disable mask
|
|
ifndef NEC_98
|
|
xor al,ACE_DTR+ACE_RTS+ACE_OUT2
|
|
out dx,al ;Set Modem Control Register
|
|
endif ; NEC_98
|
|
|
|
mov al,[si.DCB_Flags2] ;Get hardware handshake flags
|
|
rol al,1 ;Align flags as needed
|
|
rol al,1
|
|
rol al,1
|
|
and al,ACE_DTR+ACE_RTS ;Mask bits of interest
|
|
not ah ;Want inverse of disable mask
|
|
and al,ah ;al = bits to handshake with
|
|
mov HHSLines[si],al ;Save for interrupt code
|
|
|
|
.errnz fDTRFlow-00100000b
|
|
.errnz fRTSFlow-01000000b
|
|
.errnz ACE_DTR-00000001b
|
|
.errnz ACE_RTS-00000010b
|
|
|
|
mov al,[si.DCB_Flags] ;Compute the mask for the output
|
|
shl al,1 ; hardware handshake lines
|
|
and al,ACE_DSR+ACE_CTS
|
|
mov OutHHSLines[si],al
|
|
|
|
.errnz fOutXCTSFlow-00001000b
|
|
.errnz fOutXDSRFlow-00010000b
|
|
.errnz ACE_CTS-00010000b
|
|
.errnz ACE_DSR-00100000b
|
|
|
|
; Compute the queue count where XOff should be issued (or hardware
|
|
; lines dropped). This will prevent having to do it at interrupt
|
|
; time.
|
|
|
|
mov ax,QInSize[si] ;Get where they want it
|
|
sub ax,[si.DCB_XoffLim] ; and compute queue count
|
|
mov XOffPoint[si],ax
|
|
|
|
ifdef NEC_98
|
|
FSTI ; (ins:94.06.11) |
|
|
test EFlags[si], fNoFIFO ; |
|
|
jnz SetCom21_NO_AOBA ; no AOBA on board |
|
|
; (ins 94.04.11) |
|
|
; Enable FIFO if possible when baudrate >= 1200 (ins 94.04.11) |
|
|
mov dx,Port[si] ; (ins 94.04.11) |
|
|
add dx,ACE_FCR ; (ins 94.04.11) |
|
|
; (ins 94.04.11) |
|
|
mov ax, [si.DCB_BaudRate] ; (ins 94.04.11) |
|
|
cmp ah, -1 ;Q: baudrate index? (ins 94.04.11) |
|
|
jne sc_fifo ; N: baudrate >= 1200, enable FIFO |
|
|
sc_fifo: ; (ins 94.04.11) |
|
|
or [si. AOBA_flag],fFIFO_Mode ; (ins 94.04.14) |
|
|
mov al, ACE_TRIG04 OR ACE_EFIFO OR ACE_CRFIFO OR ACE_CTFIFO;(ins 94.06.07) |
|
|
out dx, al ; attempt to enable FIFO(ins 94.06.07) |
|
|
test EFlags[si], fFIFOchkd ;Q: FIFO detect been done? ; |
|
|
jnz sc_fifodone ; Y: enabled FIFO (ins 94.04.11) |
|
|
or EFlags[si], fFIFOchkd ; (ins 94.04.11) |
|
|
jmp short sc_fifodone ; (ins 94.05.12) |
|
|
sc_nofifo: ; (ins 94.04.11) |
|
|
jmp short SetCom21_NO_AOBA ; (ins 94.05.24) |
|
|
sc_fifodone: ; (ins 94.04.11) |
|
|
test [si.AOBA_flag],fFIFO_Mode ; (ins 94.04.15) |
|
|
jz SetCom21_NO_AOBA ; (ins 94.04.15) |
|
|
mov dx,Port[si] ; (ins 94.04.11) |
|
|
add dx,ACE_MSR ;--> Modem Status reg (ins 94.04.11) |
|
|
in al,dx ;Throw away 1st status read |
|
|
NEWIODELAY 2 ; <OUT 5F,AL> (ins 94.04.18) |
|
|
in al,dx ;Save 2nd for MSRWait (Clear MSR int) |
|
|
call Set8251mode ;Change to 8251 mode (ins 94.05.12) |
|
|
FSTI ; (ins 94.06.09) |
|
|
call MSR_READ_Call ;AOBA-bug ins 94.11.19 KBNES |
|
|
jmp short @f ; (ins 94.04.11) |
|
|
SetCom21_NO_AOBA: ; (ins 94.04.11) |
|
|
FSTI ; (ins 94.06.09) |
|
|
and [si.AOBA_flag],not(fFIFO_Mode) ; (mov 94.05.24) |
|
|
call MSR_READ_Call ; [QN] (ins 92.08.05) |
|
|
@@: ; (ins 94.04.11) |
|
|
else ; NEC_98
|
|
; Enable FIFO if possible when baudrate >= 4800
|
|
;
|
|
sub dl,ACE_MCR - ACE_FCR ; dx = FCR
|
|
test EFlags[si], fNoFIFO ;Q: FIFO can be enabled?
|
|
jnz sc_nofifo ; N:
|
|
mov ax, [si.DCB_BaudRate]
|
|
cmp ax, 4800
|
|
jb sc_nofifo
|
|
cmp ah, -1 ;Q: baudrate index?
|
|
jne sc_fifo ; N: baudrate >= 4800, enable FIFO
|
|
cmp ax, CBR_4800
|
|
jb sc_nofifo
|
|
%OUT this isn't correct, if lower baudrates are assigned indices above CBR_4800
|
|
|
|
sc_fifo:
|
|
mov al, ACE_TRIG14 OR ACE_EFIFO OR ACE_CRFIFO OR ACE_CTFIFO
|
|
out dx, al ; attempt to enable FIFO
|
|
test EFlags[si], fFIFOchkd ;Q: FIFO detect been done?
|
|
jnz sc_fifodone ; Y: enabled FIFO
|
|
iodelay
|
|
.errnz ACE_IIDR-ACE_FCR
|
|
in al, dx
|
|
or EFlags[si], fFIFOchkd
|
|
test al, ACE_FIFO_E2 ;Q: FIFO enabled?
|
|
jz short @F
|
|
test al, ACE_FIFO_E1 ;Q: 16550A detected?
|
|
jnz sc_fifodone ; Y: enabled FIFO
|
|
@@:
|
|
iodelay
|
|
or EFlags[si], fNoFIFO
|
|
|
|
sc_nofifo:
|
|
xor al, al
|
|
out dx, al
|
|
sc_fifodone:
|
|
|
|
sub dl,ACE_FCR-ACE_RBR ; dx -> RBR
|
|
;
|
|
; Delay for things to settle
|
|
;
|
|
push dx
|
|
cCall GetSystemMsecCount
|
|
pop dx
|
|
mov cx, ax
|
|
delay_loop:
|
|
in al, dx ;Read it once
|
|
push dx
|
|
cCall GetSystemMsecCount
|
|
pop dx
|
|
sub ax, cx
|
|
cmp ax, DELAY_TIME ;Q: Timeout reached?
|
|
ifndef WOW
|
|
jb delay_loop ; N:
|
|
endif
|
|
|
|
add dl,ACE_MSR ;--> Modem Status reg
|
|
in al,dx ;Throw away 1st status read
|
|
iodelay
|
|
in al,dx ;Save 2nd for MSRWait (Clear MSR int)
|
|
endif ; NEC_98
|
|
mov MSRShadow[si],al
|
|
|
|
; Win 3.0 didn't check hardware handshaking until the line status changed.
|
|
; Allow some apps to keep that behavior.
|
|
|
|
push dx
|
|
xor ax, ax
|
|
cCall GetAppCompatFlags,<ax>
|
|
pop dx
|
|
test ax, GACF_DELAYHWHNDSHAKECHK
|
|
jnz short sc_HHSup
|
|
|
|
;
|
|
; HACK FOR SOME MODEMS: apparently some modems set CTS, but don't set DSR
|
|
; which means that COMM.DRV won't send if the app specifies that hardware
|
|
; handshaking is based on CTS & DSR being set.
|
|
;
|
|
mov ah,OutHHSLines[si]
|
|
mov al, MSRShadow[si]
|
|
and al,ah ;Only leave bits of interest
|
|
cmp al, ah ;Q: handshaking lines ok?
|
|
je short sc_HHSup ; Y:
|
|
cmp ah, ACE_CTS OR ACE_DSR ;Q: app looking for both high?
|
|
jne short sc_HHSdown ; N: skip hack
|
|
test [si.EFlags], fUseDSR ;Q: DSR is always significant?
|
|
jnz short sc_HHSdown ; Y: skip hack
|
|
cmp al, ACE_CTS ;Q: DSR low & CTS high
|
|
jne short sc_HHSdown ; N: skip hack
|
|
and ah, NOT ACE_DSR ; Y: ignore DSR line
|
|
mov OutHHSLines[si], ah
|
|
jmp short sc_HHSup
|
|
|
|
sc_HHSdown:
|
|
or [si.HSFlag], HHSDown OR HHSAlwaysDown ; flag handshaking down
|
|
sc_HHSup:
|
|
|
|
;-----------------------------------------------------------------------;
|
|
; Now, at last, interrupts can be enabled. Don't enable the
|
|
; transmitter empty interrupt. It will be enabled by the first
|
|
; call to KickTx.
|
|
;-----------------------------------------------------------------------;
|
|
|
|
ifndef NEC_98
|
|
sub dl,ACE_MSR-ACE_IER ;--> Interrupt Enable Register
|
|
endif ; NEC_98
|
|
|
|
; flag port as being active
|
|
push cx
|
|
mov cl, [si.DCB_Id]
|
|
mov ax, 1
|
|
shl ax, cl
|
|
or [activeCOMs], ax
|
|
pop cx
|
|
|
|
ifdef NEC_98
|
|
FCLI ; (ins 94.06.09)|
|
|
mov dx,MaskFFPort[si] ;port c address (ins 92.08.xx)|
|
|
cmp [si.DCB_id],ID_COM1 ; (ins 92.08.xx)|
|
|
jne SetCom3001 ; no (ins 92.08.xx)|
|
|
in al,dx ; (ins 92.08.xx)|
|
|
jmp short SetCom3002 ; (ins 92.08.xx)|
|
|
; (ins 92.08.xx)|
|
|
SetCom3001: ; (ins 92.08.xx)|
|
|
mov al,MaskFFShadow[si] ;Read 8251 mask data (ins 92.08.xx)|
|
|
; (ins 92.08.xx)|
|
|
SetCom3002: ; (ins 92.08.xx)|
|
|
or al,MSK_RXR ;receive int enable (ins 92.08.xx)|
|
|
out dx,al ; (ins 92.08.xx)|
|
|
mov MaskFFShadow[si],al ;Write 8251 mask data (ins 92.08.xx)|
|
|
FSTI ; (ins 94.06.09)|
|
|
; (ins 92.08.xx)|
|
|
;---------------------------------------------------------------(ins 92.08.xx)|
|
|
; Close process of system timer (ins 92.08.xx)|
|
|
;---------------------------------------------------------------(ins 92.08.xx)|
|
|
push ax ; (ins 92.08.xx)|
|
|
mov ax,hTimer[si] ; Check hTimer (ins 94.04.12)|
|
|
or ax,ax ; TimerProcess Nothing? (ins 92.08.xx)|
|
|
jz SetNoKillTimer ; Y:(Nothing) (ins 92.08.xx)|
|
|
cCall KillSystemTimer,<ax> ; Kill Syatem Timer (ins 92.08.xx)|
|
|
mov hTimer[si] ,00h ; Clear hTimer (ins 94.04.12)|
|
|
SetNoKillTimer: ; (ins 92.08.xx)|
|
|
; (ins 92.08.xx)|
|
|
;---------------------------------------------------------------(ins 92.08.xx)|
|
|
; make process of system timer (ins 92.08.xx)|
|
|
;---------------------------------------------------------------(ins 92.08.xx)|
|
|
test DCB_Flags[si],fOutXCTSFlow or fOutXDSRFlow ; (ins 92.08.xx)|
|
|
; HHS Enable ? (ins 92.08.xx)|
|
|
jz SetCom3003 ; No: No Set Timer (ins 92.08.xx)|
|
|
mov ax, 200 ; create 200msec timer (ins 92.08.xx)|
|
|
push ax ; (ins 92.08.xx)|
|
|
mov ax, _INTERRUPT ; (ins 92.08.xx)|
|
|
push ax ; (ins 92.08.xx)|
|
|
mov ax,TimerProcAdr[si] ;--> timer processor (ins 92.08.xx)|
|
|
push ax ; (ins 92.08.xx)|
|
|
call CreateSystemTimer ; ax = 0, if failed (ins 92.08.xx)|
|
|
mov hTimer[si] ,ax ; Get hTimer (ins 94.04.12)|
|
|
; (ins 92.08.xx)|
|
|
SetCom3003: ; (ins 92.08.xx)|
|
|
pop ax ; (ins 92.08.xx)|
|
|
; (ins 92.08.xx)|
|
|
FCLI ; (ins 94.06.09)|
|
|
test [si.AOBA_flag],fFIFO_Mode ; (ins 94.04.15)|
|
|
jz @F ; (ins 94.04.15)|
|
|
call Set8251Mode ;FIFODisable (ins 94.05.16)|
|
|
mov dx,Port[si] ; (ins 94.04.13)|
|
|
add dx,ACE_FCR ; (ins 94.04.13)|
|
|
mov al,FIFO_Set_Buf_Clr ; (ins 94.04.15)|
|
|
out dx,al ; (ins 94.04.13)|
|
|
@@: ; (ins 94.05.12)|
|
|
else ; NEC_98
|
|
mov al,ACE_ERBFI+ACE_ELSI+ACE_EDSSI
|
|
FCLI
|
|
out dx,al ;Enable interrupts.
|
|
add dl,ACE_LSR-ACE_IER ;--> Line Status Register
|
|
iodelay
|
|
in al,dx ;Clear any Line Status interrupt
|
|
sub dl,ACE_LSR ;--> Receiver Buffer Register
|
|
iodelay
|
|
in al,dx ;Clear any Received Data interrupt
|
|
endif ; NEC_98
|
|
FSTI
|
|
jmp SetCom5 ;All done
|
|
|
|
$SETCOM endp
|
|
page
|
|
|
|
FlagNotActive proc near
|
|
push cx
|
|
mov cl, [si.DCB_Id]
|
|
mov ax, NOT 1
|
|
rol ax, cl
|
|
and [activeCOMs], ax
|
|
pop cx
|
|
ret
|
|
FlagNotActive endp
|
|
|
|
;----------------------------Private-Routine----------------------------;
|
|
;
|
|
; SetCom100
|
|
;
|
|
; Copy the given DCB into the appropriate DEB. The check has
|
|
; already been made to determine that the ID was valid, so
|
|
; that check can be skipped.
|
|
;
|
|
; Entry:
|
|
; ES:BX --> DCB
|
|
; DS:SI --> DEB
|
|
; Returns:
|
|
; DS:SI --> DEB
|
|
; ES = Data
|
|
; Error Returns:
|
|
; None
|
|
; Registers Destroyed:
|
|
; AX,CX,ES,DI,FLAGS
|
|
; History:
|
|
;-----------------------------------------------------------------------;
|
|
|
|
;------------------------------Pseudo-Code------------------------------;
|
|
; {
|
|
; }
|
|
;-----------------------------------------------------------------------;
|
|
|
|
assumes ds,Data
|
|
assumes es,nothing
|
|
|
|
SetCom100 proc near
|
|
push si ;Save DEB pointer
|
|
mov di,si
|
|
mov si,bx
|
|
push es
|
|
mov ax,ds
|
|
pop ds
|
|
assumes ds,nothing
|
|
|
|
mov es,ax
|
|
assumes es,Data
|
|
|
|
mov cx,DCBSize
|
|
cld
|
|
rep movsb
|
|
mov ds,ax
|
|
assumes ds,Data
|
|
|
|
pop si ;Restore DEB pointer
|
|
ret
|
|
|
|
SetCom100 endp
|
|
page
|
|
|
|
;----------------------------Private-Routine----------------------------;
|
|
;
|
|
; SetCom200
|
|
;
|
|
; Based on whether or not a timeout has been specified for each
|
|
; signal, set up a mask byte which is used to mask off lines for
|
|
; which we wish to detect timeouts. 0 indicates that the line is
|
|
; to be ignored.
|
|
;
|
|
; Also set up a mask to indicate those lines which are set for
|
|
; infinite timeout. 1 indicates that the line has infinite
|
|
; timeout.
|
|
;
|
|
; Entry:
|
|
; ES:BX --> DCB
|
|
; Returns:
|
|
; ES:BX --> DCB
|
|
; AH = lines to check
|
|
; AL = lines with infinite timeout
|
|
; Error Returns:
|
|
; None
|
|
; Registers Destroyed:
|
|
; AX,CX,FLAGS
|
|
; History:
|
|
;-----------------------------------------------------------------------;
|
|
|
|
;------------------------------Pseudo-Code------------------------------;
|
|
; {
|
|
; }
|
|
;-----------------------------------------------------------------------;
|
|
|
|
assumes ds,Data
|
|
assumes es,nothing
|
|
|
|
SetCom200 proc near
|
|
|
|
xor ax,ax
|
|
xor cx,cx ;Get mask of lines with timeout = 0
|
|
call SetCom210
|
|
not al ;Invert result to get lines to check
|
|
and al,ACE_CTS+ACE_DSR+ACE_RLSD
|
|
xchg ah,al
|
|
dec cx ;Get mask of infinite timeouts
|
|
|
|
SetCom210:
|
|
cmp es:[bx.DCB_RlsTimeout],cx ;Timeout set to passed value?
|
|
jne SetCom220 ; No
|
|
or al,ACE_RLSD ; Yes, show checking line
|
|
|
|
SetCom220:
|
|
cmp es:[bx.DCB_CtsTimeout],cx ;Timeout set to passed value?
|
|
jne SetCom230 ; No
|
|
or al,ACE_CTS ; Yes, show checking line
|
|
|
|
SetCom230:
|
|
cmp es:[bx.DCB_DsrTimeout],cx ;Timeout set to passed value?
|
|
jne SetCom240 ; No
|
|
or al,ACE_DSR ; Yes, show checking line
|
|
|
|
SetCom240:
|
|
ret
|
|
|
|
SetCom200 endp
|
|
page
|
|
|
|
;----------------------------Private-Routine----------------------------;
|
|
;
|
|
; SetCom300
|
|
;
|
|
; Calculate the correct baudrate divisor for the comm chip.
|
|
;
|
|
; Note that the baudrate is allowed to be any integer in the
|
|
; range 2-19200. The divisor is computed as 115,200/baudrate.
|
|
;
|
|
; Entry:
|
|
; ES:BX --> DCB
|
|
; Returns:
|
|
; ES:BX --> DCB
|
|
; CX = baudrate
|
|
; Error Returns:
|
|
; CX = 0 if error
|
|
; AX = error code if invalid baud rate
|
|
; Registers Destroyed:
|
|
; AX,CX,FLAGS
|
|
; History:
|
|
;-----------------------------------------------------------------------;
|
|
|
|
ifdef NEC_98
|
|
; If 110bps [QN] (ins 92.08.xx)
|
|
; *16 10M 1396 [QN] (ins 92.08.xx)
|
|
; 8M 1135 [QN] (ins 92.08.xx)
|
|
; *64 10M 349 [QN] (ins 92.08.xx)
|
|
; 8M 284 [QN] (ins 92.08.xx)
|
|
BaudRateByIndexTable label word ;; (93.04.12)
|
|
dw 110 ; CBR_110 ;; (93.04.12)
|
|
dw 300 ; CBR_300 ;; (93.04.12)
|
|
dw 600 ; CBR_600 ;; (93.04.12)
|
|
dw 1200 ; CBR_1200 ;; (93.04.12)
|
|
dw 2400 ; CBR_2400 ;; (93.04.12)
|
|
dw 4800 ; CBR_4800 ;; (93.04.12)
|
|
dw 9600 ; CBR_9600 ;; (93.04.12)
|
|
dw 14400 ; CBR_14400 ;; (93.04.12)
|
|
dw 19200 ; CBR_19200 ;; (93.04.12)
|
|
dw 0 ; 0FF19h (reserved) ;; (93.04.12)
|
|
dw 0 ; 0FF1Ah (reserved) ;; (93.04.12)
|
|
dw 38400 ; CBR_38400 (reserved) ;; (93.04.12)
|
|
dw 0 ; 0FF1Ch (reserved) ;; (93.04.12)
|
|
dw 0 ; 0FF1Dh (reserved) ;; (93.04.12)
|
|
dw 0 ; 0FF1Eh (reserved) ;; (93.04.12)
|
|
dw 56000 ; CBR_56000 (reserved) ;; (93.04.12)
|
|
else ; NEC_98
|
|
BaudRateByIndexTable label word
|
|
dw 1047 ; CBR_110
|
|
dw 384 ; CBR_300
|
|
dw 192 ; CBR_600
|
|
dw 96 ; CBR_1200
|
|
dw 48 ; CBR_2400
|
|
dw 24 ; CBR_4800
|
|
dw 12 ; CBR_9600
|
|
dw 9 ; CBR_14400
|
|
dw 6 ; CBR_19200
|
|
dw 0 ; 0FF19h (reserved)
|
|
dw 0 ; 0FF1Ah (reserved)
|
|
dw 3 ; CBR_38400
|
|
dw 0 ; 0FF1Ch (reserved)
|
|
dw 0 ; 0FF1Dh (reserved)
|
|
dw 0 ; 0FF1Eh (reserved)
|
|
dw 2 ; CBR_56000
|
|
endif ; NEC_98
|
|
|
|
assumes ds,Data
|
|
assumes es,nothing
|
|
|
|
SetCom300 proc near
|
|
|
|
ifdef NEC_98
|
|
push dx ; (ins 92.08.05)|
|
|
mov dx,13 ;base value of div for 8MHz(ins 92.08.05)|
|
|
; (ins 92.08.05)|
|
|
mov ax,40h ; (ins 92.08.05)|
|
|
push ds ; (ins 92.08.05)|
|
|
mov ds,ax ;DS <- 40H (ins 92.08.05)|
|
|
test byte ptr ds:[101h],80h ;clock late check (ins 92.08.05)|
|
|
pop ds ; (ins 92.08.05)|
|
|
jnz SetCom328 ;= 1 -> 8MHz (ins 92.08.05)|
|
|
; |
|
|
;In the case of 5MHz ----------------(93.03.02)-----------------(ins 93.03.02)|
|
|
xor dx,dx ; (ins 93.03.02)|
|
|
SetCom310: ; (ins 93.03.02)|
|
|
mov ax,es:DCB_BaudRate[bx] ;Save data (ins 93.03.02)|
|
|
; (ins 93.03.02)|
|
|
cmp ax, CBR_110 ;as an index? (ins 93.04.12)
|
|
jnae not_index5 ; (ins 93.04.12)
|
|
cmp ax, CBR_19200 ; (ins 93.04.12)
|
|
ja not_index5 ; (ins 93.04.12)
|
|
cmp ax, CBR_14400 ; (ins 93.04.12)
|
|
je not_index5 ; (ins 93.04.12)
|
|
|
|
push bx ; (ins 93.04.12)
|
|
mov bx, ax ; (ins 93.04.12)
|
|
sub bx, CBR_110 ; (ins 93.04.12)
|
|
shl bx, 1 ; (ins 93.04.12)
|
|
mov ax, cs:[bx+BaudRateByIndexTable]; (ins 93.04.12)
|
|
pop bx ; (ins 93.04.12)
|
|
|
|
not_index5: ; (ins 93.04.12)
|
|
cmp ax,19200 ;baudrate=75-19200bps? (ins 93.03.02)|
|
|
ja SetCom330 ; No , go to SetCom330 (ins 93.03.02)|
|
|
; (ins 93.03.02)|
|
|
cmp ax,110 ;baudrate=110 bps? (ins 93.03.02)|
|
|
je SetCom335 ;Yes, go to SetCom335 (ins 93.03.02)|
|
|
xor cx,cx ; (ins 93.03.02)|
|
|
mov cx,75 ; (ins 93.03.02)|
|
|
div cx ;dx:ax/75=ax dx(mod) (ins 93.03.02)|
|
|
cmp dx,0 ;Good data ? (ins 93.03.02)|
|
|
jne SetCom330 ; No! (ins 93.03.02)|
|
|
cmp ax,0 ;Good data ? (ins 93.03.02)|
|
|
je SetCom330 ; No,dx:ax = 0 (?!) (ins 93.03.02)|
|
|
; (ins 93.03.02)|
|
|
; It is good data!!! ; (ins 93.03.02)|
|
|
SetCom320: ; (ins 93.03.02)|
|
|
mov dx,8 ;base value of div for 5MHz(ins 93.03.02)|
|
|
mov cl,7 ; (ins 93.03.02)|
|
|
shl ax,cl ;if 19200, the most high bit is 1.(ins 93.03.02)|
|
|
SetCom327: ; (ins 93.03.02)|
|
|
shl ax,1 ; (ins 93.03.02)|
|
|
jc SetCom340 ;OK,go to return !! (ins 93.03.02)|
|
|
shl dx,1 ; (ins 93.03.02)|
|
|
jmp short SetCom327 ; (ins 93.03.02)|
|
|
; (ins 93.03.02)|
|
|
SetCom328: ;In the case of 8MHz---(ins 93.03.02)-----------(ins 93.03.02)|
|
|
mov ax,es:DCB_BaudRate[bx] ;Save data (ins 93.03.02)|
|
|
|
|
cmp ax, CBR_110 ;as an index? (ins 93.04.12)
|
|
jnae not_index8 ; (ins 93.04.12)
|
|
cmp ax, CBR_9600 ; (ins 93.04.12)
|
|
ja not_index8 ; (ins 93.04.12)
|
|
cmp ax, CBR_14400 ; (ins 93.04.12)
|
|
je not_index8 ; (ins 93.04.12)
|
|
|
|
push bx ; (ins 93.04.12)
|
|
mov bx, ax ; (ins 93.04.12)
|
|
sub bx, CBR_110 ; (ins 93.04.12)
|
|
shl bx, 1 ; (ins 93.04.12)
|
|
mov ax, cs:[bx+BaudRateByIndexTable]; (ins 93.04.12)
|
|
pop bx ; (ins 93.04.12)
|
|
not_index8: ; (ins 93.04.12)
|
|
cmp ax,9600 ;baudrate=75-9600bps? (ins 93.03.02)|
|
|
ja SetCom330 ; No , go to SetCom330 (ins 93.03.02)|
|
|
cmp ax,110 ;baudrate=110 bps? (ins 93.03.02)|
|
|
je SetCom335 ;Yes, go to SetCom335 (ins 93.03.02)|
|
|
mov cl,75 ; (ins 93.03.02)|
|
|
div cl ;ax/75=al ah(mod) (ins 93.03.02)|
|
|
cmp ah,0 ;Good data ? (ins 93.03.02)|
|
|
jne SetCom330 ; No! (ins 93.03.02)|
|
|
cmp al,0 ;Good data ? (ins 93.03.02)|
|
|
je SetCom330 ; No,ax = 0 (?!) (ins 93.03.02)|
|
|
; (ins 93.03.02)|
|
|
; It is good data!!! ; (ins 93.03.02)|
|
|
SetCom329: ; (ins 93.03.02)|
|
|
shl al,1 ; (ins 93.03.02)|
|
|
jc SetCom340 ; (ins 93.03.02)|
|
|
shl dx,1 ; (ins 93.03.02)|
|
|
jmp short SetCom329 ; (ins 93.03.02)|
|
|
; (ins 93.03.02)|
|
|
;-------(end 93.03.02)----------------------------------------- (ins 93.03.02)|
|
|
SetCom330: ; (ins 92.08.05)|
|
|
xor cx,cx ; (ins 92.08.05)|
|
|
mov ax,IE_Baudrate ;Show Unsupported BaudRate(ins 92.08.05)|
|
|
pop dx ; (ins 92.08.05)|
|
|
ret ; (ins 92.08.05)|
|
|
; (ins 92.08.05)|
|
|
SetCom335: ;for 110bps (ins 92.08.05)|
|
|
cmp dx,13 ;8MHz ? (ins 92.08.05)|
|
|
je SetCom338 ; (ins 92.08.05)|
|
|
mov cx,1396 ;Save timer value for 110bps(ins 92.08.05)|
|
|
jmp short SetCom339 ; (ins 92.08.05)|
|
|
SetCom338: ; (ins 92.08.05)|
|
|
mov cx,1135 ;Save timer value for 110bps(ins 92.08.05)|
|
|
SetCom339: ; (ins 92.08.05)|
|
|
pop dx ; (ins 92.08.05)|
|
|
ret ; (ins 92.08.05)|
|
|
SetCom340: ; (ins 92.08.05)|
|
|
mov cx,dx ;Save result (ins 92.08.05)|
|
|
pop dx ; (ins 92.08.05)|
|
|
ret ; (ins 92.08.05)|
|
|
else ; NEC_98
|
|
push dx
|
|
mov cx,es:[bx.DCB_BaudRate] ;Get requested baud rate
|
|
xor ax,ax ;Assume error
|
|
cmp cx, CBR_110 ;Q: baudrate specified as an index?
|
|
jae by_index
|
|
cmp cx,2 ;Within valid range?
|
|
jnae SetCom310 ; No, return error
|
|
|
|
mov dx,1 ;(dx:ax) = 115,200
|
|
mov ax,0C200h
|
|
div cx ;(ax) = 115,200/baud
|
|
|
|
SetCom310:
|
|
mov cx,ax ;(cx) = baud rate, or error code (0)
|
|
mov ax,IE_Baudrate ;Set error code incase bad baud
|
|
pop dx
|
|
ret
|
|
|
|
by_index:
|
|
cmp cx, CBR_56000 ;Q: above supported?
|
|
ja SetCom310 ; Y: return error
|
|
push bx
|
|
mov bx, cx
|
|
sub bx, CBR_110
|
|
shl bx, 1
|
|
mov ax, cs:[bx+BaudRateByIndexTable] ; get divisor
|
|
pop bx
|
|
jmp SetCom310 ; Y: return error
|
|
endif ; NEC_98
|
|
|
|
|
|
SetCom300 endp
|
|
page
|
|
|
|
;----------------------------Private-Routine----------------------------;
|
|
;
|
|
; SetCom400
|
|
;
|
|
; Check the line configuration (Parity, Stop bits, Byte size)
|
|
;
|
|
; Entry:
|
|
; ES:BX --> DCB
|
|
; Returns:
|
|
; ES:BX --> DCB
|
|
; 'C' clear if OK
|
|
; AL = Line Control Register
|
|
; AH = RxMask
|
|
; DI[15:8] = Flags mask (to remove parity checking)
|
|
; DI[7:0] = Error mask (to remove parity error)
|
|
; Error Returns:
|
|
; 'C' set if error
|
|
; AX = error code
|
|
; Registers Destroyed:
|
|
; AX,CX,DI,FLAGS
|
|
; History:
|
|
;-----------------------------------------------------------------------;
|
|
|
|
;------------------------------Pseudo-Code------------------------------;
|
|
; {
|
|
; }
|
|
;-----------------------------------------------------------------------;
|
|
|
|
assumes ds,Data
|
|
assumes es,nothing
|
|
|
|
SetCom400 proc near
|
|
|
|
ifdef NEC_98
|
|
assumes es,nothing
|
|
;[QN](ins 92.08.xx)---------------------------------------------------------+
|
|
; [QN]|
|
|
; (Parity, Stop bits, Byte size) Check Signal Status [QN]|
|
|
; [QN]|
|
|
; Entry: es:bx --> DCB [QN]|
|
|
; [QN]|
|
|
; Exit: es:bx --> DCB [QN]|
|
|
; 'C' clear if OK [QN]|
|
|
; al = 8251 Mode instruction [QN]|
|
|
; ah = RxMask [QN]|
|
|
; di[15:8] = Flags mask (to remove parity checking) [QN]|
|
|
; di[7:0] = Error mask (to remove parity error) [QN]|
|
|
; 'C' set if error [QN]|
|
|
; ax = error code [QN]|
|
|
; [QN]|
|
|
; Uses: ax,cx,di,flags [QN]|
|
|
; [QN]|
|
|
; 8251 mode instruction format [QN]|
|
|
; (7) (6) (5) (4) (3) (2) (1) (0) [QN]|
|
|
; <-------> <-------> <-------> <-------> [QN]|
|
|
; stop bit parity bit byte size clock(x16)=10b [QN]|
|
|
; 0 0 <----------------- none [QN]|
|
|
; 0 1 <----------------- 1 [QN]|
|
|
; 1 0 <----------------- 1.5 [QN]|
|
|
; 1 1 <----------------- 2 [QN]|
|
|
; 0 <------------------ odd parity [QN]|
|
|
; 1 <------------------ even parity [QN]|
|
|
; 0 <----------------- parity disable [QN]|
|
|
; 1 <----------------- parity enable [QN]|
|
|
; 0 0 <--------- 5 bit [QN]|
|
|
; 0 1 <--------- 6 bit [QN]|
|
|
; 1 0 <--------- 7 bit [QN]|
|
|
; 1 1 <--------- 8 bit [QN]|
|
|
; 0 0 <-sync [QN]|
|
|
; 0 1 <-async x1 [QN]|
|
|
; 1 0 <-async x16 [QN]|
|
|
; 1 1 <-async x64 [QN]|
|
|
; ins 92.08.05 [QN]|
|
|
;-----------------------------------------------------------------------[QN]|
|
|
mov ax,wo es:[bx.DCB_ByteSize] ;al=byte size, ah=parity [QN]|
|
|
mov ah,es:DCB_Parity[bx] ;ah =parity [QN]|
|
|
cmp ah,2 ;Is parity incorrect ? [QN]|
|
|
ja SetCom470 ; Yes, return error [QN]|
|
|
mov di,0FF00h+ACE_OR+ACE_PE+ACE_FE+ACE_BI; [QN]|
|
|
;ACE_OR(overrun err) [QN]|
|
|
;ACE_PE(parity err) [QN]|
|
|
;ACE_FE(framing err) [QN]|
|
|
;ACE_BI(break interrupt) [QN]|
|
|
; [QN]|
|
|
or ah,ah ;non-parity ? [QN]|
|
|
jnz SetCom410 ; No, It has parity. [QN]|
|
|
xor di,(fParity*256)+ACE_PE ; [QN]|
|
|
;Don't check parity [QN]|
|
|
SetCom410: ; [QN]|
|
|
cmp al,8 ;byte size = 8 Bit ? [QN]|
|
|
ja SetCom460 ;If bytesize > 8 Bit , error!! [QN]|
|
|
cmp al,5 ;If bytesize < 5 Bit [QN]|
|
|
jc SetCom460 ; , Error!! [QN]|
|
|
SetCom420: ; [QN]|
|
|
sub al,5 ;Shift byte size to bit1&0 [QN]|
|
|
; [QN]|
|
|
add ah,ah ;Map parity to ACE bits [QN]|
|
|
jz SetCom430 ;0=>0, 1=>1, 2=>3, 3=>5, 4=>7 [QN]|
|
|
dec ah ; [QN]|
|
|
; [QN]|
|
|
SetCom430: ; [QN]|
|
|
shl ax,1 ;Align with 8251 parity bits [QN]|
|
|
shl ax,1 ; [QN]|
|
|
shl ah,1 ; [QN]|
|
|
shl ah,1 ; [QN]|
|
|
or al,ah ;Add to byte size [QN]|
|
|
; [QN]|
|
|
mov ah,es:[bx.DCB_StopBits] ;Get # of stop bits 0=1,1/2= .GT.1 [QN]|
|
|
cmp ah,3 ;Is it out of area ? [QN]|
|
|
jnc SetCom470 ; Yes, Return Error [QN]|
|
|
inc ah ; stop1 = 00h [QN]|
|
|
ror ah,1 ; stop1.5 = 01h [QN]|
|
|
ror ah,1 ; stop2 = 10h [QN]|
|
|
; [QN]|
|
|
SetCom440: ; [QN]|
|
|
or al,ah ; [QN]|
|
|
or al,00000010b ;clock late (x16) set [QN]|
|
|
else ; NEC_98
|
|
mov ax,wo es:[bx.DCB_ByteSize] ;al = byte size, ah = parity
|
|
cmp ah,SpaceParity ;Parity out of range?
|
|
ja SetCom470 ; Yes, return error
|
|
mov di,0FF00h+ACE_OR+ACE_PE+ACE_FE+ACE_BI
|
|
or ah,ah ;Is parity "NONE"?
|
|
jnz SetCom410 ; No, something is there for parity
|
|
xor di,(fParity*256)+ACE_PE ;Disable parity checking
|
|
|
|
SetCom410:
|
|
cmp al,8 ;Byte size out of range?
|
|
ja SetCom460 ; Yes, error
|
|
|
|
SetCom420:
|
|
sub al,5 ;Shift byte size to bits 0&1
|
|
.errnz ACE_WLS-00000011b ;Word length must be these bits
|
|
jc SetCom460 ;Byte size is illegal, return error
|
|
add ah,ah ;Map parity to ACE bits
|
|
jz SetCom430 ;0=>0, 1=>1, 2=>3, 3=>5, 4=>7
|
|
dec ah
|
|
|
|
SetCom430:
|
|
shl ah,1 ;Align with 8250 parity bits
|
|
shl ah,1
|
|
shl ah,1
|
|
or al,ah ;Add to byte size
|
|
|
|
.errnz NoParity-0
|
|
.errnz OddParity-1
|
|
.errnz EvenParity-2
|
|
.errnz MarkParity-3
|
|
.errnz SpaceParity-4
|
|
.errnz ACE_PEN-00001000b
|
|
.errnz ACE_PSB-00110000b
|
|
.errnz ACE_EPS-00010000b
|
|
.errnz ACE_SP-00100000b
|
|
|
|
or al,ACE_2SB ;Assume 2 stop bits
|
|
mov ah,es:[bx.DCB_StopBits] ;Get # of stop bits 0=1,1/2= .GT. 1
|
|
or ah,ah ;Out of range?
|
|
js SetCom470 ; Yes, return error
|
|
jz SetCom440 ;One stop bit
|
|
sub ah,2
|
|
jz SetCom450 ;Two stop bits
|
|
jns SetCom470 ;Not 1.5, return error
|
|
test al,ACE_WLS ;1.5 stop bits, 5 bit words?
|
|
jnz SetCom470 ; No, illegal
|
|
.errnz OneStopBit-0
|
|
.errnz One5StopBits-1
|
|
.errnz TwoStopBits-2
|
|
.errnz ACE_5BW
|
|
|
|
SetCom440:
|
|
and al,NOT ACE_2SB ;Show 1 (or 1.5) stop bit(s)
|
|
endif ; NEC_98
|
|
|
|
|
|
; From the byte size, get a mask to be used for stripping
|
|
; off unused bits as the characters are received.
|
|
|
|
SetCom450:
|
|
push dx
|
|
mov cl,es:[bx.DCB_ByteSize] ;Get data byte size
|
|
mov dx,00FFh ;Turn into mask by shifting bits
|
|
shl dx,cl
|
|
mov ah,dh ;Return mask in ah
|
|
pop dx
|
|
clc ;Show all is fine
|
|
ret
|
|
|
|
SetCom460:
|
|
mov ax,IE_ByteSize ;Show byte size is wrong
|
|
stc ;Show error
|
|
ret
|
|
|
|
SetCom470:
|
|
mov ax,IE_Default ;Show something is wrong
|
|
stc ;Show error
|
|
ret
|
|
|
|
SetCom400 endp
|
|
page
|
|
|
|
;----------------------------------------------------------------------------
|
|
; SuspendOpenCommPorts:
|
|
;
|
|
; This routine is called from 286 Winoldaps to simply deinstall the comm port
|
|
; hooks.
|
|
;----------------------------------------------------------------------------
|
|
|
|
cProc SuspendOpenCommPorts,<FAR,PUBLIC,PASCAL>
|
|
|
|
cBegin nogen
|
|
|
|
assumes cs,Code
|
|
assumes ds,Data
|
|
|
|
%OUT not masking IRQ's
|
|
|
|
; Nothing to do under 3.1!
|
|
|
|
ret
|
|
|
|
cEnd nogen
|
|
|
|
;----------------------------------------------------------------------------;
|
|
; ReactivateOpenCommPorts: ;
|
|
; ;
|
|
; This routine reinstalls the comm hooks in real mode and reads the 8250 ;
|
|
; data and status registers to clear pending interrupts. ;
|
|
;----------------------------------------------------------------------------;
|
|
|
|
cProc ReactivateOpenCommPorts,<FAR,PASCAL,PUBLIC>,<si,di>
|
|
|
|
cBegin
|
|
call Rotate_PIC ;make comm ports highest priority
|
|
|
|
mov cx, MAXCOM+1
|
|
mov di,dataOffset COMptrs
|
|
rcp_loop:
|
|
mov si, [di]
|
|
mov dx, Port[si]
|
|
or dx, dx
|
|
jz @f
|
|
call InitAPort ;read comm port regs to clr pending ints
|
|
@@:
|
|
add di, 2
|
|
loop rcp_loop
|
|
|
|
cEnd
|
|
|
|
;----------------------------------------------------------------------------;
|
|
; InitAPort: ;
|
|
; ;
|
|
; reads the data,status & IIR registers of a port (has to be 8250!) ;
|
|
; ;
|
|
; If the port has an out queue pending, then this woutine will also start ;
|
|
; the transmit process by faking a comm interrupt. ;
|
|
;----------------------------------------------------------------------------;
|
|
|
|
public InitAPort
|
|
InitAPort proc near
|
|
|
|
ifdef NEC_98
|
|
cmp IntVecNum[si],0 ; [QN] (ins 92.08.05)
|
|
je InitAPort1 ; [QN] (ins 92.08.05)
|
|
mov dx,DataPort[si] ; [QN] (ins 92.08.05)
|
|
in al,dx ; [QN] (ins 92.08.05)
|
|
else ; NEC_98
|
|
add dl,ACE_RBR ;dx=receive buffer register
|
|
in al,dx ;read the data port
|
|
jmp short $+2 ;i/o delay
|
|
add dl,ACE_LSR - ACE_RBR ;get to the status port
|
|
in al,dx ;read it too.
|
|
jmp short $+2 ;i/o delay
|
|
add dl,ACE_IIDR - ACE_LSR ;get to the line status register
|
|
in al,dx ;read it once more
|
|
jmp short $+2 ;i/o delay
|
|
add dl,ACE_MSR - ACE_IIDR ;get to the modem status register
|
|
in al,dx ;read it once more
|
|
jmp short $+2 ;i/o delay
|
|
add dl,ACE_RBR - ACE_MSR ;get to the receive buffer register
|
|
in al,dx ;read it once more
|
|
jmp short $+2 ;i/o delay
|
|
endif ; NEC_98
|
|
call UnmaskIRQ
|
|
|
|
; now if the port has characters pending to be sent out then we must fake a
|
|
; comm interrupt.
|
|
|
|
cmp [si].QOutCount,0 ;characters pending to be sent ?
|
|
jz @f ;no.
|
|
FCLI ;disable interrupts
|
|
call FakeCOMIntFar ;fake an interrupt
|
|
FSTI ;renable interrupts
|
|
@@:
|
|
ifdef NEC_98
|
|
InitAPort1: ;(ins 92.08.xx)
|
|
endif ; NEC_98
|
|
ret
|
|
|
|
InitAPort endp
|
|
|
|
page
|
|
|
|
;----------------------------Public Routine-----------------------------;
|
|
;
|
|
; $DCBPtr - Return Pointer To DCB
|
|
;
|
|
; Returns a long pointer to the DCB for the requested device.
|
|
;
|
|
; Entry:
|
|
; AH = Device ID
|
|
; Returns:
|
|
; DX:AX = pointer to DCB.
|
|
; Error Returns:
|
|
; DX:AX = 0
|
|
; Registers Preserved:
|
|
; SI,DI,DS
|
|
; Registers Destroyed:
|
|
; BX,CX,ES,FLAGS
|
|
; History:
|
|
;-----------------------------------------------------------------------;
|
|
|
|
;------------------------------Pseudo-Code------------------------------;
|
|
; {
|
|
; }
|
|
;-----------------------------------------------------------------------;
|
|
|
|
assumes ds,Data
|
|
assumes es,nothing
|
|
|
|
public $DCBPTR
|
|
$DCBPTR proc near
|
|
|
|
push si
|
|
xor dx,dx
|
|
call GetDEB ;Get pointer to DEB
|
|
mov ax,dx
|
|
jc DCBPtr10 ;Jump if invalid device
|
|
mov ax,si ;else return value here
|
|
mov dx,ds
|
|
|
|
DCBPtr10:
|
|
pop si
|
|
ret
|
|
|
|
$DCBPTR endp
|
|
page
|
|
|
|
;----------------------------Private-Routine----------------------------;
|
|
;
|
|
; GetDEB - Get Pointer To Device's DEB
|
|
;
|
|
; Returns a pointer to appropriate DEB, based on device number.
|
|
;
|
|
; Entry:
|
|
; AH = cid
|
|
; Returns:
|
|
; 'C' clear
|
|
; 'S' set if LPT device
|
|
; DS:SI --> DEB is valid cid
|
|
; AH = cid
|
|
; Error Returns:
|
|
; 'C' set if error (cid is invalid)
|
|
; AX = 8000h
|
|
; Registers Preserved:
|
|
; BX,CX,DX,DI,DS,ES
|
|
; Registers Destroyed:
|
|
; AX,SI,FLAGS
|
|
; History:
|
|
;-----------------------------------------------------------------------;
|
|
|
|
;------------------------------Pseudo-Code------------------------------;
|
|
; {
|
|
; }
|
|
;-----------------------------------------------------------------------;
|
|
|
|
assumes ds,Data
|
|
assumes es,nothing
|
|
|
|
public GetDEB ;Public for debugging
|
|
GetDEB proc near
|
|
|
|
push cx
|
|
mov cl, ah
|
|
and cx, (NOT LPTx AND 0FFh)
|
|
test ah, ah ;Q: LPT id?
|
|
js short GetDEB10 ; Y:
|
|
.errnz LPTx - 80h
|
|
cmp ah, MAXCOM ;Q: Within range?
|
|
ja GetDEB30 ; N: return invalid ID
|
|
shl cx, 1
|
|
mov si, cx
|
|
mov si, [si+COMptrs]
|
|
jmp short GetDEB20
|
|
|
|
GetDEB10:
|
|
cmp ah, LPTx+MAXLPT ;Q: Within range?
|
|
ja GetDEB30 ; N: return invalid ID
|
|
ifdef NEC_98
|
|
cmp ah, ID_LPT1 ; (ins 940115)
|
|
je @f ; (ins 940115)
|
|
; (ins 940125)
|
|
; Don't guarantee Ex board on Highreso mode. (ins 940125)
|
|
; So return ID error if the request is LPT2/3. (ins 940125)
|
|
; (ins 940125)
|
|
; 40H:[101H] bit3 (ins 940125)
|
|
; =0 : Normal mode (ins 940125)
|
|
; =1 : Highreso mode (ins 940125)
|
|
push ax ; (ins 940125)
|
|
push ds ; (ins 940125)
|
|
mov ax,40H ; (ins 940125)
|
|
mov ds,ax ; (ins 940125)
|
|
test by ds:[101H],00001000b ; normal mode ? (ins 940125)
|
|
pop ds ; (ins 940125)
|
|
pop ax ; (ins 940125)
|
|
jz NormalRez ; Y: Next Check (ins 940125)
|
|
jmp short GetDEB30 ; N: return invalid ID (ins 940125)
|
|
NormalRez: ; (ins 940125)
|
|
push ax ; Check whether request (ins 940115)
|
|
push dx ; LPT2/3 without Ex board(ins 940115)
|
|
mov dx, Toki_ControlStatus+400h ; (ins 940115)
|
|
in al, dx ; (ins 940115)
|
|
cmp al, 0ffh ; Is there Ex board ? (ins 940115)
|
|
pop dx ; (ins 940115)
|
|
pop ax ; (ins 940115)
|
|
je GetDEB30 ; N: Error (ins 940115)
|
|
@@: ; Y: Next (ins 940115)
|
|
; (ins 931225)
|
|
cmp ah, ID_LPT3 ; Open LPT3 ? (ins 931225)
|
|
jne @f ; N: NEXT (ins 931225)
|
|
push dx ; (ins 931225)
|
|
push ax ; (ins 931225)
|
|
mov dx, 54fh ; (ins 931225)
|
|
in al, dx ; (ins 931225)
|
|
test al, 00000100b ; Ex board=LPT1/2 ? (ins 931225)
|
|
pop ax ; (ins 931225)
|
|
pop dx ; (ins 931225)
|
|
jz GetDEB30 ; Y: error (ins 931225)
|
|
@@: ;
|
|
endif ; NEC_98
|
|
mov si, DataOFFSET LPT1
|
|
jcxz GetDEB20
|
|
GetDEB15:
|
|
add si, SIZE LptDEB
|
|
loop GetDEB15
|
|
GetDEB20:
|
|
pop cx
|
|
or ah, ah ; clear Carry & set S, if LPT port
|
|
ret
|
|
|
|
GetDEB30:
|
|
pop cx
|
|
mov ax,8000h ;Set error code
|
|
stc ;Set 'C' to show error
|
|
ret
|
|
|
|
GetDEB endp
|
|
page
|
|
|
|
|
|
CvtHex proc near
|
|
; assume DS=SS
|
|
ifndef NEC_98
|
|
push si
|
|
mov cl, 4
|
|
mov si, di
|
|
xor dx, dx
|
|
cld
|
|
ch_lp:
|
|
lodsb
|
|
sub al, '0' ;Q: char < '0'
|
|
jb ch_exit ; Y: return
|
|
cmp al, 9 ;Q: char <= '9'
|
|
jbe ch_got_digit ; Y: move digit into result
|
|
sub al, 'A' - '0' ;Q: char < 'A'
|
|
jb ch_exit ; Y: return
|
|
add al, 10
|
|
cmp al, 15 ;Q: char <= 'F'
|
|
jbe ch_got_digit ; Y: move hex char into result
|
|
sub al, 10 + 'a' - 'A' ;Q: char < 'a'
|
|
jb ch_exit ; Y: return
|
|
add al, 10
|
|
cmp al, 15 ;Q: char > 'f'
|
|
ja ch_exit ; Y: return
|
|
ch_got_digit:
|
|
shl dx, cl
|
|
or dl, al
|
|
jmp ch_lp
|
|
ch_exit:
|
|
mov ax, dx
|
|
pop si
|
|
endif ; NEC_98
|
|
ret
|
|
CvtHex endp
|
|
|
|
.286
|
|
; attempt to read base from SYSTEM.INI
|
|
GetComBase proc near
|
|
push ds ; save our DS
|
|
sub sp, 6
|
|
mov di, sp
|
|
mov byte ptr ss:[di], 0
|
|
push ds
|
|
push DataOFFSET lpCommSection
|
|
push ds
|
|
push DataOFFSET lpCommBase
|
|
push ss ; temp buffer
|
|
push di
|
|
push ss ; default = temp buffer
|
|
push di
|
|
push 5
|
|
push ds
|
|
push DataOFFSET lpSYSTEMINI
|
|
mov cx, ss ; temporarily assign DS=SS
|
|
mov ds, cx ; to allow KERNEL to thunk
|
|
assumes ds,nothing
|
|
call GetPrivateProfileString ; our segment in real mode
|
|
or ax, ax
|
|
jz short gcb_exit
|
|
call CvtHex ; DS still equal to SS
|
|
gcb_exit:
|
|
add sp, 6
|
|
pop ds ; restore our DS
|
|
assumes ds,Data
|
|
ret
|
|
GetComBase endp
|
|
|
|
GetPortIRQ proc near
|
|
push ds ; save our DS
|
|
push ds
|
|
push DataOFFSET lpCommSection
|
|
push ds
|
|
push DataOFFSET lpCommIrq
|
|
push bx
|
|
mov bl, [si.DCB_Id]
|
|
cmp bl, 4
|
|
jb @f
|
|
mov bl, 4
|
|
@@:
|
|
xor bh, bh
|
|
mov bl, [bx+default_table]
|
|
mov cx, bx
|
|
pop bx
|
|
push cx ; default
|
|
push ds
|
|
push DataOFFSET lpSYSTEMINI
|
|
mov cx, ss ; temporarily assign DS=SS
|
|
mov ds, cx ; to allow KERNEL to thunk
|
|
assumes ds,nothing
|
|
call GetPrivateProfileInt ; our segment in real mode
|
|
pop ds ; restore our DS
|
|
assumes ds,Data
|
|
ret
|
|
GetPortIRQ endp
|
|
|
|
|
|
GetPortFlags proc near
|
|
mov al, [si.DCB_Id]
|
|
.erre MAXCOM LT 9 ;only single digit port numbers supported
|
|
add al, '1'
|
|
mov [CommFIFOX], al
|
|
mov [CommDSRx], al
|
|
call GetPortFIFO
|
|
call GetPortDSR
|
|
ret
|
|
GetPortFlags endp
|
|
|
|
GetPortFIFO proc near
|
|
push ds ; save our DS
|
|
push ds
|
|
push DataOFFSET lpCommSection
|
|
push ds
|
|
push DataOFFSET lpCommFifo
|
|
push 2
|
|
push ds
|
|
push DataOFFSET lpSYSTEMINI
|
|
mov cx, ss ; temporarily assign DS=SS
|
|
mov ds, cx ; to allow KERNEL to thunk
|
|
assumes ds,nothing
|
|
call GetPrivateProfileInt ; our segment in real mode
|
|
pop ds ; restore our DS
|
|
assumes ds,Data
|
|
cmp ax, 1
|
|
ja short gpf_exit ; just check at open
|
|
jb short gpf_no_fifo ; force OFF, if = 0
|
|
or EFlags[si], fFIFOchkd ; flag as checked, to force ON
|
|
jmp short gpf_exit
|
|
|
|
gpf_no_fifo:
|
|
or EFlags[si], fNoFIFO OR fFIFOchkd ; force OFF
|
|
|
|
gpf_exit:
|
|
ret
|
|
GetPortFIFO endp
|
|
|
|
GetPortDSR proc near
|
|
push ds ; save our DS
|
|
push ds
|
|
push DataOFFSET lpCommSection
|
|
push ds
|
|
push DataOFFSET lpCommDSR
|
|
push 0
|
|
push ds
|
|
push DataOFFSET lpSYSTEMINI
|
|
mov cx, ss ; temporarily assign DS=SS
|
|
mov ds, cx ; to allow KERNEL to thunk
|
|
assumes ds,nothing
|
|
call GetPrivateProfileInt ; our segment in real mode
|
|
pop ds ; restore our DS
|
|
assumes ds,Data
|
|
or ax, ax
|
|
jz short gpd_exit
|
|
or EFlags[si], fUseDSR
|
|
|
|
gpd_exit:
|
|
ret
|
|
GetPortDSR endp
|
|
|
|
|
|
; FindCOMPort
|
|
;
|
|
; DS:SI -> DEB
|
|
;
|
|
PUBLIC FindCOMPort
|
|
FindCOMPort proc near
|
|
;
|
|
; Examine BIOS data area to get base I/O addresses for COM and LPT ports
|
|
;
|
|
push bx
|
|
push cx
|
|
push es
|
|
ifndef NEC_98
|
|
mov ax, __0040H
|
|
mov es, ax
|
|
assumes es,nothing
|
|
|
|
mov al, [si.DCB_Id]
|
|
mov ah, al
|
|
.erre MAXCOM LT 9 ;only single digit port numbers supported
|
|
add ah, '1'
|
|
mov [CommBaseX], ah
|
|
mov [CommIRQX], ah
|
|
mov [CommFIFOX], ah
|
|
mov [CommDSRx], ah
|
|
|
|
cmp al, 4
|
|
jae fcp_not_phys_com
|
|
xor ah, ah
|
|
shl ax, 1
|
|
mov bx, ax
|
|
mov ax, es:[bx+RS232B]
|
|
or ax, ax
|
|
jnz fcp_got_com_base
|
|
fcp_not_phys_com:
|
|
call GetComBase
|
|
or ax, ax
|
|
jnz fcp_got_com_base
|
|
mov bl, [si.DCB_Id]
|
|
cmp bl, 2
|
|
jne fcp_invalid ; jump, if base = 0 & com port <> com3
|
|
mov ax, 3E8h ; default COM3 to 3E8h
|
|
endif ; NEC_98
|
|
fcp_got_com_base:
|
|
push ax
|
|
call GetPortIRQ
|
|
mov dx, ax
|
|
pop ax
|
|
ifndef NEC_98
|
|
or dl, dl ;Q: non-zero IRQ?
|
|
jz fcp_invalid ; N:
|
|
cmp dl, 15 ;Q: IRQ in range?
|
|
ja fcp_invalid ; N:
|
|
endif ; NEC_98
|
|
xor dh, dh
|
|
push ax
|
|
push dx
|
|
call GetPortFIFO
|
|
call GetPortDSR
|
|
pop dx
|
|
pop ax
|
|
clc
|
|
fcp_exit:
|
|
pop es
|
|
pop cx
|
|
pop bx
|
|
ret
|
|
|
|
ifndef NEC_98
|
|
fcp_invalid:
|
|
or ax, -1
|
|
mov dx, ax
|
|
stc
|
|
jmp fcp_exit
|
|
endif ; NEC_98
|
|
|
|
FindCOMPort endp
|
|
.8086
|
|
|
|
page
|
|
;--------------------------Private Routine-----------------------------;
|
|
;
|
|
; Rotate the PIC interrupt priorities so the communication ports are
|
|
; highest priority.
|
|
;
|
|
; NOTE: Only rotates priorities on master PIC.
|
|
;
|
|
;-----------------------------------------------------------------------;
|
|
|
|
assumes ds,Data
|
|
assumes es,nothing
|
|
|
|
public Rotate_PIC
|
|
|
|
Rotate_PIC proc near
|
|
|
|
ifdef NEC_98
|
|
push ax ; [QN]|
|
|
mov al, [si.DCB_Id] ; Port is 2ch/3ch ? [QN]|
|
|
or al, al ; [QN]|
|
|
jnz rotate30 ; Y: goto Return [QN]|
|
|
xor ah,ah ; [QN]|
|
|
cmp byte ptr IntVecIntcount[si],0 ; [QN]|
|
|
je rotate10 ; [QN]|
|
|
mov al,11000011b ; IRQ3 is lower [QN]|
|
|
jmp short rotate20 ; [QN]|
|
|
rotate10: ; [QN]|
|
|
mov al,11000111b ; IRQ7 is lower [QN]|
|
|
rotate20: ; [QN]|
|
|
out 00H,al ; [QN]|
|
|
rotate30: ; [QN]|
|
|
pop ax ; [QN]|
|
|
ret ; [QN]|
|
|
else ; NEC_98
|
|
push ax
|
|
push cx
|
|
push di
|
|
|
|
mov al, 8 ; 0 - 7 rotated
|
|
mov cx, MAXCOM+1
|
|
mov di, DataOFFSET IRQhooks
|
|
rp_loop:
|
|
mov ah, [di.IRQn]
|
|
cmp ah, 0 ;End of hooked IRQ list?
|
|
je rp_set
|
|
cmp [di.VecN], 0FFh ;Hooked?
|
|
je rp_next
|
|
cmp ah, 8 ;If on slave PIC, treat as IRQ2
|
|
jb @f
|
|
mov ah, 2
|
|
@@:
|
|
cmp ah, al
|
|
jae rp_next
|
|
mov al, ah ;AL = lowest hooked comm IRQ
|
|
rp_next:
|
|
add di, SIZE IRQ_Hook_Struc
|
|
loop rp_loop
|
|
|
|
rp_set:
|
|
dec al ;Setting IRQ(n-1) as the lowest
|
|
and al, 07h ; priority makes IRQn the highest
|
|
or al, 0C0h
|
|
out INTA0, al
|
|
|
|
pop di
|
|
pop cx
|
|
pop ax
|
|
ret
|
|
endif ; NEC_98
|
|
|
|
Rotate_PIC endp
|
|
|
|
ifdef NEC_98
|
|
;-----------------------------Public-Routine---------------------------;[QN]
|
|
; [QN]
|
|
; Get_PortAddr_and_INTVect - get port address and interrupt vector [QN]
|
|
; [QN]
|
|
; The interrupt vector number and the port address are returned [QN]
|
|
; to the caller. [QN]
|
|
; [QN]
|
|
; Entry: [QN]
|
|
; DS:SI --> DEB [QN]
|
|
; CX = Port base address [QN]
|
|
; DS = Data [QN]
|
|
; Returns: [QN]
|
|
; AH = IRQ number [QN]
|
|
; AL = 8259 Mask [QN]
|
|
; DI:DX --> interrupt handler [QN]
|
|
; IMRPort = IMR Port Addr [QN]
|
|
; EOIPort = EOI Port Addr [QN]
|
|
; Mask8259 = INT mask of COM1,COM2,COM3(PIC) [QN]
|
|
; IntVecNum = COM1,COM2,COM3 Vect (0CH,etc...) [QN]
|
|
; MaskFFPort [QN]
|
|
; ReadSigPort [QN]
|
|
; DataPort [QN]
|
|
; CommandPort [QN]
|
|
; StatusPort [QN]
|
|
; TimerProcAdr= Timer Process Addr (TickEntry1,TickEntry2,TickEntry3) [QN]
|
|
; [QN]
|
|
; [QN]
|
|
; Error Returns: [QN]
|
|
; None [QN]
|
|
; Registers Destroyed: [QN]
|
|
; FLAGS [QN]
|
|
; History: [QN]
|
|
;----------------------------------------------------------------------;[QN]
|
|
assumes ds,Data ; [QN]
|
|
assumes es,nothing ; [QN]
|
|
; [QN]
|
|
public Get_PortAddr_and_INTVect ; [QN]
|
|
Get_PortAddr_and_INTVect proc near; [QN]
|
|
push bx ; [QN]
|
|
push cx ; [QN]
|
|
;------------------------------------------------------------- [QN]
|
|
; Get Port Address and INT Level [QN]
|
|
;------------------------------------------------------------- [QN]
|
|
Get_Interrupt_Dat001: ; [QN]
|
|
mov ah,[si.DCB_id] ;Get device ID from DCB [QN]
|
|
cmp ah,ID_COM1 ;Com1 ? [QN]
|
|
jne Get_Interrupt_Dat002 ; No (Com2/3 select) [QN]
|
|
;-------- Com1 -------------------------- [QN]
|
|
mov IMRPort[si],02h ;Store IMR Port Addr [QN]
|
|
mov EOIPort[si],00h ;Store EOI Port Addr [QN]
|
|
mov Mask8259[si],00010000b ;INT mask of Com1(PIC) [QN]
|
|
mov IntVecNum[si],0Ch ;Store Com1 Vect (0CH) [QN]
|
|
; [QN]
|
|
mov MaskFFPort[si],035h ; [QN]
|
|
mov ReadSigPort[si],033h ; [QN]
|
|
mov DataPort[si],030h ; [QN]
|
|
mov CommandPort[si],032h ; [QN]
|
|
mov StatusPort[si],032h ; [QN]
|
|
lea ax,TickEntry1 ;(ins 92.09.25) [QN]
|
|
mov TimerProcAdr[si],ax ;(ins 92.09.25) [QN]
|
|
jmp Get_Interrupt_Dat004 ;next [QN]
|
|
;-------- Com2 or Com3 -------------------------- [QN]
|
|
Get_Interrupt_Dat002: ; [QN]
|
|
mov MaskFFPort[si],0B2h ;COM3 (CHANGE 93.03.20) [QN]
|
|
mov ReadSigPort[si],0B2h ;COM3 (CHANGE 93.03.20) [QN]
|
|
mov DataPort[si],0B9h ;COM3 (CHANGE 93.03.20) [QN]
|
|
mov CommandPort[si],0BBh ;COM3 (CHANGE 93.03.20) [QN]
|
|
mov StatusPort[si],0BBh ;COM3 (CHANGE 93.03.20) [QN]
|
|
push ax ; [QN]
|
|
lea ax,TickEntry3 ;COM3 (CHANGE 93.03.20) [QN]
|
|
mov TimerProcAdr[si],ax ;COM3 (CHANGE 93.03.20) [QN]
|
|
pop ax ; [QN]
|
|
; [QN]
|
|
cmp ah,ID_COM3 ;Com3 ? [QN]
|
|
je Get_Interrupt_Dat3 ; YES [QN]
|
|
; [QN]
|
|
mov MaskFFPort[si],0B0h ;COM2 (CHANGE 93.03.20) [QN]
|
|
mov ReadSigPort[si],0B0h ;COM2 (CHANGE 93.03.20) [QN]
|
|
mov DataPort[si],0B1h ;COM2 (CHANGE 93.03.20) [QN]
|
|
mov CommandPort[si],0B3h ;COM2 (CHANGE 93.03.20) [QN]
|
|
mov StatusPort[si],0B3h ;COM2 (CHANGE 93.03.20) [QN]
|
|
push ax ; [QN]
|
|
lea ax,TickEntry2 ;COM2 (CHANGE 93.03.20) [QN]
|
|
mov TimerProcAdr[si],ax ;COM2 (CHANGE 93.03.20) [QN]
|
|
pop ax ; [QN]
|
|
; [QN]
|
|
Get_Interrupt_Dat3: ; [QN]
|
|
mov dx,ReadSigPort[si] ;CI/CS/CD Signal Port [QN]
|
|
in al,dx ;INT Level data save [QN]
|
|
and al,03h ;Mask of BIT 0,1 for INT [QN]
|
|
push bx ;DCB Pointer save [QN]
|
|
mov bl,al ; [QN]
|
|
xor bh,bh ; [QN]
|
|
shl bx,1 ;Made of Word Pointer [QN]
|
|
call GetPortIRQ ; Get IRQ from system.ini(Registry)
|
|
mov bx,ax
|
|
;----- Not Com1(Board is available) -----
|
|
mov dx,PORT_TBL[bx] ;
|
|
mov IMRPort[si],dx ;Get IMR Port Addr
|
|
;
|
|
mov dx,PORT_EOI[bx] ;
|
|
mov EOIPort[si],dx ;EOI set resistor
|
|
;
|
|
mov dl,MASK_TBL[bx] ;
|
|
mov Mask8259[si],dl ;Get IMR Pattern
|
|
;
|
|
shr bx,1 ; byte pointer
|
|
mov dl,VECT_TBL[bx] ;
|
|
mov IntVecNum[si],dl ;Get INT Num(0b/0d/0e/11/13/14/15)h
|
|
pop bx ;
|
|
|
|
Get_Interrupt_Dat004: ; [QN]
|
|
mov al,Mask8259[si] ; al = Pic mask data [QN]
|
|
mov ah,IntVecNum[si] ; ah = Interrupt Req [QN]
|
|
mov di,cs ; di = Int Vecter Seg [QN]
|
|
; [QN]
|
|
pop cx ; [QN]
|
|
pop bx ; [QN]
|
|
ret ; [QN]
|
|
Get_PortAddr_and_INTVect endp ; [QN]
|
|
; [QN]
|
|
;(ins end 92.08.24) ; [QN]
|
|
;;;---------------------------------------------------------------------[QN]
|
|
|
|
;(ins 93.03.24)
|
|
;-----------------------------Public-Routine----------------------------;
|
|
;
|
|
; OUT5FCheck - out 5f check logic for IOrecovery time
|
|
;
|
|
; Entry:
|
|
; out5f_flag == 0 :default
|
|
; Returns:
|
|
; out5f_flag == 0 :
|
|
; out5f_flag == 1 :out5f ok
|
|
; Error Returns:
|
|
; None
|
|
; Registers Destroyed:
|
|
; es,ax
|
|
; History:
|
|
;-----------------------------------------------------------------------;
|
|
assumes ds,Data ;
|
|
assumes es,nothing ;
|
|
;
|
|
public OUT5FCheck ;
|
|
OUT5FCheck proc near ;
|
|
push es
|
|
push ax
|
|
|
|
mov AX,40h
|
|
mov ES,AX
|
|
test byte ptr ES:[BIOS_FLAG5],80h ; NESA(OUT5F support)?
|
|
jz @F
|
|
mov out5f_flag,1
|
|
jmp short OUT5FCheckEnd
|
|
@@:
|
|
test byte ptr ES:[BIOS_FLAG7],80h ; Normal(OUT5F support)?
|
|
jz OUT5FCheckEnd
|
|
mov out5f_flag,1
|
|
public OUT5FCheckEnd
|
|
OUT5FCheckEnd:
|
|
pop ax
|
|
pop es
|
|
|
|
ret
|
|
OUT5FCheck endp ;
|
|
;(ins end 93.03.24)
|
|
|
|
;(ins 94.04.11)
|
|
;-----------------------------Public-Routine----------------------------;
|
|
; AOBACheck - Check whether AOBA on board
|
|
;
|
|
; IID_Port = 0y36H using bit:6-5 PRS(1-0) register
|
|
; (y =1) for FIFO
|
|
; Entry:
|
|
;
|
|
; Returns:
|
|
; AOBA on board EFlags[si] reset fNoFIFO
|
|
; non AOBA on board EFlags[si] set fNoFIFO
|
|
; Error Returns:
|
|
; NONE
|
|
; Registers Destroyed:
|
|
; NONE
|
|
; History: QNES T-MATUDA
|
|
;-----------------------------------------------------------------------;
|
|
assumes ds,Data
|
|
assumes es,nothing
|
|
|
|
public AOBACheck
|
|
AOBACheck proc near
|
|
push ax
|
|
push dx
|
|
push cx
|
|
|
|
xor cx,cx
|
|
mov dx,130H ;
|
|
add dx,ACE_IIDR ; dx = 136H
|
|
in al,dx ;
|
|
test al,040H ;PRS(1-0) == 1x ?
|
|
jnz AOBACheck1 ;(rel 94.05.11)
|
|
in al,dx ;
|
|
AOBACheck1:
|
|
inc cx
|
|
in al,dx ;
|
|
test al,060H ;PRS(1-0) == 00 ?
|
|
jnz NoAOBA ;non AOBA on board
|
|
in al,dx ;
|
|
and al,060H ;PRS(1-0) == 10 ?
|
|
cmp al,040H ;
|
|
jne NoAOBA ;non AOBA on board
|
|
cmp cx,2 ;
|
|
jb AOBACheck1 ;check twice ?
|
|
and EFlags[si], not(fNoFIFO) ;(ins 94.05.11)
|
|
jmp short @F
|
|
NoAOBA:
|
|
or EFlags[si], fNoFIFO ;
|
|
@@:
|
|
pop cx
|
|
pop dx
|
|
pop ax
|
|
|
|
ret
|
|
AOBACheck endp
|
|
|
|
;-----------------------------Public-Routine----------------------------;
|
|
; Set8251mode - Change to 8251 mode
|
|
;
|
|
; Entry:
|
|
;
|
|
; Returns:
|
|
;
|
|
; Error Returns:
|
|
; NONE
|
|
; Registers Destroyed:
|
|
; NONE
|
|
; History: QNES T-MATUDA
|
|
;-----------------------------------------------------------------------;
|
|
assumes ds,Data
|
|
assumes es,nothing
|
|
|
|
public Set8251mode
|
|
Set8251mode proc near
|
|
push dx ;
|
|
push ax ;
|
|
mov dx,Port[si] ;
|
|
cmp dx,0130h ;(ins 94.5.31)
|
|
je Set8251mode001 ;(ins 94.5.31)
|
|
pop ax ;(ins 94.5.31)
|
|
pop dx ;(ins 94.5.31)
|
|
ret ;(ins 94.5.31)
|
|
Set8251mode001: ;(ins 94.5.31)
|
|
add dx,ACE_FCR ;
|
|
mov al,0 ;set 8251 mode <94.5.16>
|
|
out dx,al ;
|
|
NEWIODELAY 3 ;(ins 94.04.25)
|
|
PUSH CX
|
|
Set8251mode002: ; (94.05.19)
|
|
mov cx,1000h ; (94.05.19)
|
|
Set8251mode003: ; (94.05.19)
|
|
in al,dx ; (94.05.19)
|
|
NEWIODELAY 3 ; (94.05.19)
|
|
test al,ACE_EFIFO ; (94.05.19)
|
|
jz @f ; (94.05.19)
|
|
loop Set8251mode003 ; (94.05.19)
|
|
; (94.05.19)
|
|
mov dx,132h ; (94.05.19)
|
|
in al,dx ; (94.05.19)
|
|
NEWIODELAY 3 ; (94.05.19)
|
|
mov dx,130h ; (94.05.19)
|
|
in al,dx ; (94.05.19)
|
|
NEWIODELAY 3 ; (94.05.19)
|
|
mov dx,138h ; (94.05.19)
|
|
jmp Set8251mode002 ; (94.05.19)
|
|
@@: ; (94.05.19)
|
|
POP CX
|
|
pop ax ;
|
|
pop dx ;
|
|
ret
|
|
Set8251mode endp
|
|
|
|
;-----------------------------Public-Routine----------------------------;
|
|
; SetFIFOmode - Change to FIFO mode
|
|
;
|
|
; Entry:
|
|
;
|
|
; Returns:
|
|
;
|
|
; Error Returns:
|
|
; NONE
|
|
; Registers Destroyed:
|
|
; NONE
|
|
; History: QNES T-MATUDA
|
|
;-----------------------------------------------------------------------;
|
|
assumes ds,Data
|
|
assumes es,nothing
|
|
|
|
public SetFIFOmode
|
|
SetFIFOmode proc near
|
|
push dx ;
|
|
push ax ;
|
|
mov dx,Port[si] ;
|
|
add dx,ACE_FCR ;
|
|
in al, dx ;(ins 94.5.16)
|
|
NEWIODELAY 3 ; <OUT 5F,AL> (ins 94.04.25) |
|
|
test al, 01h ;(ins 94.5.16)
|
|
jz running_8251_mode ;(ins 94.5.16)
|
|
pop ax ;(ins 94.5.16)
|
|
pop dx ;(ins 94.5.16)
|
|
ret ;(ins 94.5.16)
|
|
running_8251_mode: ;(ins 94.5.16)
|
|
;Set_FIFO == RLInt_Enable + ACE_TRIG14 + ACE_EFIFO
|
|
mov al,Set_FIFO ;set FIFO mode
|
|
out dx,al ;
|
|
NEWIODELAY 3 ; <OUT 5F,AL> (ins 94.04.25) |
|
|
pop ax ;
|
|
pop dx ;
|
|
ret
|
|
SetFIFOmode endp
|
|
|
|
public Set8251modeFar
|
|
Set8251modeFar proc far
|
|
call Set8251mode
|
|
ret
|
|
Set8251modeFar endp
|
|
|
|
public SetFIFOmodeFar
|
|
SetFIFOmodeFar proc far
|
|
call SetFIFOmode
|
|
ret
|
|
SetFIFOmodeFar endp
|
|
|
|
;(ins end 94.04.11)
|
|
|
|
TOMOE_PAT DB 16 DUP('PATCH !!') ;PATCH AREA (ins 92.11.11)
|
|
endif ; NEC_98
|
|
|
|
ifdef DEBUG
|
|
public InitCom10, InitCom20, InitCom40, InitCom50, InitCom59
|
|
public InitCom60, InitCom70, InitCom80, InitCom90, InitCom100
|
|
public TermCom10, TermCom15, TermCom20, TermCom30
|
|
ifdef NEC_98
|
|
public TermCom60, Terminate5, Terminate10, Terminate20
|
|
else ; NEC_98
|
|
public TermCom60, Terminate5, Terminate10, Terminate20, Terminate30
|
|
endif ; NEC_98
|
|
public Terminate45, Terminate49, Terminate50
|
|
public SetQue10
|
|
public SetCom5, SetCom10, SetCom20, SetCom210, SetCom220, SetCom230
|
|
public SetCom240, SetCom310, SetCom410, SetCom420, SetCom430
|
|
public SetCom440, SetCom450, SetCom460, SetCom470
|
|
public GetDEB10, GetDEB20, GetDEB30
|
|
public DCBPtr10
|
|
endif
|
|
|
|
sEnd code
|
|
|
|
page
|
|
|
|
createSeg _INIT,init,word,public,CODE
|
|
sBegin init
|
|
assumes cs,init
|
|
|
|
|
|
|
|
;------------------------------------------------------------------------------
|
|
;------------------------------------------------------------------------------
|
|
IBMmodel proc near
|
|
ifndef NEC_98
|
|
push ax
|
|
push bx
|
|
push es
|
|
|
|
mov ah, 0c0h
|
|
int 15h
|
|
jc IBMmodel_exit
|
|
|
|
assumes es,nothing
|
|
|
|
cmp by es:[bx+2], 0f8h ; PS/2 80
|
|
je IBMmodel_exit ; return carry clear
|
|
|
|
cmp by es:[bx+2], 0fch ; AT or PS/2 50 or 60
|
|
jne OldBios ; assume OldBios
|
|
|
|
cmp by es:[bx+3], 04h ; PS/2 50
|
|
je IBMmodel_exit ; return carry clear
|
|
cmp by es:[bx+3], 05h ; PS/2 60
|
|
je IBMmodel_exit ; return carry clear
|
|
|
|
OldBios:
|
|
stc
|
|
|
|
IBMmodel_exit:
|
|
pop es
|
|
pop bx
|
|
pop ax
|
|
endif ; NEC_98
|
|
ret
|
|
|
|
IBMmodel endp
|
|
|
|
cProc LoadLib, <FAR,PUBLIC,NODATA>,<si,di>
|
|
cBegin
|
|
|
|
ifndef NEC_98
|
|
push ds
|
|
mov ax, __F000H
|
|
mov ds, ax
|
|
assumes ds, ROMBios
|
|
mov al, [MachineID]
|
|
pop ds
|
|
assumes ds,Data
|
|
mov [$MachineID], al
|
|
|
|
call IBMmodel ;Q: PS/2?
|
|
jc @F ; N:
|
|
mov [default_table+2], 3 ; Y: change COM3 default IRQ to 3
|
|
@@:
|
|
|
|
push ds
|
|
mov ax, __0040H
|
|
mov ds, ax
|
|
assumes ds,nothing
|
|
cmp word ptr ds:[RS232B], 2F8h ;Q: COM2 base in COM1 slot?
|
|
pop ds
|
|
assumes ds,Data
|
|
jne @F ; N: leave IRQ default as 4
|
|
mov [default_table], 3 ; Y: change IRQ default to 3
|
|
@@:
|
|
endif ; NEC_98
|
|
|
|
mov [fVPICD], -1 ; assume no
|
|
|
|
xor di, di
|
|
mov es, di
|
|
mov ax, GET386API
|
|
mov bx, VPICD
|
|
int MULTIPLEX
|
|
mov wo [lpfnVPICD], di
|
|
mov wo [lpfnVPICD+2], es
|
|
mov ax, es
|
|
or ax, di
|
|
jz short no_VPICD ; jump if no bimodel services available
|
|
;
|
|
; version check VPICD
|
|
;
|
|
mov ax, VPICD_API_Get_Ver
|
|
call [lpfnVPICD]
|
|
%OUT version check VPICD
|
|
|
|
mov [fVPICD], 1 ; flag services are available
|
|
|
|
IFDEF No_DOSX_Bimodal_Services
|
|
jmp short skip_dosx_stuff
|
|
|
|
no_VPICD:
|
|
mov ax, __WinFlags
|
|
and al, WF_PMODE or WF_WIN286
|
|
cmp al, WF_PMODE or WF_WIN286
|
|
jne skip_dosx_stuff
|
|
|
|
.286
|
|
mov ax, Int31_Get_Version SHL 8
|
|
int 31h
|
|
test bl, 10b ;Q: processor goes to real mode
|
|
; for int reflection?
|
|
jz skip_dosx_stuff ; N:
|
|
mov [Using_DPMI], 0FFh ; Y: flag use of DPMI
|
|
|
|
mov ax, ds
|
|
cCall GetSelectorBase,<ax> ;DX:AX = segment of selector
|
|
shr ax, 4
|
|
shl dl, 4
|
|
or ah, dl ;AX now points to interrupt *segment*
|
|
push ax ;save on stack
|
|
mov ax, _INTERRUPT ;write data SEGMENT into _INTERRUPT
|
|
cCall AllocCStoDSAlias,<ax> ; code segment -- requires a data alias
|
|
mov es, ax
|
|
pop ax
|
|
mov es:[RM_IntDataSeg],ax
|
|
push ds
|
|
push es
|
|
mov ax, ds
|
|
mov es, ax
|
|
mov ax, _INTERRUPT
|
|
mov ds, ax
|
|
mov ax, (Int31_Trans_Serv SHL 8) + Trans_Call_Back
|
|
mov si, IntCodeOFFSET Entry_From_RM
|
|
mov di, DataOFFSET RM_Call_Struc
|
|
int 31h
|
|
pop es
|
|
pop ds
|
|
mov ax, 0
|
|
jnc @f
|
|
jmp short LoadExit
|
|
@@:
|
|
mov wo es:[RM_CallBack], dx
|
|
mov wo es:[RM_CallBack+2], cx
|
|
cCall FreeSelector,<es> ;don't need CS alias any longer
|
|
.8086
|
|
skip_dosx_stuff:
|
|
ELSE
|
|
no_VPICD:
|
|
ENDIF
|
|
|
|
;
|
|
; find base values for LPT ports
|
|
;
|
|
ifndef NEC_98
|
|
mov cx, __0040h
|
|
mov es, cx
|
|
mov cx, MAXLPT+1
|
|
mov si, DataOFFSET LPT1
|
|
ll_initl_lp:
|
|
mov bx, [si.BIOSPortLoc]
|
|
or bx, bx
|
|
jz ll_not_phys_lpt
|
|
mov ax, es:[bx]
|
|
or ah, ah ;Q: lpt redirected, or 0?
|
|
jz ll_not_phys_lpt ; Y:
|
|
cmp bx, LPTB ;Q: first LPT?
|
|
je ll_got_lpt_base ; Y:
|
|
cmp ax, es:[bx-2] ;Q: base same as previous (redirected)?
|
|
jne ll_got_lpt_base ; N: must be real
|
|
ll_not_phys_lpt:
|
|
%OUT attempt to read base from SYSTEM.INI
|
|
|
|
ll_got_lpt_base:
|
|
mov [si.Port], ax
|
|
loop ll_initl_lp
|
|
endif ; NEC_98
|
|
|
|
;
|
|
; create system timer for signalling chars in receive buffer
|
|
;
|
|
|
|
ifndef WOW
|
|
mov ax, 100 ; create 100msec timer
|
|
push ax
|
|
mov ax, _INTERRUPT
|
|
push ax
|
|
mov ax, IntCodeOFFSET TimerProc
|
|
push ax
|
|
call CreateSystemTimer ; ax = 0, if failed
|
|
%OUT should I display an error message here?
|
|
|
|
endif
|
|
assumes es,nothing
|
|
LoadExit:
|
|
cEnd
|
|
|
|
sEnd init
|
|
|
|
End LoadLib
|