2020-04-23 02:23:59 +02:00
|
|
|
/*
|
2022-11-13 16:37:58 -05:00
|
|
|
* 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.
|
2020-04-23 02:23:59 +02:00
|
|
|
*
|
2022-11-13 16:37:58 -05:00
|
|
|
* This file is part of the 86Box distribution.
|
2020-04-23 02:23:59 +02:00
|
|
|
*
|
2022-11-13 16:37:58 -05:00
|
|
|
* Universal Serial Bus emulation (currently dummy UHCI and
|
|
|
|
|
* OHCI).
|
2020-04-23 02:23:59 +02:00
|
|
|
*
|
|
|
|
|
*
|
|
|
|
|
*
|
2022-11-13 16:37:58 -05:00
|
|
|
* Authors: Miran Grca, <mgrca8@gmail.com>
|
2020-04-23 02:23:59 +02:00
|
|
|
*
|
2022-11-13 16:37:58 -05:00
|
|
|
* Copyright 2020 Miran Grca.
|
2020-04-23 02:23:59 +02:00
|
|
|
*/
|
|
|
|
|
#include <stdarg.h>
|
2017-09-25 04:31:20 -04:00
|
|
|
#include <stdio.h>
|
2017-01-22 00:22:14 -06:00
|
|
|
#include <stdint.h>
|
2020-04-23 02:23:59 +02:00
|
|
|
#include <stdlib.h>
|
2017-09-25 04:31:20 -04:00
|
|
|
#include <string.h>
|
2023-05-07 23:07:03 +06:00
|
|
|
#include <stdbool.h>
|
2017-09-25 04:31:20 -04:00
|
|
|
#include <wchar.h>
|
2023-05-08 16:51:16 +06:00
|
|
|
#include <assert.h>
|
2020-04-23 02:23:59 +02:00
|
|
|
#define HAVE_STDARG_H
|
|
|
|
|
#include <86box/86box.h>
|
|
|
|
|
#include <86box/device.h>
|
2020-03-29 14:24:42 +02:00
|
|
|
#include <86box/io.h>
|
|
|
|
|
#include <86box/mem.h>
|
2023-05-01 16:55:49 +06:00
|
|
|
#include <86box/timer.h>
|
2020-03-29 14:24:42 +02:00
|
|
|
#include <86box/usb.h>
|
2023-05-07 01:16:18 +06:00
|
|
|
#include <86box/dma.h>
|
2023-06-28 13:46:28 -04:00
|
|
|
#include <86box/plat_unused.h>
|
2017-01-22 00:22:14 -06:00
|
|
|
|
2020-04-23 02:23:59 +02:00
|
|
|
#ifdef ENABLE_USB_LOG
|
|
|
|
|
int usb_do_log = ENABLE_USB_LOG;
|
2017-01-22 00:22:14 -06:00
|
|
|
|
2020-04-23 02:23:59 +02:00
|
|
|
static void
|
|
|
|
|
usb_log(const char *fmt, ...)
|
|
|
|
|
{
|
|
|
|
|
va_list ap;
|
|
|
|
|
|
|
|
|
|
if (usb_do_log) {
|
2022-09-18 17:11:43 -04:00
|
|
|
va_start(ap, fmt);
|
|
|
|
|
pclog_ex(fmt, ap);
|
|
|
|
|
va_end(ap);
|
2020-04-23 02:23:59 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#else
|
2022-09-18 17:11:43 -04:00
|
|
|
# define usb_log(fmt, ...)
|
2020-04-23 02:23:59 +02:00
|
|
|
#endif
|
|
|
|
|
|
2023-04-30 23:01:15 +06:00
|
|
|
/* OHCI registers */
|
2023-06-28 13:46:28 -04:00
|
|
|
enum {
|
2023-05-07 02:53:04 +02:00
|
|
|
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 */
|
|
|
|
|
};
|
|
|
|
|
|
2023-06-28 13:46:28 -04:00
|
|
|
enum {
|
2023-05-07 02:53:04 +02:00
|
|
|
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
|
2023-04-30 23:01:15 +06:00
|
|
|
};
|
|
|
|
|
|
2023-05-05 23:26:41 +06:00
|
|
|
/* OHCI HcInterruptEnable/Disable bits */
|
2023-06-28 13:46:28 -04:00
|
|
|
enum {
|
2023-05-05 23:26:41 +06:00
|
|
|
OHCI_HcInterruptEnable_SO = 1 << 0,
|
|
|
|
|
OHCI_HcInterruptEnable_WDH = 1 << 1,
|
|
|
|
|
OHCI_HcInterruptEnable_SF = 1 << 2,
|
|
|
|
|
OHCI_HcInterruptEnable_RD = 1 << 3,
|
|
|
|
|
OHCI_HcInterruptEnable_UE = 1 << 4,
|
|
|
|
|
OHCI_HcInterruptEnable_HNO = 1 << 5,
|
|
|
|
|
OHCI_HcInterruptEnable_RHSC = 1 << 6,
|
|
|
|
|
};
|
|
|
|
|
|
2023-05-07 23:07:03 +06:00
|
|
|
/* OHCI HcControl bits */
|
2023-06-28 13:46:28 -04:00
|
|
|
enum {
|
2023-05-07 23:07:03 +06:00
|
|
|
OHCI_HcControl_ControlBulkServiceRatio = 1 << 0,
|
|
|
|
|
OHCI_HcControl_PeriodicListEnable = 1 << 1,
|
|
|
|
|
OHCI_HcControl_IsochronousEnable = 1 << 2,
|
|
|
|
|
OHCI_HcControl_ControlListEnable = 1 << 3,
|
|
|
|
|
OHCI_HcControl_BulkListEnable = 1 << 4
|
|
|
|
|
};
|
|
|
|
|
|
2023-05-08 22:30:28 +06:00
|
|
|
usb_t* usb_device_inst = NULL;
|
|
|
|
|
|
2023-04-30 23:01:15 +06:00
|
|
|
static void
|
2023-05-07 02:53:04 +02:00
|
|
|
usb_interrupt_ohci(usb_t *dev, uint32_t level)
|
2023-04-30 23:01:15 +06:00
|
|
|
{
|
2023-05-07 02:53:04 +02:00
|
|
|
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))
|
2023-05-05 00:28:08 +06:00
|
|
|
return;
|
|
|
|
|
|
2023-05-07 02:53:04 +02:00
|
|
|
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);
|
2023-04-30 23:01:15 +06:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-04-23 02:23:59 +02:00
|
|
|
static uint8_t
|
2023-06-28 13:46:28 -04:00
|
|
|
uhci_reg_read(uint16_t addr, void *priv)
|
2020-04-23 02:23:59 +02:00
|
|
|
{
|
2023-06-28 13:46:28 -04:00
|
|
|
usb_t *dev = (usb_t *) priv;
|
2023-05-11 03:02:36 -04:00
|
|
|
uint8_t ret;
|
|
|
|
|
uint8_t *regs = dev->uhci_io;
|
2020-04-23 02:23:59 +02:00
|
|
|
|
2020-11-16 00:01:21 +01:00
|
|
|
addr &= 0x0000001f;
|
2020-10-13 19:25:40 -03:00
|
|
|
|
2020-11-16 00:01:21 +01:00
|
|
|
ret = regs[addr];
|
2020-04-23 02:23:59 +02:00
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2023-06-28 13:46:28 -04:00
|
|
|
uhci_reg_write(uint16_t addr, uint8_t val, void *priv)
|
2020-04-23 02:23:59 +02:00
|
|
|
{
|
2023-06-28 13:46:28 -04:00
|
|
|
usb_t *dev = (usb_t *) priv;
|
2020-11-16 00:01:21 +01:00
|
|
|
uint8_t *regs = dev->uhci_io;
|
|
|
|
|
|
|
|
|
|
addr &= 0x0000001f;
|
|
|
|
|
|
|
|
|
|
switch (addr) {
|
2022-09-18 17:11:43 -04:00
|
|
|
case 0x02:
|
|
|
|
|
regs[0x02] &= ~(val & 0x3f);
|
|
|
|
|
break;
|
|
|
|
|
case 0x04:
|
|
|
|
|
regs[0x04] = (val & 0x0f);
|
|
|
|
|
break;
|
|
|
|
|
case 0x09:
|
|
|
|
|
regs[0x09] = (val & 0xf0);
|
|
|
|
|
break;
|
|
|
|
|
case 0x0a:
|
|
|
|
|
case 0x0b:
|
|
|
|
|
regs[addr] = val;
|
|
|
|
|
break;
|
|
|
|
|
case 0x0c:
|
|
|
|
|
regs[0x0c] = (val & 0x7f);
|
|
|
|
|
break;
|
2023-06-28 13:46:28 -04:00
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
break;
|
2020-11-16 00:01:21 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2023-06-28 13:46:28 -04:00
|
|
|
uhci_reg_writew(uint16_t addr, uint16_t val, void *priv)
|
2020-11-16 00:01:21 +01:00
|
|
|
{
|
2023-06-28 13:46:28 -04:00
|
|
|
usb_t *dev = (usb_t *) priv;
|
2020-11-16 00:01:21 +01:00
|
|
|
uint16_t *regs = (uint16_t *) dev->uhci_io;
|
|
|
|
|
|
|
|
|
|
addr &= 0x0000001f;
|
|
|
|
|
|
|
|
|
|
switch (addr) {
|
2022-09-18 17:11:43 -04:00
|
|
|
case 0x00:
|
|
|
|
|
if ((val & 0x0001) && !(regs[0x00] & 0x0001))
|
|
|
|
|
regs[0x01] &= ~0x20;
|
|
|
|
|
else if (!(val & 0x0001))
|
|
|
|
|
regs[0x01] |= 0x20;
|
|
|
|
|
regs[0x00] = (val & 0x00ff);
|
|
|
|
|
break;
|
|
|
|
|
case 0x06:
|
|
|
|
|
regs[0x03] = (val & 0x07ff);
|
|
|
|
|
break;
|
|
|
|
|
case 0x10:
|
|
|
|
|
case 0x12:
|
|
|
|
|
regs[addr >> 1] = ((regs[addr >> 1] & 0xedbb) | (val & 0x1244)) & ~(val & 0x080a);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
2023-06-28 13:46:28 -04:00
|
|
|
uhci_reg_write(addr, val & 0xff, priv);
|
|
|
|
|
uhci_reg_write(addr + 1, (val >> 8) & 0xff, priv);
|
2022-09-18 17:11:43 -04:00
|
|
|
break;
|
2020-11-16 00:01:21 +01:00
|
|
|
}
|
2020-04-23 02:23:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
uhci_update_io_mapping(usb_t *dev, uint8_t base_l, uint8_t base_h, int enable)
|
|
|
|
|
{
|
|
|
|
|
if (dev->uhci_enable && (dev->uhci_io_base != 0x0000))
|
2022-09-18 17:11:43 -04:00
|
|
|
io_removehandler(dev->uhci_io_base, 0x20, uhci_reg_read, NULL, NULL, uhci_reg_write, uhci_reg_writew, NULL, dev);
|
2020-04-23 02:23:59 +02:00
|
|
|
|
|
|
|
|
dev->uhci_io_base = base_l | (base_h << 8);
|
2022-09-18 17:11:43 -04:00
|
|
|
dev->uhci_enable = enable;
|
2020-04-23 02:23:59 +02:00
|
|
|
|
|
|
|
|
if (dev->uhci_enable && (dev->uhci_io_base != 0x0000))
|
2022-09-18 17:11:43 -04:00
|
|
|
io_sethandler(dev->uhci_io_base, 0x20, uhci_reg_read, NULL, NULL, uhci_reg_write, uhci_reg_writew, NULL, dev);
|
2020-04-23 02:23:59 +02:00
|
|
|
}
|
|
|
|
|
|
2023-05-07 23:07:03 +06:00
|
|
|
typedef struct
|
|
|
|
|
{
|
|
|
|
|
uint32_t HccaInterrruptTable[32];
|
|
|
|
|
uint16_t HccaFrameNumber;
|
|
|
|
|
uint16_t HccaPad1;
|
|
|
|
|
uint32_t HccaDoneHead;
|
|
|
|
|
} usb_hcca_t;
|
|
|
|
|
|
|
|
|
|
/* Transfer descriptors */
|
|
|
|
|
typedef struct
|
|
|
|
|
{
|
2023-05-08 16:51:16 +06:00
|
|
|
uint32_t Control;
|
2023-05-07 23:07:03 +06:00
|
|
|
uint32_t CBP;
|
|
|
|
|
uint32_t NextTD;
|
|
|
|
|
uint32_t BE;
|
|
|
|
|
} usb_td_t;
|
|
|
|
|
|
|
|
|
|
/* Endpoint descriptors */
|
|
|
|
|
typedef struct
|
|
|
|
|
{
|
2023-05-08 16:51:16 +06:00
|
|
|
uint32_t Control;
|
2023-05-07 23:07:03 +06:00
|
|
|
uint32_t TailP;
|
2023-05-08 16:51:16 +06:00
|
|
|
uint32_t HeadP;
|
2023-05-07 23:07:03 +06:00
|
|
|
uint32_t NextED;
|
|
|
|
|
} usb_ed_t;
|
|
|
|
|
|
|
|
|
|
#define ENDPOINT_DESC_LIMIT 32
|
|
|
|
|
|
2020-04-23 02:23:59 +02:00
|
|
|
static uint8_t
|
2023-06-28 13:46:28 -04:00
|
|
|
ohci_mmio_read(uint32_t addr, void *priv)
|
2020-04-23 02:23:59 +02:00
|
|
|
{
|
2023-06-28 13:46:28 -04:00
|
|
|
usb_t *dev = (usb_t *) priv;
|
2020-04-23 02:23:59 +02:00
|
|
|
uint8_t ret = 0x00;
|
2023-05-07 02:53:04 +02:00
|
|
|
#ifdef ENABLE_USB_LOG
|
|
|
|
|
uint32_t old_addr = addr;
|
|
|
|
|
#endif
|
2017-08-24 01:14:39 -04:00
|
|
|
|
2020-04-23 02:23:59 +02:00
|
|
|
addr &= 0x00000fff;
|
2017-08-24 01:14:39 -04:00
|
|
|
|
2023-05-07 02:53:04 +02:00
|
|
|
ret = dev->ohci_mmio[addr >> 2].b[addr & 3];
|
2020-04-23 02:23:59 +02:00
|
|
|
|
2023-05-05 22:19:04 +06:00
|
|
|
switch (addr) {
|
|
|
|
|
case 0x101:
|
|
|
|
|
ret = (ret & 0xfe) | (!!mem_a20_key);
|
|
|
|
|
break;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcRhPortStatus1 + 1:
|
|
|
|
|
case OHCI_aHcRhPortStatus2 + 1:
|
|
|
|
|
case OHCI_aHcRhPortStatus3 + 1:
|
2023-05-05 22:19:04 +06:00
|
|
|
ret |= 0x1;
|
|
|
|
|
break;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcInterruptDisable:
|
|
|
|
|
case OHCI_aHcInterruptDisable + 1:
|
|
|
|
|
case OHCI_aHcInterruptDisable + 2:
|
|
|
|
|
case OHCI_aHcInterruptDisable + 3:
|
|
|
|
|
ret = dev->ohci_mmio[OHCI_HcInterruptEnable].b[addr & 3];
|
2023-05-05 22:19:04 +06:00
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2021-07-04 17:40:39 +02:00
|
|
|
if (addr == 0x101)
|
2022-09-18 17:11:43 -04:00
|
|
|
ret = (ret & 0xfe) | (!!mem_a20_key);
|
2021-07-04 17:40:39 +02:00
|
|
|
|
2023-05-07 02:53:04 +02:00
|
|
|
#ifdef ENABLE_USB_LOG
|
|
|
|
|
usb_log("[R] %08X = %04X\n", old_addr, ret);
|
|
|
|
|
#endif
|
|
|
|
|
|
2020-04-23 02:23:59 +02:00
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-07 02:53:04 +02:00
|
|
|
static uint16_t
|
2023-06-28 13:46:28 -04:00
|
|
|
ohci_mmio_readw(uint32_t addr, void *priv)
|
2023-05-07 02:53:04 +02:00
|
|
|
{
|
2023-06-28 13:46:28 -04:00
|
|
|
return ohci_mmio_read(addr, priv) | (ohci_mmio_read(addr + 1, priv) << 8);
|
2023-05-07 02:53:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static uint32_t
|
2023-06-28 13:46:28 -04:00
|
|
|
ohci_mmio_readl(uint32_t addr, void *priv)
|
2023-05-07 02:53:04 +02:00
|
|
|
{
|
2023-06-28 13:46:28 -04:00
|
|
|
return ohci_mmio_readw(addr, priv) | (ohci_mmio_readw(addr + 2, priv) << 16);
|
2023-05-07 02:53:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-06 23:19:55 +06:00
|
|
|
void
|
2023-05-07 02:53:04 +02:00
|
|
|
ohci_set_interrupt(usb_t *dev, uint8_t bit)
|
2023-05-06 23:19:55 +06:00
|
|
|
{
|
2023-05-07 02:53:04 +02:00
|
|
|
if (!(dev->ohci_mmio[OHCI_HcInterruptEnable].b[3] & 0x80))
|
2023-05-06 23:19:55 +06:00
|
|
|
return;
|
|
|
|
|
|
2023-05-07 02:53:04 +02:00
|
|
|
if (!(dev->ohci_mmio[OHCI_HcInterruptEnable].b[0] & bit))
|
2023-05-06 23:19:55 +06:00
|
|
|
return;
|
|
|
|
|
|
2023-05-07 02:53:04 +02:00
|
|
|
if (dev->ohci_mmio[OHCI_HcInterruptDisable].b[0] & bit)
|
2023-05-06 23:19:55 +06:00
|
|
|
return;
|
|
|
|
|
|
2023-05-07 02:53:04 +02:00
|
|
|
dev->ohci_mmio[OHCI_HcInterruptStatus].b[0] |= bit;
|
|
|
|
|
|
2023-05-08 16:51:16 +06:00
|
|
|
/* TODO: Does setting UnrecoverableError also assert PERR# on any emulated USB chipsets? */
|
|
|
|
|
|
2023-05-07 02:53:04 +02:00
|
|
|
ohci_update_irq(dev);
|
2023-05-06 23:19:55 +06:00
|
|
|
}
|
|
|
|
|
|
2023-06-11 14:10:47 +02:00
|
|
|
/* TODO: Actually use this function somewhere. */
|
|
|
|
|
#if 0
|
2023-05-08 00:50:00 +06:00
|
|
|
/* Next two functions ported over from QEMU. */
|
|
|
|
|
static int ohci_copy_td_input(usb_t* dev, usb_td_t *td,
|
|
|
|
|
uint8_t *buf, int len)
|
|
|
|
|
{
|
2023-05-11 03:02:36 -04:00
|
|
|
uint32_t ptr;
|
|
|
|
|
uint32_t n;
|
2023-05-08 00:50:00 +06:00
|
|
|
|
|
|
|
|
ptr = td->CBP;
|
|
|
|
|
n = 0x1000 - (ptr & 0xfff);
|
|
|
|
|
if (n > len) {
|
|
|
|
|
n = len;
|
|
|
|
|
}
|
|
|
|
|
dma_bm_write(ptr, buf, n, 1);
|
|
|
|
|
if (n == len) {
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
ptr = td->BE & ~0xfffu;
|
|
|
|
|
buf += n;
|
|
|
|
|
dma_bm_write(ptr, buf, len - n, 1);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2023-06-11 14:10:47 +02:00
|
|
|
#endif
|
2023-05-08 00:50:00 +06:00
|
|
|
|
2023-06-28 13:46:28 -04:00
|
|
|
static int ohci_copy_td_output(UNUSED(usb_t* dev), usb_td_t *td,
|
2023-05-08 00:50:00 +06:00
|
|
|
uint8_t *buf, int len)
|
|
|
|
|
{
|
2023-05-11 03:02:36 -04:00
|
|
|
uint32_t ptr;
|
|
|
|
|
uint32_t n;
|
2023-05-08 00:50:00 +06:00
|
|
|
|
|
|
|
|
ptr = td->CBP;
|
|
|
|
|
n = 0x1000 - (ptr & 0xfff);
|
|
|
|
|
if (n > len) {
|
|
|
|
|
n = len;
|
|
|
|
|
}
|
|
|
|
|
dma_bm_read(ptr, buf, n, 1);
|
|
|
|
|
if (n == len) {
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
ptr = td->BE & ~0xfffu;
|
|
|
|
|
buf += n;
|
|
|
|
|
dma_bm_read(ptr, buf, len - n, 1);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-08 16:51:16 +06:00
|
|
|
#define OHCI_TD_DIR(val) ((val >> 19) & 3)
|
|
|
|
|
#define OHCI_ED_DIR(val) ((val >> 11) & 3)
|
|
|
|
|
|
2023-05-08 00:50:00 +06:00
|
|
|
uint8_t
|
|
|
|
|
ohci_service_transfer_desc(usb_t* dev, usb_ed_t* endpoint_desc)
|
|
|
|
|
{
|
2023-05-11 03:02:36 -04:00
|
|
|
uint32_t td_addr = endpoint_desc->HeadP & ~0xf;
|
2023-05-08 16:51:16 +06:00
|
|
|
usb_td_t td;
|
2023-05-11 03:02:36 -04:00
|
|
|
uint8_t dir;
|
|
|
|
|
uint8_t pid_token = 255;
|
|
|
|
|
uint32_t len = 0;
|
|
|
|
|
uint32_t pktlen = 0;
|
2023-05-08 16:51:16 +06:00
|
|
|
uint32_t actual_length = 0;
|
|
|
|
|
uint32_t i = 0;
|
|
|
|
|
uint8_t device_result = 0;
|
|
|
|
|
usb_device_t* target = NULL;
|
|
|
|
|
|
|
|
|
|
dma_bm_read(td_addr, (uint8_t*)&td, sizeof(usb_td_t), 4);
|
|
|
|
|
|
|
|
|
|
switch (dir = OHCI_ED_DIR(endpoint_desc->Control)) {
|
|
|
|
|
case 1:
|
|
|
|
|
case 2:
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
dir = OHCI_TD_DIR(td.Control);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch (dir) {
|
|
|
|
|
case 0: /* Setup */
|
|
|
|
|
pid_token = USB_PID_SETUP;
|
|
|
|
|
break;
|
|
|
|
|
case 1: /* OUT */
|
|
|
|
|
pid_token = USB_PID_OUT;
|
|
|
|
|
break;
|
|
|
|
|
case 2: /* IN */
|
|
|
|
|
pid_token = USB_PID_IN;
|
|
|
|
|
break;
|
2023-05-10 17:28:40 +06:00
|
|
|
default:
|
|
|
|
|
return 1;
|
2023-05-08 16:51:16 +06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (td.CBP && td.BE) {
|
|
|
|
|
if ((td.CBP & 0xfffff000) != (td.BE & 0xfffff000)) {
|
|
|
|
|
len = (td.BE & 0xfff) + 0x1001 - (td.CBP & 0xfff);
|
|
|
|
|
} else {
|
|
|
|
|
if (td.CBP > td.BE) {
|
|
|
|
|
ohci_set_interrupt(dev, OHCI_HcInterruptEnable_UE);
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
len = (td.BE - td.CBP) + 1;
|
|
|
|
|
}
|
|
|
|
|
if (len > sizeof(dev->ohci_usb_buf)) {
|
|
|
|
|
len = sizeof(dev->ohci_usb_buf);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pktlen = len;
|
|
|
|
|
if (len && pid_token != USB_PID_IN) {
|
|
|
|
|
pktlen = (endpoint_desc->Control >> 16) & 0xFFF;
|
|
|
|
|
if (pktlen > len) {
|
|
|
|
|
pktlen = len;
|
|
|
|
|
}
|
|
|
|
|
ohci_copy_td_output(dev, &td, dev->ohci_usb_buf, pktlen);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < 2; i++) {
|
|
|
|
|
if (!dev->ohci_devices[i])
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
assert(dev->ohci_devices[i]->device_get_address != NULL);
|
|
|
|
|
|
|
|
|
|
if (dev->ohci_devices[i]->device_get_address(dev->ohci_devices[i]->priv) != (endpoint_desc->Control & 0x7f))
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
target = dev->ohci_devices[i];
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!target)
|
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
|
|
device_result = target->device_process(target->priv, dev->ohci_usb_buf, &actual_length, pid_token, (endpoint_desc->Control & 0x780) >> 7, !(endpoint_desc->Control & (1 << 18)));
|
|
|
|
|
|
|
|
|
|
if ((actual_length == pktlen) || (pid_token == USB_PID_IN && (endpoint_desc->Control & (1 << 18)) && device_result == USB_ERROR_NO_ERROR)) {
|
|
|
|
|
if (len == actual_length) {
|
|
|
|
|
td.CBP = 0;
|
|
|
|
|
} else {
|
|
|
|
|
if ((td.CBP & 0xfff) + actual_length > 0xfff) {
|
|
|
|
|
td.CBP = (td.BE & ~0xfff) + ((td.CBP + actual_length) & 0xfff);
|
|
|
|
|
} else {
|
|
|
|
|
td.CBP += actual_length;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
td.Control |= (1 << 25); /* dataToggle[1] */
|
|
|
|
|
td.Control ^= (1 << 24); /* dataToggle[0] */
|
|
|
|
|
td.Control &= ~0xFC000000; /* Set both ErrorCount and ConditionCode to 0. */
|
|
|
|
|
|
|
|
|
|
if (pid_token != USB_PID_IN && len != actual_length) {
|
|
|
|
|
goto exit_no_retire;
|
|
|
|
|
}
|
2023-05-08 00:50:00 +06:00
|
|
|
|
2023-05-08 16:51:16 +06:00
|
|
|
endpoint_desc->HeadP &= ~0x2;
|
|
|
|
|
if (td.Control & (1 << 24)) {
|
|
|
|
|
endpoint_desc->HeadP |= 0x2;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
if (actual_length != 0xFFFFFFFF && actual_length >= 0) {
|
|
|
|
|
td.Control &= ~0xF0000000;
|
|
|
|
|
td.Control |= 0x90000000;
|
|
|
|
|
} else {
|
|
|
|
|
switch (device_result) {
|
|
|
|
|
case USB_ERROR_NAK:
|
|
|
|
|
return 1;
|
2023-06-28 13:46:28 -04:00
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
break;
|
2023-05-08 16:51:16 +06:00
|
|
|
}
|
|
|
|
|
dev->ohci_interrupt_counter = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
endpoint_desc->HeadP |= 0x1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
endpoint_desc->HeadP &= 0xf;
|
|
|
|
|
endpoint_desc->HeadP |= td.NextTD & ~0xf;
|
|
|
|
|
td.NextTD = dev->ohci_mmio[OHCI_HcDoneHead].l;
|
|
|
|
|
dev->ohci_mmio[OHCI_HcDoneHead].l = td_addr;
|
|
|
|
|
i = (td.Control >> 21) & 7;
|
|
|
|
|
if (i < dev->ohci_interrupt_counter) {
|
|
|
|
|
dev->ohci_interrupt_counter = i;
|
|
|
|
|
}
|
|
|
|
|
exit_no_retire:
|
|
|
|
|
dma_bm_write(td_addr, (uint8_t*)&td, sizeof(usb_td_t), 4);
|
|
|
|
|
return !(td.Control & 0xF0000000);
|
2023-05-08 00:50:00 +06:00
|
|
|
}
|
|
|
|
|
|
2023-05-07 23:07:03 +06:00
|
|
|
uint8_t
|
|
|
|
|
ohci_service_endpoint_desc(usb_t* dev, uint32_t head)
|
|
|
|
|
{
|
|
|
|
|
usb_ed_t endpoint_desc;
|
2023-05-08 00:50:00 +06:00
|
|
|
uint8_t active = 0;
|
|
|
|
|
uint32_t next = 0;
|
|
|
|
|
uint32_t limit_counter = 0;
|
|
|
|
|
|
|
|
|
|
if (head == 0)
|
|
|
|
|
return 0;
|
2023-05-07 23:07:03 +06:00
|
|
|
|
2023-05-11 03:02:36 -04:00
|
|
|
for (uint32_t cur = head; cur && limit_counter++ < ENDPOINT_DESC_LIMIT; cur = next) {
|
2023-05-08 00:50:00 +06:00
|
|
|
dma_bm_read(cur, (uint8_t*)&endpoint_desc, sizeof(usb_ed_t), 4);
|
|
|
|
|
|
2023-05-11 03:02:36 -04:00
|
|
|
next = endpoint_desc.NextED & ~0xFu;
|
2023-05-08 00:50:00 +06:00
|
|
|
|
2023-05-08 16:51:16 +06:00
|
|
|
if ((endpoint_desc.Control & (1 << 13)) || (endpoint_desc.HeadP & (1 << 0)))
|
2023-05-08 00:50:00 +06:00
|
|
|
continue;
|
|
|
|
|
|
2023-05-08 16:51:16 +06:00
|
|
|
if (endpoint_desc.Control & 0x8000) {
|
2023-05-08 00:50:00 +06:00
|
|
|
fatal("OHCI: Isochronous transfers not implemented!\n");
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-08 16:51:16 +06:00
|
|
|
active = 1;
|
|
|
|
|
|
2023-05-11 03:02:36 -04:00
|
|
|
while ((endpoint_desc.HeadP & ~0xFu) != endpoint_desc.TailP) {
|
2023-05-08 00:50:00 +06:00
|
|
|
ohci_service_transfer_desc(dev, &endpoint_desc);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
dma_bm_write(cur, (uint8_t*)&endpoint_desc, sizeof(usb_ed_t), 4);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return active;
|
2023-05-07 23:07:03 +06:00
|
|
|
}
|
|
|
|
|
|
2023-05-07 00:33:15 +06:00
|
|
|
void
|
|
|
|
|
ohci_end_of_frame(usb_t* dev)
|
|
|
|
|
{
|
2023-05-07 16:55:15 +06:00
|
|
|
usb_hcca_t hcca;
|
2023-05-08 17:13:34 +06:00
|
|
|
if (dev->ohci_initial_start)
|
|
|
|
|
return;
|
2023-05-07 16:55:15 +06:00
|
|
|
dma_bm_read(dev->ohci_mmio[OHCI_HcHCCA].l, (uint8_t*)&hcca, sizeof(usb_hcca_t), 4);
|
|
|
|
|
|
2023-05-07 23:07:03 +06:00
|
|
|
if (dev->ohci_mmio[OHCI_HcControl].l & OHCI_HcControl_PeriodicListEnable) {
|
|
|
|
|
ohci_service_endpoint_desc(dev, hcca.HccaInterrruptTable[dev->ohci_mmio[OHCI_HcFmNumber].l & 0x1f]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ((dev->ohci_mmio[OHCI_HcControl].l & OHCI_HcControl_ControlListEnable)
|
|
|
|
|
&& (dev->ohci_mmio[OHCI_HcCommandStatus].l & 0x2)) {
|
|
|
|
|
uint8_t result = ohci_service_endpoint_desc(dev, dev->ohci_mmio[OHCI_HcControlHeadED].l);
|
|
|
|
|
if (!result) {
|
|
|
|
|
dev->ohci_mmio[OHCI_HcControlHeadED].l = 0;
|
|
|
|
|
dev->ohci_mmio[OHCI_HcCommandStatus].l &= ~0x2;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ((dev->ohci_mmio[OHCI_HcControl].l & OHCI_HcControl_BulkListEnable)
|
|
|
|
|
&& (dev->ohci_mmio[OHCI_HcCommandStatus].l & 0x4)) {
|
|
|
|
|
uint8_t result = ohci_service_endpoint_desc(dev, dev->ohci_mmio[OHCI_HcBulkHeadED].l);
|
|
|
|
|
if (!result) {
|
|
|
|
|
dev->ohci_mmio[OHCI_HcBulkHeadED].l = 0;
|
|
|
|
|
dev->ohci_mmio[OHCI_HcCommandStatus].l &= ~0x4;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (dev->ohci_interrupt_counter == 0 && !(dev->ohci_mmio[OHCI_HcInterruptStatus].l & OHCI_HcInterruptEnable_WDH)) {
|
|
|
|
|
if (dev->ohci_mmio[OHCI_HcDoneHead].l == 0) {
|
|
|
|
|
fatal("OHCI: HcDoneHead is still NULL!");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (dev->ohci_mmio[OHCI_HcInterruptStatus].l & dev->ohci_mmio[OHCI_HcInterruptEnable].l) {
|
|
|
|
|
dev->ohci_mmio[OHCI_HcDoneHead].l |= 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
hcca.HccaDoneHead = dev->ohci_mmio[OHCI_HcDoneHead].l;
|
|
|
|
|
dev->ohci_mmio[OHCI_HcDoneHead].l = 0;
|
|
|
|
|
dev->ohci_interrupt_counter = 7;
|
|
|
|
|
ohci_set_interrupt(dev, OHCI_HcInterruptEnable_WDH);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (dev->ohci_interrupt_counter != 0 && dev->ohci_interrupt_counter != 7) {
|
|
|
|
|
dev->ohci_interrupt_counter--;
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-07 02:53:04 +02:00
|
|
|
dev->ohci_mmio[OHCI_HcFmNumber].w[0]++;
|
2023-05-07 23:07:03 +06:00
|
|
|
hcca.HccaFrameNumber = dev->ohci_mmio[OHCI_HcFmNumber].w[0];
|
2023-05-07 16:55:15 +06:00
|
|
|
|
|
|
|
|
dma_bm_write(dev->ohci_mmio[OHCI_HcHCCA].l, (uint8_t*)&hcca, sizeof(usb_hcca_t), 4);
|
2023-05-07 00:33:15 +06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
ohci_start_of_frame(usb_t* dev)
|
|
|
|
|
{
|
2023-05-08 17:13:34 +06:00
|
|
|
dev->ohci_initial_start = 0;
|
2023-05-07 00:33:15 +06:00
|
|
|
ohci_set_interrupt(dev, OHCI_HcInterruptEnable_SO);
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-01 16:55:49 +06:00
|
|
|
void
|
|
|
|
|
ohci_update_frame_counter(void* priv)
|
|
|
|
|
{
|
|
|
|
|
usb_t *dev = (usb_t *) priv;
|
2023-05-06 23:19:55 +06:00
|
|
|
|
2023-05-07 02:53:04 +02:00
|
|
|
dev->ohci_mmio[OHCI_HcFmRemaining].w[0] &= 0x3fff;
|
|
|
|
|
if (dev->ohci_mmio[OHCI_HcFmRemaining].w[0] == 0) {
|
2023-05-07 00:33:15 +06:00
|
|
|
ohci_end_of_frame(dev);
|
2023-05-07 02:53:04 +02:00
|
|
|
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);
|
2023-05-07 00:33:15 +06:00
|
|
|
ohci_start_of_frame(dev);
|
2023-05-07 23:07:03 +06:00
|
|
|
timer_on_auto(&dev->ohci_frame_timer, 1. / 12.);
|
2023-05-07 01:05:35 +06:00
|
|
|
return;
|
2023-05-06 23:19:55 +06:00
|
|
|
}
|
2023-05-07 23:07:03 +06:00
|
|
|
dev->ohci_mmio[OHCI_HcFmRemaining].w[0]--;
|
|
|
|
|
timer_on_auto(&dev->ohci_frame_timer, 1. / 12.);
|
2023-05-03 16:14:15 +06:00
|
|
|
}
|
|
|
|
|
|
2023-05-01 16:55:49 +06:00
|
|
|
void
|
|
|
|
|
ohci_port_reset_callback(void* priv)
|
|
|
|
|
{
|
|
|
|
|
usb_t *dev = (usb_t *) priv;
|
|
|
|
|
|
2023-05-07 02:53:04 +02:00
|
|
|
dev->ohci_mmio[OHCI_HcRhPortStatus1].b[0] &= ~0x10;
|
|
|
|
|
dev->ohci_mmio[OHCI_HcRhPortStatus1].b[2] |= 0x10;
|
2023-05-01 16:55:49 +06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
ohci_port_reset_callback_2(void* priv)
|
|
|
|
|
{
|
|
|
|
|
usb_t *dev = (usb_t *) priv;
|
|
|
|
|
|
2023-05-07 02:53:04 +02:00
|
|
|
dev->ohci_mmio[OHCI_HcRhPortStatus2].b[0] &= ~0x10;
|
|
|
|
|
dev->ohci_mmio[OHCI_HcRhPortStatus2].b[2] |= 0x10;
|
2023-05-01 16:55:49 +06:00
|
|
|
}
|
2023-05-07 02:53:04 +02:00
|
|
|
|
2023-05-08 16:51:16 +06:00
|
|
|
static void
|
|
|
|
|
ohci_soft_reset(usb_t* dev)
|
|
|
|
|
{
|
|
|
|
|
uint32_t old_HcControl = (dev->ohci_mmio[OHCI_HcControl].l & 0x100) | 0xc0;
|
|
|
|
|
memset(dev->ohci_mmio, 0x00, 4096);
|
|
|
|
|
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;
|
|
|
|
|
dev->ohci_mmio[OHCI_HcFmInterval].l = 0x27782edf; /* FrameInterval = 11999, FSLargestDataPacket = 10104 */
|
|
|
|
|
dev->ohci_mmio[OHCI_HcLSThreshold].l = 0x628;
|
|
|
|
|
dev->ohci_mmio[OHCI_HcInterruptEnable].l |= (1 << 31);
|
|
|
|
|
dev->ohci_mmio[OHCI_HcControl].l = old_HcControl;
|
|
|
|
|
dev->ohci_interrupt_counter = 7;
|
|
|
|
|
ohci_update_irq(dev);
|
|
|
|
|
}
|
|
|
|
|
|
2020-04-23 02:23:59 +02:00
|
|
|
static void
|
2023-06-28 13:46:28 -04:00
|
|
|
ohci_mmio_write(uint32_t addr, uint8_t val, void *priv)
|
2020-04-23 02:23:59 +02:00
|
|
|
{
|
2023-06-28 13:46:28 -04:00
|
|
|
usb_t *dev = (usb_t *) priv;
|
2020-05-15 07:23:52 +02:00
|
|
|
uint8_t old;
|
2020-04-23 02:23:59 +02:00
|
|
|
|
2023-05-07 02:53:04 +02:00
|
|
|
#ifdef ENABLE_USB_LOG
|
|
|
|
|
usb_log("[W] %08X = %04X\n", addr, val);
|
|
|
|
|
#endif
|
|
|
|
|
|
2020-04-23 02:23:59 +02:00
|
|
|
addr &= 0x00000fff;
|
|
|
|
|
|
|
|
|
|
switch (addr) {
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcControl:
|
2023-05-08 16:51:16 +06:00
|
|
|
old = dev->ohci_mmio[OHCI_HcControl].b[0];
|
2023-05-09 00:14:15 +06:00
|
|
|
#ifdef ENABLE_USB_LOG
|
|
|
|
|
usb_log("OHCI: OHCI state 0x%X\n", (val & 0xc0));
|
|
|
|
|
#endif
|
2022-09-18 17:11:43 -04:00
|
|
|
if ((val & 0xc0) == 0x00) {
|
|
|
|
|
/* UsbReset */
|
2023-05-07 02:53:04 +02:00
|
|
|
dev->ohci_mmio[OHCI_HcRhPortStatus1].b[2] = dev->ohci_mmio[OHCI_HcRhPortStatus2].b[2] = 0x16;
|
2023-05-08 16:51:16 +06:00
|
|
|
for (int i = 0; i < 2; i++) {
|
|
|
|
|
if (dev->ohci_devices[i]) {
|
|
|
|
|
dev->ohci_devices[i]->device_reset(dev->ohci_devices[i]->priv);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else if ((val & 0xc0) == 0x80 && (old & 0xc0) != (val & 0xc0)) {
|
|
|
|
|
dev->ohci_mmio[OHCI_HcFmRemaining].l = 0;
|
|
|
|
|
dev->ohci_initial_start = 1;
|
|
|
|
|
timer_on_auto(&dev->ohci_frame_timer, 1000.);
|
2022-09-18 17:11:43 -04:00
|
|
|
}
|
|
|
|
|
break;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcCommandStatus:
|
2022-09-18 17:11:43 -04:00
|
|
|
/* bit OwnershipChangeRequest triggers an ownership change (SMM <-> OS) */
|
|
|
|
|
if (val & 0x08) {
|
2023-05-07 02:53:04 +02:00
|
|
|
dev->ohci_mmio[OHCI_HcInterruptStatus].b[3] = 0x40;
|
2023-05-10 17:09:13 +06:00
|
|
|
if ((dev->ohci_mmio[OHCI_HcInterruptEnable].b[3] & 0x40) == 0x40) {
|
2022-09-18 17:11:43 -04:00
|
|
|
smi_raise();
|
2023-05-10 17:09:13 +06:00
|
|
|
}
|
2022-09-18 17:11:43 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* bit HostControllerReset must be cleared for the controller to be seen as initialized */
|
|
|
|
|
if (val & 0x01) {
|
2023-05-08 16:51:16 +06:00
|
|
|
ohci_soft_reset(dev);
|
2023-05-16 00:52:39 +02:00
|
|
|
|
2022-09-18 17:11:43 -04:00
|
|
|
val &= ~0x01;
|
|
|
|
|
}
|
|
|
|
|
break;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcHCCA:
|
2023-04-30 00:59:11 +06:00
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
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);
|
2023-05-05 23:50:50 +06:00
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcInterruptEnable + 1:
|
|
|
|
|
case OHCI_aHcInterruptEnable + 2:
|
2023-05-05 23:50:50 +06:00
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
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);
|
2023-05-05 23:50:50 +06:00
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
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);
|
2023-05-05 23:50:50 +06:00
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcInterruptDisable + 1:
|
|
|
|
|
case OHCI_aHcInterruptDisable + 2:
|
2023-05-05 23:50:50 +06:00
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
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);
|
2023-05-05 23:50:50 +06:00
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcInterruptStatus:
|
|
|
|
|
dev->ohci_mmio[addr >> 2].b[addr & 3] &= ~(val & 0x7f);
|
2022-09-18 17:11:43 -04:00
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcInterruptStatus + 1:
|
|
|
|
|
case OHCI_aHcInterruptStatus + 2:
|
2022-09-18 17:11:43 -04:00
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcInterruptStatus + 3:
|
|
|
|
|
dev->ohci_mmio[addr >> 2].b[addr & 3] &= ~(val & 0x40);
|
2022-09-18 17:11:43 -04:00
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcFmRemaining + 3:
|
|
|
|
|
dev->ohci_mmio[addr >> 2].b[addr & 3] = (val & 0x80);
|
2022-09-18 17:11:43 -04:00
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcFmRemaining + 1:
|
|
|
|
|
case OHCI_aHcPeriodicStart + 1:
|
|
|
|
|
dev->ohci_mmio[addr >> 2].b[addr & 3] = (val & 0x3f);
|
2022-09-18 17:11:43 -04:00
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcLSThreshold + 1:
|
|
|
|
|
dev->ohci_mmio[addr >> 2].b[addr & 3] = (val & 0x0f);
|
2022-09-18 17:11:43 -04:00
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
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:
|
2022-09-18 17:11:43 -04:00
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcRhDescriptorA + 1:
|
|
|
|
|
dev->ohci_mmio[addr >> 2].b[addr & 3] = (val & 0x1b);
|
2022-09-18 17:11:43 -04:00
|
|
|
if (val & 0x02) {
|
2023-05-07 02:53:04 +02:00
|
|
|
dev->ohci_mmio[OHCI_HcRhPortStatus1].b[1] |= 0x01;
|
|
|
|
|
dev->ohci_mmio[OHCI_HcRhPortStatus2].b[1] |= 0x01;
|
2022-09-18 17:11:43 -04:00
|
|
|
}
|
|
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcRhDescriptorA + 3:
|
|
|
|
|
dev->ohci_mmio[addr >> 2].b[addr & 3] = (val & 0x03);
|
2022-09-18 17:11:43 -04:00
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcRhDescriptorB:
|
|
|
|
|
case OHCI_aHcRhDescriptorB + 2:
|
|
|
|
|
dev->ohci_mmio[addr >> 2].b[addr & 3] = (val & 0x06);
|
2023-04-29 22:43:29 +06:00
|
|
|
if ((addr == OHCI_HcRhDescriptorB) && !(val & 0x04)) {
|
2023-05-07 02:53:04 +02:00
|
|
|
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;
|
2022-09-18 17:11:43 -04:00
|
|
|
}
|
2023-04-29 22:43:29 +06:00
|
|
|
if ((addr == OHCI_HcRhDescriptorB) && !(val & 0x02)) {
|
2023-05-07 02:53:04 +02:00
|
|
|
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;
|
2022-09-18 17:11:43 -04:00
|
|
|
}
|
|
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcRhDescriptorB + 1:
|
|
|
|
|
case OHCI_aHcRhDescriptorB + 3:
|
2022-09-18 17:11:43 -04:00
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcRhStatus:
|
2022-09-18 17:11:43 -04:00
|
|
|
if (val & 0x01) {
|
2023-05-07 02:53:04 +02:00
|
|
|
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;
|
2022-09-18 17:11:43 -04:00
|
|
|
}
|
2023-05-07 02:53:04 +02:00
|
|
|
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;
|
2022-09-18 17:11:43 -04:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcRhStatus + 1:
|
2022-09-18 17:11:43 -04:00
|
|
|
if (val & 0x80)
|
2023-05-07 02:53:04 +02:00
|
|
|
dev->ohci_mmio[addr >> 2].b[addr & 3] |= 0x80;
|
2022-09-18 17:11:43 -04:00
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcRhStatus + 2:
|
|
|
|
|
dev->ohci_mmio[addr >> 2].b[addr & 3] &= ~(val & 0x02);
|
2022-09-18 17:11:43 -04:00
|
|
|
if (val & 0x01) {
|
2023-05-07 02:53:04 +02:00
|
|
|
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;
|
2022-09-18 17:11:43 -04:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcRhStatus + 3:
|
2022-09-18 17:11:43 -04:00
|
|
|
if (val & 0x80)
|
2023-05-07 02:53:04 +02:00
|
|
|
dev->ohci_mmio[OHCI_HcRhStatus].b[1] &= ~0x80;
|
2022-09-18 17:11:43 -04:00
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcRhPortStatus1:
|
|
|
|
|
case OHCI_aHcRhPortStatus2:
|
|
|
|
|
old = dev->ohci_mmio[addr >> 2].b[addr & 3];
|
2022-09-18 17:11:43 -04:00
|
|
|
|
|
|
|
|
if (val & 0x10) {
|
|
|
|
|
if (old & 0x01) {
|
2023-05-07 02:53:04 +02:00
|
|
|
dev->ohci_mmio[addr >> 2].b[addr & 3] |= 0x10;
|
|
|
|
|
timer_on_auto(&dev->ohci_port_reset_timer[(addr - OHCI_aHcRhPortStatus1) / 4], 10000.);
|
2023-05-08 16:51:16 +06:00
|
|
|
if (dev->ohci_devices[(addr - OHCI_aHcRhPortStatus1) >> 2])
|
|
|
|
|
dev->ohci_devices[(addr - OHCI_aHcRhPortStatus1) >> 2]->device_reset(dev->ohci_devices[(addr - OHCI_aHcRhPortStatus1) >> 2]->priv);
|
2022-09-18 17:11:43 -04:00
|
|
|
} else
|
2023-05-07 02:53:04 +02:00
|
|
|
dev->ohci_mmio[(addr + 2) >> 2].b[(addr + 2) & 3] |= 0x01;
|
2022-09-18 17:11:43 -04:00
|
|
|
}
|
|
|
|
|
if (val & 0x08)
|
2023-05-07 02:53:04 +02:00
|
|
|
dev->ohci_mmio[addr >> 2].b[addr & 3] &= ~0x04;
|
2023-05-08 22:30:28 +06:00
|
|
|
if (val & 0x04) {
|
|
|
|
|
if (old & 0x01)
|
|
|
|
|
dev->ohci_mmio[addr >> 2].b[addr & 3] |= 0x04;
|
|
|
|
|
else
|
|
|
|
|
dev->ohci_mmio[(addr + 2) >> 2].b[(addr + 2) & 3] |= 0x01;
|
|
|
|
|
}
|
2022-09-18 17:11:43 -04:00
|
|
|
if (val & 0x02) {
|
|
|
|
|
if (old & 0x01)
|
2023-05-07 02:53:04 +02:00
|
|
|
dev->ohci_mmio[addr >> 2].b[addr & 3] |= 0x02;
|
2022-09-18 17:11:43 -04:00
|
|
|
else
|
2023-05-07 02:53:04 +02:00
|
|
|
dev->ohci_mmio[(addr + 2) >> 2].b[(addr + 2) & 3] |= 0x01;
|
2022-09-18 17:11:43 -04:00
|
|
|
}
|
|
|
|
|
if (val & 0x01) {
|
|
|
|
|
if (old & 0x01)
|
2023-05-07 02:53:04 +02:00
|
|
|
dev->ohci_mmio[addr >> 2].b[addr & 3] &= ~0x02;
|
2022-09-18 17:11:43 -04:00
|
|
|
else
|
2023-05-07 02:53:04 +02:00
|
|
|
dev->ohci_mmio[(addr + 2) >> 2].b[(addr + 2) & 3] |= 0x01;
|
2022-09-18 17:11:43 -04:00
|
|
|
}
|
|
|
|
|
|
2023-05-07 02:53:04 +02:00
|
|
|
if (!(dev->ohci_mmio[addr >> 2].b[addr & 3] & 0x04) && (old & 0x04))
|
|
|
|
|
dev->ohci_mmio[(addr + 2) >> 2].b[(addr + 2) & 3] |= 0x04;
|
2023-06-28 13:46:28 -04:00
|
|
|
#if 0
|
|
|
|
|
if (!(dev->ohci_mmio[addr >> 2].b[addr & 3] & 0x02))
|
|
|
|
|
dev->ohci_mmio[(addr + 2) >> 2].b[(addr + 2) & 3] |= 0x02;
|
|
|
|
|
#endif
|
2022-09-18 17:11:43 -04:00
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
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;
|
2022-09-18 17:11:43 -04:00
|
|
|
}
|
2023-05-07 02:53:04 +02:00
|
|
|
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;
|
2022-09-18 17:11:43 -04:00
|
|
|
}
|
|
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
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;
|
2022-09-18 17:11:43 -04:00
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcRhPortStatus1 + 2:
|
|
|
|
|
case OHCI_aHcRhPortStatus2 + 2:
|
|
|
|
|
dev->ohci_mmio[addr >> 2].b[addr & 3] &= ~(val & 0x1f);
|
2022-09-18 17:11:43 -04:00
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
case OHCI_aHcRhPortStatus1 + 3:
|
|
|
|
|
case OHCI_aHcRhPortStatus2 + 3:
|
2022-09-18 17:11:43 -04:00
|
|
|
return;
|
2023-05-07 02:53:04 +02:00
|
|
|
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);
|
2023-05-05 22:32:20 +06:00
|
|
|
return;
|
2023-06-28 13:46:28 -04:00
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
break;
|
2020-04-23 02:23:59 +02:00
|
|
|
}
|
|
|
|
|
|
2023-05-07 02:53:04 +02:00
|
|
|
dev->ohci_mmio[addr >> 2].b[addr & 3] = val;
|
2020-04-23 02:23:59 +02:00
|
|
|
}
|
2023-05-07 02:53:04 +02:00
|
|
|
|
|
|
|
|
static void
|
2023-06-28 13:46:28 -04:00
|
|
|
ohci_mmio_writew(uint32_t addr, uint16_t val, void *priv)
|
2023-05-07 02:53:04 +02:00
|
|
|
{
|
2023-06-28 13:46:28 -04:00
|
|
|
ohci_mmio_write(addr, val & 0xff, priv);
|
|
|
|
|
ohci_mmio_write(addr + 1, val >> 8, priv);
|
2023-05-07 02:53:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2023-06-28 13:46:28 -04:00
|
|
|
ohci_mmio_writel(uint32_t addr, uint32_t val, void *priv)
|
2023-05-07 02:53:04 +02:00
|
|
|
{
|
2023-06-28 13:46:28 -04:00
|
|
|
ohci_mmio_writew(addr, val & 0xffff, priv);
|
|
|
|
|
ohci_mmio_writew(addr + 2, val >> 16, priv);
|
2023-05-07 02:53:04 +02:00
|
|
|
}
|
|
|
|
|
|
2020-04-23 02:23:59 +02:00
|
|
|
void
|
|
|
|
|
ohci_update_mem_mapping(usb_t *dev, uint8_t base1, uint8_t base2, uint8_t base3, int enable)
|
2017-01-22 00:22:14 -06:00
|
|
|
{
|
2020-04-23 02:23:59 +02:00
|
|
|
if (dev->ohci_enable && (dev->ohci_mem_base != 0x00000000))
|
2022-09-18 17:11:43 -04:00
|
|
|
mem_mapping_disable(&dev->ohci_mmio_mapping);
|
2020-04-23 02:23:59 +02:00
|
|
|
|
|
|
|
|
dev->ohci_mem_base = ((base1 << 8) | (base2 << 16) | (base3 << 24)) & 0xfffff000;
|
2022-09-18 17:11:43 -04:00
|
|
|
dev->ohci_enable = enable;
|
2020-04-23 02:23:59 +02:00
|
|
|
|
2023-05-07 12:42:39 +02:00
|
|
|
if (dev->ohci_enable && (dev->ohci_mem_base != 0x00000000))
|
2022-09-18 17:11:43 -04:00
|
|
|
mem_mapping_set_addr(&dev->ohci_mmio_mapping, dev->ohci_mem_base, 0x1000);
|
2023-05-07 02:53:04 +02:00
|
|
|
|
|
|
|
|
usb_log("ohci_update_mem_mapping(): OHCI %sabled at %08X\n", dev->ohci_enable ? "en" : "dis", dev->ohci_mem_base);
|
2017-01-22 00:22:14 -06:00
|
|
|
}
|
|
|
|
|
|
2023-05-05 22:46:18 +06:00
|
|
|
uint8_t
|
|
|
|
|
usb_attach_device(usb_t *dev, usb_device_t* device, uint8_t bus_type)
|
|
|
|
|
{
|
2023-05-08 16:51:16 +06:00
|
|
|
switch (bus_type) {
|
|
|
|
|
case USB_BUS_OHCI:
|
|
|
|
|
{
|
2023-06-28 13:46:28 -04:00
|
|
|
for (uint8_t i = 0; i < 2; i++) {
|
2023-05-08 16:51:16 +06:00
|
|
|
if (!dev->ohci_devices[i]) {
|
2023-05-08 22:30:28 +06:00
|
|
|
uint32_t old = dev->ohci_mmio[OHCI_HcRhPortStatus1 + (4 * i)].l;
|
2023-05-08 16:51:16 +06:00
|
|
|
dev->ohci_devices[i] = device;
|
|
|
|
|
dev->ohci_mmio[OHCI_HcRhPortStatus1 + (4 * i)].b[0] |= 0x1;
|
|
|
|
|
if ((dev->ohci_mmio[OHCI_HcControl].b[0] & 0xc0) == 0xc0) {
|
|
|
|
|
ohci_set_interrupt(dev, OHCI_HcInterruptEnable_RD);
|
|
|
|
|
}
|
2023-05-08 22:30:28 +06:00
|
|
|
if (old != dev->ohci_mmio[OHCI_HcRhPortStatus1 + (4 * i)].l) {
|
|
|
|
|
dev->ohci_mmio[OHCI_HcRhPortStatus1 + (4 * i)].b[2] |= 0x1;
|
|
|
|
|
ohci_set_interrupt(dev, OHCI_HcInterruptEnable_RHSC);
|
|
|
|
|
}
|
2023-05-08 16:51:16 +06:00
|
|
|
return i;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
2023-06-28 13:46:28 -04:00
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
break;
|
2023-05-08 16:51:16 +06:00
|
|
|
}
|
2023-06-28 13:46:28 -04:00
|
|
|
return 0xff;
|
2023-05-05 22:46:18 +06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
usb_detach_device(usb_t *dev, uint8_t port, uint8_t bus_type)
|
|
|
|
|
{
|
2023-05-08 16:51:16 +06:00
|
|
|
switch (bus_type) {
|
|
|
|
|
case USB_BUS_OHCI:
|
|
|
|
|
{
|
2023-05-08 22:30:28 +06:00
|
|
|
if (port > 2)
|
|
|
|
|
return;
|
|
|
|
|
if (dev->ohci_devices[port]) {
|
|
|
|
|
uint32_t old = dev->ohci_mmio[OHCI_HcRhPortStatus1 + (4 * port)].l;
|
|
|
|
|
dev->ohci_devices[port] = NULL;
|
|
|
|
|
if (dev->ohci_mmio[OHCI_HcRhPortStatus1 + (4 * port)].b[0] & 0x1) {
|
|
|
|
|
dev->ohci_mmio[OHCI_HcRhPortStatus1 + (4 * port)].b[0] &= ~0x1;
|
|
|
|
|
dev->ohci_mmio[OHCI_HcRhPortStatus1 + (4 * port)].b[2] |= 0x1;
|
|
|
|
|
}
|
|
|
|
|
if (dev->ohci_mmio[OHCI_HcRhPortStatus1 + (4 * port)].b[0] & 0x2) {
|
|
|
|
|
dev->ohci_mmio[OHCI_HcRhPortStatus1 + (4 * port)].b[0] &= ~0x2;
|
|
|
|
|
dev->ohci_mmio[OHCI_HcRhPortStatus1 + (4 * port)].b[2] |= 0x2;
|
2023-05-08 16:51:16 +06:00
|
|
|
}
|
2023-05-08 22:30:28 +06:00
|
|
|
if (old != dev->ohci_mmio[OHCI_HcRhPortStatus1 + (4 * port)].l)
|
|
|
|
|
ohci_set_interrupt(dev, OHCI_HcInterruptEnable_RHSC);
|
|
|
|
|
return;
|
2023-05-08 16:51:16 +06:00
|
|
|
}
|
2023-05-08 22:30:28 +06:00
|
|
|
|
2023-05-08 16:51:16 +06:00
|
|
|
}
|
|
|
|
|
break;
|
2023-06-28 13:46:28 -04:00
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
break;
|
2023-05-08 16:51:16 +06:00
|
|
|
}
|
|
|
|
|
return;
|
2023-05-05 22:46:18 +06:00
|
|
|
}
|
|
|
|
|
|
2020-11-16 00:01:21 +01:00
|
|
|
static void
|
|
|
|
|
usb_reset(void *priv)
|
|
|
|
|
{
|
|
|
|
|
usb_t *dev = (usb_t *) priv;
|
|
|
|
|
|
2023-05-07 02:53:04 +02:00
|
|
|
memset(dev->uhci_io, 0x00, sizeof(dev->uhci_io));
|
2020-11-16 00:01:21 +01:00
|
|
|
dev->uhci_io[0x0c] = 0x40;
|
|
|
|
|
dev->uhci_io[0x10] = dev->uhci_io[0x12] = 0x80;
|
|
|
|
|
|
2023-05-08 16:51:16 +06:00
|
|
|
ohci_soft_reset(dev);
|
|
|
|
|
dev->ohci_mmio[OHCI_HcControl].l = 0x00;
|
2020-11-16 00:01:21 +01:00
|
|
|
|
|
|
|
|
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;
|
2023-05-07 02:53:04 +02:00
|
|
|
|
|
|
|
|
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);
|
2020-11-16 00:01:21 +01:00
|
|
|
}
|
|
|
|
|
|
2020-04-23 02:23:59 +02:00
|
|
|
static void
|
|
|
|
|
usb_close(void *priv)
|
2017-01-22 00:22:14 -06:00
|
|
|
{
|
2020-04-23 02:23:59 +02:00
|
|
|
usb_t *dev = (usb_t *) priv;
|
|
|
|
|
|
|
|
|
|
free(dev);
|
2017-01-22 00:22:14 -06:00
|
|
|
}
|
|
|
|
|
|
2020-04-23 02:23:59 +02:00
|
|
|
static void *
|
2023-06-28 13:46:28 -04:00
|
|
|
usb_init_ext(UNUSED(const device_t *info), void *params)
|
2017-01-22 00:22:14 -06:00
|
|
|
{
|
2020-04-23 02:23:59 +02:00
|
|
|
usb_t *dev;
|
|
|
|
|
|
2023-05-07 14:09:48 +02:00
|
|
|
dev = (usb_t *) calloc(1, sizeof(usb_t));
|
2022-09-18 17:11:43 -04:00
|
|
|
if (dev == NULL)
|
|
|
|
|
return (NULL);
|
2020-04-23 02:23:59 +02:00
|
|
|
|
2023-05-07 02:53:04 +02:00
|
|
|
dev->usb_params = (usb_params_t *) params;
|
2023-04-30 23:01:15 +06:00
|
|
|
|
2023-05-07 02:53:04 +02:00
|
|
|
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,
|
2022-09-18 17:11:43 -04:00
|
|
|
NULL, MEM_MAPPING_EXTERNAL, dev);
|
2023-05-07 02:53:04 +02:00
|
|
|
|
|
|
|
|
mem_mapping_disable(&dev->ohci_mmio_mapping);
|
|
|
|
|
|
2023-05-01 16:55:49 +06:00
|
|
|
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);
|
2023-05-07 02:53:04 +02:00
|
|
|
|
2020-11-16 00:01:21 +01:00
|
|
|
usb_reset(dev);
|
2020-04-23 02:23:59 +02:00
|
|
|
|
2023-05-08 22:30:28 +06:00
|
|
|
usb_device_inst = dev;
|
|
|
|
|
|
2020-04-23 02:23:59 +02:00
|
|
|
return dev;
|
2017-05-05 01:49:42 +02:00
|
|
|
}
|
2020-04-23 02:23:59 +02:00
|
|
|
|
2022-03-13 09:15:11 -04:00
|
|
|
const device_t usb_device = {
|
2022-09-18 17:11:43 -04:00
|
|
|
.name = "Universal Serial Bus",
|
2022-03-13 09:15:11 -04:00
|
|
|
.internal_name = "usb",
|
2023-04-30 23:01:15 +06:00
|
|
|
.flags = DEVICE_PCI | DEVICE_EXTPARAMS,
|
2022-09-18 17:11:43 -04:00
|
|
|
.local = 0,
|
2023-04-30 23:01:15 +06:00
|
|
|
.init_ext = usb_init_ext,
|
2022-09-18 17:11:43 -04:00
|
|
|
.close = usb_close,
|
|
|
|
|
.reset = usb_reset,
|
2022-03-13 09:15:11 -04:00
|
|
|
{ .available = NULL },
|
|
|
|
|
.speed_changed = NULL,
|
2022-09-18 17:11:43 -04:00
|
|
|
.force_redraw = NULL,
|
|
|
|
|
.config = NULL
|
2020-04-23 02:23:59 +02:00
|
|
|
};
|