WinUAE/pcem/keyboard_at.cpp
2022-01-08 13:30:55 +02:00

850 lines
34 KiB
C++

#include "ibm.h"
#include "io.h"
#include "mem.h"
#include "pic.h"
#include "pit.h"
#include "sound.h"
#include "sound_speaker.h"
#include "t3100e.h"
#include "timer.h"
#include "video.h"
#include "x86.h"
#include "xi8088.h"
#include "keyboard.h"
#include "keyboard_at.h"
#ifdef UAE
extern bool ps2_mouse_supported;
extern uint8_t x86_get_jumpers(void);
#endif
#define STAT_PARITY 0x80
#define STAT_RTIMEOUT 0x40
#define STAT_TTIMEOUT 0x20
#define STAT_MFULL 0x20
#define STAT_LOCK 0x10
#define STAT_CD 0x08
#define STAT_SYSFLAG 0x04
#define STAT_IFULL 0x02
#define STAT_OFULL 0x01
#define PS2_REFRESH_TIME (16 * TIMER_USEC)
#define RESET_DELAY_TIME (100 * 10) /*600ms*/
struct
{
int initialised;
int want60;
int wantirq, wantirq12;
uint8_t command;
uint8_t status;
uint8_t mem[0x20];
uint8_t out;
int out_new, out_delayed;
int scancode_set;
int translate;
int next_is_release;
uint8_t input_port;
uint8_t output_port;
uint8_t key_command;
int key_wantdata;
int last_irq;
void (*mouse_write)(uint8_t val, void *p);
void *mouse_p;
pc_timer_t refresh_timer;
int refresh;
int is_ps2;
pc_timer_t send_delay_timer;
int reset_delay;
} keyboard_at;
/*Translation table taken from https://www.win.tue.nl/~aeb/linux/kbd/scancodes-10.html#ss10.3*/
static uint8_t at_translation[256] =
{
0xff, 0x43, 0x41, 0x3f, 0x3d, 0x3b, 0x3c, 0x58, 0x64, 0x44, 0x42, 0x40, 0x3e, 0x0f, 0x29, 0x59,
0x65, 0x38, 0x2a, 0x70, 0x1d, 0x10, 0x02, 0x5a, 0x66, 0x71, 0x2c, 0x1f, 0x1e, 0x11, 0x03, 0x5b,
0x67, 0x2e, 0x2d, 0x20, 0x12, 0x05, 0x04, 0x5c, 0x68, 0x39, 0x2f, 0x21, 0x14, 0x13, 0x06, 0x5d,
0x69, 0x31, 0x30, 0x23, 0x22, 0x15, 0x07, 0x5e, 0x6a, 0x72, 0x32, 0x24, 0x16, 0x08, 0x09, 0x5f,
0x6b, 0x33, 0x25, 0x17, 0x18, 0x0b, 0x0a, 0x60, 0x6c, 0x34, 0x35, 0x26, 0x27, 0x19, 0x0c, 0x61,
0x6d, 0x73, 0x28, 0x74, 0x1a, 0x0d, 0x62, 0x6e, 0x3a, 0x36, 0x1c, 0x1b, 0x75, 0x2b, 0x63, 0x76,
0x55, 0x56, 0x77, 0x78, 0x79, 0x7a, 0x0e, 0x7b, 0x7c, 0x4f, 0x7d, 0x4b, 0x47, 0x7e, 0x7f, 0x6f,
0x52, 0x53, 0x50, 0x4c, 0x4d, 0x48, 0x01, 0x45, 0x57, 0x4e, 0x51, 0x4a, 0x37, 0x49, 0x46, 0x54,
0x80, 0x81, 0x82, 0x41, 0x54, 0x85, 0x86, 0x87, 0x88, 0x89, 0x8a, 0x8b, 0x8c, 0x8d, 0x8e, 0x8f,
0x90, 0x91, 0x92, 0x93, 0x94, 0x95, 0x96, 0x97, 0x98, 0x99, 0x9a, 0x9b, 0x9c, 0x9d, 0x9e, 0x9f,
0xa0, 0xa1, 0xa2, 0xa3, 0xa4, 0xa5, 0xa6, 0xa7, 0xa8, 0xa9, 0xaa, 0xab, 0xac, 0xad, 0xae, 0xaf,
0xb0, 0xb1, 0xb2, 0xb3, 0xb4, 0xb5, 0xb6, 0xb7, 0xb8, 0xb9, 0xba, 0xbb, 0xbc, 0xbd, 0xbe, 0xbf,
0xc0, 0xc1, 0xc2, 0xc3, 0xc4, 0xc5, 0xc6, 0xc7, 0xc8, 0xc9, 0xca, 0xcb, 0xcc, 0xcd, 0xce, 0xcf,
0xd0, 0xd1, 0xd2, 0xd3, 0xd4, 0xd5, 0xd6, 0xd7, 0xd8, 0xd9, 0xda, 0xdb, 0xdc, 0xdd, 0xde, 0xdf,
0xe0, 0xe1, 0xe2, 0xe3, 0xe4, 0xe5, 0xe6, 0xe7, 0xe8, 0xe9, 0xea, 0xeb, 0xec, 0xed, 0xee, 0xef,
0xf0, 0xf1, 0xf2, 0xf3, 0xf4, 0xf5, 0xf6, 0xf7, 0xf8, 0xf9, 0xfa, 0xfb, 0xfc, 0xfd, 0xfe, 0xff
};
static uint8_t key_ctrl_queue[16];
static int key_ctrl_queue_start = 0, key_ctrl_queue_end = 0;
static uint8_t key_queue[16];
static int key_queue_start = 0, key_queue_end = 0;
static uint8_t mouse_queue[16];
int mouse_queue_start = 0, mouse_queue_end = 0;
void keyboard_at_adddata_keyboard(uint8_t val);
void keyboard_at_poll()
{
timer_advance_u64(&keyboard_at.send_delay_timer, (100 * TIMER_USEC));
if (keyboard_at.out_new != -1 && !keyboard_at.last_irq)
{
keyboard_at.wantirq = 0;
if (keyboard_at.out_new & 0x100)
{
if (mouse_scan)
{
// pclog("keyboard_at : take IRQ12\n");
if (keyboard_at.mem[0] & 0x02)
picint(0x1000);
keyboard_at.out = keyboard_at.out_new & 0xff;
keyboard_at.out_new = -1;
keyboard_at.status |= STAT_OFULL;
keyboard_at.status &= ~STAT_IFULL;
keyboard_at.status |= STAT_MFULL;
keyboard_at.last_irq = 0x1000;
}
else
{
// pclog("keyboard_at: suppressing IRQ12\n");
keyboard_at.out_new = -1;
}
}
else
{
if (keyboard_at.mem[0] & 0x01)
picint(2);
keyboard_at.out = keyboard_at.out_new & 0xff;
keyboard_at.out_new = -1;
keyboard_at.status |= STAT_OFULL;
keyboard_at.status &= ~STAT_IFULL;
keyboard_at.status &= ~STAT_MFULL;
// pclog("keyboard_at : take IRQ1\n");
keyboard_at.last_irq = 2;
}
}
if (keyboard_at.out_new == -1 && !(keyboard_at.status & STAT_OFULL) &&
key_ctrl_queue_start != key_ctrl_queue_end)
{
keyboard_at.out_new = key_ctrl_queue[key_ctrl_queue_start] | 0x200;
key_ctrl_queue_start = (key_ctrl_queue_start + 1) & 0xf;
}
else if (!(keyboard_at.status & STAT_OFULL) && keyboard_at.out_new == -1 &&
keyboard_at.out_delayed != -1)
{
keyboard_at.out_new = keyboard_at.out_delayed;
keyboard_at.out_delayed = -1;
}
else if (!(keyboard_at.status & STAT_OFULL) && keyboard_at.out_new == -1 &&
!(keyboard_at.mem[0] & 0x10) && keyboard_at.out_delayed != -1)
{
keyboard_at.out_new = keyboard_at.out_delayed;
keyboard_at.out_delayed = -1;
}
else if (!(keyboard_at.status & STAT_OFULL) && keyboard_at.out_new == -1 && /*!(keyboard_at.mem[0] & 0x20) &&*/
mouse_queue_start != mouse_queue_end)
{
keyboard_at.out_new = mouse_queue[mouse_queue_start] | 0x100;
mouse_queue_start = (mouse_queue_start + 1) & 0xf;
}
else if (!(keyboard_at.status & STAT_OFULL) && keyboard_at.out_new == -1 &&
!(keyboard_at.mem[0] & 0x10) && key_queue_start != key_queue_end)
{
keyboard_at.out_new = key_queue[key_queue_start];
key_queue_start = (key_queue_start + 1) & 0xf;
}
if (keyboard_at.reset_delay)
{
keyboard_at.reset_delay--;
if (!keyboard_at.reset_delay)
keyboard_at_adddata_keyboard(0xaa);
}
}
void keyboard_at_adddata(uint8_t val)
{
key_ctrl_queue[key_ctrl_queue_end] = val;
key_ctrl_queue_end = (key_ctrl_queue_end + 1) & 0xf;
if (!(keyboard_at.out_new & 0x300))
{
keyboard_at.out_delayed = keyboard_at.out_new;
keyboard_at.out_new = -1;
}
}
void keyboard_at_adddata_keyboard(uint8_t val)
{
if (keyboard_at.reset_delay)
return;
/* if (val == 0x1c)
{
key_1c++;
if (key_1c == 4)
output = 3;
}*/
/* Test for T3100E 'Fn' key (Right Alt / Right Ctrl) */
if (romset == ROM_T3100E && (pcem_key[0xb8] || pcem_key[0x9d]))
{
switch (val)
{
case 0x4f: t3100e_notify_set(0x01); break; /* End */
case 0x50: t3100e_notify_set(0x02); break; /* Down */
case 0x51: t3100e_notify_set(0x03); break; /* PgDn */
case 0x52: t3100e_notify_set(0x04); break; /* Ins */
case 0x53: t3100e_notify_set(0x05); break; /* Del */
case 0x54: t3100e_notify_set(0x06); break; /* SysRQ */
case 0x45: t3100e_notify_set(0x07); break; /* NumLock */
case 0x46: t3100e_notify_set(0x08); break; /* ScrLock */
case 0x47: t3100e_notify_set(0x09); break; /* Home */
case 0x48: t3100e_notify_set(0x0A); break; /* Up */
case 0x49: t3100e_notify_set(0x0B); break; /* PgUp */
case 0x4A: t3100e_notify_set(0x0C); break; /* Keypad -*/
case 0x4B: t3100e_notify_set(0x0D); break; /* Left */
case 0x4C: t3100e_notify_set(0x0E); break; /* KP 5 */
case 0x4D: t3100e_notify_set(0x0F); break; /* Right */
}
}
if (keyboard_at.translate)
{
if (val == 0xf0)
{
keyboard_at.next_is_release = 1;
return;
}
else
{
val = at_translation[val];
if (keyboard_at.next_is_release)
val |= 0x80;
keyboard_at.next_is_release = 0;
}
}
key_queue[key_queue_end] = val;
key_queue_end = (key_queue_end + 1) & 0xf;
// pclog("keyboard_at : %02X added to key queue\n", val);
return;
}
void keyboard_at_adddata_mouse(uint8_t val)
{
mouse_queue[mouse_queue_end] = val;
mouse_queue_end = (mouse_queue_end + 1) & 0xf;
// pclog("keyboard_at : %02X added to mouse queue\n", val);
return;
}
void keyboard_at_write(uint16_t port, uint8_t val, void *priv)
{
// pclog("keyboard_at : write %04X %02X %i %02X\n", port, val, keyboard_at.key_wantdata, ram[8]);
if (romset == ROM_XI8088 && port == 0x63)
port = 0x61;
/* if (ram[8] == 0xc3)
{
output = 3;
}*/
switch (port)
{
case 0x60:
if (keyboard_at.want60)
{
/*Write to controller*/
keyboard_at.want60 = 0;
switch (keyboard_at.command)
{
case 0x60: case 0x61: case 0x62: case 0x63:
case 0x64: case 0x65: case 0x66: case 0x67:
case 0x68: case 0x69: case 0x6a: case 0x6b:
case 0x6c: case 0x6d: case 0x6e: case 0x6f:
case 0x70: case 0x71: case 0x72: case 0x73:
case 0x74: case 0x75: case 0x76: case 0x77:
case 0x78: case 0x79: case 0x7a: case 0x7b:
case 0x7c: case 0x7d: case 0x7e: case 0x7f:
keyboard_at.mem[keyboard_at.command & 0x1f] = val;
if (keyboard_at.command == 0x60)
{
if ((val & 1) && (keyboard_at.status & STAT_OFULL))
keyboard_at.wantirq = 1;
if (!(val & 1) && keyboard_at.wantirq)
keyboard_at.wantirq = 0;
#ifdef UAE
if (ps2_mouse_supported)
#endif
mouse_scan = !(val & 0x20);
#ifndef UAE // A2386SX sets this bit but if translation is allowed, keyboard stops working. Purpose unknown.
keyboard_at.translate = val & 0x40;
#endif
}
break;
case 0xb6: /* T3100e - set colour/mono switch */
if (romset == ROM_T3100E)
t3100e_mono_set(val);
break;
case 0xcb: /*AMI - set keyboard mode*/
break;
case 0xcf: /*??? - sent by MegaPC BIOS*/
break;
case 0xd1: /*Write output port*/
// pclog("Write output port - %02X %02X %04X:%04X\n", keyboard_at.output_port, val, CS, pc);
if ((keyboard_at.output_port ^ val) & 0x02) /*A20 enable change*/
{
mem_a20_key = val & 0x02;
mem_a20_recalc();
// pclog("Rammask change to %08X %02X\n", rammask, val & 0x02);
flushmmucache();
}
keyboard_at.output_port = val;
break;
case 0xd2: /*Write to keyboard output buffer*/
keyboard_at_adddata(val);
break;
case 0xd3: /*Write to mouse output buffer*/
#ifdef UAE
if (ps2_mouse_supported)
#endif
keyboard_at_adddata_mouse(val);
break;
case 0xd4: /*Write to mouse*/
#ifdef UAE
if (ps2_mouse_supported)
#endif
if (keyboard_at.mouse_write)
{
keyboard_at.mouse_write(val, keyboard_at.mouse_p);
/*Implicitly enable mouse*/
mouse_scan = 1;
keyboard_at.mem[0] &= ~0x20;
}
break;
default:
pclog("Bad AT keyboard controller 0060 write %02X command %02X\n", val, keyboard_at.command);
// dumpregs();
// exit(-1);
}
}
else
{
/*Write to keyboard*/
keyboard_at.mem[0] &= ~0x10;
if (keyboard_at.key_wantdata)
{
keyboard_at.key_wantdata = 0;
switch (keyboard_at.key_command)
{
case 0xed: /*Set/reset LEDs*/
keyboard_at_adddata_keyboard(0xfa);
break;
case 0xf0: /*Set scancode set*/
switch (val)
{
case 0: /*Read current set*/
keyboard_at_adddata_keyboard(0xfa);
switch (keyboard_at.scancode_set)
{
case SCANCODE_SET_1:
keyboard_at_adddata_keyboard(0x01);
break;
case SCANCODE_SET_2:
keyboard_at_adddata_keyboard(0x02);
break;
case SCANCODE_SET_3:
keyboard_at_adddata_keyboard(0x03);
break;
}
break;
case 1:
keyboard_at.scancode_set = SCANCODE_SET_1;
keyboard_set_scancode_set(SCANCODE_SET_1);
keyboard_at_adddata_keyboard(0xfa);
break;
case 2:
keyboard_at.scancode_set = SCANCODE_SET_2;
keyboard_set_scancode_set(SCANCODE_SET_2);
keyboard_at_adddata_keyboard(0xfa);
break;
case 3:
keyboard_at.scancode_set = SCANCODE_SET_3;
keyboard_set_scancode_set(SCANCODE_SET_3);
keyboard_at_adddata_keyboard(0xfa);
break;
default:
keyboard_at_adddata_keyboard(0xfe);
break;
}
break;
case 0xf3: /*Set typematic rate/delay*/
keyboard_at_adddata_keyboard(0xfa);
break;
default:
pclog("Bad AT keyboard 0060 write %02X command %02X\n", val, keyboard_at.key_command);
// dumpregs();
// exit(-1);
}
}
else
{
keyboard_at.key_command = val;
switch (val)
{
case 0x05: /*??? - sent by NT 4.0*/
keyboard_at_adddata_keyboard(0xfe);
break;
case 0xed: /*Set/reset LEDs*/
keyboard_at.key_wantdata = 1;
keyboard_at_adddata_keyboard(0xfa);
break;
case 0xee: /*Diagnostic echo*/
keyboard_at_adddata_keyboard(0xee);
break;
case 0xf0: /*Set scancode set*/
keyboard_at.key_wantdata = 1;
keyboard_at_adddata_keyboard(0xfa);
break;
case 0xf2: /*Read ID*/
keyboard_at_adddata_keyboard(0xfa);
#ifndef UAE // A2286/A2386 does not want these codes
keyboard_at_adddata_keyboard(0xab);
keyboard_at_adddata_keyboard(0x83);
#endif
break;
case 0xf3: /*Set typematic rate/delay*/
keyboard_at.key_wantdata = 1;
keyboard_at_adddata_keyboard(0xfa);
break;
case 0xf4: /*Enable keyboard*/
keyboard_scan = 1;
keyboard_at_adddata_keyboard(0xfa);
break;
case 0xf5: /*Disable keyboard*/
keyboard_scan = 0;
keyboard_at_adddata_keyboard(0xfa);
break;
case 0xff: /*Reset*/
key_queue_start = key_queue_end = 0; /*Clear key queue*/
keyboard_at_adddata_keyboard(0xfa);
keyboard_at.reset_delay = RESET_DELAY_TIME;
break;
default:
pclog("Bad AT keyboard command %02X\n", val);
keyboard_at_adddata_keyboard(0xfe);
// dumpregs();
// exit(-1);
}
}
}
break;
case 0x61:
ppi.pb = val;
speaker_update();
speaker_gated = val & 1;
speaker_enable = val & 2;
if (speaker_enable)
was_speaker_enable = 1;
pit_set_gate((PIT*)priv, 2, val & 1);
if (romset == ROM_XI8088)
{
if (val & 0x04)
xi8088_turbo_set(1);
else
xi8088_turbo_set(0);
}
break;
case 0x64:
keyboard_at.want60 = 0;
keyboard_at.command = val;
/*New controller command*/
switch (val)
{
case 0x20: case 0x21: case 0x22: case 0x23:
case 0x24: case 0x25: case 0x26: case 0x27:
case 0x28: case 0x29: case 0x2a: case 0x2b:
case 0x2c: case 0x2d: case 0x2e: case 0x2f:
case 0x30: case 0x31: case 0x32: case 0x33:
case 0x34: case 0x35: case 0x36: case 0x37:
case 0x38: case 0x39: case 0x3a: case 0x3b:
case 0x3c: case 0x3d: case 0x3e: case 0x3f:
keyboard_at_adddata(keyboard_at.mem[val & 0x1f]);
break;
case 0x60: case 0x61: case 0x62: case 0x63:
case 0x64: case 0x65: case 0x66: case 0x67:
case 0x68: case 0x69: case 0x6a: case 0x6b:
case 0x6c: case 0x6d: case 0x6e: case 0x6f:
case 0x70: case 0x71: case 0x72: case 0x73:
case 0x74: case 0x75: case 0x76: case 0x77:
case 0x78: case 0x79: case 0x7a: case 0x7b:
case 0x7c: case 0x7d: case 0x7e: case 0x7f:
keyboard_at.want60 = 1;
break;
case 0xa1: /*AMI - get controlled version*/
break;
case 0xa7: /*Disable mouse port*/
mouse_scan = 0;
keyboard_at.mem[0] |= 0x20;
break;
case 0xa8: /*Enable mouse port*/
#ifdef UAE
if (ps2_mouse_supported) {
#endif
mouse_scan = 1;
keyboard_at.mem[0] &= ~0x20;
#ifdef UAE
}
#endif
break;
case 0xa9: /*Test mouse port*/
#ifdef UAE
if (ps2_mouse_supported)
#endif
keyboard_at_adddata(0x00); /*no error*/
break;
case 0xaa: /*Self-test*/
if (!keyboard_at.initialised)
{
keyboard_at.initialised = 1;
key_ctrl_queue_start = key_ctrl_queue_end = 0;
keyboard_at.status &= ~STAT_OFULL;
}
/* T3100e expects STAT_IFULL to be set immediately
* after sending 0xAA */
if(romset == ROM_T3100E) keyboard_at.status |= STAT_IFULL;
keyboard_at.status |= STAT_SYSFLAG;
keyboard_at.mem[0] |= 0x04;
keyboard_at_adddata(0x55);
/*Self-test also resets the output port, enabling A20*/
if (!(keyboard_at.output_port & 0x02))
{
mem_a20_key = 2;
mem_a20_recalc();
// pclog("Rammask change to %08X %02X\n", rammask, val & 0x02);
flushmmucache();
}
keyboard_at.output_port = 0xcf;
break;
case 0xab: /*Interface test*/
keyboard_at_adddata(0x00); /*no error*/
break;
case 0xad: /*Disable keyboard*/
keyboard_at.mem[0] |= 0x10;
break;
case 0xae: /*Enable keyboard*/
keyboard_at.mem[0] &= ~0x10;
#ifdef UAE
if (!keyboard_at.initialised)
keyboard_at_adddata(0x00); // A2286 BIOS requires data in output buffer after keyboard enable
#endif
break;
case 0xb0: /* T3100e: Turbo on */
if (romset == ROM_T3100E)
t3100e_turbo_set(1);
break;
case 0xb1: /* T3100e: Turbo off */
if (romset == ROM_T3100E)
t3100e_turbo_set(0);
break;
case 0xb2: /* T3100e: Select external display */
if (romset == ROM_T3100E)
t3100e_display_set(0x00);
break;
case 0xb3: /* T3100e: Select internal display */
if (romset == ROM_T3100E)
t3100e_display_set(0x01);
break;
case 0xb4: /* T3100e: Get configuration / status */
if (romset == ROM_T3100E)
keyboard_at_adddata(t3100e_config_get());
break;
case 0xb5: /* T3100e: Get colour / mono byte */
if (romset == ROM_T3100E)
keyboard_at_adddata(t3100e_mono_get());
break;
case 0xb6: /* T3100e: Set colour / mono byte */
if (romset == ROM_T3100E)
keyboard_at.want60 = 1;
break;
case 0xba:
if (romset == ROM_ENDEAVOR || romset == ROM_ZAPPA || romset == ROM_ITAUTEC_INFOWAYM)
keyboard_at_adddata(0);
break;
/* T3100e commands not implemented:
* 0xB7: Emulate PS/2 keyboard
* 0xB8: Emulate AT keyboard */
case 0xbb: /* T3100e: Read 'Fn' key.
Return it for right Ctrl and right
Alt; on the real T3100e, these keystrokes
could only be generated using 'Fn'. */
if (romset == ROM_T3100E)
{
if (pcem_key[0xb8] || /* Right Alt */
pcem_key[0x9d]) /* Right Ctrl */
{
keyboard_at_adddata(0x04);
}
else keyboard_at_adddata(0x00);
}
break;
case 0xbc: /* T3100e: Reset Fn+Key notification */
if (romset == ROM_T3100E)
t3100e_notify_set(0x00);
break;
case 0xc0: /*Read input port*/
/* The T3100e returns all bits set except bit 6 which
* is set by t3100e_mono_set() */
if (romset == ROM_T3100E)
keyboard_at.input_port = (t3100e_mono_get() & 1) ? 0xFF : 0xBF;
#ifdef UAE
keyboard_at_adddata(x86_get_jumpers());
#else
keyboard_at_adddata(keyboard_at.input_port | 4);
#endif
keyboard_at.input_port = ((keyboard_at.input_port + 1) & 3) | (keyboard_at.input_port & 0xfc);
break;
case 0xc9: /*AMI - block P22 and P23 ??? */
break;
case 0xca: /*AMI - read keyboard mode*/
if (romset == ROM_GA686BX) /*TODO*/
keyboard_at_adddata(0x01); /*PS2 mode*/
else
keyboard_at_adddata(0x00); /*ISA mode*/
break;
case 0xcb: /*AMI - set keyboard mode*/
keyboard_at.want60 = 1;
break;
case 0xcf: /*??? - sent by MegaPC BIOS*/
keyboard_at.want60 = 1;
break;
case 0xd0: /*Read output port*/
keyboard_at_adddata(keyboard_at.output_port);
break;
case 0xd1: /*Write output port*/
keyboard_at.want60 = 1;
break;
case 0xd3: /*Write mouse output buffer*/
#ifdef UAE
if (ps2_mouse_supported)
#endif
keyboard_at.want60 = 1;
break;
case 0xd4: /*Write to mouse*/
#ifdef UAE
if (ps2_mouse_supported)
#endif
keyboard_at.want60 = 1;
break;
case 0xe0: /*Read test inputs*/
keyboard_at_adddata(0x00);
break;
// AWARD BIOS: called after turbo ON/OFF in the
// Hyundai Super-286TR and probably other AWARD 286
// bioses. Related to the Turbo LEDs. Exact function
// can't be confirmed because the system in question
// has no such leds.
//case 0xe8: /* Turbo ON, always followed by 0xba */
//case 0xe9: /* Turbo OFF */
//break;
case 0xef: /*??? - sent by AMI486*/
break;
case 0xf0: case 0xf2: case 0xf4: case 0xf6:
case 0xf8: case 0xfa: case 0xfc: case 0xfe: /*Pulse output port - pin 0 selected - x86 reset*/
softresetx86(); /*Pulse reset!*/
cpu_set_edx();
break;
case 0xff: /*Pulse output port - but no pins selected - sent by MegaPC BIOS*/
break;
default:
pclog("Bad AT keyboard controller command %02X\n", val);
// dumpregs();
// exit(-1);
}
}
}
uint8_t keyboard_at_read(uint16_t port, void *priv)
{
uint8_t temp = 0xff;
if (romset != ROM_IBMAT && romset != ROM_IBMXT286)
cycles -= ISA_CYCLES(8);
// if (port != 0x61) pclog("keyboard_at : read %04X ", port);
if (romset == ROM_XI8088 && port == 0x63)
port = 0x61;
switch (port)
{
case 0x60:
temp = keyboard_at.out;
keyboard_at.status &= ~(STAT_OFULL/* | STAT_MFULL*/);
picintc(keyboard_at.last_irq);
keyboard_at.last_irq = 0;
break;
case 0x61:
temp = ppi.pb & ~0xe0;
if (ppispeakon)
temp |= 0x20;
if (keyboard_at.is_ps2)
{
if (keyboard_at.refresh)
temp |= 0x10;
else
temp &= ~0x10;
}
if (romset == ROM_XI8088)
{
if (xi8088_turbo_get())
temp |= 0x04;
else
temp &= ~0x04;
}
break;
case 0x64:
temp = keyboard_at.status & ~4;
if (keyboard_at.mem[0] & 0x04)
temp |= 0x04;
keyboard_at.status &= ~(STAT_RTIMEOUT/* | STAT_TTIMEOUT*/);
break;
}
// if (port != 0x61) pclog("%02X %08X\n", temp, rammask);
return temp;
}
void keyboard_at_reset()
{
keyboard_at.initialised = 0;
keyboard_at.status = STAT_LOCK | STAT_CD;
keyboard_at.mem[0] = 0x11;
keyboard_at.wantirq = 0;
keyboard_at.output_port = 0xcf;
if (romset == ROM_XI8088)
keyboard_at.input_port = (video_is_mda()) ? 0xb0 : 0xf0;
else
keyboard_at.input_port = (video_is_mda()) ? 0xf0 : 0xb0;
keyboard_at.out_new = -1;
keyboard_at.out_delayed = -1;
keyboard_at.last_irq = 0;
keyboard_at.key_wantdata = 0;
keyboard_scan = 1;
}
static void at_refresh(void *p)
{
keyboard_at.refresh = !keyboard_at.refresh;
timer_advance_u64(&keyboard_at.refresh_timer, PS2_REFRESH_TIME);
}
// C++ FIX
static void keyboard_at_poll_2(void *p)
{
keyboard_at_poll();
}
// END C++
void keyboard_at_init()
{
//return;
memset(&keyboard_at, 0, sizeof(keyboard_at));
io_sethandler(0x0060, 0x0005, keyboard_at_read, NULL, NULL, keyboard_at_write, NULL, NULL, &pit);
keyboard_at_reset();
keyboard_send = keyboard_at_adddata_keyboard;
keyboard_poll = keyboard_at_poll;
keyboard_at.mouse_write = NULL;
keyboard_at.mouse_p = NULL;
keyboard_at.is_ps2 = 0;
keyboard_set_scancode_set(SCANCODE_SET_2);
keyboard_at.scancode_set = SCANCODE_SET_2;
timer_add(&keyboard_at.send_delay_timer, keyboard_at_poll_2, NULL, 1);
}
void keyboard_at_set_mouse(void (*mouse_write)(uint8_t val, void *p), void *p)
{
keyboard_at.mouse_write = mouse_write;
keyboard_at.mouse_p = p;
}
void keyboard_at_init_ps2()
{
timer_add(&keyboard_at.refresh_timer, at_refresh, NULL, 1);
keyboard_at.is_ps2 = 1;
}