Fixed Cacodemon345's OHCI mess and implemented proper OHCI IRQ updating, fixes the Gigabyte GA-5AX POST.
This commit is contained in:
523
src/usb.c
523
src/usb.c
@@ -51,30 +51,58 @@ usb_log(const char *fmt, ...)
|
||||
/* OHCI registers */
|
||||
enum
|
||||
{
|
||||
OHCI_HcRevision = 0x00,
|
||||
OHCI_HcControl = 0x04,
|
||||
OHCI_HcCommandStatus = 0x08,
|
||||
OHCI_HcInterruptStatus = 0x0C,
|
||||
OHCI_HcInterruptEnable = 0x10,
|
||||
OHCI_HcInterruptDisable = 0x14,
|
||||
OHCI_HcHCCA = 0x18,
|
||||
OHCI_HcPeriodCurrentED = 0x1C,
|
||||
OHCI_HcControlHeadED = 0x20,
|
||||
OHCI_HcControlCurrentED = 0x24,
|
||||
OHCI_HcBulkHeadED = 0x28,
|
||||
OHCI_HcBulkCurrentED = 0x2C,
|
||||
OHCI_HcDoneHead = 0x30,
|
||||
OHCI_HcFmInterval = 0x34,
|
||||
OHCI_HcFmRemaining = 0x38,
|
||||
OHCI_HcFmNumber = 0x3C,
|
||||
OHCI_HcPeriodicStart = 0x40,
|
||||
OHCI_HcLSThreshold = 0x44,
|
||||
OHCI_HcRhDescriptorA = 0x48,
|
||||
OHCI_HcRhDescriptorB = 0x4C,
|
||||
OHCI_HcRhStatus = 0x50,
|
||||
OHCI_HcRhPortStatus1 = 0x54,
|
||||
OHCI_HcRhPortStatus2 = 0x58,
|
||||
OHCI_HcRhPortStatus3 = 0x5C
|
||||
OHCI_HcRevision = 0x00 /* 0x00 */,
|
||||
OHCI_HcControl = 0x01 /* 0x04 */,
|
||||
OHCI_HcCommandStatus = 0x02 /* 0x08 */,
|
||||
OHCI_HcInterruptStatus = 0x03 /* 0x0c */,
|
||||
OHCI_HcInterruptEnable = 0x04 /* 0x10 */,
|
||||
OHCI_HcInterruptDisable = 0x05 /* 0x14 */,
|
||||
OHCI_HcHCCA = 0x06 /* 0x18 */,
|
||||
OHCI_HcPeriodCurrentED = 0x07 /* 0x1c */,
|
||||
OHCI_HcControlHeadED = 0x08 /* 0x20 */,
|
||||
OHCI_HcControlCurrentED = 0x09 /* 0x24 */,
|
||||
OHCI_HcBulkHeadED = 0x0a /* 0x28 */,
|
||||
OHCI_HcBulkCurrentED = 0x0b /* 0x2c */,
|
||||
OHCI_HcDoneHead = 0x0c /* 0x30 */,
|
||||
OHCI_HcFmInterval = 0x0d /* 0x34 */,
|
||||
OHCI_HcFmRemaining = 0x0e /* 0x38 */,
|
||||
OHCI_HcFmNumber = 0x0f /* 0x3c */,
|
||||
OHCI_HcPeriodicStart = 0x10 /* 0x40 */,
|
||||
OHCI_HcLSThreshold = 0x11 /* 0x44 */,
|
||||
OHCI_HcRhDescriptorA = 0x12 /* 0x48 */,
|
||||
OHCI_HcRhDescriptorB = 0x13 /* 0x4c */,
|
||||
OHCI_HcRhStatus = 0x14 /* 0x50 */,
|
||||
OHCI_HcRhPortStatus1 = 0x15 /* 0x54 */,
|
||||
OHCI_HcRhPortStatus2 = 0x16 /* 0x58 */,
|
||||
OHCI_HcRhPortStatus3 = 0x17 /* 0x5c */
|
||||
};
|
||||
|
||||
enum
|
||||
{
|
||||
OHCI_aHcRevision = 0x00,
|
||||
OHCI_aHcControl = 0x04,
|
||||
OHCI_aHcCommandStatus = 0x08,
|
||||
OHCI_aHcInterruptStatus = 0x0c,
|
||||
OHCI_aHcInterruptEnable = 0x10,
|
||||
OHCI_aHcInterruptDisable = 0x14,
|
||||
OHCI_aHcHCCA = 0x18,
|
||||
OHCI_aHcPeriodCurrentED = 0x1c,
|
||||
OHCI_aHcControlHeadED = 0x20,
|
||||
OHCI_aHcControlCurrentED = 0x24,
|
||||
OHCI_aHcBulkHeadED = 0x28,
|
||||
OHCI_aHcBulkCurrentED = 0x2c,
|
||||
OHCI_aHcDoneHead = 0x30,
|
||||
OHCI_aHcFmInterval = 0x34,
|
||||
OHCI_aHcFmRemaining = 0x38,
|
||||
OHCI_aHcFmNumber = 0x3c,
|
||||
OHCI_aHcPeriodicStart = 0x40,
|
||||
OHCI_aHcLSThreshold = 0x44,
|
||||
OHCI_aHcRhDescriptorA = 0x48,
|
||||
OHCI_aHcRhDescriptorB = 0x4c,
|
||||
OHCI_aHcRhStatus = 0x50,
|
||||
OHCI_aHcRhPortStatus1 = 0x54,
|
||||
OHCI_aHcRhPortStatus2 = 0x58,
|
||||
OHCI_aHcRhPortStatus3 = 0x5c
|
||||
};
|
||||
|
||||
/* OHCI HcInterruptEnable/Disable bits */
|
||||
@@ -90,18 +118,17 @@ enum
|
||||
};
|
||||
|
||||
static void
|
||||
usb_interrupt_ohci(usb_t* usb)
|
||||
usb_interrupt_ohci(usb_t *dev, uint32_t level)
|
||||
{
|
||||
if (usb->ohci_mmio[OHCI_HcControl + 1] & 1) {
|
||||
if (usb->usb_params && usb->usb_params->smi_handle && !usb->usb_params->smi_handle(usb, usb->usb_params->parent_priv))
|
||||
if (dev->ohci_mmio[OHCI_HcControl].b[1] & 1) {
|
||||
if (dev->usb_params && dev->usb_params->smi_handle && !dev->usb_params->smi_handle(dev, dev->usb_params->parent_priv))
|
||||
return;
|
||||
|
||||
smi_raise();
|
||||
}
|
||||
else if (usb->usb_params != NULL) {
|
||||
if (usb->usb_params->parent_priv != NULL && usb->usb_params->raise_interrupt != NULL) {
|
||||
usb->usb_params->raise_interrupt(usb, usb->usb_params->parent_priv);
|
||||
}
|
||||
if (level)
|
||||
smi_raise();
|
||||
} else if (dev->usb_params != NULL) {
|
||||
if ((dev->usb_params->parent_priv != NULL) && (dev->usb_params->update_interrupt != NULL))
|
||||
dev->usb_params->update_interrupt(dev, dev->usb_params->parent_priv);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -194,26 +221,28 @@ ohci_mmio_read(uint32_t addr, void *p)
|
||||
{
|
||||
usb_t *dev = (usb_t *) p;
|
||||
uint8_t ret = 0x00;
|
||||
#ifdef ENABLE_USB_LOG
|
||||
uint32_t old_addr = addr;
|
||||
#endif
|
||||
|
||||
addr &= 0x00000fff;
|
||||
|
||||
ret = dev->ohci_mmio[addr];
|
||||
ret = dev->ohci_mmio[addr >> 2].b[addr & 3];
|
||||
|
||||
switch (addr) {
|
||||
case 0x101:
|
||||
ret = (ret & 0xfe) | (!!mem_a20_key);
|
||||
break;
|
||||
case OHCI_HcRhPortStatus1 + 1:
|
||||
case OHCI_HcRhPortStatus2 + 1:
|
||||
case OHCI_HcRhPortStatus3 + 1:
|
||||
case OHCI_aHcRhPortStatus1 + 1:
|
||||
case OHCI_aHcRhPortStatus2 + 1:
|
||||
case OHCI_aHcRhPortStatus3 + 1:
|
||||
ret |= 0x1;
|
||||
break;
|
||||
case OHCI_HcInterruptDisable:
|
||||
case OHCI_HcInterruptDisable + 1:
|
||||
case OHCI_HcInterruptDisable + 2:
|
||||
case OHCI_HcInterruptDisable + 3:
|
||||
ret = dev->ohci_mmio[OHCI_HcInterruptEnable + (addr - OHCI_HcInterruptDisable)];
|
||||
break;
|
||||
case OHCI_aHcInterruptDisable:
|
||||
case OHCI_aHcInterruptDisable + 1:
|
||||
case OHCI_aHcInterruptDisable + 2:
|
||||
case OHCI_aHcInterruptDisable + 3:
|
||||
ret = dev->ohci_mmio[OHCI_HcInterruptEnable].b[addr & 3];
|
||||
default:
|
||||
break;
|
||||
}
|
||||
@@ -221,30 +250,58 @@ ohci_mmio_read(uint32_t addr, void *p)
|
||||
if (addr == 0x101)
|
||||
ret = (ret & 0xfe) | (!!mem_a20_key);
|
||||
|
||||
#ifdef ENABLE_USB_LOG
|
||||
usb_log("[R] %08X = %04X\n", old_addr, ret);
|
||||
#endif
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
void
|
||||
ohci_set_interrupt(usb_t* usb, uint8_t bit)
|
||||
static uint16_t
|
||||
ohci_mmio_readw(uint32_t addr, void *p)
|
||||
{
|
||||
if (!(usb->ohci_mmio[OHCI_HcInterruptEnable + 3] & 0x80))
|
||||
return ohci_mmio_read(addr, p) | (ohci_mmio_read(addr + 1, p) << 8);
|
||||
}
|
||||
|
||||
static uint32_t
|
||||
ohci_mmio_readl(uint32_t addr, void *p)
|
||||
{
|
||||
return ohci_mmio_readw(addr, p) | (ohci_mmio_readw(addr + 2, p) << 16);
|
||||
}
|
||||
|
||||
static void
|
||||
ohci_update_irq(usb_t *dev)
|
||||
{
|
||||
uint32_t level = !!(dev->ohci_mmio[OHCI_HcInterruptStatus].l & dev->ohci_mmio[OHCI_HcInterruptEnable].l);
|
||||
|
||||
if (level != dev->irq_level) {
|
||||
dev->irq_level = level;
|
||||
usb_interrupt_ohci(dev, level);
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
ohci_set_interrupt(usb_t *dev, uint8_t bit)
|
||||
{
|
||||
if (!(dev->ohci_mmio[OHCI_HcInterruptEnable].b[3] & 0x80))
|
||||
return;
|
||||
|
||||
if (!(usb->ohci_mmio[OHCI_HcInterruptEnable] & bit))
|
||||
if (!(dev->ohci_mmio[OHCI_HcInterruptEnable].b[0] & bit))
|
||||
return;
|
||||
|
||||
if (usb->ohci_mmio[OHCI_HcInterruptDisable] & bit)
|
||||
if (dev->ohci_mmio[OHCI_HcInterruptDisable].b[0] & bit)
|
||||
return;
|
||||
|
||||
usb->ohci_mmio[OHCI_HcInterruptStatus] |= bit;
|
||||
usb_interrupt_ohci(usb);
|
||||
dev->ohci_mmio[OHCI_HcInterruptStatus].b[0] |= bit;
|
||||
|
||||
ohci_update_irq(dev);
|
||||
}
|
||||
|
||||
void
|
||||
ohci_end_of_frame(usb_t* dev)
|
||||
{
|
||||
/* TODO: Put endpoint and transfer descriptor processing here. */
|
||||
dev->ohci_mmio_w[OHCI_HcFmNumber / 2]++;
|
||||
dev->ohci_mmio[OHCI_HcFmNumber].w[0]++;
|
||||
}
|
||||
|
||||
void
|
||||
@@ -258,17 +315,17 @@ ohci_update_frame_counter(void* priv)
|
||||
{
|
||||
usb_t *dev = (usb_t *) priv;
|
||||
|
||||
dev->ohci_mmio_w[OHCI_HcFmRemaining / 2] &= 0x3fff;
|
||||
if (dev->ohci_mmio_w[OHCI_HcFmRemaining / 2] == 0) {
|
||||
dev->ohci_mmio[OHCI_HcFmRemaining].w[0] &= 0x3fff;
|
||||
if (dev->ohci_mmio[OHCI_HcFmRemaining].w[0] == 0) {
|
||||
ohci_end_of_frame(dev);
|
||||
dev->ohci_mmio_w[OHCI_HcFmRemaining / 2] = dev->ohci_mmio_w[OHCI_HcFmInterval / 2] & 0x3fff;
|
||||
dev->ohci_mmio_l[OHCI_HcFmRemaining / 4] &= ~(1 << 31);
|
||||
dev->ohci_mmio_l[OHCI_HcFmRemaining / 4] |= dev->ohci_mmio_l[OHCI_HcFmInterval / 4] & (1 << 31);
|
||||
dev->ohci_mmio[OHCI_HcFmRemaining].w[0] = dev->ohci_mmio[OHCI_HcFmInterval].w[0] & 0x3fff;
|
||||
dev->ohci_mmio[OHCI_HcFmRemaining].l &= ~(1 << 31);
|
||||
dev->ohci_mmio[OHCI_HcFmRemaining].l |= dev->ohci_mmio[OHCI_HcFmInterval].l & (1 << 31);
|
||||
ohci_start_of_frame(dev);
|
||||
timer_on_auto(&dev->ohci_frame_timer, 1. / (12. * 1000. * 1000.));
|
||||
return;
|
||||
}
|
||||
if (dev->ohci_mmio_w[OHCI_HcFmRemaining / 2]) dev->ohci_mmio_w[OHCI_HcFmRemaining / 2]--;
|
||||
if (dev->ohci_mmio[OHCI_HcFmRemaining].w[0]) dev->ohci_mmio[OHCI_HcFmRemaining].w[0]--;
|
||||
timer_on_auto(&dev->ohci_frame_timer, 1. / (12. * 1000. * 1000.));
|
||||
}
|
||||
|
||||
@@ -288,8 +345,8 @@ ohci_port_reset_callback(void* priv)
|
||||
{
|
||||
usb_t *dev = (usb_t *) priv;
|
||||
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1] &= ~0x10;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1 + 2] |= 0x10;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1].b[0] &= ~0x10;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1].b[2] |= 0x10;
|
||||
}
|
||||
|
||||
void
|
||||
@@ -297,244 +354,260 @@ ohci_port_reset_callback_2(void* priv)
|
||||
{
|
||||
usb_t *dev = (usb_t *) priv;
|
||||
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2] &= ~0x10;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2 + 2] |= 0x10;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2].b[0] &= ~0x10;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2].b[2] |= 0x10;
|
||||
}
|
||||
|
||||
static void
|
||||
ohci_mmio_write(uint32_t addr, uint8_t val, void *p)
|
||||
{
|
||||
usb_t *dev = (usb_t *) p;
|
||||
uint8_t old;
|
||||
|
||||
#ifdef ENABLE_USB_LOG
|
||||
usb_log("[W] %08X = %04X\n", addr, val);
|
||||
#endif
|
||||
|
||||
addr &= 0x00000fff;
|
||||
|
||||
switch (addr) {
|
||||
case OHCI_HcControl:
|
||||
case OHCI_aHcControl:
|
||||
if ((val & 0xc0) == 0x00) {
|
||||
/* UsbReset */
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1 + 2] = dev->ohci_mmio[OHCI_HcRhPortStatus2 + 2] = 0x16;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1].b[2] = dev->ohci_mmio[OHCI_HcRhPortStatus2].b[2] = 0x16;
|
||||
}
|
||||
break;
|
||||
case OHCI_HcCommandStatus:
|
||||
case OHCI_aHcCommandStatus:
|
||||
/* bit OwnershipChangeRequest triggers an ownership change (SMM <-> OS) */
|
||||
if (val & 0x08) {
|
||||
dev->ohci_mmio[OHCI_HcInterruptStatus + 3] = 0x40;
|
||||
if ((dev->ohci_mmio[OHCI_HcInterruptEnable + 3] & 0xc0) == 0xc0)
|
||||
dev->ohci_mmio[OHCI_HcInterruptStatus].b[3] = 0x40;
|
||||
if ((dev->ohci_mmio[OHCI_HcInterruptEnable].b[3] & 0xc0) == 0xc0)
|
||||
smi_raise();
|
||||
}
|
||||
|
||||
/* bit HostControllerReset must be cleared for the controller to be seen as initialized */
|
||||
if (val & 0x01) {
|
||||
memset(dev->ohci_mmio, 0x00, 4096);
|
||||
dev->ohci_mmio[OHCI_HcRevision] = 0x10;
|
||||
dev->ohci_mmio[OHCI_HcRevision + 1] = 0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhDescriptorA] = 0x02;
|
||||
dev->ohci_mmio[OHCI_HcRevision].b[0] = 0x10;
|
||||
dev->ohci_mmio[OHCI_HcRevision].b[1] = 0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhDescriptorA].b[0] = 0x02;
|
||||
val &= ~0x01;
|
||||
}
|
||||
break;
|
||||
case OHCI_HcHCCA:
|
||||
case OHCI_aHcHCCA:
|
||||
return;
|
||||
case OHCI_HcInterruptEnable:
|
||||
dev->ohci_mmio[addr] = (val & 0x7f);
|
||||
dev->ohci_mmio[OHCI_HcInterruptDisable] &= ~(val & 0x7f);
|
||||
case OHCI_aHcInterruptEnable:
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] = (val & 0x7f);
|
||||
dev->ohci_mmio[OHCI_HcInterruptDisable].b[0] &= ~(val & 0x7f);
|
||||
ohci_update_irq(dev);
|
||||
return;
|
||||
case OHCI_HcInterruptEnable + 1:
|
||||
case OHCI_HcInterruptEnable + 2:
|
||||
case OHCI_aHcInterruptEnable + 1:
|
||||
case OHCI_aHcInterruptEnable + 2:
|
||||
return;
|
||||
case OHCI_HcInterruptEnable + 3:
|
||||
dev->ohci_mmio[addr] = (val & 0x40);
|
||||
dev->ohci_mmio[addr] |= (val & 0x80);
|
||||
if (val & 0x80)
|
||||
dev->ohci_mmio[OHCI_HcInterruptDisable + 3] &= ~0x80;
|
||||
if (val & 0x40)
|
||||
dev->ohci_mmio[OHCI_HcInterruptDisable + 3] &= ~0x40;
|
||||
case OHCI_aHcInterruptEnable + 3:
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] = (val & 0xc0);
|
||||
dev->ohci_mmio[OHCI_HcInterruptDisable].b[3] &= ~(val & 0xc0);
|
||||
ohci_update_irq(dev);
|
||||
return;
|
||||
case OHCI_HcInterruptDisable:
|
||||
dev->ohci_mmio[addr] = (val & 0x7f);
|
||||
dev->ohci_mmio[OHCI_HcInterruptEnable] &= ~(val & 0x7f);
|
||||
case OHCI_aHcInterruptDisable:
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] = (val & 0x7f);
|
||||
dev->ohci_mmio[OHCI_HcInterruptEnable].b[0] &= ~(val & 0x7f);
|
||||
ohci_update_irq(dev);
|
||||
return;
|
||||
case OHCI_HcInterruptDisable + 1:
|
||||
case OHCI_HcInterruptDisable + 2:
|
||||
case OHCI_aHcInterruptDisable + 1:
|
||||
case OHCI_aHcInterruptDisable + 2:
|
||||
return;
|
||||
case OHCI_HcInterruptDisable + 3:
|
||||
dev->ohci_mmio[addr] = (val & 0x40);
|
||||
dev->ohci_mmio[addr] |= (val & 0x80);
|
||||
if (val & 0x80)
|
||||
dev->ohci_mmio[OHCI_HcInterruptEnable + 3] &= ~0x80;
|
||||
if (val & 0x40)
|
||||
dev->ohci_mmio[OHCI_HcInterruptEnable + 3] &= ~0x40;
|
||||
case OHCI_aHcInterruptDisable + 3:
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] = (val & 0xc0);
|
||||
dev->ohci_mmio[OHCI_HcInterruptEnable].b[3] &= ~(val & 0xc0);
|
||||
ohci_update_irq(dev);
|
||||
return;
|
||||
case OHCI_HcInterruptStatus:
|
||||
dev->ohci_mmio[addr] &= ~(val & 0x7f);
|
||||
case OHCI_aHcInterruptStatus:
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] &= ~(val & 0x7f);
|
||||
return;
|
||||
case OHCI_HcInterruptStatus + 1:
|
||||
case OHCI_HcInterruptStatus + 2:
|
||||
case OHCI_aHcInterruptStatus + 1:
|
||||
case OHCI_aHcInterruptStatus + 2:
|
||||
return;
|
||||
case OHCI_HcInterruptStatus + 3:
|
||||
dev->ohci_mmio[addr] &= ~(val & 0x40);
|
||||
case OHCI_aHcInterruptStatus + 3:
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] &= ~(val & 0x40);
|
||||
return;
|
||||
case OHCI_HcFmRemaining + 3:
|
||||
dev->ohci_mmio[addr] = (val & 0x80);
|
||||
case OHCI_aHcFmRemaining + 3:
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] = (val & 0x80);
|
||||
return;
|
||||
case OHCI_HcFmRemaining + 1:
|
||||
case OHCI_HcPeriodicStart + 1:
|
||||
dev->ohci_mmio[addr] = (val & 0x3f);
|
||||
case OHCI_aHcFmRemaining + 1:
|
||||
case OHCI_aHcPeriodicStart + 1:
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] = (val & 0x3f);
|
||||
return;
|
||||
case OHCI_HcLSThreshold + 1:
|
||||
dev->ohci_mmio[addr] = (val & 0x0f);
|
||||
case OHCI_aHcLSThreshold + 1:
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] = (val & 0x0f);
|
||||
return;
|
||||
case OHCI_HcFmRemaining + 2:
|
||||
case OHCI_HcFmNumber + 2:
|
||||
case OHCI_HcFmNumber + 3:
|
||||
case OHCI_HcPeriodicStart + 2:
|
||||
case OHCI_HcPeriodicStart + 3:
|
||||
case OHCI_HcLSThreshold + 2:
|
||||
case OHCI_HcLSThreshold + 3:
|
||||
case OHCI_HcRhDescriptorA:
|
||||
case OHCI_HcRhDescriptorA + 2:
|
||||
case OHCI_aHcFmRemaining + 2:
|
||||
case OHCI_aHcFmNumber + 2:
|
||||
case OHCI_aHcFmNumber + 3:
|
||||
case OHCI_aHcPeriodicStart + 2:
|
||||
case OHCI_aHcPeriodicStart + 3:
|
||||
case OHCI_aHcLSThreshold + 2:
|
||||
case OHCI_aHcLSThreshold + 3:
|
||||
case OHCI_aHcRhDescriptorA:
|
||||
case OHCI_aHcRhDescriptorA + 2:
|
||||
return;
|
||||
case OHCI_HcRhDescriptorA + 1:
|
||||
dev->ohci_mmio[addr] = (val & 0x1b);
|
||||
case OHCI_aHcRhDescriptorA + 1:
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] = (val & 0x1b);
|
||||
if (val & 0x02) {
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1 + 1] |= 0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2 + 1] |= 0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1].b[1] |= 0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2].b[1] |= 0x01;
|
||||
}
|
||||
return;
|
||||
case OHCI_HcRhDescriptorA + 3:
|
||||
dev->ohci_mmio[addr] = (val & 0x03);
|
||||
case OHCI_aHcRhDescriptorA + 3:
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] = (val & 0x03);
|
||||
return;
|
||||
case OHCI_HcRhDescriptorB:
|
||||
case OHCI_HcRhDescriptorB + 2:
|
||||
dev->ohci_mmio[addr] = (val & 0x06);
|
||||
case OHCI_aHcRhDescriptorB:
|
||||
case OHCI_aHcRhDescriptorB + 2:
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] = (val & 0x06);
|
||||
if ((addr == OHCI_HcRhDescriptorB) && !(val & 0x04)) {
|
||||
if (!(dev->ohci_mmio[OHCI_HcRhPortStatus2] & 0x01))
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2 + 2] |= 0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2] |= 0x01;
|
||||
if (!(dev->ohci_mmio[OHCI_HcRhPortStatus2].b[0] & 0x01))
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2].b[2] |= 0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2].b[0] |= 0x01;
|
||||
}
|
||||
if ((addr == OHCI_HcRhDescriptorB) && !(val & 0x02)) {
|
||||
if (!(dev->ohci_mmio[OHCI_HcRhPortStatus1] & 0x01))
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1 + 2] |= 0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1] |= 0x01;
|
||||
if (!(dev->ohci_mmio[OHCI_HcRhPortStatus1].b[0] & 0x01))
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1].b[2] |= 0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1].b[0] |= 0x01;
|
||||
}
|
||||
return;
|
||||
case OHCI_HcRhDescriptorB + 1:
|
||||
case OHCI_HcRhDescriptorB + 3:
|
||||
case OHCI_aHcRhDescriptorB + 1:
|
||||
case OHCI_aHcRhDescriptorB + 3:
|
||||
return;
|
||||
case OHCI_HcRhStatus:
|
||||
case OHCI_aHcRhStatus:
|
||||
if (val & 0x01) {
|
||||
if ((dev->ohci_mmio[OHCI_HcRhDescriptorA + 1] & 0x03) == 0x00) {
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1 + 1] &= ~0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1] &= ~0x17;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1 + 2] &= ~0x17;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2 + 1] &= ~0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2] &= ~0x17;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2 + 2] &= ~0x17;
|
||||
} else if ((dev->ohci_mmio[OHCI_HcRhDescriptorA + 1] & 0x03) == 0x01) {
|
||||
if (!(dev->ohci_mmio[OHCI_HcRhDescriptorB + 2] & 0x02)) {
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1 + 1] &= ~0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1] &= ~0x17;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1 + 2] &= ~0x17;
|
||||
if ((dev->ohci_mmio[OHCI_HcRhDescriptorA].b[1] & 0x03) == 0x00) {
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1].b[1] &= ~0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1].b[0] &= ~0x17;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1].b[2] &= ~0x17;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2].b[1] &= ~0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2].b[0] &= ~0x17;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2].b[2] &= ~0x17;
|
||||
} else if ((dev->ohci_mmio[OHCI_HcRhDescriptorA].b[1] & 0x03) == 0x01) {
|
||||
if (!(dev->ohci_mmio[OHCI_HcRhDescriptorB].b[2] & 0x02)) {
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1].b[1] &= ~0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1].b[0] &= ~0x17;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1].b[2] &= ~0x17;
|
||||
}
|
||||
if (!(dev->ohci_mmio[OHCI_HcRhDescriptorB + 2] & 0x04)) {
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2 + 1] &= ~0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2] &= ~0x17;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2 + 2] &= ~0x17;
|
||||
if (!(dev->ohci_mmio[OHCI_HcRhDescriptorB].b[2] & 0x04)) {
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2].b[1] &= ~0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2].b[0] &= ~0x17;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2].b[2] &= ~0x17;
|
||||
}
|
||||
}
|
||||
}
|
||||
return;
|
||||
case OHCI_HcRhStatus + 1:
|
||||
case OHCI_aHcRhStatus + 1:
|
||||
if (val & 0x80)
|
||||
dev->ohci_mmio[addr] |= 0x80;
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] |= 0x80;
|
||||
return;
|
||||
case OHCI_HcRhStatus + 2:
|
||||
dev->ohci_mmio[addr] &= ~(val & 0x02);
|
||||
case OHCI_aHcRhStatus + 2:
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] &= ~(val & 0x02);
|
||||
if (val & 0x01) {
|
||||
if ((dev->ohci_mmio[OHCI_HcRhDescriptorA + 1] & 0x03) == 0x00) {
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1 + 1] |= 0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2 + 1] |= 0x01;
|
||||
} else if ((dev->ohci_mmio[OHCI_HcRhDescriptorA + 1] & 0x03) == 0x01) {
|
||||
if (!(dev->ohci_mmio[OHCI_HcRhDescriptorB + 2] & 0x02))
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1 + 1] |= 0x01;
|
||||
if (!(dev->ohci_mmio[OHCI_HcRhDescriptorB + 2] & 0x04))
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2 + 1] |= 0x01;
|
||||
if ((dev->ohci_mmio[OHCI_HcRhDescriptorA].b[1] & 0x03) == 0x00) {
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1].b[1] |= 0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2].b[1] |= 0x01;
|
||||
} else if ((dev->ohci_mmio[OHCI_HcRhDescriptorA].b[1] & 0x03) == 0x01) {
|
||||
if (!(dev->ohci_mmio[OHCI_HcRhDescriptorB].b[2] & 0x02))
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1].b[1] |= 0x01;
|
||||
if (!(dev->ohci_mmio[OHCI_HcRhDescriptorB].b[2] & 0x04))
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2].b[1] |= 0x01;
|
||||
}
|
||||
}
|
||||
return;
|
||||
case OHCI_HcRhStatus + 3:
|
||||
case OHCI_aHcRhStatus + 3:
|
||||
if (val & 0x80)
|
||||
dev->ohci_mmio[OHCI_HcRhStatus + 1] &= ~0x80;
|
||||
dev->ohci_mmio[OHCI_HcRhStatus].b[1] &= ~0x80;
|
||||
return;
|
||||
case OHCI_HcRhPortStatus1:
|
||||
case OHCI_HcRhPortStatus2:
|
||||
old = dev->ohci_mmio[addr];
|
||||
case OHCI_aHcRhPortStatus1:
|
||||
case OHCI_aHcRhPortStatus2:
|
||||
old = dev->ohci_mmio[addr >> 2].b[addr & 3];
|
||||
|
||||
if (val & 0x10) {
|
||||
if (old & 0x01) {
|
||||
dev->ohci_mmio[addr] |= 0x10;
|
||||
timer_on_auto(&dev->ohci_port_reset_timer[(addr - OHCI_HcRhPortStatus1) / 4], 10000.);
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] |= 0x10;
|
||||
timer_on_auto(&dev->ohci_port_reset_timer[(addr - OHCI_aHcRhPortStatus1) / 4], 10000.);
|
||||
} else
|
||||
dev->ohci_mmio[addr + 2] |= 0x01;
|
||||
dev->ohci_mmio[(addr + 2) >> 2].b[(addr + 2) & 3] |= 0x01;
|
||||
}
|
||||
if (val & 0x08)
|
||||
dev->ohci_mmio[addr] &= ~0x04;
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] &= ~0x04;
|
||||
if (val & 0x04)
|
||||
dev->ohci_mmio[addr] |= 0x04;
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] |= 0x04;
|
||||
if (val & 0x02) {
|
||||
if (old & 0x01)
|
||||
dev->ohci_mmio[addr] |= 0x02;
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] |= 0x02;
|
||||
else
|
||||
dev->ohci_mmio[addr + 2] |= 0x01;
|
||||
dev->ohci_mmio[(addr + 2) >> 2].b[(addr + 2) & 3] |= 0x01;
|
||||
}
|
||||
if (val & 0x01) {
|
||||
if (old & 0x01)
|
||||
dev->ohci_mmio[addr] &= ~0x02;
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] &= ~0x02;
|
||||
else
|
||||
dev->ohci_mmio[addr + 2] |= 0x01;
|
||||
dev->ohci_mmio[(addr + 2) >> 2].b[(addr + 2) & 3] |= 0x01;
|
||||
}
|
||||
|
||||
if (!(dev->ohci_mmio[addr] & 0x04) && (old & 0x04))
|
||||
dev->ohci_mmio[addr + 2] |= 0x04;
|
||||
/* if (!(dev->ohci_mmio[addr] & 0x02))
|
||||
dev->ohci_mmio[addr + 2] |= 0x02; */
|
||||
if (!(dev->ohci_mmio[addr >> 2].b[addr & 3] & 0x04) && (old & 0x04))
|
||||
dev->ohci_mmio[(addr + 2) >> 2].b[(addr + 2) & 3] |= 0x04;
|
||||
/* if (!(dev->ohci_mmio[addr >> 2].b[addr & 3] & 0x02))
|
||||
dev->ohci_mmio[(addr + 2) >> 2].b[(addr + 2) & 3] |= 0x02; */
|
||||
return;
|
||||
case OHCI_HcRhPortStatus1 + 1:
|
||||
if ((val & 0x02) && ((dev->ohci_mmio[OHCI_HcRhDescriptorA + 1] & 0x03) == 0x00) && (dev->ohci_mmio[OHCI_HcRhDescriptorB + 2] & 0x02)) {
|
||||
dev->ohci_mmio[addr] &= ~0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1] &= ~0x17;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1 + 2] &= ~0x17;
|
||||
case OHCI_aHcRhPortStatus1 + 1:
|
||||
if ((val & 0x02) && ((dev->ohci_mmio[OHCI_HcRhDescriptorA].b[1] & 0x03) == 0x00) && (dev->ohci_mmio[OHCI_HcRhDescriptorB].b[2] & 0x02)) {
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] &= ~0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1].b[0] &= ~0x17;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus1].b[2] &= ~0x17;
|
||||
}
|
||||
if ((val & 0x01) && ((dev->ohci_mmio[OHCI_HcRhDescriptorA + 1] & 0x03) == 0x00) && (dev->ohci_mmio[OHCI_HcRhDescriptorB + 2] & 0x02)) {
|
||||
dev->ohci_mmio[addr] |= 0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2] &= ~0x17;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2 + 2] &= ~0x17;
|
||||
if ((val & 0x01) && ((dev->ohci_mmio[OHCI_HcRhDescriptorA].b[1] & 0x03) == 0x00) && (dev->ohci_mmio[OHCI_HcRhDescriptorB].b[2] & 0x02)) {
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] |= 0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2].b[0] &= ~0x17;
|
||||
dev->ohci_mmio[OHCI_HcRhPortStatus2].b[2] &= ~0x17;
|
||||
}
|
||||
return;
|
||||
case OHCI_HcRhPortStatus2 + 1:
|
||||
if ((val & 0x02) && ((dev->ohci_mmio[OHCI_HcRhDescriptorA + 1] & 0x03) == 0x00) && (dev->ohci_mmio[OHCI_HcRhDescriptorB + 2] & 0x04))
|
||||
dev->ohci_mmio[addr] &= ~0x01;
|
||||
if ((val & 0x01) && ((dev->ohci_mmio[OHCI_HcRhDescriptorA + 1] & 0x03) == 0x00) && (dev->ohci_mmio[OHCI_HcRhDescriptorB + 2] & 0x04))
|
||||
dev->ohci_mmio[addr] |= 0x01;
|
||||
case OHCI_aHcRhPortStatus2 + 1:
|
||||
if ((val & 0x02) && ((dev->ohci_mmio[OHCI_HcRhDescriptorA].b[1] & 0x03) == 0x00) && (dev->ohci_mmio[OHCI_HcRhDescriptorB].b[2] & 0x04))
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] &= ~0x01;
|
||||
if ((val & 0x01) && ((dev->ohci_mmio[OHCI_HcRhDescriptorA].b[1] & 0x03) == 0x00) && (dev->ohci_mmio[OHCI_HcRhDescriptorB].b[2] & 0x04))
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] |= 0x01;
|
||||
return;
|
||||
case OHCI_HcRhPortStatus1 + 2:
|
||||
case OHCI_HcRhPortStatus2 + 2:
|
||||
dev->ohci_mmio[addr] &= ~(val & 0x1f);
|
||||
case OHCI_aHcRhPortStatus1 + 2:
|
||||
case OHCI_aHcRhPortStatus2 + 2:
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] &= ~(val & 0x1f);
|
||||
return;
|
||||
case OHCI_HcRhPortStatus1 + 3:
|
||||
case OHCI_HcRhPortStatus2 + 3:
|
||||
case OHCI_aHcRhPortStatus1 + 3:
|
||||
case OHCI_aHcRhPortStatus2 + 3:
|
||||
return;
|
||||
case OHCI_HcDoneHead:
|
||||
case OHCI_HcBulkCurrentED:
|
||||
case OHCI_HcBulkHeadED:
|
||||
case OHCI_HcControlCurrentED:
|
||||
case OHCI_HcControlHeadED:
|
||||
case OHCI_HcPeriodCurrentED:
|
||||
dev->ohci_mmio[addr] = (val & 0xf0);
|
||||
case OHCI_aHcDoneHead:
|
||||
case OHCI_aHcBulkCurrentED:
|
||||
case OHCI_aHcBulkHeadED:
|
||||
case OHCI_aHcControlCurrentED:
|
||||
case OHCI_aHcControlHeadED:
|
||||
case OHCI_aHcPeriodCurrentED:
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] = (val & 0xf0);
|
||||
return;
|
||||
}
|
||||
|
||||
dev->ohci_mmio[addr] = val;
|
||||
dev->ohci_mmio[addr >> 2].b[addr & 3] = val;
|
||||
}
|
||||
|
||||
static void
|
||||
ohci_mmio_writew(uint32_t addr, uint16_t val, void *p)
|
||||
{
|
||||
ohci_mmio_write(addr, val & 0xff, p);
|
||||
ohci_mmio_write(addr + 1, val >> 8, p);
|
||||
}
|
||||
|
||||
static void
|
||||
ohci_mmio_writel(uint32_t addr, uint32_t val, void *p)
|
||||
{
|
||||
ohci_mmio_writew(addr, val & 0xffff, p);
|
||||
ohci_mmio_writew(addr + 2, val >> 16, p);
|
||||
}
|
||||
|
||||
void
|
||||
ohci_update_mem_mapping(usb_t *dev, uint8_t base1, uint8_t base2, uint8_t base3, int enable)
|
||||
{
|
||||
@@ -544,8 +617,13 @@ ohci_update_mem_mapping(usb_t *dev, uint8_t base1, uint8_t base2, uint8_t base3,
|
||||
dev->ohci_mem_base = ((base1 << 8) | (base2 << 16) | (base3 << 24)) & 0xfffff000;
|
||||
dev->ohci_enable = enable;
|
||||
|
||||
if (dev->ohci_enable && (dev->ohci_mem_base != 0x00000000))
|
||||
if (dev->ohci_enable && (dev->ohci_mem_base != 0x00000000)) {
|
||||
dev->ohci_mmio_mapping.flags = MEM_MAPPING_EXTERNAL;
|
||||
mem_mapping_set_addr(&dev->ohci_mmio_mapping, dev->ohci_mem_base, 0x1000);
|
||||
mem_mapping_enable(&dev->ohci_mmio_mapping);
|
||||
}
|
||||
|
||||
usb_log("ohci_update_mem_mapping(): OHCI %sabled at %08X\n", dev->ohci_enable ? "en" : "dis", dev->ohci_mem_base);
|
||||
}
|
||||
|
||||
uint8_t
|
||||
@@ -565,21 +643,24 @@ usb_reset(void *priv)
|
||||
{
|
||||
usb_t *dev = (usb_t *) priv;
|
||||
|
||||
memset(dev->uhci_io, 0x00, 128);
|
||||
memset(dev->uhci_io, 0x00, sizeof(dev->uhci_io));
|
||||
dev->uhci_io[0x0c] = 0x40;
|
||||
dev->uhci_io[0x10] = dev->uhci_io[0x12] = 0x80;
|
||||
|
||||
memset(dev->ohci_mmio, 0x00, 4096);
|
||||
dev->ohci_mmio[OHCI_HcRevision] = 0x10;
|
||||
dev->ohci_mmio[OHCI_HcRevision + 1] = 0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhDescriptorA] = 0x02;
|
||||
dev->ohci_mmio[OHCI_HcRhDescriptorA + 1] = 0x02;
|
||||
memset(dev->ohci_mmio, 0x00, sizeof(dev->ohci_mmio));
|
||||
dev->ohci_mmio[OHCI_HcRevision].b[0] = 0x10;
|
||||
dev->ohci_mmio[OHCI_HcRevision].b[1] = 0x01;
|
||||
dev->ohci_mmio[OHCI_HcRhDescriptorA].b[0] = 0x02;
|
||||
dev->ohci_mmio[OHCI_HcRhDescriptorA].b[1] = 0x02;
|
||||
|
||||
io_removehandler(dev->uhci_io_base, 0x20, uhci_reg_read, NULL, NULL, uhci_reg_write, uhci_reg_writew, NULL, dev);
|
||||
dev->uhci_enable = 0;
|
||||
|
||||
mem_mapping_disable(&dev->ohci_mmio_mapping);
|
||||
dev->ohci_enable = 0;
|
||||
|
||||
usb_log("usb_reset(): OHCI %sabled at %08X\n", dev->ohci_enable ? "en" : "dis", dev->ohci_mem_base);
|
||||
usb_log("usb_reset(): map = %08X\n", &dev->ohci_mmio_mapping);
|
||||
}
|
||||
|
||||
static void
|
||||
@@ -591,7 +672,7 @@ usb_close(void *priv)
|
||||
}
|
||||
|
||||
static void *
|
||||
usb_init_ext(const device_t *info, void* params)
|
||||
usb_init_ext(const device_t *info, void *params)
|
||||
{
|
||||
usb_t *dev;
|
||||
|
||||
@@ -600,16 +681,20 @@ usb_init_ext(const device_t *info, void* params)
|
||||
return (NULL);
|
||||
memset(dev, 0x00, sizeof(usb_t));
|
||||
|
||||
dev->usb_params = (usb_params_t*)params;
|
||||
dev->usb_params = (usb_params_t *) params;
|
||||
|
||||
mem_mapping_add(&dev->ohci_mmio_mapping, 0, 0,
|
||||
ohci_mmio_read, NULL, NULL,
|
||||
ohci_mmio_write, NULL, NULL,
|
||||
mem_mapping_add(&dev->ohci_mmio_mapping, 0, 0x1000,
|
||||
ohci_mmio_read, ohci_mmio_readw, ohci_mmio_readl,
|
||||
ohci_mmio_write, ohci_mmio_writew, ohci_mmio_writel,
|
||||
NULL, MEM_MAPPING_EXTERNAL, dev);
|
||||
|
||||
mem_mapping_disable(&dev->ohci_mmio_mapping);
|
||||
|
||||
timer_add(&dev->ohci_frame_timer, ohci_update_frame_counter, dev, 0); /* Unused for now, to be used for frame counting. */
|
||||
timer_add(&dev->ohci_port_reset_timer[0], ohci_port_reset_callback, dev, 0);
|
||||
timer_add(&dev->ohci_port_reset_timer[1], ohci_port_reset_callback_2, dev, 0);
|
||||
timer_add(&dev->ohci_interrupt_desc_poll_timer, ohci_poll_interrupt_descriptors, dev, 0);
|
||||
|
||||
usb_reset(dev);
|
||||
|
||||
return dev;
|
||||
|
||||
Reference in New Issue
Block a user