209 lines
6.2 KiB
C
209 lines
6.2 KiB
C
|
|
#include "ibm.h"
|
||
|
|
#include "device.h"
|
||
|
|
#include "io.h"
|
||
|
|
#include "mem.h"
|
||
|
|
#include "nmi.h"
|
||
|
|
#include "pic.h"
|
||
|
|
#include "sound.h"
|
||
|
|
#include "sound_sn76489.h"
|
||
|
|
#include "sound_speaker.h"
|
||
|
|
#include "timer.h"
|
||
|
|
|
||
|
|
#include "keyboard.h"
|
||
|
|
#include "keyboard_pcjr.h"
|
||
|
|
|
||
|
|
#define STAT_PARITY 0x80
|
||
|
|
#define STAT_RTIMEOUT 0x40
|
||
|
|
#define STAT_TTIMEOUT 0x20
|
||
|
|
#define STAT_LOCK 0x10
|
||
|
|
#define STAT_CD 0x08
|
||
|
|
#define STAT_SYSFLAG 0x04
|
||
|
|
#define STAT_IFULL 0x02
|
||
|
|
#define STAT_OFULL 0x01
|
||
|
|
|
||
|
|
struct
|
||
|
|
{
|
||
|
|
int latched;
|
||
|
|
int data;
|
||
|
|
|
||
|
|
int serial_data[44];
|
||
|
|
int serial_pos;
|
||
|
|
|
||
|
|
uint8_t pa;
|
||
|
|
uint8_t pb;
|
||
|
|
} keyboard_pcjr;
|
||
|
|
|
||
|
|
static uint8_t key_queue[16];
|
||
|
|
static int key_queue_start = 0, key_queue_end = 0;
|
||
|
|
|
||
|
|
void keyboard_pcjr_poll()
|
||
|
|
{
|
||
|
|
keybsenddelay += (220 * TIMER_USEC);
|
||
|
|
|
||
|
|
|
||
|
|
if (key_queue_start != key_queue_end && !keyboard_pcjr.serial_pos && !keyboard_pcjr.latched)
|
||
|
|
{
|
||
|
|
int c;
|
||
|
|
int p = 0;
|
||
|
|
uint8_t key = key_queue[key_queue_start];
|
||
|
|
|
||
|
|
// pclog("Reading %02X from the key queue at %i\n", key, key_queue_start);
|
||
|
|
key_queue_start = (key_queue_start + 1) & 0xf;
|
||
|
|
|
||
|
|
keyboard_pcjr.latched = 1;
|
||
|
|
|
||
|
|
keyboard_pcjr.serial_data[0] = 1; /*Start bit*/
|
||
|
|
keyboard_pcjr.serial_data[1] = 0;
|
||
|
|
|
||
|
|
for (c = 0; c < 8; c++)
|
||
|
|
{
|
||
|
|
if (key & (1 << c))
|
||
|
|
{
|
||
|
|
keyboard_pcjr.serial_data[(c + 1) * 2] = 1;
|
||
|
|
keyboard_pcjr.serial_data[(c + 1) * 2 + 1] = 0;
|
||
|
|
p++;
|
||
|
|
}
|
||
|
|
else
|
||
|
|
{
|
||
|
|
keyboard_pcjr.serial_data[(c + 1) * 2] = 0;
|
||
|
|
keyboard_pcjr.serial_data[(c + 1) * 2 + 1] = 1;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
if (p & 1) /*Parity*/
|
||
|
|
{
|
||
|
|
keyboard_pcjr.serial_data[9 * 2] = 1;
|
||
|
|
keyboard_pcjr.serial_data[9 * 2 + 1] = 0;
|
||
|
|
}
|
||
|
|
else
|
||
|
|
{
|
||
|
|
keyboard_pcjr.serial_data[9 * 2] = 0;
|
||
|
|
keyboard_pcjr.serial_data[9 * 2 + 1] = 1;
|
||
|
|
}
|
||
|
|
|
||
|
|
for (c = 0; c < 11; c++) /*11 stop bits*/
|
||
|
|
{
|
||
|
|
keyboard_pcjr.serial_data[(c + 10) * 2] = 0;
|
||
|
|
keyboard_pcjr.serial_data[(c + 10) * 2 + 1] = 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
keyboard_pcjr.serial_pos++;
|
||
|
|
}
|
||
|
|
|
||
|
|
if (keyboard_pcjr.serial_pos)
|
||
|
|
{
|
||
|
|
keyboard_pcjr.data = keyboard_pcjr.serial_data[keyboard_pcjr.serial_pos - 1];
|
||
|
|
nmi = keyboard_pcjr.data;
|
||
|
|
keyboard_pcjr.serial_pos++;
|
||
|
|
if (keyboard_pcjr.serial_pos == 42+1)
|
||
|
|
keyboard_pcjr.serial_pos = 0;
|
||
|
|
// pclog("Keyboard poll %i %i\n", keyboard_pcjr.data, keyboard_pcjr.serial_pos);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
void keyboard_pcjr_adddata(uint8_t val)
|
||
|
|
{
|
||
|
|
key_queue[key_queue_end] = val;
|
||
|
|
// pclog("keyboard_pcjr : %02X added to key queue at %i\n", val, key_queue_end);
|
||
|
|
key_queue_end = (key_queue_end + 1) & 0xf;
|
||
|
|
return;
|
||
|
|
}
|
||
|
|
|
||
|
|
void keyboard_pcjr_write(uint16_t port, uint8_t val, void *priv)
|
||
|
|
{
|
||
|
|
// pclog("keyboard_pcjr : write %04X %02X %02X\n", port, val, keyboard_pcjr.pb);
|
||
|
|
/* if (ram[8] == 0xc3)
|
||
|
|
{
|
||
|
|
output = 3;
|
||
|
|
}*/
|
||
|
|
switch (port)
|
||
|
|
{
|
||
|
|
case 0x60:
|
||
|
|
keyboard_pcjr.pa = val;
|
||
|
|
break;
|
||
|
|
|
||
|
|
case 0x61:
|
||
|
|
keyboard_pcjr.pb = val;
|
||
|
|
|
||
|
|
timer_process();
|
||
|
|
timer_update_outstanding();
|
||
|
|
|
||
|
|
speaker_update();
|
||
|
|
speaker_gated = val & 1;
|
||
|
|
speaker_enable = val & 2;
|
||
|
|
if (speaker_enable)
|
||
|
|
was_speaker_enable = 1;
|
||
|
|
pit_set_gate(2, val & 1);
|
||
|
|
sn76489_mute = speaker_mute = 1;
|
||
|
|
switch (val & 0x60)
|
||
|
|
{
|
||
|
|
case 0x00:
|
||
|
|
speaker_mute = 0;
|
||
|
|
break;
|
||
|
|
case 0x60:
|
||
|
|
sn76489_mute = 0;
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
break;
|
||
|
|
|
||
|
|
case 0xa0:
|
||
|
|
nmi_mask = val & 0x80;
|
||
|
|
pit_set_using_timer(1, !(val & 0x20));
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
uint8_t keyboard_pcjr_read(uint16_t port, void *priv)
|
||
|
|
{
|
||
|
|
uint8_t temp;
|
||
|
|
// pclog("keyboard_pcjr : read %04X ", port);
|
||
|
|
switch (port)
|
||
|
|
{
|
||
|
|
case 0x60:
|
||
|
|
temp = keyboard_pcjr.pa;
|
||
|
|
break;
|
||
|
|
|
||
|
|
case 0x61:
|
||
|
|
temp = keyboard_pcjr.pb;
|
||
|
|
break;
|
||
|
|
|
||
|
|
case 0x62:
|
||
|
|
temp = (keyboard_pcjr.latched ? 1 : 0);
|
||
|
|
temp |= 0x02; /*Modem card not installed*/
|
||
|
|
temp |= (ppispeakon ? 0x10 : 0);
|
||
|
|
temp |= (ppispeakon ? 0x20 : 0);
|
||
|
|
temp |= (keyboard_pcjr.data ? 0x40: 0);
|
||
|
|
// temp |= 0x04;
|
||
|
|
if (keyboard_pcjr.data)
|
||
|
|
temp |= 0x40;
|
||
|
|
break;
|
||
|
|
|
||
|
|
case 0xa0:
|
||
|
|
keyboard_pcjr.latched = 0;
|
||
|
|
break;
|
||
|
|
|
||
|
|
default:
|
||
|
|
pclog("\nBad XT keyboard read %04X\n", port);
|
||
|
|
//dumpregs();
|
||
|
|
//exit(-1);
|
||
|
|
}
|
||
|
|
// pclog("%02X\n", temp);
|
||
|
|
return temp;
|
||
|
|
}
|
||
|
|
|
||
|
|
void keyboard_pcjr_reset()
|
||
|
|
{
|
||
|
|
}
|
||
|
|
|
||
|
|
void keyboard_pcjr_init()
|
||
|
|
{
|
||
|
|
//return;
|
||
|
|
io_sethandler(0x0060, 0x0004, keyboard_pcjr_read, NULL, NULL, keyboard_pcjr_write, NULL, NULL, NULL);
|
||
|
|
io_sethandler(0x00a0, 0x0008, keyboard_pcjr_read, NULL, NULL, keyboard_pcjr_write, NULL, NULL, NULL);
|
||
|
|
keyboard_pcjr_reset();
|
||
|
|
keyboard_send = keyboard_pcjr_adddata;
|
||
|
|
keyboard_poll = keyboard_pcjr_poll;
|
||
|
|
|
||
|
|
timer_add(keyboard_pcjr_poll, &keybsenddelay, TIMER_ALWAYS_ENABLED, NULL);
|
||
|
|
}
|