mirror of
https://github.com/dragonpilot/dragonpilot.git
synced 2026-02-27 09:33:56 +08:00
30c7ca8a5 bump version to 1.5.3 9403dbebe Need to fix wifi test before re-enabling. 0812362b5 GPS UART fix until boardd is refactored (#294) ffbdb87a8 python2 -> 3 fixes to pedal flasher (#292) 78b75ef59 Added build type to release version strings 736c2cbf7 Fixed sending of bytes over PandaSerial 0894b28f1 Fixed USB power mode on black (#291) 4b3358c92 patch to be able to switch from EON to PC with a Panda that has EON b… (#290) a95c44a71 Made setting of NOOUTPUT on no heartbeat more efficient (#287) 948683688 UART instability fix with high interrupt load (#283) 9a9e9d47b Fix usb_power_mode missing initialization (#289) af0960ad3 DFU fix (#288) 70219d7bb match safety enum in cereal (#285) a338d3932 Fix build for jenkins test 78ef4a6eb Stop charge (#284) 5266a4028 Fix typo (#286) f4787ec5a Revert "turn on CDP when ignition switches on (#281)" d37daee97 Revert "NONE and CLIENT should be the same thing in white/grey pandas" e97b283e7 NONE and CLIENT should be the same thing in white/grey pandas 8c1df559f turn on CDP when ignition switches on (#281) 847a35d42 Fix bullet points fac027716 Misra update (#280) 5a04df6b1 Added description of regression tests to README c4aabae59 Fixed some python3 bugs in the test scripts and PandaSerial 9af0cb353 Bump version c4ac3d63b Disable GPS load switching on black pandas 078ee588c This is the correct table, actually 578b95ee3 Misra table of coverage added d383a2625 bump panda b98ca010d fix sdk build in python3 env (#279) 63d3dc7d3 Set python3 env before runnign get_sdk, so we know if it fails e951d79c0 legacy code we don't control can remain python2 11b715118 Merge pull request #276 from commaai/python3 9893a842a Merge pull request #277 from zorrobyte/patch-1 d3268690c Revert "revert back esptool to python2 and force to build esptools with python2" 875e76012 revert back esptool to python2 and force to build esptools with python2 9c40e6240 needed to install python3 ed2ac87cf Also moved safety tests to python3 6842b2d2c move esptool sdk installation before python3 env is set. Kind of a cheat b5a2cabcd this hopefully fixes build test 628050955 Fixes safety replay 2c220b623 this fixes language regr test fdbe789b8 use python 3 in Docker container ee1ae4f86 Better hash print 0de9ef73c Revert "Final 2to3 on the whole repo" c92fd3bc9 Final 2to3 on the whole repo 5f2bc4460 better b2a30fdbd make works! b74005d10 fix sign.py fe727706b read file as byte and no tab before sleep 32a344ef6 Update README.md 2dc34096a 2to3 applied ffa68ef71 undo unnecessary brackets for print dbc248027 Fix all the prints with 2to3, some need to be undo 5a7aeba0f xrange is gone 982c4c928 one more python3 env 1e2412a29 env python -> env python3 git-subtree-dir: panda git-subtree-split: 30c7ca8a53a3adb05d23d7cfe64fb716a656ef1a
1019 lines
36 KiB
C
1019 lines
36 KiB
C
// IRQs: OTG_FS
|
||
|
||
typedef union {
|
||
uint16_t w;
|
||
struct BW {
|
||
uint8_t msb;
|
||
uint8_t lsb;
|
||
}
|
||
bw;
|
||
}
|
||
uint16_t_uint8_t;
|
||
|
||
typedef union _USB_Setup {
|
||
uint32_t d8[2];
|
||
struct _SetupPkt_Struc
|
||
{
|
||
uint8_t bmRequestType;
|
||
uint8_t bRequest;
|
||
uint16_t_uint8_t wValue;
|
||
uint16_t_uint8_t wIndex;
|
||
uint16_t_uint8_t wLength;
|
||
} b;
|
||
}
|
||
USB_Setup_TypeDef;
|
||
|
||
void usb_init(void);
|
||
int usb_cb_control_msg(USB_Setup_TypeDef *setup, uint8_t *resp, bool hardwired);
|
||
int usb_cb_ep1_in(void *usbdata, int len, bool hardwired);
|
||
void usb_cb_ep2_out(void *usbdata, int len, bool hardwired);
|
||
void usb_cb_ep3_out(void *usbdata, int len, bool hardwired);
|
||
void usb_cb_enumeration_complete(void);
|
||
|
||
// **** supporting defines ****
|
||
|
||
typedef struct
|
||
{
|
||
__IO uint32_t HPRT;
|
||
}
|
||
USB_OTG_HostPortTypeDef;
|
||
|
||
USB_OTG_GlobalTypeDef *USBx = USB_OTG_FS;
|
||
|
||
#define USBx_HOST ((USB_OTG_HostTypeDef *)((uint32_t)USBx + USB_OTG_HOST_BASE))
|
||
#define USBx_HOST_PORT ((USB_OTG_HostPortTypeDef *)((uint32_t)USBx + USB_OTG_HOST_PORT_BASE))
|
||
#define USBx_DEVICE ((USB_OTG_DeviceTypeDef *)((uint32_t)USBx + USB_OTG_DEVICE_BASE))
|
||
#define USBx_INEP(i) ((USB_OTG_INEndpointTypeDef *)((uint32_t)USBx + USB_OTG_IN_ENDPOINT_BASE + ((i) * USB_OTG_EP_REG_SIZE)))
|
||
#define USBx_OUTEP(i) ((USB_OTG_OUTEndpointTypeDef *)((uint32_t)USBx + USB_OTG_OUT_ENDPOINT_BASE + ((i) * USB_OTG_EP_REG_SIZE)))
|
||
#define USBx_DFIFO(i) *(__IO uint32_t *)((uint32_t)USBx + USB_OTG_FIFO_BASE + ((i) * USB_OTG_FIFO_SIZE))
|
||
#define USBx_PCGCCTL *(__IO uint32_t *)((uint32_t)USBx + USB_OTG_PCGCCTL_BASE)
|
||
|
||
#define USB_REQ_GET_STATUS 0x00
|
||
#define USB_REQ_CLEAR_FEATURE 0x01
|
||
#define USB_REQ_SET_FEATURE 0x03
|
||
#define USB_REQ_SET_ADDRESS 0x05
|
||
#define USB_REQ_GET_DESCRIPTOR 0x06
|
||
#define USB_REQ_SET_DESCRIPTOR 0x07
|
||
#define USB_REQ_GET_CONFIGURATION 0x08
|
||
#define USB_REQ_SET_CONFIGURATION 0x09
|
||
#define USB_REQ_GET_INTERFACE 0x0A
|
||
#define USB_REQ_SET_INTERFACE 0x0B
|
||
#define USB_REQ_SYNCH_FRAME 0x0C
|
||
|
||
#define USB_DESC_TYPE_DEVICE 0x01
|
||
#define USB_DESC_TYPE_CONFIGURATION 0x02
|
||
#define USB_DESC_TYPE_STRING 0x03
|
||
#define USB_DESC_TYPE_INTERFACE 0x04
|
||
#define USB_DESC_TYPE_ENDPOINT 0x05
|
||
#define USB_DESC_TYPE_DEVICE_QUALIFIER 0x06
|
||
#define USB_DESC_TYPE_OTHER_SPEED_CONFIGURATION 0x07
|
||
#define USB_DESC_TYPE_BINARY_OBJECT_STORE 0x0f
|
||
|
||
// offsets for configuration strings
|
||
#define STRING_OFFSET_LANGID 0x00
|
||
#define STRING_OFFSET_IMANUFACTURER 0x01
|
||
#define STRING_OFFSET_IPRODUCT 0x02
|
||
#define STRING_OFFSET_ISERIAL 0x03
|
||
#define STRING_OFFSET_ICONFIGURATION 0x04
|
||
#define STRING_OFFSET_IINTERFACE 0x05
|
||
|
||
// WebUSB requests
|
||
#define WEBUSB_REQ_GET_URL 0x02
|
||
|
||
// WebUSB types
|
||
#define WEBUSB_DESC_TYPE_URL 0x03
|
||
#define WEBUSB_URL_SCHEME_HTTPS 0x01
|
||
#define WEBUSB_URL_SCHEME_HTTP 0x00
|
||
|
||
// WinUSB requests
|
||
#define WINUSB_REQ_GET_COMPATID_DESCRIPTOR 0x04
|
||
#define WINUSB_REQ_GET_EXT_PROPS_OS 0x05
|
||
#define WINUSB_REQ_GET_DESCRIPTOR 0x07
|
||
|
||
#define STS_GOUT_NAK 1
|
||
#define STS_DATA_UPDT 2
|
||
#define STS_XFER_COMP 3
|
||
#define STS_SETUP_COMP 4
|
||
#define STS_SETUP_UPDT 6
|
||
|
||
#define USBD_FS_TRDT_VALUE 5U
|
||
|
||
#define USB_OTG_SPEED_FULL 3
|
||
|
||
uint8_t resp[MAX_RESP_LEN];
|
||
|
||
// for the repeating interfaces
|
||
#define DSCR_INTERFACE_LEN 9
|
||
#define DSCR_ENDPOINT_LEN 7
|
||
#define DSCR_CONFIG_LEN 9
|
||
#define DSCR_DEVICE_LEN 18
|
||
|
||
// endpoint types
|
||
#define ENDPOINT_TYPE_CONTROL 0
|
||
#define ENDPOINT_TYPE_ISO 1
|
||
#define ENDPOINT_TYPE_BULK 2
|
||
#define ENDPOINT_TYPE_INT 3
|
||
|
||
// These are arbitrary values used in bRequest
|
||
#define MS_VENDOR_CODE 0x20
|
||
#define WEBUSB_VENDOR_CODE 0x30
|
||
|
||
// BOS constants
|
||
#define BINARY_OBJECT_STORE_DESCRIPTOR_LENGTH 0x05
|
||
#define BINARY_OBJECT_STORE_DESCRIPTOR 0x0F
|
||
#define WINUSB_PLATFORM_DESCRIPTOR_LENGTH 0x9E
|
||
|
||
// Convert machine byte order to USB byte order
|
||
#define TOUSBORDER(num)\
|
||
((num) & 0xFFU), (((num) >> 8) & 0xFFU)
|
||
|
||
// take in string length and return the first 2 bytes of a string descriptor
|
||
#define STRING_DESCRIPTOR_HEADER(size)\
|
||
(((((size) * 2) + 2) & 0xFF) | 0x0300)
|
||
|
||
uint8_t device_desc[] = {
|
||
DSCR_DEVICE_LEN, USB_DESC_TYPE_DEVICE, //Length, Type
|
||
0x10, 0x02, // bcdUSB max version of USB supported (2.1)
|
||
0xFF, 0xFF, 0xFF, 0x40, // Class, Subclass, Protocol, Max Packet Size
|
||
TOUSBORDER(USB_VID), // idVendor
|
||
TOUSBORDER(USB_PID), // idProduct
|
||
#ifdef STM32F4
|
||
0x00, 0x23, // bcdDevice
|
||
#else
|
||
0x00, 0x22, // bcdDevice
|
||
#endif
|
||
0x01, 0x02, // Manufacturer, Product
|
||
0x03, 0x01 // Serial Number, Num Configurations
|
||
};
|
||
|
||
uint8_t device_qualifier[] = {
|
||
0x0a, USB_DESC_TYPE_DEVICE_QUALIFIER, //Length, Type
|
||
0x10, 0x02, // bcdUSB max version of USB supported (2.1)
|
||
0xFF, 0xFF, 0xFF, 0x40, // bDeviceClass, bDeviceSubClass, bDeviceProtocol, bMaxPacketSize0
|
||
0x01, 0x00 // bNumConfigurations, bReserved
|
||
};
|
||
|
||
#define ENDPOINT_RCV 0x80
|
||
#define ENDPOINT_SND 0x00
|
||
|
||
uint8_t configuration_desc[] = {
|
||
DSCR_CONFIG_LEN, USB_DESC_TYPE_CONFIGURATION, // Length, Type,
|
||
TOUSBORDER(0x0045U), // Total Len (uint16)
|
||
0x01, 0x01, STRING_OFFSET_ICONFIGURATION, // Num Interface, Config Value, Configuration
|
||
0xc0, 0x32, // Attributes, Max Power
|
||
// interface 0 ALT 0
|
||
DSCR_INTERFACE_LEN, USB_DESC_TYPE_INTERFACE, // Length, Type
|
||
0x00, 0x00, 0x03, // Index, Alt Index idx, Endpoint count
|
||
0XFF, 0xFF, 0xFF, // Class, Subclass, Protocol
|
||
0x00, // Interface
|
||
// endpoint 1, read CAN
|
||
DSCR_ENDPOINT_LEN, USB_DESC_TYPE_ENDPOINT, // Length, Type
|
||
ENDPOINT_RCV | 1, ENDPOINT_TYPE_BULK, // Endpoint Num/Direction, Type
|
||
TOUSBORDER(0x0040U), // Max Packet (0x0040)
|
||
0x00, // Polling Interval (NA)
|
||
// endpoint 2, send serial
|
||
DSCR_ENDPOINT_LEN, USB_DESC_TYPE_ENDPOINT, // Length, Type
|
||
ENDPOINT_SND | 2, ENDPOINT_TYPE_BULK, // Endpoint Num/Direction, Type
|
||
TOUSBORDER(0x0040U), // Max Packet (0x0040)
|
||
0x00, // Polling Interval
|
||
// endpoint 3, send CAN
|
||
DSCR_ENDPOINT_LEN, USB_DESC_TYPE_ENDPOINT, // Length, Type
|
||
ENDPOINT_SND | 3, ENDPOINT_TYPE_BULK, // Endpoint Num/Direction, Type
|
||
TOUSBORDER(0x0040U), // Max Packet (0x0040)
|
||
0x00, // Polling Interval
|
||
// interface 0 ALT 1
|
||
DSCR_INTERFACE_LEN, USB_DESC_TYPE_INTERFACE, // Length, Type
|
||
0x00, 0x01, 0x03, // Index, Alt Index idx, Endpoint count
|
||
0XFF, 0xFF, 0xFF, // Class, Subclass, Protocol
|
||
0x00, // Interface
|
||
// endpoint 1, read CAN
|
||
DSCR_ENDPOINT_LEN, USB_DESC_TYPE_ENDPOINT, // Length, Type
|
||
ENDPOINT_RCV | 1, ENDPOINT_TYPE_INT, // Endpoint Num/Direction, Type
|
||
TOUSBORDER(0x0040U), // Max Packet (0x0040)
|
||
0x05, // Polling Interval (5 frames)
|
||
// endpoint 2, send serial
|
||
DSCR_ENDPOINT_LEN, USB_DESC_TYPE_ENDPOINT, // Length, Type
|
||
ENDPOINT_SND | 2, ENDPOINT_TYPE_BULK, // Endpoint Num/Direction, Type
|
||
TOUSBORDER(0x0040U), // Max Packet (0x0040)
|
||
0x00, // Polling Interval
|
||
// endpoint 3, send CAN
|
||
DSCR_ENDPOINT_LEN, USB_DESC_TYPE_ENDPOINT, // Length, Type
|
||
ENDPOINT_SND | 3, ENDPOINT_TYPE_BULK, // Endpoint Num/Direction, Type
|
||
TOUSBORDER(0x0040U), // Max Packet (0x0040)
|
||
0x00, // Polling Interval
|
||
};
|
||
|
||
// STRING_DESCRIPTOR_HEADER is for uint16 string descriptors
|
||
// it takes in a string length, which is bytes/2 because unicode
|
||
uint16_t string_language_desc[] = {
|
||
STRING_DESCRIPTOR_HEADER(1),
|
||
0x0409 // american english
|
||
};
|
||
|
||
// these strings are all uint16's so that we don't need to spam ,0 after every character
|
||
uint16_t string_manufacturer_desc[] = {
|
||
STRING_DESCRIPTOR_HEADER(8),
|
||
'c', 'o', 'm', 'm', 'a', '.', 'a', 'i'
|
||
};
|
||
|
||
uint16_t string_product_desc[] = {
|
||
STRING_DESCRIPTOR_HEADER(5),
|
||
'p', 'a', 'n', 'd', 'a'
|
||
};
|
||
|
||
// default serial number when we're not a panda
|
||
uint16_t string_serial_desc[] = {
|
||
STRING_DESCRIPTOR_HEADER(4),
|
||
'n', 'o', 'n', 'e'
|
||
};
|
||
|
||
// a string containing the default configuration index
|
||
uint16_t string_configuration_desc[] = {
|
||
STRING_DESCRIPTOR_HEADER(2),
|
||
'0', '1' // "01"
|
||
};
|
||
|
||
// WCID (auto install WinUSB driver)
|
||
// https://github.com/pbatard/libwdi/wiki/WCID-Devices
|
||
// https://docs.microsoft.com/en-us/windows-hardware/drivers/usbcon/winusb-installation#automatic-installation-of--winusb-without-an-inf-file
|
||
// WinUSB 1.0 descriptors, this is mostly used by Windows XP
|
||
uint8_t string_238_desc[] = {
|
||
0x12, USB_DESC_TYPE_STRING, // bLength, bDescriptorType
|
||
'M',0, 'S',0, 'F',0, 'T',0, '1',0, '0',0, '0',0, // qwSignature (MSFT100)
|
||
MS_VENDOR_CODE, 0x00 // bMS_VendorCode, bPad
|
||
};
|
||
uint8_t winusb_ext_compatid_os_desc[] = {
|
||
0x28, 0x00, 0x00, 0x00, // dwLength
|
||
0x00, 0x01, // bcdVersion
|
||
0x04, 0x00, // wIndex
|
||
0x01, // bCount
|
||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // Reserved
|
||
0x00, // bFirstInterfaceNumber
|
||
0x00, // Reserved
|
||
'W', 'I', 'N', 'U', 'S', 'B', 0x00, 0x00, // compatible ID (WINUSB)
|
||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // subcompatible ID (none)
|
||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00 // Reserved
|
||
};
|
||
uint8_t winusb_ext_prop_os_desc[] = {
|
||
0x8e, 0x00, 0x00, 0x00, // dwLength
|
||
0x00, 0x01, // bcdVersion
|
||
0x05, 0x00, // wIndex
|
||
0x01, 0x00, // wCount
|
||
// first property
|
||
0x84, 0x00, 0x00, 0x00, // dwSize
|
||
0x01, 0x00, 0x00, 0x00, // dwPropertyDataType
|
||
0x28, 0x00, // wPropertyNameLength
|
||
'D',0, 'e',0, 'v',0, 'i',0, 'c',0, 'e',0, 'I',0, 'n',0, 't',0, 'e',0, 'r',0, 'f',0, 'a',0, 'c',0, 'e',0, 'G',0, 'U',0, 'I',0, 'D',0, 0, 0, // bPropertyName (DeviceInterfaceGUID)
|
||
0x4e, 0x00, 0x00, 0x00, // dwPropertyDataLength
|
||
'{',0, 'c',0, 'c',0, 'e',0, '5',0, '2',0, '9',0, '1',0, 'c',0, '-',0, 'a',0, '6',0, '9',0, 'f',0, '-',0, '4',0 ,'9',0 ,'9',0 ,'5',0 ,'-',0, 'a',0, '4',0, 'c',0, '2',0, '-',0, '2',0, 'a',0, 'e',0, '5',0, '7',0, 'a',0, '5',0, '1',0, 'a',0, 'd',0, 'e',0, '9',0, '}',0, 0, 0, // bPropertyData ({CCE5291C-A69F-4995-A4C2-2AE57A51ADE9})
|
||
};
|
||
|
||
/*
|
||
Binary Object Store descriptor used to expose WebUSB (and more WinUSB) metadata
|
||
comments are from the wicg spec
|
||
References used:
|
||
https://wicg.github.io/webusb/#webusb-platform-capability-descriptor
|
||
https://github.com/sowbug/weblight/blob/192ad7a0e903542e2aa28c607d98254a12a6399d/firmware/webusb.c
|
||
https://os.mbed.com/users/larsgk/code/USBDevice_WebUSB/file/1d8a6665d607/WebUSBDevice/
|
||
|
||
*/
|
||
uint8_t binary_object_store_desc[] = {
|
||
// BOS header
|
||
BINARY_OBJECT_STORE_DESCRIPTOR_LENGTH, // bLength, this is only the length of the header
|
||
BINARY_OBJECT_STORE_DESCRIPTOR, // bDescriptorType
|
||
0x40, 0x00, // wTotalLength (LSB, MSB)
|
||
0x03, // bNumDeviceCaps (USB 2.0 + WebUSB + WinUSB)
|
||
|
||
// -------------------------------------------------
|
||
// USB 2.0 extension descriptor
|
||
0x07, // bLength, Descriptor size
|
||
0x10, // bDescriptorType, Device Capability Descriptor Type
|
||
0x02, // bDevCapabilityType, USB 2.0 extension capability type
|
||
0x00, 0x00, 0x00, 0x00, // bmAttributes, LIBUSB_BM_LPM_SUPPORT = 2 and its the only option
|
||
|
||
// -------------------------------------------------
|
||
// WebUSB descriptor
|
||
// header
|
||
0x18, // bLength, Size of this descriptor. Must be set to 24.
|
||
0x10, // bDescriptorType, DEVICE CAPABILITY descriptor
|
||
0x05, // bDevCapabilityType, PLATFORM capability
|
||
0x00, // bReserved, This field is reserved and shall be set to zero.
|
||
|
||
// PlatformCapabilityUUID, Must be set to {3408b638-09a9-47a0-8bfd-a0768815b665}.
|
||
0x38, 0xB6, 0x08, 0x34,
|
||
0xA9, 0x09, 0xA0, 0x47,
|
||
0x8B, 0xFD, 0xA0, 0x76,
|
||
0x88, 0x15, 0xB6, 0x65,
|
||
// </PlatformCapabilityUUID>
|
||
|
||
0x00, 0x01, // bcdVersion, Protocol version supported. Must be set to 0x0100.
|
||
WEBUSB_VENDOR_CODE, // bVendorCode, bRequest value used for issuing WebUSB requests.
|
||
// there used to be a concept of "allowed origins", but it was removed from the spec
|
||
// it was intended to be a security feature, but then the entire security model relies on domain ownership
|
||
// https://github.com/WICG/webusb/issues/49
|
||
// other implementations use various other indexed to leverate this no-longer-valid feature. we wont.
|
||
// the spec says we *must* reply to index 0x03 with the url, so we'll hint that that's the right index
|
||
0x03, // iLandingPage, URL descriptor index of the device’s landing page.
|
||
|
||
// -------------------------------------------------
|
||
// WinUSB descriptor
|
||
// header
|
||
0x1C, // Descriptor size (28 bytes)
|
||
0x10, // Descriptor type (Device Capability)
|
||
0x05, // Capability type (Platform)
|
||
0x00, // Reserved
|
||
|
||
// MS OS 2.0 Platform Capability ID (D8DD60DF-4589-4CC7-9CD2-659D9E648A9F)
|
||
// Indicates the device supports the Microsoft OS 2.0 descriptor
|
||
0xDF, 0x60, 0xDD, 0xD8,
|
||
0x89, 0x45, 0xC7, 0x4C,
|
||
0x9C, 0xD2, 0x65, 0x9D,
|
||
0x9E, 0x64, 0x8A, 0x9F,
|
||
|
||
0x00, 0x00, 0x03, 0x06, // Windows version, currently set to 8.1 (0x06030000)
|
||
|
||
WINUSB_PLATFORM_DESCRIPTOR_LENGTH, 0x00, // MS OS 2.0 descriptor size (word)
|
||
MS_VENDOR_CODE, 0x00 // vendor code, no alternate enumeration
|
||
};
|
||
|
||
uint8_t webusb_url_descriptor[] = {
|
||
0x14, /* bLength */
|
||
WEBUSB_DESC_TYPE_URL, // bDescriptorType
|
||
WEBUSB_URL_SCHEME_HTTPS, // bScheme
|
||
'u', 's', 'b', 'p', 'a', 'n', 'd', 'a', '.', 'c', 'o', 'm', 'm', 'a', '.', 'a', 'i'
|
||
};
|
||
|
||
// WinUSB 2.0 descriptor. This is what modern systems use
|
||
// https://github.com/sowbug/weblight/blob/192ad7a0e903542e2aa28c607d98254a12a6399d/firmware/webusb.c
|
||
// http://janaxelson.com/files/ms_os_20_descriptors.c
|
||
// https://books.google.com/books?id=pkefBgAAQBAJ&pg=PA353&lpg=PA353
|
||
uint8_t winusb_20_desc[WINUSB_PLATFORM_DESCRIPTOR_LENGTH] = {
|
||
// Microsoft OS 2.0 descriptor set header (table 10)
|
||
0x0A, 0x00, // Descriptor size (10 bytes)
|
||
0x00, 0x00, // MS OS 2.0 descriptor set header
|
||
|
||
0x00, 0x00, 0x03, 0x06, // Windows version (8.1) (0x06030000)
|
||
WINUSB_PLATFORM_DESCRIPTOR_LENGTH, 0x00, // Total size of MS OS 2.0 descriptor set
|
||
|
||
// Microsoft OS 2.0 compatible ID descriptor
|
||
0x14, 0x00, // Descriptor size (20 bytes)
|
||
0x03, 0x00, // MS OS 2.0 compatible ID descriptor
|
||
'W', 'I', 'N', 'U', 'S', 'B', 0x00, 0x00, // compatible ID (WINUSB)
|
||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // Sub-compatible ID
|
||
|
||
// Registry property descriptor
|
||
0x80, 0x00, // Descriptor size (130 bytes)
|
||
0x04, 0x00, // Registry Property descriptor
|
||
0x01, 0x00, // Strings are null-terminated Unicode
|
||
0x28, 0x00, // Size of Property Name (40 bytes) "DeviceInterfaceGUID"
|
||
|
||
// bPropertyName (DeviceInterfaceGUID)
|
||
'D', 0x00, 'e', 0x00, 'v', 0x00, 'i', 0x00, 'c', 0x00, 'e', 0x00, 'I', 0x00, 'n', 0x00,
|
||
't', 0x00, 'e', 0x00, 'r', 0x00, 'f', 0x00, 'a', 0x00, 'c', 0x00, 'e', 0x00, 'G', 0x00,
|
||
'U', 0x00, 'I', 0x00, 'D', 0x00, 0x00, 0x00,
|
||
|
||
0x4E, 0x00, // Size of Property Data (78 bytes)
|
||
|
||
// Vendor-defined property data: {CCE5291C-A69F-4995-A4C2-2AE57A51ADE9}
|
||
'{', 0x00, 'c', 0x00, 'c', 0x00, 'e', 0x00, '5', 0x00, '2', 0x00, '9', 0x00, '1', 0x00, // 16
|
||
'c', 0x00, '-', 0x00, 'a', 0x00, '6', 0x00, '9', 0x00, 'f', 0x00, '-', 0x00, '4', 0x00, // 32
|
||
'9', 0x00, '9', 0x00, '5', 0x00, '-', 0x00, 'a', 0x00, '4', 0x00, 'c', 0x00, '2', 0x00, // 48
|
||
'-', 0x00, '2', 0x00, 'a', 0x00, 'e', 0x00, '5', 0x00, '7', 0x00, 'a', 0x00, '5', 0x00, // 64
|
||
'1', 0x00, 'a', 0x00, 'd', 0x00, 'e', 0x00, '9', 0x00, '}', 0x00, 0x00, 0x00 // 78 bytes
|
||
};
|
||
|
||
// current packet
|
||
USB_Setup_TypeDef setup;
|
||
uint8_t usbdata[0x100];
|
||
uint8_t* ep0_txdata = NULL;
|
||
uint16_t ep0_txlen = 0;
|
||
|
||
// Store the current interface alt setting.
|
||
int current_int0_alt_setting = 0;
|
||
|
||
// packet read and write
|
||
|
||
void *USB_ReadPacket(void *dest, uint16_t len) {
|
||
uint32_t *dest_copy = (uint32_t *)dest;
|
||
uint32_t count32b = (len + 3U) / 4U;
|
||
|
||
for (uint32_t i = 0; i < count32b; i++) {
|
||
*dest_copy = USBx_DFIFO(0);
|
||
dest_copy++;
|
||
}
|
||
return ((void *)dest_copy);
|
||
}
|
||
|
||
void USB_WritePacket(const void *src, uint16_t len, uint32_t ep) {
|
||
#ifdef DEBUG_USB
|
||
puts("writing ");
|
||
hexdump(src, len);
|
||
#endif
|
||
|
||
uint8_t numpacket = (len + (MAX_RESP_LEN - 1U)) / MAX_RESP_LEN;
|
||
uint32_t count32b = 0;
|
||
count32b = (len + 3U) / 4U;
|
||
|
||
// TODO: revisit this
|
||
USBx_INEP(ep)->DIEPTSIZ = ((numpacket << 19) & USB_OTG_DIEPTSIZ_PKTCNT) |
|
||
(len & USB_OTG_DIEPTSIZ_XFRSIZ);
|
||
USBx_INEP(ep)->DIEPCTL |= (USB_OTG_DIEPCTL_CNAK | USB_OTG_DIEPCTL_EPENA);
|
||
|
||
// load the FIFO
|
||
const uint32_t *src_copy = (const uint32_t *)src;
|
||
for (uint32_t i = 0; i < count32b; i++) {
|
||
USBx_DFIFO(ep) = *src_copy;
|
||
src_copy++;
|
||
}
|
||
}
|
||
|
||
// IN EP 0 TX FIFO has a max size of 127 bytes (much smaller than the rest)
|
||
// so use TX FIFO empty interrupt to send larger amounts of data
|
||
void USB_WritePacket_EP0(uint8_t *src, uint16_t len) {
|
||
#ifdef DEBUG_USB
|
||
puts("writing ");
|
||
hexdump(src, len);
|
||
#endif
|
||
|
||
uint16_t wplen = MIN(len, 0x40);
|
||
USB_WritePacket(src, wplen, 0);
|
||
|
||
if (wplen < len) {
|
||
ep0_txdata = &src[wplen];
|
||
ep0_txlen = len - wplen;
|
||
USBx_DEVICE->DIEPEMPMSK |= 1;
|
||
} else {
|
||
USBx_OUTEP(0)->DOEPCTL |= USB_OTG_DOEPCTL_CNAK;
|
||
}
|
||
}
|
||
|
||
void usb_reset(void) {
|
||
// unmask endpoint interrupts, so many sets
|
||
USBx_DEVICE->DAINT = 0xFFFFFFFF;
|
||
USBx_DEVICE->DAINTMSK = 0xFFFFFFFF;
|
||
//USBx_DEVICE->DOEPMSK = (USB_OTG_DOEPMSK_STUPM | USB_OTG_DOEPMSK_XFRCM | USB_OTG_DOEPMSK_EPDM);
|
||
//USBx_DEVICE->DIEPMSK = (USB_OTG_DIEPMSK_TOM | USB_OTG_DIEPMSK_XFRCM | USB_OTG_DIEPMSK_EPDM | USB_OTG_DIEPMSK_ITTXFEMSK);
|
||
//USBx_DEVICE->DIEPMSK = (USB_OTG_DIEPMSK_TOM | USB_OTG_DIEPMSK_XFRCM | USB_OTG_DIEPMSK_EPDM);
|
||
|
||
// all interrupts for debugging
|
||
USBx_DEVICE->DIEPMSK = 0xFFFFFFFF;
|
||
USBx_DEVICE->DOEPMSK = 0xFFFFFFFF;
|
||
|
||
// clear interrupts
|
||
USBx_INEP(0)->DIEPINT = 0xFF;
|
||
USBx_OUTEP(0)->DOEPINT = 0xFF;
|
||
|
||
// unset the address
|
||
USBx_DEVICE->DCFG &= ~USB_OTG_DCFG_DAD;
|
||
|
||
// set up USB FIFOs
|
||
// RX start address is fixed to 0
|
||
USBx->GRXFSIZ = 0x40;
|
||
|
||
// 0x100 to offset past GRXFSIZ
|
||
USBx->DIEPTXF0_HNPTXFSIZ = (0x40U << 16) | 0x40U;
|
||
|
||
// EP1, massive
|
||
USBx->DIEPTXF[0] = (0x40U << 16) | 0x80U;
|
||
|
||
// flush TX fifo
|
||
USBx->GRSTCTL = USB_OTG_GRSTCTL_TXFFLSH | USB_OTG_GRSTCTL_TXFNUM_4;
|
||
while ((USBx->GRSTCTL & USB_OTG_GRSTCTL_TXFFLSH) == USB_OTG_GRSTCTL_TXFFLSH);
|
||
// flush RX FIFO
|
||
USBx->GRSTCTL = USB_OTG_GRSTCTL_RXFFLSH;
|
||
while ((USBx->GRSTCTL & USB_OTG_GRSTCTL_RXFFLSH) == USB_OTG_GRSTCTL_RXFFLSH);
|
||
|
||
// no global NAK
|
||
USBx_DEVICE->DCTL |= USB_OTG_DCTL_CGINAK;
|
||
|
||
// ready to receive setup packets
|
||
USBx_OUTEP(0)->DOEPTSIZ = USB_OTG_DOEPTSIZ_STUPCNT | (USB_OTG_DOEPTSIZ_PKTCNT & (1U << 19)) | (3U << 3);
|
||
}
|
||
|
||
char to_hex_char(int a) {
|
||
char ret;
|
||
if (a < 10) {
|
||
ret = '0' + a;
|
||
} else {
|
||
ret = 'a' + (a - 10);
|
||
}
|
||
return ret;
|
||
}
|
||
|
||
void usb_setup(void) {
|
||
int resp_len;
|
||
// setup packet is ready
|
||
switch (setup.b.bRequest) {
|
||
case USB_REQ_SET_CONFIGURATION:
|
||
// enable other endpoints, has to be here?
|
||
USBx_INEP(1)->DIEPCTL = (0x40U & USB_OTG_DIEPCTL_MPSIZ) | (2U << 18) | (1U << 22) |
|
||
USB_OTG_DIEPCTL_SD0PID_SEVNFRM | USB_OTG_DIEPCTL_USBAEP;
|
||
USBx_INEP(1)->DIEPINT = 0xFF;
|
||
|
||
USBx_OUTEP(2)->DOEPTSIZ = (1U << 19) | 0x40U;
|
||
USBx_OUTEP(2)->DOEPCTL = (0x40U & USB_OTG_DOEPCTL_MPSIZ) | (2U << 18) |
|
||
USB_OTG_DOEPCTL_SD0PID_SEVNFRM | USB_OTG_DOEPCTL_USBAEP;
|
||
USBx_OUTEP(2)->DOEPINT = 0xFF;
|
||
|
||
USBx_OUTEP(3)->DOEPTSIZ = (1U << 19) | 0x40U;
|
||
USBx_OUTEP(3)->DOEPCTL = (0x40U & USB_OTG_DOEPCTL_MPSIZ) | (2U << 18) |
|
||
USB_OTG_DOEPCTL_SD0PID_SEVNFRM | USB_OTG_DOEPCTL_USBAEP;
|
||
USBx_OUTEP(3)->DOEPINT = 0xFF;
|
||
|
||
// mark ready to receive
|
||
USBx_OUTEP(2)->DOEPCTL |= USB_OTG_DOEPCTL_EPENA | USB_OTG_DOEPCTL_CNAK;
|
||
USBx_OUTEP(3)->DOEPCTL |= USB_OTG_DOEPCTL_EPENA | USB_OTG_DOEPCTL_CNAK;
|
||
|
||
USB_WritePacket(0, 0, 0);
|
||
USBx_OUTEP(0)->DOEPCTL |= USB_OTG_DOEPCTL_CNAK;
|
||
break;
|
||
case USB_REQ_SET_ADDRESS:
|
||
// set now?
|
||
USBx_DEVICE->DCFG |= ((setup.b.wValue.w & 0x7fU) << 4);
|
||
|
||
#ifdef DEBUG_USB
|
||
puts(" set address\n");
|
||
#endif
|
||
|
||
// TODO: this isn't enumeration complete
|
||
// moved here to work better on OS X
|
||
usb_cb_enumeration_complete();
|
||
|
||
USB_WritePacket(0, 0, 0);
|
||
USBx_OUTEP(0)->DOEPCTL |= USB_OTG_DOEPCTL_CNAK;
|
||
|
||
break;
|
||
case USB_REQ_GET_DESCRIPTOR:
|
||
switch (setup.b.wValue.bw.lsb) {
|
||
case USB_DESC_TYPE_DEVICE:
|
||
//puts(" writing device descriptor\n");
|
||
|
||
// setup transfer
|
||
USB_WritePacket(device_desc, MIN(sizeof(device_desc), setup.b.wLength.w), 0);
|
||
USBx_OUTEP(0)->DOEPCTL |= USB_OTG_DOEPCTL_CNAK;
|
||
|
||
//puts("D");
|
||
break;
|
||
case USB_DESC_TYPE_CONFIGURATION:
|
||
USB_WritePacket(configuration_desc, MIN(sizeof(configuration_desc), setup.b.wLength.w), 0);
|
||
USBx_OUTEP(0)->DOEPCTL |= USB_OTG_DOEPCTL_CNAK;
|
||
break;
|
||
case USB_DESC_TYPE_DEVICE_QUALIFIER:
|
||
USB_WritePacket(device_qualifier, MIN(sizeof(device_qualifier), setup.b.wLength.w), 0);
|
||
USBx_OUTEP(0)->DOEPCTL |= USB_OTG_DOEPCTL_CNAK;
|
||
break;
|
||
case USB_DESC_TYPE_STRING:
|
||
switch (setup.b.wValue.bw.msb) {
|
||
case STRING_OFFSET_LANGID:
|
||
USB_WritePacket((uint8_t*)string_language_desc, MIN(sizeof(string_language_desc), setup.b.wLength.w), 0);
|
||
break;
|
||
case STRING_OFFSET_IMANUFACTURER:
|
||
USB_WritePacket((uint8_t*)string_manufacturer_desc, MIN(sizeof(string_manufacturer_desc), setup.b.wLength.w), 0);
|
||
break;
|
||
case STRING_OFFSET_IPRODUCT:
|
||
USB_WritePacket((uint8_t*)string_product_desc, MIN(sizeof(string_product_desc), setup.b.wLength.w), 0);
|
||
break;
|
||
case STRING_OFFSET_ISERIAL:
|
||
#ifdef UID_BASE
|
||
resp[0] = 0x02 + (12 * 4);
|
||
resp[1] = 0x03;
|
||
|
||
// 96 bits = 12 bytes
|
||
for (int i = 0; i < 12; i++){
|
||
uint8_t cc = ((uint8_t *)UID_BASE)[i];
|
||
resp[2 + (i * 4) + 0] = to_hex_char((cc >> 4) & 0xFU);
|
||
resp[2 + (i * 4) + 1] = '\0';
|
||
resp[2 + (i * 4) + 2] = to_hex_char((cc >> 0) & 0xFU);
|
||
resp[2 + (i * 4) + 3] = '\0';
|
||
}
|
||
|
||
USB_WritePacket(resp, MIN(resp[0], setup.b.wLength.w), 0);
|
||
#else
|
||
USB_WritePacket((const uint8_t *)string_serial_desc, MIN(sizeof(string_serial_desc), setup.b.wLength.w), 0);
|
||
#endif
|
||
break;
|
||
case STRING_OFFSET_ICONFIGURATION:
|
||
USB_WritePacket((uint8_t*)string_configuration_desc, MIN(sizeof(string_configuration_desc), setup.b.wLength.w), 0);
|
||
break;
|
||
case 238:
|
||
USB_WritePacket((uint8_t*)string_238_desc, MIN(sizeof(string_238_desc), setup.b.wLength.w), 0);
|
||
break;
|
||
default:
|
||
// nothing
|
||
USB_WritePacket(0, 0, 0);
|
||
break;
|
||
}
|
||
USBx_OUTEP(0)->DOEPCTL |= USB_OTG_DOEPCTL_CNAK;
|
||
break;
|
||
case USB_DESC_TYPE_BINARY_OBJECT_STORE:
|
||
USB_WritePacket(binary_object_store_desc, MIN(sizeof(binary_object_store_desc), setup.b.wLength.w), 0);
|
||
USBx_OUTEP(0)->DOEPCTL |= USB_OTG_DOEPCTL_CNAK;
|
||
break;
|
||
default:
|
||
// nothing here?
|
||
USB_WritePacket(0, 0, 0);
|
||
USBx_OUTEP(0)->DOEPCTL |= USB_OTG_DOEPCTL_CNAK;
|
||
break;
|
||
}
|
||
break;
|
||
case USB_REQ_GET_STATUS:
|
||
// empty resp?
|
||
resp[0] = 0;
|
||
resp[1] = 0;
|
||
USB_WritePacket((void*)&resp, 2, 0);
|
||
USBx_OUTEP(0)->DOEPCTL |= USB_OTG_DOEPCTL_CNAK;
|
||
break;
|
||
case USB_REQ_SET_INTERFACE:
|
||
// Store the alt setting number for IN EP behavior.
|
||
current_int0_alt_setting = setup.b.wValue.w;
|
||
USB_WritePacket(0, 0, 0);
|
||
USBx_OUTEP(0)->DOEPCTL |= USB_OTG_DOEPCTL_CNAK;
|
||
break;
|
||
case WEBUSB_VENDOR_CODE:
|
||
switch (setup.b.wIndex.w) {
|
||
case WEBUSB_REQ_GET_URL:
|
||
USB_WritePacket(webusb_url_descriptor, MIN(sizeof(webusb_url_descriptor), setup.b.wLength.w), 0);
|
||
USBx_OUTEP(0)->DOEPCTL |= USB_OTG_DOEPCTL_CNAK;
|
||
break;
|
||
default:
|
||
// probably asking for allowed origins, which was removed from the spec
|
||
USB_WritePacket(0, 0, 0);
|
||
USBx_OUTEP(0)->DOEPCTL |= USB_OTG_DOEPCTL_CNAK;
|
||
break;
|
||
}
|
||
break;
|
||
case MS_VENDOR_CODE:
|
||
switch (setup.b.wIndex.w) {
|
||
// winusb 2.0 descriptor from BOS
|
||
case WINUSB_REQ_GET_DESCRIPTOR:
|
||
USB_WritePacket_EP0((uint8_t*)winusb_20_desc, MIN(sizeof(winusb_20_desc), setup.b.wLength.w));
|
||
break;
|
||
// Extended Compat ID OS Descriptor
|
||
case WINUSB_REQ_GET_COMPATID_DESCRIPTOR:
|
||
USB_WritePacket_EP0((uint8_t*)winusb_ext_compatid_os_desc, MIN(sizeof(winusb_ext_compatid_os_desc), setup.b.wLength.w));
|
||
break;
|
||
// Extended Properties OS Descriptor
|
||
case WINUSB_REQ_GET_EXT_PROPS_OS:
|
||
USB_WritePacket_EP0((uint8_t*)winusb_ext_prop_os_desc, MIN(sizeof(winusb_ext_prop_os_desc), setup.b.wLength.w));
|
||
break;
|
||
default:
|
||
USB_WritePacket_EP0(0, 0);
|
||
}
|
||
break;
|
||
default:
|
||
resp_len = usb_cb_control_msg(&setup, resp, 1);
|
||
USB_WritePacket(resp, MIN(resp_len, setup.b.wLength.w), 0);
|
||
USBx_OUTEP(0)->DOEPCTL |= USB_OTG_DOEPCTL_CNAK;
|
||
}
|
||
}
|
||
|
||
void usb_init(void) {
|
||
// full speed PHY, do reset and remove power down
|
||
/*puth(USBx->GRSTCTL);
|
||
puts(" resetting PHY\n");*/
|
||
while ((USBx->GRSTCTL & USB_OTG_GRSTCTL_AHBIDL) == 0);
|
||
//puts("AHB idle\n");
|
||
|
||
// reset PHY here
|
||
USBx->GRSTCTL |= USB_OTG_GRSTCTL_CSRST;
|
||
while ((USBx->GRSTCTL & USB_OTG_GRSTCTL_CSRST) == USB_OTG_GRSTCTL_CSRST);
|
||
//puts("reset done\n");
|
||
|
||
// internal PHY, force device mode
|
||
USBx->GUSBCFG = USB_OTG_GUSBCFG_PHYSEL | USB_OTG_GUSBCFG_FDMOD;
|
||
|
||
// slowest timings
|
||
USBx->GUSBCFG |= ((USBD_FS_TRDT_VALUE << 10) & USB_OTG_GUSBCFG_TRDT);
|
||
|
||
// power up the PHY
|
||
#ifdef STM32F4
|
||
USBx->GCCFG = USB_OTG_GCCFG_PWRDWN;
|
||
|
||
//USBx->GCCFG |= USB_OTG_GCCFG_VBDEN | USB_OTG_GCCFG_SDEN |USB_OTG_GCCFG_PDEN | USB_OTG_GCCFG_DCDEN;
|
||
|
||
/* B-peripheral session valid override enable*/
|
||
USBx->GOTGCTL |= USB_OTG_GOTGCTL_BVALOVAL;
|
||
USBx->GOTGCTL |= USB_OTG_GOTGCTL_BVALOEN;
|
||
#else
|
||
USBx->GCCFG = USB_OTG_GCCFG_PWRDWN | USB_OTG_GCCFG_NOVBUSSENS;
|
||
#endif
|
||
|
||
// be a device, slowest timings
|
||
//USBx->GUSBCFG = USB_OTG_GUSBCFG_FDMOD | USB_OTG_GUSBCFG_PHYSEL | USB_OTG_GUSBCFG_TRDT | USB_OTG_GUSBCFG_TOCAL;
|
||
//USBx->GUSBCFG |= (uint32_t)((USBD_FS_TRDT_VALUE << 10) & USB_OTG_GUSBCFG_TRDT);
|
||
//USBx->GUSBCFG = USB_OTG_GUSBCFG_PHYSEL | USB_OTG_GUSBCFG_TRDT | USB_OTG_GUSBCFG_TOCAL;
|
||
|
||
// **** for debugging, doesn't seem to work ****
|
||
//USBx->GUSBCFG |= USB_OTG_GUSBCFG_CTXPKT;
|
||
|
||
// reset PHY clock
|
||
USBx_PCGCCTL = 0;
|
||
|
||
// enable the fancy OTG things
|
||
// DCFG_FRAME_INTERVAL_80 is 0
|
||
//USBx->GUSBCFG |= USB_OTG_GUSBCFG_HNPCAP | USB_OTG_GUSBCFG_SRPCAP;
|
||
USBx_DEVICE->DCFG |= USB_OTG_SPEED_FULL | USB_OTG_DCFG_NZLSOHSK;
|
||
|
||
//USBx_DEVICE->DCFG = USB_OTG_DCFG_NZLSOHSK | USB_OTG_DCFG_DSPD;
|
||
//USBx_DEVICE->DCFG = USB_OTG_DCFG_DSPD;
|
||
|
||
// clear pending interrupts
|
||
USBx->GINTSTS = 0xBFFFFFFFU;
|
||
|
||
// setup USB interrupts
|
||
// all interrupts except TXFIFO EMPTY
|
||
//USBx->GINTMSK = 0xFFFFFFFF & ~(USB_OTG_GINTMSK_NPTXFEM | USB_OTG_GINTMSK_PTXFEM | USB_OTG_GINTSTS_SOF | USB_OTG_GINTSTS_EOPF);
|
||
//USBx->GINTMSK = 0xFFFFFFFF & ~(USB_OTG_GINTMSK_NPTXFEM | USB_OTG_GINTMSK_PTXFEM);
|
||
USBx->GINTMSK = USB_OTG_GINTMSK_USBRST | USB_OTG_GINTMSK_ENUMDNEM | USB_OTG_GINTMSK_OTGINT |
|
||
USB_OTG_GINTMSK_RXFLVLM | USB_OTG_GINTMSK_GONAKEFFM | USB_OTG_GINTMSK_GINAKEFFM |
|
||
USB_OTG_GINTMSK_OEPINT | USB_OTG_GINTMSK_IEPINT | USB_OTG_GINTMSK_USBSUSPM |
|
||
USB_OTG_GINTMSK_CIDSCHGM | USB_OTG_GINTMSK_SRQIM | USB_OTG_GINTMSK_MMISM;
|
||
|
||
USBx->GAHBCFG = USB_OTG_GAHBCFG_GINT;
|
||
|
||
// DCTL startup value is 2 on new chip, 0 on old chip
|
||
USBx_DEVICE->DCTL = 0;
|
||
|
||
// enable the IRQ
|
||
NVIC_EnableIRQ(OTG_FS_IRQn);
|
||
}
|
||
|
||
// ***************************** USB port *****************************
|
||
|
||
void usb_irqhandler(void) {
|
||
//USBx->GINTMSK = 0;
|
||
|
||
unsigned int gintsts = USBx->GINTSTS;
|
||
unsigned int gotgint = USBx->GOTGINT;
|
||
unsigned int daint = USBx_DEVICE->DAINT;
|
||
|
||
// gintsts SUSPEND? 04008428
|
||
#ifdef DEBUG_USB
|
||
puth(gintsts);
|
||
puts(" ");
|
||
/*puth(USBx->GCCFG);
|
||
puts(" ");*/
|
||
puth(gotgint);
|
||
puts(" ep ");
|
||
puth(daint);
|
||
puts(" USB interrupt!\n");
|
||
#endif
|
||
|
||
if ((gintsts & USB_OTG_GINTSTS_CIDSCHG) != 0) {
|
||
puts("connector ID status change\n");
|
||
}
|
||
|
||
if ((gintsts & USB_OTG_GINTSTS_ESUSP) != 0) {
|
||
puts("ESUSP detected\n");
|
||
}
|
||
|
||
if ((gintsts & USB_OTG_GINTSTS_USBRST) != 0) {
|
||
puts("USB reset\n");
|
||
usb_reset();
|
||
}
|
||
|
||
if ((gintsts & USB_OTG_GINTSTS_ENUMDNE) != 0) {
|
||
puts("enumeration done");
|
||
// Full speed, ENUMSPD
|
||
//puth(USBx_DEVICE->DSTS);
|
||
puts("\n");
|
||
}
|
||
|
||
if ((gintsts & USB_OTG_GINTSTS_OTGINT) != 0) {
|
||
puts("OTG int:");
|
||
puth(USBx->GOTGINT);
|
||
puts("\n");
|
||
|
||
// getting ADTOCHG
|
||
//USBx->GOTGINT = USBx->GOTGINT;
|
||
}
|
||
|
||
// RX FIFO first
|
||
if ((gintsts & USB_OTG_GINTSTS_RXFLVL) != 0) {
|
||
// 1. Read the Receive status pop register
|
||
volatile unsigned int rxst = USBx->GRXSTSP;
|
||
int status = (rxst & USB_OTG_GRXSTSP_PKTSTS) >> 17;
|
||
|
||
#ifdef DEBUG_USB
|
||
puts(" RX FIFO:");
|
||
puth(rxst);
|
||
puts(" status: ");
|
||
puth(status);
|
||
puts(" len: ");
|
||
puth((rxst & USB_OTG_GRXSTSP_BCNT) >> 4);
|
||
puts("\n");
|
||
#endif
|
||
|
||
if (status == STS_DATA_UPDT) {
|
||
int endpoint = (rxst & USB_OTG_GRXSTSP_EPNUM);
|
||
int len = (rxst & USB_OTG_GRXSTSP_BCNT) >> 4;
|
||
(void)USB_ReadPacket(&usbdata, len);
|
||
#ifdef DEBUG_USB
|
||
puts(" data ");
|
||
puth(len);
|
||
puts("\n");
|
||
hexdump(&usbdata, len);
|
||
#endif
|
||
|
||
if (endpoint == 2) {
|
||
usb_cb_ep2_out(usbdata, len, 1);
|
||
}
|
||
|
||
if (endpoint == 3) {
|
||
usb_cb_ep3_out(usbdata, len, 1);
|
||
}
|
||
} else if (status == STS_SETUP_UPDT) {
|
||
(void)USB_ReadPacket(&setup, 8);
|
||
#ifdef DEBUG_USB
|
||
puts(" setup ");
|
||
hexdump(&setup, 8);
|
||
puts("\n");
|
||
#endif
|
||
} else {
|
||
// status is neither STS_DATA_UPDT or STS_SETUP_UPDT, skip
|
||
}
|
||
}
|
||
|
||
/*if (gintsts & USB_OTG_GINTSTS_HPRTINT) {
|
||
// host
|
||
puts("HPRT:");
|
||
puth(USBx_HOST_PORT->HPRT);
|
||
puts("\n");
|
||
if (USBx_HOST_PORT->HPRT & USB_OTG_HPRT_PCDET) {
|
||
USBx_HOST_PORT->HPRT |= USB_OTG_HPRT_PRST;
|
||
USBx_HOST_PORT->HPRT |= USB_OTG_HPRT_PCDET;
|
||
}
|
||
|
||
}*/
|
||
|
||
if ((gintsts & USB_OTG_GINTSTS_BOUTNAKEFF) || (gintsts & USB_OTG_GINTSTS_GINAKEFF)) {
|
||
// no global NAK, why is this getting set?
|
||
#ifdef DEBUG_USB
|
||
puts("GLOBAL NAK\n");
|
||
#endif
|
||
USBx_DEVICE->DCTL |= USB_OTG_DCTL_CGONAK | USB_OTG_DCTL_CGINAK;
|
||
}
|
||
|
||
if ((gintsts & USB_OTG_GINTSTS_SRQINT) != 0) {
|
||
// we want to do "A-device host negotiation protocol" since we are the A-device
|
||
/*puts("start request\n");
|
||
puth(USBx->GOTGCTL);
|
||
puts("\n");*/
|
||
//USBx->GUSBCFG |= USB_OTG_GUSBCFG_FDMOD;
|
||
//USBx_HOST_PORT->HPRT = USB_OTG_HPRT_PPWR | USB_OTG_HPRT_PENA;
|
||
//USBx->GOTGCTL |= USB_OTG_GOTGCTL_SRQ;
|
||
}
|
||
|
||
// out endpoint hit
|
||
if ((gintsts & USB_OTG_GINTSTS_OEPINT) != 0) {
|
||
#ifdef DEBUG_USB
|
||
puts(" 0:");
|
||
puth(USBx_OUTEP(0)->DOEPINT);
|
||
puts(" 2:");
|
||
puth(USBx_OUTEP(2)->DOEPINT);
|
||
puts(" 3:");
|
||
puth(USBx_OUTEP(3)->DOEPINT);
|
||
puts(" ");
|
||
puth(USBx_OUTEP(3)->DOEPCTL);
|
||
puts(" 4:");
|
||
puth(USBx_OUTEP(4)->DOEPINT);
|
||
puts(" OUT ENDPOINT\n");
|
||
#endif
|
||
|
||
if ((USBx_OUTEP(2)->DOEPINT & USB_OTG_DOEPINT_XFRC) != 0) {
|
||
#ifdef DEBUG_USB
|
||
puts(" OUT2 PACKET XFRC\n");
|
||
#endif
|
||
USBx_OUTEP(2)->DOEPTSIZ = (1U << 19) | 0x40U;
|
||
USBx_OUTEP(2)->DOEPCTL |= USB_OTG_DOEPCTL_EPENA | USB_OTG_DOEPCTL_CNAK;
|
||
}
|
||
|
||
if ((USBx_OUTEP(3)->DOEPINT & USB_OTG_DOEPINT_XFRC) != 0) {
|
||
#ifdef DEBUG_USB
|
||
puts(" OUT3 PACKET XFRC\n");
|
||
#endif
|
||
USBx_OUTEP(3)->DOEPTSIZ = (1U << 19) | 0x40U;
|
||
USBx_OUTEP(3)->DOEPCTL |= USB_OTG_DOEPCTL_EPENA | USB_OTG_DOEPCTL_CNAK;
|
||
} else if ((USBx_OUTEP(3)->DOEPINT & 0x2000) != 0) {
|
||
#ifdef DEBUG_USB
|
||
puts(" OUT3 PACKET WTF\n");
|
||
#endif
|
||
// if NAK was set trigger this, unknown interrupt
|
||
USBx_OUTEP(3)->DOEPTSIZ = (1U << 19) | 0x40U;
|
||
USBx_OUTEP(3)->DOEPCTL |= USB_OTG_DOEPCTL_CNAK;
|
||
} else if ((USBx_OUTEP(3)->DOEPINT) != 0) {
|
||
puts("OUTEP3 error ");
|
||
puth(USBx_OUTEP(3)->DOEPINT);
|
||
puts("\n");
|
||
} else {
|
||
// USBx_OUTEP(3)->DOEPINT is 0, ok to skip
|
||
}
|
||
|
||
if ((USBx_OUTEP(0)->DOEPINT & USB_OTG_DIEPINT_XFRC) != 0) {
|
||
// ready for next packet
|
||
USBx_OUTEP(0)->DOEPTSIZ = USB_OTG_DOEPTSIZ_STUPCNT | (USB_OTG_DOEPTSIZ_PKTCNT & (1U << 19)) | (1U << 3);
|
||
}
|
||
|
||
// respond to setup packets
|
||
if ((USBx_OUTEP(0)->DOEPINT & USB_OTG_DOEPINT_STUP) != 0) {
|
||
usb_setup();
|
||
}
|
||
|
||
USBx_OUTEP(0)->DOEPINT = USBx_OUTEP(0)->DOEPINT;
|
||
USBx_OUTEP(2)->DOEPINT = USBx_OUTEP(2)->DOEPINT;
|
||
USBx_OUTEP(3)->DOEPINT = USBx_OUTEP(3)->DOEPINT;
|
||
}
|
||
|
||
// interrupt endpoint hit (Page 1221)
|
||
if ((gintsts & USB_OTG_GINTSTS_IEPINT) != 0) {
|
||
#ifdef DEBUG_USB
|
||
puts(" ");
|
||
puth(USBx_INEP(0)->DIEPINT);
|
||
puts(" ");
|
||
puth(USBx_INEP(1)->DIEPINT);
|
||
puts(" IN ENDPOINT\n");
|
||
#endif
|
||
|
||
// Should likely check the EP of the IN request even if there is
|
||
// only one IN endpoint.
|
||
|
||
// No need to set NAK in OTG_DIEPCTL0 when nothing to send,
|
||
// Appears USB core automatically sets NAK. WritePacket clears it.
|
||
|
||
// Handle the two interface alternate settings. Setting 0 has EP1
|
||
// as bulk. Setting 1 has EP1 as interrupt. The code to handle
|
||
// these two EP variations are very similar and can be
|
||
// restructured for smaller code footprint. Keeping split out for
|
||
// now for clarity.
|
||
|
||
//TODO add default case. Should it NAK?
|
||
switch (current_int0_alt_setting) {
|
||
case 0: ////// Bulk config
|
||
// *** IN token received when TxFIFO is empty
|
||
if ((USBx_INEP(1)->DIEPINT & USB_OTG_DIEPMSK_ITTXFEMSK) != 0) {
|
||
#ifdef DEBUG_USB
|
||
puts(" IN PACKET QUEUE\n");
|
||
#endif
|
||
// TODO: always assuming max len, can we get the length?
|
||
USB_WritePacket((void *)resp, usb_cb_ep1_in(resp, 0x40, 1), 1);
|
||
}
|
||
break;
|
||
|
||
case 1: ////// Interrupt config
|
||
// *** IN token received when TxFIFO is empty
|
||
if ((USBx_INEP(1)->DIEPINT & USB_OTG_DIEPMSK_ITTXFEMSK) != 0) {
|
||
#ifdef DEBUG_USB
|
||
puts(" IN PACKET QUEUE\n");
|
||
#endif
|
||
// TODO: always assuming max len, can we get the length?
|
||
int len = usb_cb_ep1_in(resp, 0x40, 1);
|
||
if (len > 0) {
|
||
USB_WritePacket((void *)resp, len, 1);
|
||
}
|
||
}
|
||
break;
|
||
default:
|
||
puts("current_int0_alt_setting value invalid\n");
|
||
break;
|
||
}
|
||
|
||
if ((USBx_INEP(0)->DIEPINT & USB_OTG_DIEPMSK_ITTXFEMSK) != 0) {
|
||
#ifdef DEBUG_USB
|
||
puts(" IN PACKET QUEUE\n");
|
||
#endif
|
||
|
||
if ((ep0_txlen != 0U) && ((USBx_INEP(0)->DTXFSTS & USB_OTG_DTXFSTS_INEPTFSAV) >= 0x40U)) {
|
||
uint16_t len = MIN(ep0_txlen, 0x40);
|
||
USB_WritePacket(ep0_txdata, len, 0);
|
||
ep0_txdata = &ep0_txdata[len];
|
||
ep0_txlen -= len;
|
||
if (ep0_txlen == 0U) {
|
||
ep0_txdata = NULL;
|
||
USBx_DEVICE->DIEPEMPMSK &= ~1;
|
||
USBx_OUTEP(0)->DOEPCTL |= USB_OTG_DOEPCTL_CNAK;
|
||
}
|
||
}
|
||
}
|
||
|
||
// clear interrupts
|
||
USBx_INEP(0)->DIEPINT = USBx_INEP(0)->DIEPINT; // Why ep0?
|
||
USBx_INEP(1)->DIEPINT = USBx_INEP(1)->DIEPINT;
|
||
}
|
||
|
||
// clear all interrupts we handled
|
||
USBx_DEVICE->DAINT = daint;
|
||
USBx->GOTGINT = gotgint;
|
||
USBx->GINTSTS = gintsts;
|
||
|
||
//USBx->GINTMSK = 0xFFFFFFFF & ~(USB_OTG_GINTMSK_NPTXFEM | USB_OTG_GINTMSK_PTXFEM | USB_OTG_GINTSTS_SOF | USB_OTG_GINTSTS_EOPF);
|
||
}
|
||
|
||
void OTG_FS_IRQHandler(void) {
|
||
NVIC_DisableIRQ(OTG_FS_IRQn);
|
||
//__disable_irq();
|
||
usb_irqhandler();
|
||
//__enable_irq();
|
||
NVIC_EnableIRQ(OTG_FS_IRQn);
|
||
}
|
||
|