mirror of
https://github.com/86Box/86Box.git
synced 2025-01-22 09:11:50 -05:00
More updates regarding the PC-98x1:
1.Fixed I/O step when using interleaving I/O. 2. Add more and fixes of the existing incomplete PC-98x1 core.
This commit is contained in:
parent
d4cf722d7d
commit
9910b6f3cd
15 changed files with 640 additions and 526 deletions
80
src/dma.c
80
src/dma.c
|
@ -37,6 +37,7 @@ dma_t dma[8];
|
||||||
uint8_t dma_e;
|
uint8_t dma_e;
|
||||||
uint8_t dma_m;
|
uint8_t dma_m;
|
||||||
|
|
||||||
|
static int dma_pc98 = 0;
|
||||||
static uint8_t dmaregs[3][16];
|
static uint8_t dmaregs[3][16];
|
||||||
static int dma_wp[2];
|
static int dma_wp[2];
|
||||||
static uint8_t dma_stat;
|
static uint8_t dma_stat;
|
||||||
|
@ -455,9 +456,13 @@ dma_sg_int_status_read(UNUSED(uint16_t addr), UNUSED(void *priv))
|
||||||
static uint8_t
|
static uint8_t
|
||||||
dma_read(uint16_t addr, UNUSED(void *priv))
|
dma_read(uint16_t addr, UNUSED(void *priv))
|
||||||
{
|
{
|
||||||
int channel = (addr >> 1) & 3;
|
int channel;
|
||||||
uint8_t temp;
|
uint8_t temp;
|
||||||
|
|
||||||
|
if (dma_pc98)
|
||||||
|
addr >>= 1;
|
||||||
|
|
||||||
|
channel = (addr >> 1) & 3;
|
||||||
switch (addr & 0xf) {
|
switch (addr & 0xf) {
|
||||||
case 0:
|
case 0:
|
||||||
case 2:
|
case 2:
|
||||||
|
@ -499,7 +504,12 @@ dma_read(uint16_t addr, UNUSED(void *priv))
|
||||||
static void
|
static void
|
||||||
dma_write(uint16_t addr, uint8_t val, UNUSED(void *priv))
|
dma_write(uint16_t addr, uint8_t val, UNUSED(void *priv))
|
||||||
{
|
{
|
||||||
int channel = (addr >> 1) & 3;
|
int channel;
|
||||||
|
|
||||||
|
if (dma_pc98)
|
||||||
|
addr >>= 1;
|
||||||
|
|
||||||
|
channel = (addr >> 1) & 3;
|
||||||
|
|
||||||
dmaregs[0][addr & 0xf] = val;
|
dmaregs[0][addr & 0xf] = val;
|
||||||
switch (addr & 0xf) {
|
switch (addr & 0xf) {
|
||||||
|
@ -943,6 +953,27 @@ dma_page_write(uint16_t addr, uint8_t val, UNUSED(void *priv))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
pc98x1_dma_page_write(uint16_t addr, uint8_t val, UNUSED(void *priv))
|
||||||
|
{
|
||||||
|
addr = ((addr >> 1) + 1) & 0x03;
|
||||||
|
dmaregs[2][addr] = val;
|
||||||
|
|
||||||
|
if (addr < 8) {
|
||||||
|
dma[addr].page_l = val;
|
||||||
|
|
||||||
|
if (addr > 4) {
|
||||||
|
dma[addr].page = val & 0xfe;
|
||||||
|
dma[addr].ab = (dma[addr].ab & 0xff01ffff & dma_mask) | (dma[addr].page << 16);
|
||||||
|
dma[addr].ac = (dma[addr].ac & 0xff01ffff & dma_mask) | (dma[addr].page << 16);
|
||||||
|
} else {
|
||||||
|
dma[addr].page = dma_at ? val : val & 0xf;
|
||||||
|
dma[addr].ab = (dma[addr].ab & 0xff00ffff & dma_mask) | (dma[addr].page << 16);
|
||||||
|
dma[addr].ac = (dma[addr].ac & 0xff00ffff & dma_mask) | (dma[addr].page << 16);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static uint8_t
|
static uint8_t
|
||||||
dma_page_read(uint16_t addr, UNUSED(void *priv))
|
dma_page_read(uint16_t addr, UNUSED(void *priv))
|
||||||
{
|
{
|
||||||
|
@ -994,6 +1025,28 @@ dma_page_read(uint16_t addr, UNUSED(void *priv))
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static uint8_t
|
||||||
|
pc98x1_dma_page_read(uint16_t addr, UNUSED(void *priv))
|
||||||
|
{
|
||||||
|
uint8_t ret = 0xff;
|
||||||
|
|
||||||
|
addr = ((addr >> 1) + 1) & 0x03;
|
||||||
|
ret = dmaregs[2][addr];
|
||||||
|
|
||||||
|
if (addr < 8)
|
||||||
|
ret = dma[addr].page_l;
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
pc98x1_dma_auto_increment_bank_write(UNUSED(uint16_t addr), uint8_t val, UNUSED(void *priv))
|
||||||
|
{
|
||||||
|
uint8_t bounds[4] = {0, 0x0f, 0, 0xff};
|
||||||
|
|
||||||
|
dma[val & 3].bound = bounds[(val >> 2) & 3];
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
dma_high_page_write(uint16_t addr, uint8_t val, UNUSED(void *priv))
|
dma_high_page_write(uint16_t addr, uint8_t val, UNUSED(void *priv))
|
||||||
{
|
{
|
||||||
|
@ -1171,6 +1224,17 @@ dma_init(void)
|
||||||
dma_ps2.is_ps2 = 0;
|
dma_ps2.is_ps2 = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
dma_pc98_init(void)
|
||||||
|
{
|
||||||
|
dma_pc98 = 1;
|
||||||
|
io_sethandler_interleaved(0x0001, 8, dma_read, NULL, NULL, dma_write, NULL, NULL, NULL);
|
||||||
|
io_sethandler_interleaved(0x0011, 8, dma_read, NULL, NULL, dma_write, NULL, NULL, NULL);
|
||||||
|
io_sethandler_interleaved(0x0021, 4, pc98x1_dma_page_read, NULL, NULL, pc98x1_dma_page_write, NULL, NULL, NULL);
|
||||||
|
io_sethandler(0x0029, 1, NULL, NULL, NULL, pc98x1_dma_auto_increment_bank_write, NULL, NULL, NULL);
|
||||||
|
dma_ps2.is_ps2 = 0;
|
||||||
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
dma16_init(void)
|
dma16_init(void)
|
||||||
{
|
{
|
||||||
|
@ -1432,6 +1496,9 @@ dma_channel_read(int channel)
|
||||||
dma_retreat(dma_c);
|
dma_retreat(dma_c);
|
||||||
else
|
else
|
||||||
dma_c->ac = (dma_c->ac & 0xffff0000 & dma_mask) | ((dma_c->ac - 1) & 0xffff);
|
dma_c->ac = (dma_c->ac & 0xffff0000 & dma_mask) | ((dma_c->ac - 1) & 0xffff);
|
||||||
|
|
||||||
|
if (dma_pc98 && dma_c->bound)
|
||||||
|
dma_c->page_l = ((dma_c->page_l - 1) & dma_c->bound) | (dma_c->page_l & (~dma_c->bound));
|
||||||
} else {
|
} else {
|
||||||
if (dma_ps2.is_ps2)
|
if (dma_ps2.is_ps2)
|
||||||
dma_c->ac++;
|
dma_c->ac++;
|
||||||
|
@ -1439,6 +1506,9 @@ dma_channel_read(int channel)
|
||||||
dma_advance(dma_c);
|
dma_advance(dma_c);
|
||||||
else
|
else
|
||||||
dma_c->ac = (dma_c->ac & 0xffff0000 & dma_mask) | ((dma_c->ac + 1) & 0xffff);
|
dma_c->ac = (dma_c->ac & 0xffff0000 & dma_mask) | ((dma_c->ac + 1) & 0xffff);
|
||||||
|
|
||||||
|
if (dma_pc98 && dma_c->bound)
|
||||||
|
dma_c->page_l = ((dma_c->page_l + 1) & dma_c->bound) | (dma_c->page_l & (~dma_c->bound));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
temp = _dma_readw(dma_c->ac, dma_c);
|
temp = _dma_readw(dma_c->ac, dma_c);
|
||||||
|
@ -1519,6 +1589,9 @@ dma_channel_write(int channel, uint16_t val)
|
||||||
dma_retreat(dma_c);
|
dma_retreat(dma_c);
|
||||||
else
|
else
|
||||||
dma_c->ac = (dma_c->ac & 0xffff0000 & dma_mask) | ((dma_c->ac - 1) & 0xffff);
|
dma_c->ac = (dma_c->ac & 0xffff0000 & dma_mask) | ((dma_c->ac - 1) & 0xffff);
|
||||||
|
|
||||||
|
if (dma_pc98 && dma_c->bound)
|
||||||
|
dma_c->page_l = ((dma_c->page_l - 1) & dma_c->bound) | (dma_c->page_l & (~dma_c->bound));
|
||||||
} else {
|
} else {
|
||||||
if (dma_ps2.is_ps2)
|
if (dma_ps2.is_ps2)
|
||||||
dma_c->ac++;
|
dma_c->ac++;
|
||||||
|
@ -1526,6 +1599,9 @@ dma_channel_write(int channel, uint16_t val)
|
||||||
dma_advance(dma_c);
|
dma_advance(dma_c);
|
||||||
else
|
else
|
||||||
dma_c->ac = (dma_c->ac & 0xffff0000 & dma_mask) | ((dma_c->ac + 1) & 0xffff);
|
dma_c->ac = (dma_c->ac & 0xffff0000 & dma_mask) | ((dma_c->ac + 1) & 0xffff);
|
||||||
|
|
||||||
|
if (dma_pc98 && dma_c->bound)
|
||||||
|
dma_c->page_l = ((dma_c->page_l + 1) & dma_c->bound) | (dma_c->page_l & (~dma_c->bound));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
_dma_writew(dma_c->ac, val, dma_c);
|
_dma_writew(dma_c->ac, val, dma_c);
|
||||||
|
|
|
@ -59,6 +59,7 @@ typedef struct dma_t {
|
||||||
uint8_t ext_mode;
|
uint8_t ext_mode;
|
||||||
uint8_t page_l;
|
uint8_t page_l;
|
||||||
uint8_t page_h;
|
uint8_t page_h;
|
||||||
|
uint8_t bound;
|
||||||
uint8_t pad;
|
uint8_t pad;
|
||||||
uint16_t cb;
|
uint16_t cb;
|
||||||
uint16_t io_addr;
|
uint16_t io_addr;
|
||||||
|
@ -81,6 +82,7 @@ extern uint8_t dma_e;
|
||||||
extern uint8_t dma_m;
|
extern uint8_t dma_m;
|
||||||
|
|
||||||
extern void dma_init(void);
|
extern void dma_init(void);
|
||||||
|
extern void dma_pc98_init(void);
|
||||||
extern void dma16_init(void);
|
extern void dma16_init(void);
|
||||||
extern void ps2_dma_init(void);
|
extern void ps2_dma_init(void);
|
||||||
extern void dma_reset(void);
|
extern void dma_reset(void);
|
||||||
|
|
|
@ -338,7 +338,6 @@ typedef struct _machine_ {
|
||||||
|
|
||||||
/*ToDo: preliminary, to improve.*/
|
/*ToDo: preliminary, to improve.*/
|
||||||
typedef struct _machine_pc98_ {
|
typedef struct _machine_pc98_ {
|
||||||
char *font_rom;
|
|
||||||
char *hdd_rom;
|
char *hdd_rom;
|
||||||
char *pci_rom;
|
char *pci_rom;
|
||||||
char *sound_rom;
|
char *sound_rom;
|
||||||
|
@ -408,6 +407,7 @@ extern uint32_t machine_handle_gpio_acpi(uint8_t write, uint32_t val);
|
||||||
|
|
||||||
/* Initialization functions for boards and systems. */
|
/* Initialization functions for boards and systems. */
|
||||||
extern void machine_common_init(const machine_t *);
|
extern void machine_common_init(const machine_t *);
|
||||||
|
extern void machine_pc98_common_init(const machine_t *);
|
||||||
|
|
||||||
/* m_amstrad.c */
|
/* m_amstrad.c */
|
||||||
extern int machine_pc1512_init(const machine_t *);
|
extern int machine_pc1512_init(const machine_t *);
|
||||||
|
|
|
@ -81,6 +81,10 @@ extern void pic_init_pcjr(void);
|
||||||
extern void pic2_init(void);
|
extern void pic2_init(void);
|
||||||
extern void pic_reset(void);
|
extern void pic_reset(void);
|
||||||
|
|
||||||
|
extern void pic_pc98_init(void);
|
||||||
|
extern void pic2_pc98_init(void);
|
||||||
|
extern void pic_pc98_reset(void);
|
||||||
|
|
||||||
extern uint8_t pic_read_icw(uint8_t pic_id, uint8_t icw);
|
extern uint8_t pic_read_icw(uint8_t pic_id, uint8_t icw);
|
||||||
extern uint8_t pic_read_ocw(uint8_t pic_id, uint8_t ocw);
|
extern uint8_t pic_read_ocw(uint8_t pic_id, uint8_t ocw);
|
||||||
extern int picint_is_level(int irq);
|
extern int picint_is_level(int irq);
|
||||||
|
|
|
@ -135,6 +135,7 @@ extern uint8_t pit_read_reg(void *priv, uint8_t reg);
|
||||||
|
|
||||||
#ifdef EMU_DEVICE_H
|
#ifdef EMU_DEVICE_H
|
||||||
extern const device_t i8253_device;
|
extern const device_t i8253_device;
|
||||||
|
extern const device_t i8253_pc98_device;
|
||||||
extern const device_t i8254_device;
|
extern const device_t i8254_device;
|
||||||
extern const device_t i8254_sec_device;
|
extern const device_t i8254_sec_device;
|
||||||
extern const device_t i8254_ext_io_device;
|
extern const device_t i8254_ext_io_device;
|
||||||
|
|
|
@ -75,6 +75,7 @@ extern const pit_intf_t pit_fast_intf;
|
||||||
|
|
||||||
#ifdef EMU_DEVICE_H
|
#ifdef EMU_DEVICE_H
|
||||||
extern const device_t i8253_fast_device;
|
extern const device_t i8253_fast_device;
|
||||||
|
extern const device_t i8253_fast_pc98_device;
|
||||||
extern const device_t i8254_fast_device;
|
extern const device_t i8254_fast_device;
|
||||||
extern const device_t i8254_sec_fast_device;
|
extern const device_t i8254_sec_fast_device;
|
||||||
extern const device_t i8254_ext_io_fast_device;
|
extern const device_t i8254_ext_io_fast_device;
|
||||||
|
|
|
@ -142,8 +142,10 @@ typedef struct pc98x1_vid_t {
|
||||||
double clock;
|
double clock;
|
||||||
} pc98x1_vid_t;
|
} pc98x1_vid_t;
|
||||||
|
|
||||||
# ifdef EMU_DEVICE_H
|
extern void pc98x1_font_init(pc98x1_vid_t *dev, char *s);
|
||||||
|
|
||||||
|
#ifdef EMU_DEVICE_H
|
||||||
extern const device_t pc98x1_vid_device;
|
extern const device_t pc98x1_vid_device;
|
||||||
# endif // EMU_DEVICE_H
|
#endif // EMU_DEVICE_H
|
||||||
|
|
||||||
#endif /*VIDEO_PC98X1_EGC_H*/
|
#endif /*VIDEO_PC98X1_EGC_H*/
|
||||||
|
|
|
@ -122,10 +122,8 @@ extern void upd7220_init(upd7220_t *dev, void *priv,
|
||||||
void (*vram_write)(uint32_t addr, uint8_t val, void *priv));
|
void (*vram_write)(uint32_t addr, uint8_t val, void *priv));
|
||||||
|
|
||||||
extern void upd7220_recalctimings(upd7220_t *dev);
|
extern void upd7220_recalctimings(upd7220_t *dev);
|
||||||
extern void upd7220_param_write(uint16_t addr, uint8_t value, void *priv);
|
extern void upd7220_write(uint16_t addr, uint8_t value, void *priv);
|
||||||
extern uint8_t upd7220_statreg_read(uint16_t addr, void *priv);
|
extern uint8_t upd7220_read(uint16_t addr, void *priv);
|
||||||
extern void upd7220_cmdreg_write(uint16_t addr, uint8_t value, void *priv);
|
|
||||||
extern uint8_t upd7220_data_read(uint16_t addr, void *priv);
|
|
||||||
extern void upd7220_reset(upd7220_t *dev);
|
extern void upd7220_reset(upd7220_t *dev);
|
||||||
|
|
||||||
#endif /*VIDEO_UPD7220_H*/
|
#endif /*VIDEO_UPD7220_H*/
|
||||||
|
|
4
src/io.c
4
src/io.c
|
@ -121,7 +121,7 @@ io_sethandler_common(uint16_t base, int size,
|
||||||
io_t *p;
|
io_t *p;
|
||||||
io_t *q = NULL;
|
io_t *q = NULL;
|
||||||
|
|
||||||
for (int c = 0; c < size; c += step) {
|
for (int c = 0; c < (size * step); c += step) {
|
||||||
p = io_last[base + c];
|
p = io_last[base + c];
|
||||||
q = (io_t *) malloc(sizeof(io_t));
|
q = (io_t *) malloc(sizeof(io_t));
|
||||||
memset(q, 0, sizeof(io_t));
|
memset(q, 0, sizeof(io_t));
|
||||||
|
@ -161,7 +161,7 @@ io_removehandler_common(uint16_t base, int size,
|
||||||
io_t *p;
|
io_t *p;
|
||||||
io_t *q;
|
io_t *q;
|
||||||
|
|
||||||
for (int c = 0; c < size; c += step) {
|
for (int c = 0; c < (size * step); c += step) {
|
||||||
p = io[base + c];
|
p = io[base + c];
|
||||||
if (!p)
|
if (!p)
|
||||||
continue;
|
continue;
|
||||||
|
|
|
@ -82,7 +82,6 @@ machine_init_ex(int m)
|
||||||
machine_snd = NULL;
|
machine_snd = NULL;
|
||||||
|
|
||||||
is_vpc = 0;
|
is_vpc = 0;
|
||||||
machine_pc98.init = 0;
|
|
||||||
|
|
||||||
standalone_gameport_type = NULL;
|
standalone_gameport_type = NULL;
|
||||||
gameport_instance_id = 0;
|
gameport_instance_id = 0;
|
||||||
|
@ -185,3 +184,21 @@ machine_common_init(UNUSED(const machine_t *model))
|
||||||
|
|
||||||
pit_common_init(pit_type, pit_irq0_timer, NULL);
|
pit_common_init(pit_type, pit_irq0_timer, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
machine_pc98_common_init(UNUSED(const machine_t *model))
|
||||||
|
{
|
||||||
|
uint8_t cpu_requires_fast_pit = is486 || (!is286 && is8086 && (cpu_s->rspeed >= 8000000));
|
||||||
|
cpu_requires_fast_pit = cpu_requires_fast_pit && !cpu_16bitbus;
|
||||||
|
|
||||||
|
/* System devices first. */
|
||||||
|
pic_pc98_init();
|
||||||
|
dma_pc98_init();
|
||||||
|
|
||||||
|
int pit_type = PIT_8253;
|
||||||
|
/* Select fast PIT if needed */
|
||||||
|
if (((pit_mode == -1) && cpu_requires_fast_pit) || (pit_mode == 1))
|
||||||
|
pit_type += 2;
|
||||||
|
|
||||||
|
pit_common_init(pit_type, pit_irq0_timer, NULL);
|
||||||
|
}
|
||||||
|
|
68
src/pic.c
68
src/pic.c
|
@ -279,6 +279,32 @@ pic_reset(void)
|
||||||
pic_pci = 0;
|
pic_pci = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
pic_pc98_reset(void)
|
||||||
|
{
|
||||||
|
memset(&pic, 0, sizeof(pic_t));
|
||||||
|
memset(&pic2, 0, sizeof(pic_t));
|
||||||
|
|
||||||
|
pic.is_master = 1;
|
||||||
|
pic.interrupt = pic2.interrupt = 0x17;
|
||||||
|
|
||||||
|
pic.slaves[7] = &pic2;
|
||||||
|
|
||||||
|
if (tmr_inited)
|
||||||
|
timer_on_auto(&pic_timer, 0.0);
|
||||||
|
memset(&pic_timer, 0x00, sizeof(pc_timer_t));
|
||||||
|
timer_add(&pic_timer, pic_callback, &pic, 0);
|
||||||
|
tmr_inited = 1;
|
||||||
|
|
||||||
|
update_pending = pic_update_pending_at;
|
||||||
|
pic.at = pic2.at = 0;
|
||||||
|
|
||||||
|
smi_irq_mask = smi_irq_status = 0x0000;
|
||||||
|
|
||||||
|
shadow = 0;
|
||||||
|
pic_pci = 0;
|
||||||
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
pic_set_shadow(int sh)
|
pic_set_shadow(int sh)
|
||||||
{
|
{
|
||||||
|
@ -637,17 +663,32 @@ pic_reset_hard(void)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
pic_pc98_reset_hard(void)
|
||||||
|
{
|
||||||
|
pic_pc98_reset();
|
||||||
|
|
||||||
|
/* Explicitly reset the latches. */
|
||||||
|
kbd_latch = mouse_latch = 0;
|
||||||
|
latched_irqs = 0x0000;
|
||||||
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
pic_init(void)
|
pic_init(void)
|
||||||
{
|
{
|
||||||
pic_reset_hard();
|
pic_reset_hard();
|
||||||
|
|
||||||
shadow = 0;
|
shadow = 0;
|
||||||
if (machine_pc98.init) {
|
io_sethandler(0x0020, 0x0002, pic_read, NULL, NULL, pic_write, NULL, NULL, &pic);
|
||||||
io_sethandler_interleaved(0x0000, 0x0001, pic_read, NULL, NULL, pic_write, NULL, NULL, &pic);
|
}
|
||||||
io_sethandler_interleaved(0x0002, 0x0001, pic_read, NULL, NULL, pic_write, NULL, NULL, &pic);
|
|
||||||
} else
|
void
|
||||||
io_sethandler(0x0020, 0x0002, pic_read, NULL, NULL, pic_write, NULL, NULL, &pic);
|
pic_pc98_init(void)
|
||||||
|
{
|
||||||
|
pic_pc98_reset_hard();
|
||||||
|
|
||||||
|
shadow = 0;
|
||||||
|
io_sethandler_interleaved(0x0000, 0x0002, pic_read, NULL, NULL, pic_write, NULL, NULL, &pic);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
@ -662,14 +703,15 @@ pic_init_pcjr(void)
|
||||||
void
|
void
|
||||||
pic2_init(void)
|
pic2_init(void)
|
||||||
{
|
{
|
||||||
if (machine_pc98.init) {
|
io_sethandler(0x00a0, 0x0002, pic_read, NULL, NULL, pic_write, NULL, NULL, &pic2);
|
||||||
io_sethandler_interleaved(0x0008, 0x0001, pic_read, NULL, NULL, pic_write, NULL, NULL, &pic2);
|
pic.slaves[2] = &pic2;
|
||||||
io_sethandler_interleaved(0x000a, 0x0001, pic_read, NULL, NULL, pic_write, NULL, NULL, &pic2);
|
}
|
||||||
pic.slaves[7] = &pic2;
|
|
||||||
} else {
|
void
|
||||||
io_sethandler(0x00a0, 0x0002, pic_read, NULL, NULL, pic_write, NULL, NULL, &pic2);
|
pic2_pc98_init(void)
|
||||||
pic.slaves[2] = &pic2;
|
{
|
||||||
}
|
io_sethandler_interleaved(0x0008, 0x0002, pic_read, NULL, NULL, pic_write, NULL, NULL, &pic2);
|
||||||
|
pic.slaves[7] = &pic2;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
|
31
src/pit.c
31
src/pit.c
|
@ -885,6 +885,23 @@ pit_init(const device_t *info)
|
||||||
return dev;
|
return dev;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void *
|
||||||
|
pit_pc98_init(const device_t *info)
|
||||||
|
{
|
||||||
|
pit_t *dev = (pit_t *) malloc(sizeof(pit_t));
|
||||||
|
pit_reset(dev);
|
||||||
|
|
||||||
|
timer_add(&dev->callback_timer, pit_timer_over, (void *) dev, 0);
|
||||||
|
timer_set_delay_u64(&dev->callback_timer, PITCONST >> 1ULL);
|
||||||
|
|
||||||
|
dev->flags = info->local;
|
||||||
|
|
||||||
|
io_sethandler_interleaved(0x0071, 0x0004, pit_read, NULL, NULL, pit_write, NULL, NULL, dev);
|
||||||
|
io_sethandler_interleaved(0x3fd9, 0x0004, pit_read, NULL, NULL, pit_write, NULL, NULL, dev);
|
||||||
|
|
||||||
|
return dev;
|
||||||
|
}
|
||||||
|
|
||||||
const device_t i8253_device = {
|
const device_t i8253_device = {
|
||||||
.name = "Intel 8253/8253-5 Programmable Interval Timer",
|
.name = "Intel 8253/8253-5 Programmable Interval Timer",
|
||||||
.internal_name = "i8253",
|
.internal_name = "i8253",
|
||||||
|
@ -899,6 +916,20 @@ const device_t i8253_device = {
|
||||||
.config = NULL
|
.config = NULL
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const device_t i8253_pc98_device = {
|
||||||
|
.name = "Intel 8253/8253-5 Programmable Interval Timer (NEC PC-98x1)",
|
||||||
|
.internal_name = "i8253_pc98",
|
||||||
|
.flags = DEVICE_CBUS | DEVICE_PIT,
|
||||||
|
.local = PIT_8253,
|
||||||
|
.init = pit_pc98_init,
|
||||||
|
.close = pit_close,
|
||||||
|
.reset = NULL,
|
||||||
|
{ .available = NULL },
|
||||||
|
.speed_changed = NULL,
|
||||||
|
.force_redraw = NULL,
|
||||||
|
.config = NULL
|
||||||
|
};
|
||||||
|
|
||||||
const device_t i8254_device = {
|
const device_t i8254_device = {
|
||||||
.name = "Intel 8254 Programmable Interval Timer",
|
.name = "Intel 8254 Programmable Interval Timer",
|
||||||
.internal_name = "i8254",
|
.internal_name = "i8254",
|
||||||
|
|
|
@ -691,6 +691,25 @@ pitf_init(const device_t *info)
|
||||||
return dev;
|
return dev;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void *
|
||||||
|
pitf_pc98_init(const device_t *info)
|
||||||
|
{
|
||||||
|
pitf_t *dev = (pitf_t *) malloc(sizeof(pitf_t));
|
||||||
|
pitf_reset(dev);
|
||||||
|
|
||||||
|
dev->flags = info->local;
|
||||||
|
|
||||||
|
for (int i = 0; i < 3; i++) {
|
||||||
|
ctrf_t *ctr = &dev->counters[i];
|
||||||
|
timer_add(&ctr->timer, pitf_timer_over, (void *) ctr, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
io_sethandler_interleaved(0x0071, 0x0004, pitf_read, NULL, NULL, pitf_write, NULL, NULL, dev);
|
||||||
|
io_sethandler_interleaved(0x3fd9, 0x0004, pitf_read, NULL, NULL, pitf_write, NULL, NULL, dev);
|
||||||
|
|
||||||
|
return dev;
|
||||||
|
}
|
||||||
|
|
||||||
const device_t i8253_fast_device = {
|
const device_t i8253_fast_device = {
|
||||||
.name = "Intel 8253/8253-5 Programmable Interval Timer",
|
.name = "Intel 8253/8253-5 Programmable Interval Timer",
|
||||||
.internal_name = "i8253_fast",
|
.internal_name = "i8253_fast",
|
||||||
|
@ -705,6 +724,20 @@ const device_t i8253_fast_device = {
|
||||||
.config = NULL
|
.config = NULL
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const device_t i8253_fast_pc98_device = {
|
||||||
|
.name = "Intel 8253/8253-5 Programmable Interval Timer (NEC PC-98x1)",
|
||||||
|
.internal_name = "i8253_pc98_fast",
|
||||||
|
.flags = DEVICE_CBUS | DEVICE_PIT,
|
||||||
|
.local = PIT_8253,
|
||||||
|
.init = pitf_pc98_init,
|
||||||
|
.close = pitf_close,
|
||||||
|
.reset = NULL,
|
||||||
|
{ .available = NULL },
|
||||||
|
.speed_changed = NULL,
|
||||||
|
.force_redraw = NULL,
|
||||||
|
.config = NULL
|
||||||
|
};
|
||||||
|
|
||||||
const device_t i8254_fast_device = {
|
const device_t i8254_fast_device = {
|
||||||
.name = "Intel 8254 Programmable Interval Timer",
|
.name = "Intel 8254 Programmable Interval Timer",
|
||||||
.internal_name = "i8254_fast",
|
.internal_name = "i8254_fast",
|
||||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -994,71 +994,63 @@ upd7220_process_cmd(upd7220_t *dev)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* i/o */
|
/* i/o */
|
||||||
|
|
||||||
void
|
void
|
||||||
upd7220_param_write(uint16_t addr, uint8_t value, void *priv)
|
upd7220_write(uint16_t addr, uint8_t value, void *priv)
|
||||||
{
|
{
|
||||||
/* ioport 0x60(chr), 0xa0(gfx) */
|
|
||||||
upd7220_t *dev = (upd7220_t *) priv;
|
upd7220_t *dev = (upd7220_t *) priv;
|
||||||
|
|
||||||
if (dev->cmdreg != -1) {
|
/* ioport 0x62(chr), 0xa2(gfx) */
|
||||||
if (dev->params_count < 16)
|
if (addr & 1) {
|
||||||
dev->params[dev->params_count++] = value;
|
if (dev->cmdreg != -1)
|
||||||
|
upd7220_process_cmd(dev);
|
||||||
|
|
||||||
|
dev->cmdreg = value;
|
||||||
|
dev->params_count = 0;
|
||||||
upd7220_check_cmd(dev);
|
upd7220_check_cmd(dev);
|
||||||
if (dev->cmdreg == -1)
|
} else {
|
||||||
dev->params_count = 0;
|
/* ioport 0x60(chr), 0xa0(gfx) */
|
||||||
|
if (dev->cmdreg != -1) {
|
||||||
|
if (dev->params_count < 16)
|
||||||
|
dev->params[dev->params_count++] = value;
|
||||||
|
|
||||||
|
upd7220_check_cmd(dev);
|
||||||
|
if (dev->cmdreg == -1)
|
||||||
|
dev->params_count = 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
uint8_t
|
uint8_t
|
||||||
upd7220_statreg_read(uint16_t addr, void *priv)
|
upd7220_read(uint16_t addr, void *priv)
|
||||||
{
|
{
|
||||||
/* ioport 0x60(chr), 0xa0(gfx) */
|
|
||||||
upd7220_t *dev = (upd7220_t *) priv;
|
upd7220_t *dev = (upd7220_t *) priv;
|
||||||
pc98x1_vid_t *vid = (pc98x1_vid_t *) dev->priv;
|
pc98x1_vid_t *vid = (pc98x1_vid_t *) dev->priv;
|
||||||
uint8_t value = dev->statreg | vid->vsync;
|
uint8_t value;
|
||||||
|
|
||||||
|
/* ioport 0x62(chr), 0xa2(gfx) */
|
||||||
|
if (addr & 1)
|
||||||
|
value = upd7220_fifo_read(dev);
|
||||||
|
else {
|
||||||
|
/* ioport 0x60(chr), 0xa0(gfx) */
|
||||||
|
value = dev->statreg | vid->vsync;
|
||||||
#if 0
|
#if 0
|
||||||
if (dev->params_count == 0)
|
if (dev->params_count == 0)
|
||||||
#endif
|
#endif
|
||||||
value |= GDC_STAT_EMPTY;
|
value |= GDC_STAT_EMPTY;
|
||||||
|
|
||||||
if (dev->params_count == 16)
|
if (dev->params_count == 16)
|
||||||
value |= GDC_STAT_FULL;
|
value |= GDC_STAT_FULL;
|
||||||
|
|
||||||
if (dev->data_count > 0)
|
if (dev->data_count > 0)
|
||||||
value |= GDC_STAT_DRDY;
|
value |= GDC_STAT_DRDY;
|
||||||
|
|
||||||
dev->statreg &= ~(GDC_STAT_DMA | GDC_STAT_DRAW);
|
dev->statreg &= ~(GDC_STAT_DMA | GDC_STAT_DRAW);
|
||||||
/* toggle hblank bit */
|
/* toggle hblank bit */
|
||||||
dev->statreg ^= GDC_STAT_HBLANK;
|
dev->statreg ^= GDC_STAT_HBLANK;
|
||||||
|
}
|
||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
|
||||||
upd7220_cmdreg_write(uint16_t addr, uint8_t value, void *priv)
|
|
||||||
{
|
|
||||||
/* ioport 0x62(chr), 0xa2(gfx) */
|
|
||||||
upd7220_t *dev = (upd7220_t *) priv;
|
|
||||||
|
|
||||||
if (dev->cmdreg != -1)
|
|
||||||
upd7220_process_cmd(dev);
|
|
||||||
|
|
||||||
dev->cmdreg = value;
|
|
||||||
dev->params_count = 0;
|
|
||||||
upd7220_check_cmd(dev);
|
|
||||||
}
|
|
||||||
|
|
||||||
uint8_t
|
|
||||||
upd7220_data_read(uint16_t addr, void *priv)
|
|
||||||
{
|
|
||||||
/* ioport 0x62(chr), 0xa2(gfx) */
|
|
||||||
upd7220_t *dev = (upd7220_t *) priv;
|
|
||||||
|
|
||||||
return upd7220_fifo_read(dev);
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
void
|
||||||
upd7220_reset(upd7220_t *dev)
|
upd7220_reset(upd7220_t *dev)
|
||||||
{
|
{
|
||||||
|
|
Loading…
Reference in a new issue