Files
86Box/src/chipset/opti822.c

388 lines
9.2 KiB
C
Raw Normal View History

/*
* 86Box A hypervisor and IBM PC system emulator that specializes in
* running old operating systems and software designed for IBM
* PC systems and compatibles from 1981 through fairly recent
* system designs based on the PCI bus.
*
* This file is part of the 86Box distribution.
*
2022-10-31 05:44:32 +01:00
* Implementation of the OPTi 82C822 VESA Local Bus to PCI
* Bridge Interface.
*
*
*
2022-10-31 05:44:32 +01:00
* Authors: Miran Grca, <mgrca8@gmail.com>
*
* Copyright 2022 Miran Grca.
*/
#include <stdarg.h>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <wchar.h>
#define HAVE_STDARG_H
#include <86box/86box.h>
#include <86box/device.h>
2022-10-31 05:44:32 +01:00
#include <86box/io.h>
#include <86box/apm.h>
#include <86box/dma.h>
#include <86box/mem.h>
2022-10-31 05:44:32 +01:00
#include <86box/smram.h>
#include <86box/pci.h>
2022-10-31 05:44:32 +01:00
#include <86box/timer.h>
#include <86box/pic.h>
#include <86box/pit.h>
#include <86box/port_92.h>
#include <86box/hdc_ide.h>
#include <86box/hdc.h>
#include <86box/machine.h>
#include <86box/chipset.h>
2022-10-31 05:44:32 +01:00
#include <86box/spd.h>
2022-10-31 05:44:32 +01:00
#define MEM_STATE_SHADOW_R 0x01
#define MEM_STATE_SHADOW_W 0x02
#define MEM_STATE_SMRAM 0x04
2022-10-31 05:44:32 +01:00
typedef struct
{
uint8_t irq_convert,
pci_regs[256];
} opti822_t;
#define ENABLE_OPTI822_LOG 1
#ifdef ENABLE_OPTI822_LOG
int opti822_do_log = ENABLE_OPTI822_LOG;
2022-10-31 05:44:32 +01:00
static void
opti822_log(const char *fmt, ...)
{
va_list ap;
2022-09-18 17:12:38 -04:00
if (opti822_do_log) {
2022-10-31 05:44:32 +01:00
va_start(ap, fmt);
pclog_ex(fmt, ap);
va_end(ap);
}
}
#else
2022-10-31 05:44:32 +01:00
#define opti822_log(fmt, ...)
#endif
2022-10-31 05:44:32 +01:00
/* NOTE: We cheat here. The real ALi M1435 uses a level to edge triggered IRQ converter
when the most siginificant bit is set. We work around that by manipulating the
emulated PIC's ELCR register. */
static void
opti822_update_irqs(opti822_t *dev, int set)
{
2022-10-31 05:44:32 +01:00
uint8_t val;
int i, reg;
int shift, irq;
int irq_map[8] = { -1, 5, 9, 10, 11, 12, 14, 15 };
pic_t *temp_pic;
dev->irq_convert = (dev->pci_regs[0x53] & 0x08);
for (i = 0; i < 4; i++) {
reg = 0x80 + (i >> 1);
shift = (i & 1) << 2;
val = (dev->pci_regs[reg] >> shift) & 0x0f;
irq = irq_map[val & 0x07];
if (irq == -1)
continue;
temp_pic = (irq >= 8) ? &pic2 : &pic;
irq &= 7;
if (dev->irq_convert && set && (val & 0x08))
temp_pic->elcr |= (1 << irq);
else
temp_pic->elcr &= ~(1 << irq);
}
}
2022-10-31 05:44:32 +01:00
static void
2022-10-31 05:44:32 +01:00
opti822_pci_write(int func, int addr, uint8_t val, void *priv)
{
2022-09-18 17:12:38 -04:00
opti822_t *dev = (opti822_t *) priv;
2022-10-31 05:44:32 +01:00
int irq, irq_map[8] = { -1, 5, 9, 10, 11, 12, 14, 15 };
2022-09-18 17:12:38 -04:00
2022-10-31 05:44:32 +01:00
opti822_log("opti822_write(%02X, %02X, %02X)\n", func, addr, val);
if (func > 0)
return;
2022-09-18 17:12:38 -04:00
2022-10-31 05:44:32 +01:00
switch (addr) {
/* Command Register */
case 0x04:
dev->pci_regs[addr] = (val & 0x40) | 0x07;
2022-09-18 17:12:38 -04:00
break;
2022-10-31 05:44:32 +01:00
/* Status Register */
case 0x06:
if (!(dev->pci_regs[0x52] & 0x04))
dev->pci_regs[addr] = (val & 0x80);
break;
case 0x07:
dev->pci_regs[addr] &= ~(val & 0xf9);
2022-09-18 17:12:38 -04:00
break;
2022-10-31 05:44:32 +01:00
/* Master Latency Timer Register */
case 0x0d:
dev->pci_regs[addr] = val;
2022-09-18 17:12:38 -04:00
break;
case 0x40:
2022-10-31 05:44:32 +01:00
dev->pci_regs[addr] = (val & 0xc0) | 0x01;
2022-09-18 17:12:38 -04:00
break;
case 0x41:
2022-10-31 05:44:32 +01:00
/* TODO: Bit 15th enable the PCI Bridge when 1. */
dev->pci_regs[addr] = val & 0xcf;
2022-09-18 17:12:38 -04:00
break;
case 0x42:
2022-10-31 05:44:32 +01:00
dev->pci_regs[addr] = val & 0xf8;
2022-09-18 17:12:38 -04:00
break;
case 0x43:
2022-10-31 05:44:32 +01:00
dev->pci_regs[addr] = val;
2022-09-18 17:12:38 -04:00
break;
2022-10-31 05:44:32 +01:00
/* TODO: We do not currently allow separate shadow RAM mapping for PCI. */
2022-09-18 17:12:38 -04:00
case 0x45:
2022-10-31 05:44:32 +01:00
dev->pci_regs[addr] = val;
break;
2022-09-18 17:12:38 -04:00
case 0x46:
2022-10-31 05:44:32 +01:00
dev->pci_regs[addr] = val;
break;
2022-09-18 17:12:38 -04:00
case 0x47:
2022-10-31 05:44:32 +01:00
dev->pci_regs[addr] = val;
break;
/* Memory hole stuff. */
case 0x48 ... 0x51:
dev->pci_regs[addr] = val;
break;
2022-09-18 17:12:38 -04:00
case 0x52:
2022-10-31 05:44:32 +01:00
dev->pci_regs[addr] = val;
break;
2022-09-18 17:12:38 -04:00
case 0x53:
2022-10-31 05:44:32 +01:00
dev->pci_regs[addr] = val;
opti822_update_irqs(dev, 0);
opti822_update_irqs(dev, 1);
break;
case 0x54 ... 0x57:
dev->pci_regs[addr] = val;
2022-09-18 17:12:38 -04:00
break;
case 0x58:
2022-10-31 05:44:32 +01:00
dev->pci_regs[addr] = val & 0xfc;
break;
case 0x59 ... 0x5b:
dev->pci_regs[addr] = val;
2022-09-18 17:12:38 -04:00
break;
2022-10-31 05:44:32 +01:00
case 0x5c ... 0x5f:
dev->pci_regs[addr] = val;
2022-09-18 17:12:38 -04:00
break;
case 0x60:
2022-10-31 05:44:32 +01:00
dev->pci_regs[addr] = val & 0xfc;
break;
case 0x61 ... 0x63:
dev->pci_regs[addr] = val;
2022-09-18 17:12:38 -04:00
break;
2022-10-31 05:44:32 +01:00
case 0x64 ... 0x67:
dev->pci_regs[addr] = val;
2022-09-18 17:12:38 -04:00
break;
case 0x68:
2022-10-31 05:44:32 +01:00
dev->pci_regs[addr] = val & 0xfc;
break;
case 0x69 ... 0x6b:
dev->pci_regs[addr] = val;
2022-09-18 17:12:38 -04:00
break;
2022-10-31 05:44:32 +01:00
case 0x6c ... 0x6f:
dev->pci_regs[addr] = val;
2022-09-18 17:12:38 -04:00
break;
case 0x70:
2022-10-31 05:44:32 +01:00
dev->pci_regs[addr] = val & 0xfc;
2022-09-18 17:12:38 -04:00
break;
2022-10-31 05:44:32 +01:00
case 0x71 ... 0x73:
dev->pci_regs[addr] = val;
2022-09-18 17:12:38 -04:00
break;
case 0x74:
2022-10-31 05:44:32 +01:00
dev->pci_regs[addr] = val & 0xf8;
2022-09-18 17:12:38 -04:00
break;
2022-10-31 05:44:32 +01:00
/* ROMCS# and NVMCS# stuff. */
2022-09-18 17:12:38 -04:00
case 0x75:
2022-10-31 05:44:32 +01:00
dev->pci_regs[addr] = val;
break;
2022-09-18 17:12:38 -04:00
case 0x76:
2022-10-31 05:44:32 +01:00
dev->pci_regs[addr] = val;
2022-09-18 17:12:38 -04:00
break;
case 0x77:
2022-10-31 05:44:32 +01:00
dev->pci_regs[addr] = val;
2022-09-18 17:12:38 -04:00
break;
2022-10-31 05:44:32 +01:00
/* Enabling of memory blocks at ISA bus. */
2022-09-18 17:12:38 -04:00
case 0x78:
2022-10-31 05:44:32 +01:00
dev->pci_regs[addr] = val;
2022-09-18 17:12:38 -04:00
break;
case 0x79:
2022-10-31 05:44:32 +01:00
dev->pci_regs[addr] = val & 0xfc;
2022-09-18 17:12:38 -04:00
break;
case 0x7a:
2022-10-31 05:44:32 +01:00
dev->pci_regs[addr] = val;
break;
case 0x7b ... 0x7c:
dev->pci_regs[addr] = val;
break;
case 0x7d ... 0x7e:
dev->pci_regs[addr] = val;
2022-09-18 17:12:38 -04:00
break;
case 0x7f:
2022-10-31 05:44:32 +01:00
dev->pci_regs[addr] = val & 0x03;
2022-09-18 17:12:38 -04:00
break;
2022-10-31 05:44:32 +01:00
case 0x80 ... 0x81:
dev->pci_regs[addr] = val;
break;
2022-09-18 17:12:38 -04:00
case 0x82:
2022-10-31 05:44:32 +01:00
dev->pci_regs[addr] = val;
break;
case 0x84 ... 0x85:
dev->pci_regs[addr] = val;
break;
2022-09-18 17:12:38 -04:00
case 0x86:
2022-10-31 05:44:32 +01:00
dev->pci_regs[addr] = val;
2022-09-18 17:12:38 -04:00
break;
2022-10-31 05:44:32 +01:00
case 0x88 ... 0x8f:
dev->pci_regs[addr] = val;
opti822_update_irqs(dev, 0);
irq = irq_map[val & 0x07];
if (irq >= 0) {
opti822_log("Set IRQ routing: INT %c%c -> %02X\n", 0x41 + ((addr & 0x01) << 1), ((addr & 0x06) >> 1) + 1, irq);
pci_set_irq_routing(PCI_INTA + ((addr & 0x07) << 1), irq);
} else {
opti822_log("Set IRQ routing: INT %c%c -> FF\n", 0x41 + ((addr & 0x01) << 1), ((addr & 0x06) >> 1) + 1);
pci_set_irq_routing(PCI_INTA + ((addr & 0x07) << 1), PCI_IRQ_DISABLED);
}
irq = irq_map[(val >> 4) & 0x07];
if (irq >= 0) {
opti822_log("Set IRQ routing: INT %c%c -> %02X\n", 0x42 + ((addr & 0x01) << 1), ((addr & 0x06) >> 1) + 1, irq);
pci_set_irq_routing(PCI_INTB + ((addr & 0x07) << 1), irq);
} else {
opti822_log("Set IRQ routing: INT %c%c -> FF\n", 0x42 + ((addr & 0x01) << 1), ((addr & 0x06) >> 1) + 1);
pci_set_irq_routing(PCI_INTB + ((addr & 0x07) << 1), PCI_IRQ_DISABLED);
}
opti822_update_irqs(dev, 1);
break;
}
}
2022-10-31 05:44:32 +01:00
static uint8_t
2022-10-31 05:44:32 +01:00
opti822_pci_read(int func, int addr, void *priv)
{
2022-09-18 17:12:38 -04:00
opti822_t *dev = (opti822_t *) priv;
2022-10-31 05:44:32 +01:00
uint8_t ret;
ret = 0xff;
if (func == 0)
ret = dev->pci_regs[addr];
opti822_log("opti822_read(%02X, %02X) = %02X\n", func, addr, ret);
return ret;
}
2022-10-31 05:44:32 +01:00
static void
opti822_reset(void *priv)
{
2022-09-18 17:12:38 -04:00
opti822_t *dev = (opti822_t *) priv;
2022-10-31 05:44:32 +01:00
pci_set_pmc(0);
memset(dev->pci_regs, 0, 256);
dev->pci_regs[0x00] = 0x45; dev->pci_regs[0x01] = 0x10; /*OPTi*/
dev->pci_regs[0x02] = 0x22; dev->pci_regs[0x03] = 0xc8; /*82C822 PCIB*/
dev->pci_regs[0x04] = 0x07;
dev->pci_regs[0x06] = 0x80;
dev->pci_regs[0x07] = 0x02;
dev->pci_regs[0x08] = 0x01;
dev->pci_regs[0x0b] = 0x06;
dev->pci_regs[0x0d] = 0x20;
dev->pci_regs[0x40] = 0x01; dev->pci_regs[0x41] = 0x0c;
dev->pci_regs[0x43] = 0x02;
dev->pci_regs[0x52] = 0x06;
dev->pci_regs[0x53] = 0x90;
dev->irq_convert = 0;
pci_set_irq_routing(PCI_INTA, PCI_IRQ_DISABLED);
pci_set_irq_routing(PCI_INTB, PCI_IRQ_DISABLED);
pci_set_irq_routing(PCI_INTC, PCI_IRQ_DISABLED);
pci_set_irq_routing(PCI_INTD, PCI_IRQ_DISABLED);
}
2022-10-31 05:44:32 +01:00
static void
2022-10-31 05:44:32 +01:00
opti822_close(void *p)
{
2022-10-31 05:44:32 +01:00
opti822_t *dev = (opti822_t *)p;
free(dev);
}
2022-10-31 05:44:32 +01:00
static void *
opti822_init(const device_t *info)
{
2022-09-18 17:12:38 -04:00
opti822_t *dev = (opti822_t *) malloc(sizeof(opti822_t));
memset(dev, 0, sizeof(opti822_t));
2022-10-31 05:44:32 +01:00
pci_add_card(PCI_ADD_NORTHBRIDGE, opti822_pci_read, opti822_pci_write, dev);
opti822_reset(dev);
return dev;
}
const device_t opti822_device = {
2022-09-18 17:12:38 -04:00
.name = "OPTi 82C822 PCIB",
2022-03-13 09:21:08 -04:00
.internal_name = "opti822",
2022-09-18 17:12:38 -04:00
.flags = DEVICE_PCI,
.local = 0,
.init = opti822_init,
.close = opti822_close,
.reset = opti822_reset,
2022-03-13 09:21:08 -04:00
{ .available = NULL },
.speed_changed = NULL,
2022-09-18 17:12:38 -04:00
.force_redraw = NULL,
.config = NULL
2022-03-13 09:21:08 -04:00
};