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:
TC1995 2024-02-13 01:29:00 +01:00
parent d4cf722d7d
commit 9910b6f3cd
15 changed files with 640 additions and 526 deletions

View file

@ -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);

View file

@ -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);

View file

@ -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 *);

View file

@ -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);

View file

@ -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;

View file

@ -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;

View file

@ -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*/

View file

@ -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*/

View file

@ -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;

View file

@ -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);
}

View file

@ -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

View file

@ -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",

View file

@ -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

View file

@ -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)
{ {