480 lines
10 KiB
NASM
Raw Normal View History

2001-01-01 00:00:00 +01:00
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