480 lines
10 KiB
NASM
480 lines
10 KiB
NASM
|
title "Keyboard Detection"
|
|||
|
;++
|
|||
|
;
|
|||
|
; Copyright (c) 1989 Microsoft Corporation
|
|||
|
;
|
|||
|
; Module Name:
|
|||
|
;
|
|||
|
; cpu.asm
|
|||
|
;
|
|||
|
; Abstract:
|
|||
|
;
|
|||
|
; This module implements the assembley code necessary to determine
|
|||
|
; keyboard.
|
|||
|
;
|
|||
|
; Author:
|
|||
|
;
|
|||
|
; Shie-Lin Tzong (shielint) 16-Dec-1991. Most of the code is extracted
|
|||
|
; from Win31 Setup.
|
|||
|
;
|
|||
|
; Environment:
|
|||
|
;
|
|||
|
; 80x86 Real Mode.
|
|||
|
;
|
|||
|
; Revision History:
|
|||
|
;
|
|||
|
;
|
|||
|
;--
|
|||
|
|
|||
|
extrn _READ_PORT_UCHAR:proc
|
|||
|
|
|||
|
;
|
|||
|
; Now define the needed equates.
|
|||
|
;
|
|||
|
|
|||
|
TRUE equ -1
|
|||
|
FALSE equ 0
|
|||
|
|
|||
|
;
|
|||
|
; equates for ports -- these are used in the keyboard detection module.
|
|||
|
;
|
|||
|
|
|||
|
ack_port equ 20h ; 8259 acknowledge port
|
|||
|
eoi equ 20h ; 8259 end of interrupt
|
|||
|
|
|||
|
kb_data equ 60h
|
|||
|
kb_ctl equ 61h
|
|||
|
kb_command equ 64h ;
|
|||
|
kb_status equ 64h ; status port -- bit 1: ok to write
|
|||
|
|
|||
|
ENABLE_KEYBOARD_COMMAND EQU 0AEH
|
|||
|
DISABLE_KEYBOARD_COMMAND EQU 0ADH
|
|||
|
|
|||
|
.386
|
|||
|
|
|||
|
_TEXT SEGMENT PARA USE16 PUBLIC 'CODE'
|
|||
|
ASSUME CS: _TEXT, DS:NOTHING, SS:NOTHING
|
|||
|
|
|||
|
|
|||
|
;++
|
|||
|
;
|
|||
|
; BOOLEAN
|
|||
|
; IsEnhancedKeyboard (
|
|||
|
; VOID
|
|||
|
; )
|
|||
|
;
|
|||
|
; Routine Description:
|
|||
|
;
|
|||
|
; Function looks at bios data area 40:96 to see if an enhanced keyboard
|
|||
|
; is present.
|
|||
|
;
|
|||
|
; Arguments:
|
|||
|
;
|
|||
|
; None.
|
|||
|
;
|
|||
|
; Return Value:
|
|||
|
;
|
|||
|
; TRUE if Enhanced keyboard is present. Else a value of FALSE is returned.
|
|||
|
;
|
|||
|
;--
|
|||
|
|
|||
|
Public _IsEnhancedKeyboard
|
|||
|
_IsEnhancedKeyboard proc near
|
|||
|
|
|||
|
push es
|
|||
|
|
|||
|
mov ax,40h
|
|||
|
mov es,ax
|
|||
|
xor ax,ax
|
|||
|
mov al,byte ptr es:[96h]
|
|||
|
and al,00010000b
|
|||
|
|
|||
|
pop es
|
|||
|
ret
|
|||
|
|
|||
|
_IsEnhancedKeyboard endp
|
|||
|
|
|||
|
|
|||
|
;++
|
|||
|
;
|
|||
|
; SHORT
|
|||
|
; GetKeyboardIdBytes (
|
|||
|
; PCHAR IdBuffer,
|
|||
|
; SHORT Length
|
|||
|
; )
|
|||
|
;
|
|||
|
; Routine Description:
|
|||
|
;
|
|||
|
; This routine returns keyboard identification bytes.
|
|||
|
;
|
|||
|
; Note that this function accepts one argument. The arg is a pointer to a
|
|||
|
; character buffer allocated to hold five (five) bytes. Upon return from
|
|||
|
; this function the buffer will contain between 1 and 5 bytes of keyboard
|
|||
|
; ID information. The number of valid ID bytes in the buffer is returned
|
|||
|
; in AX as a C proc return value.
|
|||
|
;
|
|||
|
; Arguments:
|
|||
|
;
|
|||
|
; None.
|
|||
|
;
|
|||
|
; Return Value:
|
|||
|
;
|
|||
|
; Id bytes are stored in IdBuffer and the length of Id bytes is returned.
|
|||
|
;
|
|||
|
;--
|
|||
|
|
|||
|
KeybID EQU [bp + 4] ; parameters
|
|||
|
ReqByte EQU [bp + 6]
|
|||
|
|
|||
|
nKeyID EQU [bp - 2] ; Local variables
|
|||
|
AckByte EQU [bp - 3]
|
|||
|
|
|||
|
public _GetKeyboardIdBytes
|
|||
|
_GetKeyboardIdBytes proc near
|
|||
|
|
|||
|
push bp
|
|||
|
mov bp, sp
|
|||
|
sub sp, 4 ; space for local variables
|
|||
|
push bx
|
|||
|
push es
|
|||
|
|
|||
|
;
|
|||
|
; First I initialize needed local vars. Next I find out if machine is AT
|
|||
|
; type or non AT type for the purpose of selecting proper acknowledgement
|
|||
|
; byte so I can talk to the interrupt controler.
|
|||
|
;
|
|||
|
|
|||
|
mov bx, KeybID ; Initialize base pointer to buffer.
|
|||
|
mov word ptr nKeyID, 0 ; Initialize count to zero.
|
|||
|
mov byte ptr AckByte, 20h ; for all but AT-like. acknowledge for
|
|||
|
; interrupt controller. 61h for AT's.
|
|||
|
mov ax, 0fff0h ; look into 0FFF0:0FE location.
|
|||
|
mov es, ax ; this is where the model byte is.
|
|||
|
mov al, byte ptr es:[0feh]
|
|||
|
cmp al, 0fch ; is it AT-like?
|
|||
|
jne UnlikeAT
|
|||
|
mov byte ptr AckByte, 61h
|
|||
|
call _Empty8042
|
|||
|
if 0
|
|||
|
;
|
|||
|
; Disable keyboard is a right thing to do. But, it turned out
|
|||
|
; this causes some keyboards to fail GetId.
|
|||
|
;
|
|||
|
|
|||
|
call DisableKeyboard
|
|||
|
endif
|
|||
|
|
|||
|
UnlikeAT:
|
|||
|
|
|||
|
;
|
|||
|
; Now, let's see if we can get some ID bytes from the keyboard controler.
|
|||
|
;
|
|||
|
|
|||
|
mov ah, ReqByte ; AT: send second command.
|
|||
|
mov dx, 60h ; write to data port
|
|||
|
call Write8042 ; Output command byte to keyboard, bytes in AH
|
|||
|
call ReadKeyboard ; Get byte from keyboard, byte in AL if No CF
|
|||
|
jc gotNoByte
|
|||
|
mov [bx], al ; save a byte. remember bx is pointer.
|
|||
|
inc word ptr nKeyID
|
|||
|
call ReadKeyboard ; Get byte from keyboard, byte in AL if No CF
|
|||
|
jc gotNoByte
|
|||
|
mov [bx]+1, al ; save a byte. remember bx is pointer.
|
|||
|
inc word ptr nKeyID
|
|||
|
call ReadKeyboard ; check for extra bytes.
|
|||
|
jc gotNoByte
|
|||
|
mov [bx]+2, al ; save a byte. remember bx is pointer.
|
|||
|
inc word ptr nKeyID
|
|||
|
|
|||
|
gotNoByte:
|
|||
|
mov al, AckByte
|
|||
|
out ack_port, al
|
|||
|
call EnableKeyboard
|
|||
|
call _Empty8042
|
|||
|
mov ax, nKeyID ; Return number of valid ID bytes obtained.
|
|||
|
|
|||
|
pop es
|
|||
|
pop bx
|
|||
|
mov sp, bp
|
|||
|
pop bp
|
|||
|
ret
|
|||
|
_GetKeyboardIdBytes endp
|
|||
|
|
|||
|
;++
|
|||
|
;
|
|||
|
; UCHAR
|
|||
|
; ReadKeyboard (
|
|||
|
; VOID
|
|||
|
; )
|
|||
|
;
|
|||
|
; Routine Description:
|
|||
|
;
|
|||
|
; This routine reads character from keyboard.
|
|||
|
;
|
|||
|
; It is assumed that a command (05H or 0F2H) has been set to request
|
|||
|
; that the keyboard send an identification byte.
|
|||
|
; It is also assumed that interrupts are disabled.
|
|||
|
;
|
|||
|
; Arguments:
|
|||
|
;
|
|||
|
; None.
|
|||
|
;
|
|||
|
; Return Value:
|
|||
|
;
|
|||
|
; If a character is returned, the carry bit is reset.
|
|||
|
; If no character is returned, the carry bit is set.
|
|||
|
;
|
|||
|
;--
|
|||
|
|
|||
|
public ReadKeyboard
|
|||
|
ReadKeyboard proc near
|
|||
|
|
|||
|
push cx
|
|||
|
push bx
|
|||
|
mov bx, 2 ; set outer timeout for double nested.
|
|||
|
cli
|
|||
|
|
|||
|
inner_timeout:
|
|||
|
xor cx, cx ; set inner timeout for double nested.
|
|||
|
|
|||
|
kbiwait:
|
|||
|
in al, kb_status ; wait for port ready
|
|||
|
test al, 1 ; ready?
|
|||
|
jnz kbiAvailable
|
|||
|
loop kbiwait
|
|||
|
|
|||
|
dec bx ; decrement outer timeout loop.
|
|||
|
jnz inner_timeout ; zero out inner loop again.
|
|||
|
stc
|
|||
|
jmp short no_byte
|
|||
|
|
|||
|
kbiAvailable: ; we received a byte.
|
|||
|
mov cx,100
|
|||
|
|
|||
|
;
|
|||
|
; We need to let some time elapse before we try to read the data
|
|||
|
; because of a problem running this code in the DOS box under
|
|||
|
; OS/2.
|
|||
|
;
|
|||
|
|
|||
|
wait_for_data:
|
|||
|
loop wait_for_data
|
|||
|
|
|||
|
in al, kb_data ; get data byte.
|
|||
|
clc
|
|||
|
|
|||
|
no_byte:
|
|||
|
sti
|
|||
|
pop bx
|
|||
|
pop cx
|
|||
|
ret
|
|||
|
|
|||
|
ReadKeyboard endp
|
|||
|
|
|||
|
|
|||
|
;++
|
|||
|
;
|
|||
|
; UCHAR
|
|||
|
; Write8042 (
|
|||
|
; USHORT Port,
|
|||
|
; UCHAR Command
|
|||
|
; )
|
|||
|
;
|
|||
|
; Routine Description:
|
|||
|
;
|
|||
|
; This routine writes command byte to keyboard.
|
|||
|
;
|
|||
|
; It is assumed that a command (05H or 0F2H) has been set to request
|
|||
|
; that the keyboard send an identification byte.
|
|||
|
; It is also assumed that interrupts are disabled.
|
|||
|
;
|
|||
|
; Arguments:
|
|||
|
;
|
|||
|
; Port (dx) - Port to write data to
|
|||
|
; Command (ah) - to be written to keyboard.
|
|||
|
;
|
|||
|
; Return Value:
|
|||
|
;
|
|||
|
; None.
|
|||
|
;
|
|||
|
;--
|
|||
|
public Write8042
|
|||
|
Write8042 proc near
|
|||
|
|
|||
|
push cx
|
|||
|
push bx
|
|||
|
mov bx, 4 ; set outer timeout for double nested.
|
|||
|
cli
|
|||
|
|
|||
|
reset_inner:
|
|||
|
xor cx, cx ; set inner timeout for double dested.
|
|||
|
|
|||
|
koutwait:
|
|||
|
in al, kb_status ; get 8042 status
|
|||
|
test al, 10b ; can we output a byte?
|
|||
|
jz ok_to_send
|
|||
|
loop koutwait
|
|||
|
|
|||
|
dec bx ; decrement outer timeout loop.
|
|||
|
jnz reset_inner ; zero out inner loop again.
|
|||
|
jmp short nosiree ; timeout expired, don't try to send.
|
|||
|
|
|||
|
ok_to_send:
|
|||
|
call _Empty8042
|
|||
|
mov al, ah ; ok, send the byte
|
|||
|
out dx, al
|
|||
|
|
|||
|
nosiree:
|
|||
|
sti
|
|||
|
pop bx
|
|||
|
pop cx
|
|||
|
ret
|
|||
|
|
|||
|
Write8042 endp
|
|||
|
|
|||
|
;++
|
|||
|
;
|
|||
|
; UCHAR
|
|||
|
; GetKeyboardFlags (
|
|||
|
; VOID
|
|||
|
; )
|
|||
|
;
|
|||
|
; Routine Description:
|
|||
|
;
|
|||
|
; This routine returns the ROM BIOS flags byte that describes the state
|
|||
|
; of the various keyboard toggles and shift keys.
|
|||
|
;
|
|||
|
; Arguments:
|
|||
|
;
|
|||
|
; None.
|
|||
|
;
|
|||
|
; Return Value:
|
|||
|
;
|
|||
|
; Keyboard ROM BIOS Flags byte.
|
|||
|
;
|
|||
|
;--
|
|||
|
|
|||
|
public _GetKeyboardFlags
|
|||
|
_GetKeyboardFlags proc near
|
|||
|
|
|||
|
mov ah, 02
|
|||
|
int 16h
|
|||
|
ret
|
|||
|
|
|||
|
_GetKeyboardFlags endp
|
|||
|
|
|||
|
|
|||
|
;++
|
|||
|
;
|
|||
|
; VOID
|
|||
|
; EnableKeyboard (
|
|||
|
; VOID
|
|||
|
; )
|
|||
|
;
|
|||
|
; Routine Description:
|
|||
|
;
|
|||
|
; This routine enables 8042 keyboard interface.
|
|||
|
;
|
|||
|
; Arguments:
|
|||
|
;
|
|||
|
; None.
|
|||
|
;
|
|||
|
; Return Value:
|
|||
|
;
|
|||
|
; None.
|
|||
|
;
|
|||
|
;--
|
|||
|
|
|||
|
public EnableKeyboard
|
|||
|
EnableKeyboard proc near
|
|||
|
|
|||
|
mov ah, ENABLE_KEYBOARD_COMMAND
|
|||
|
mov dx, kb_command
|
|||
|
call Write8042
|
|||
|
ret
|
|||
|
|
|||
|
EnableKeyboard endp
|
|||
|
|
|||
|
;++
|
|||
|
;
|
|||
|
; VOID
|
|||
|
; DisableKeyboard (
|
|||
|
; VOID
|
|||
|
; )
|
|||
|
;
|
|||
|
; Routine Description:
|
|||
|
;
|
|||
|
; This routine disables 8042 keyboard interface.
|
|||
|
;
|
|||
|
; Arguments:
|
|||
|
;
|
|||
|
; None.
|
|||
|
;
|
|||
|
; Return Value:
|
|||
|
;
|
|||
|
; None.
|
|||
|
;
|
|||
|
;--
|
|||
|
|
|||
|
public DisableKeyboard
|
|||
|
DisableKeyboard proc near
|
|||
|
|
|||
|
mov ah, DISABLE_KEYBOARD_COMMAND
|
|||
|
mov dx, kb_command
|
|||
|
call Write8042
|
|||
|
ret
|
|||
|
|
|||
|
DisableKeyboard endp
|
|||
|
|
|||
|
;++
|
|||
|
;
|
|||
|
; VOID
|
|||
|
; Empty8042 (
|
|||
|
; VOID
|
|||
|
; )
|
|||
|
;
|
|||
|
; Routine Description:
|
|||
|
;
|
|||
|
; This routine drains the i8042 controller's output buffer. This gets
|
|||
|
; rid of stale data that may have resulted from the user hitting a key
|
|||
|
; or moving the mouse, prior to the execution of keyboard initialization.
|
|||
|
;
|
|||
|
; Arguments:
|
|||
|
;
|
|||
|
; None.
|
|||
|
;
|
|||
|
; Return Value:
|
|||
|
;
|
|||
|
; None.
|
|||
|
;
|
|||
|
;--
|
|||
|
|
|||
|
public _Empty8042
|
|||
|
_Empty8042 proc near
|
|||
|
|
|||
|
push ax
|
|||
|
push dx
|
|||
|
pushf
|
|||
|
cli
|
|||
|
E8Check:
|
|||
|
in al, kb_status ; wait for port ready
|
|||
|
test al, 1 ; ready?
|
|||
|
jz E8Exit
|
|||
|
|
|||
|
mov dx, kb_data
|
|||
|
push dx
|
|||
|
call _READ_PORT_UCHAR ; use this call to delay I/O
|
|||
|
add sp, 2
|
|||
|
jmp E8Check
|
|||
|
E8Exit:
|
|||
|
popf
|
|||
|
pop dx
|
|||
|
pop ax
|
|||
|
ret
|
|||
|
|
|||
|
_Empty8042 endp
|
|||
|
|
|||
|
_TEXT ends
|
|||
|
|
|||
|
END
|