-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathusb.h
76 lines (62 loc) · 3.29 KB
/
usb.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
#pragma once
/*
Small USB driver for STM32F103, providing 1 device with 1 configuration
with 1 interface with 1 setting, providing endpoint 0 for control and
endpoint 0x1/0x81 with bidirectional bulk transfers, with a buffer
size of 64 bytes in both directions.
By sending a slightly bigger descriptor, this shows up as a CDC/ACM device
aka a serial port, under MacOS and (hopefully) Linux without the need for extra drivers.
No form of line control, baud rate, parity etc setting is implemented, you just get
the full USB bandwith to/from your microcontroller with the abi defined in usb.h
The device has VID 0483 (STMicroelectronics) and PID 5740 (Virtual Com Port).
This implementation only uses the USB_LP_CAN_RX0_IRQn interrupt.
The low level driver requires the following entries in the linker script:
USB = 0x40005C00
USB_PMA = 0x40006000
*/
#include <stddef.h> // for size_t
#include <stdint.h> // for uintX_t
// usb_init() (re-)initializes the usb device.
//
// Before calling this, be sure to enable the USB peripheral, e.g.
//
// RCC.APB1ENR |= RCC_APB1ENR_USBEN;
//
// To enable handling of the protocol messages exchanged with the usb host,
// the user must call NVIC_EnableIRQ(USB_LP_CAN1_RX0_IRQn) and provide an
// implementation of USB_LP_CAN1_RX0_IRQ_Handler() that calls usb_recv()
// or call usb_recv() repeatedly by other means, e.g. in the main loop.
void usb_init();
// usb_shutdown() puts the USB transceiver in power-down mode.
//
// Before calling this the user may call NVIC_DisableIRQ
// (USB_LP_CAN1_RX0_IRQn) to ensure no further transactions are being
// handled. After a call to usb_shutdown the USB peripheral may be powered
// down completely, e.g.
//
// RCC.APB1ENR &= ~RCC_APB1ENR_USBEN;
//
void usb_shutdown();
// Cf. the Visible States Universal Serial Bus Specification Revision 2.0 Section 9.1.1
// In the DEFAULT state, the device address is zero and only endpoint 0 is configured.
// On a resume(wakeup) event, the SUSPENDED state returns to whatever the state was before.
enum usb_state_t {
USB_UNATTACHED, // The built-in USB peripheral not yet communicating with the host.
USB_DEFAULT, // the host has issued a bus-reset to the peripheral.
USB_ADDRESS, // the host has assigned a device address to the peripheral
USB_CONFIGURED, // the host has issued a successful configuration request to the peripheral
USB_SUSPENDED, // the peripheral is in the suspended state due to bus inactivity
};
// usb_state() returns the current state of the built-in USB peripheral.
enum usb_state_t usb_state();
const char* usb_state_str(enum usb_state_t s);
// usb_recv() tries to copy up to sz bytes from the last received message from
// the USB receive buffer to buf. This function is meant to be called from
// USB_LP_CAN1_RX0_IRQ_Handler() or the main loop, as it drives the entire protocol stack. If
// more than sz bytes are available, the unread portion of the buffer is
// discarded. To avoid this call usb_recv with sz >= 64.
size_t usb_recv(uint8_t* buf, size_t sz);
// usb_send tries to copy buf[:min(len,64)] to a free USB transmit buffer and
// schedules it for transmission. If no buffer is available usb_send returns
// zero, otherwise it returns the number of bytes that could be copied.
size_t usb_send(const uint8_t* buf, size_t len);