update doc

This commit is contained in:
sakumisu
2022-09-17 14:00:04 +08:00
parent 34291a42a1
commit 0412b6b685
6 changed files with 124 additions and 177 deletions

View File

@@ -4,7 +4,7 @@
设备协议栈主要负责枚举和驱动加载,枚举这边就不说了,驱动加载,也就是接口驱动加载,主要是依靠 `usbd_add_interface` 函数,记录传入的接口驱动保存到链表中,当主机进行类请求时就可以查找链表进行访问了。
在调用 `usbd_desc_register` 以后需要进行接口注册和端点注册,口诀如下:
- 有多少个接口就调用多少次 `usbd_add_interface`,参数填各个 class alloc 出来的 intf如果没有 alloc 的intf 表示不需要加载。
- 有多少个接口就调用多少次 `usbd_add_interface`,参数填相关 `xxx_alloc_intf`, 如果没有支持的,手动创建一个填入
- 有多少个端点就调用多少次 `usbd_add_endpoint`,当中断完成时,会调用到注册的端点回调中。
CORE

View File

@@ -15,11 +15,12 @@ CLASS 驱动信息结构体
.. code-block:: C
struct usbh_class_info {
uint8_t class; /* Base device class code */
uint8_t subclass; /* Sub-class, depends on base class. Eg. */
uint8_t protocol; /* Protocol, depends on base class. Eg. */
uint16_t vid; /* Vendor ID (for vendor/product specific devices) */
uint16_t pid; /* Product ID (for vendor/product specific devices) */
uint8_t match_flags; /* Used for product specific matches; range is inclusive */
uint8_t class; /* Base device class code */
uint8_t subclass; /* Sub-class, depends on base class. Eg. */
uint8_t protocol; /* Protocol, depends on base class. Eg. */
uint16_t vid; /* Vendor ID (for vendor/product specific devices) */
uint16_t pid; /* Product ID (for vendor/product specific devices) */
const struct usbh_class_driver *class_driver;
};
@@ -28,99 +29,90 @@ CLASS 驱动信息结构体
.. code-block:: C
typedef struct usbh_endpoint {
struct usbh_endpoint {
struct usb_endpoint_descriptor ep_desc;
} usbh_endpoint_t;
};
接口备用结构体
""""""""""""""""""""""""""""""""""""
.. code-block:: C
struct usbh_interface_altsetting {
struct usb_interface_descriptor intf_desc;
struct usbh_endpoint ep[CONFIG_USBHOST_MAX_ENDPOINTS];
};
接口结构体
""""""""""""""""""""""""""""""""""""
.. code-block:: C
typedef struct usbh_interface {
struct usb_interface_descriptor intf_desc;
struct usbh_endpoint ep[CONFIG_USBHOST_EP_NUM];
struct usbh_interface {
struct usbh_interface_altsetting altsetting[CONFIG_USBHOST_MAX_INTF_ALTSETTINGS];
uint8_t altsetting_num;
char devname[CONFIG_USBHOST_DEV_NAMELEN];
struct usbh_class_driver *class_driver;
void *priv;
} usbh_interface_t;
};
配置结构体
""""""""""""""""""""""""""""""""""""
.. code-block:: C
typedef struct usbh_configuration {
struct usbh_configuration {
struct usb_configuration_descriptor config_desc;
struct usbh_interface intf[CONFIG_USBHOST_INTF_NUM];
} usbh_configuration_t;
struct usbh_interface intf[CONFIG_USBHOST_MAX_INTERFACES];
};
hubport 结构体
""""""""""""""""""""""""""""""""""""
.. code-block:: C
typedef struct usbh_hubport {
bool connected; /* True: device connected; false: disconnected */
bool port_change; /* True: port changed; false: port do not change */
uint8_t port; /* Hub port index */
uint8_t dev_addr; /* device address */
uint8_t speed; /* device speed */
usbh_epinfo_t ep0; /* control ep info */
struct usbh_hubport {
bool connected; /* True: device connected; false: disconnected */
uint8_t port; /* Hub port index */
uint8_t dev_addr; /* device address */
uint8_t speed; /* device speed */
usbh_pipe_t ep0; /* control ep pipe info */
struct usb_device_descriptor device_desc;
struct usbh_configuration config;
const char *iManufacturer;
const char *iProduct;
const char *iSerialNumber;
#if 0
uint8_t* config_desc;
uint8_t* raw_config_desc;
#endif
struct usb_setup_packet *setup;
struct usbh_hub *parent; /*if NULL, is roothub*/
} usbh_hubport_t;
USB_MEM_ALIGNX struct usb_setup_packet setup;
struct usbh_hub *parent;
};
hub 结构体
""""""""""""""""""""""""""""""""""""
.. code-block:: C
typedef struct usbh_hub {
struct usbh_hub {
usb_slist_t list;
uint8_t index; /* Hub index */
uint8_t nports; /* Hub port number */
uint8_t dev_addr; /* Hub device address */
usbh_epinfo_t intin;
uint8_t *int_buffer;
struct hub_port_status *port_status;
bool connected;
bool is_roothub;
uint8_t index;
uint8_t hub_addr;
usbh_pipe_t intin;
USB_MEM_ALIGNX uint8_t int_buffer[1];
struct usbh_urb intin_urb;
struct usb_hub_descriptor hub_desc;
struct usbh_hubport child[CONFIG_USBHOST_EHPORTS];
struct usbh_hubport *parent; /* Parent hub port */
struct usb_work work;
} usbh_hub_t;
usbh_event_notify_handler
""""""""""""""""""""""""""""""""""""
``usbh_event_notify_handler`` 是 USB 中断中的核心,主要用于处理 **设备连接****设备断开** 中断,从而唤醒线程去执行枚举。
.. code-block:: C
void usbh_event_notify_handler(uint8_t event, uint8_t rhport);
- **event** 中断事件
- **rhport** roothub 端口号
其中 ``event`` 有如下类型:
.. code-block:: C
enum usbh_event_type {
USBH_EVENT_ATTACHED,
USBH_EVENT_REMOVED,
struct usbh_hubport child[CONFIG_USBHOST_MAX_EHPORTS];
struct usbh_hubport *parent;
usb_slist_t hub_event_list;
};
usbh_initialize
""""""""""""""""""""""""""""""""""""
``usbh_initialize`` 用来初始化 usb 主机协议栈,包括:创建插拔检测用的信号量和枚举线程、高低工作队列、初始化 roothub 端点0 配置,初始化 usb 主机控制器。
``usbh_initialize`` 用来初始化 usb 主机协议栈,包括:初始化 usb 主机控制器,创建 roothub 设备,创建 hub 检测线程
.. code-block:: C

View File

@@ -140,112 +140,100 @@ usb_hc_init
- **return** 返回 0 表示正确,其他表示错误
usbh_get_port_connect_status
usbh_roothub_control
""""""""""""""""""""""""""""""""""""
``usbh_get_port_connect_status`` 获取当前 hubport 连接状态。 **此函数不对用户开放**
``usbh_roothub_control`` 用来对 roothub 发起请求, **此函数不对用户开放**
.. code-block:: C
int usbh_get_port_connect_status(const uint8_t port);
int usbh_roothub_control(struct usb_setup_packet *setup, uint8_t *buf);
- **port** 端口号
- **return** 返回 1 表示连接0 表示未连接
usbh_reset_port
""""""""""""""""""""""""""""""""""""
``usbh_reset_port`` 复位指定的 hubport **此函数不对用户开放**
.. code-block:: C
int usbh_reset_port(const uint8_t port);
- **port** 端口号
- **setup** 请求
- **buf** 接收缓冲区
- **return** 返回 0 表示正确,其他表示错误
usbh_get_port_speed
usbh_ep0_pipe_reconfigure
""""""""""""""""""""""""""""""""""""
``usbh_get_port_speed`` 获取当前 hubport 上连接的设备速度**此函数不对用户开放**
``usbh_ep0_pipe_reconfigure`` 重新设置端点 0 的 pipe 属性**此函数不对用户开放**
.. code-block:: C
int usbh_get_port_speed(const uint8_t port);
int usbh_ep0_pipe_reconfigure(usbh_pipe_t pipe, uint8_t dev_addr, uint8_t ep_mps, uint8_t speed);
- **port** 端口号
- **return** 返回 1 表示低速2 表示全速3 表示高速
usbh_ep0_reconfigure
""""""""""""""""""""""""""""""""""""
``usbh_ep0_reconfigure`` 重新设置端点 0 的属性。 **此函数不对用户开放**
.. code-block:: C
int usbh_ep0_reconfigure(usbh_epinfo_t ep, uint8_t dev_addr, uint8_t ep_mps, uint8_t speed);
- **ep** 端点信息
- **pipe** pipe 句柄
- **dev_addr** 端点所在设备地址
- **ep_mps** 端点最大包长
- **speed** 端点所在设备的速度
- **return** 返回 0 表示正确,其他表示错误
usbh_ep_alloc
usbh_pipe_alloc
""""""""""""""""""""""""""""""""""""
``usbh_ep_alloc`` 为端点分配相关属性,初始化相关寄存器,并保存相关信息到 **ep** 句柄中**此函数不对用户开放**
``usbh_pipe_alloc`` 为端点分配 pipe**此函数不对用户开放**
.. code-block:: C
int usbh_ep_alloc(usbh_epinfo_t *ep, const struct usbh_endpoint_cfg *ep_cfg);
int usbh_pipe_alloc(usbh_pipe_t *pipe, const struct usbh_endpoint_cfg *ep_cfg);
- **ep** 端点信息
- **pipe** pipe 句柄
- **ep_cfg** 端点初始化需要的一些信息
- **return** 返回 0 表示正确,其他表示错误
usbh_ep_free
usbh_pipe_free
""""""""""""""""""""""""""""""""""""
``usbh_ep_free`` 释放端点的一些属性。 **此函数不对用户开放**
``usbh_pipe_free`` 释放端点的一些属性。 **此函数不对用户开放**
.. code-block:: C
int usbh_ep_free(usbh_epinfo_t ep);
int usbh_pipe_free(usbh_pipe_t pipe);
- **ep** 端点信息
- **pipe** 端点信息
- **return** 返回 0 表示正确,其他表示错误
usbh_control_transfer
usbh_submit_urb
""""""""""""""""""""""""""""""""""""
``usbh_control_transfer`` 对端点 0 进行控制传输,并且 **此函数为阻塞式传输,默认超时时间 5s****此函数对用户开放**
``usbh_submit_urb`` 对某个地址上的端点进行数据请求**此函数对用户开放**
.. code-block:: C
int usbh_control_transfer(usbh_epinfo_t ep, struct usb_setup_packet *setup, uint8_t *buffer);
int usbh_submit_urb(struct usbh_urb *urb);
- **ep** 端点信息
- **setup** setup 包
- **buffer** 要发送或者读取的数据缓冲区,为 NULL 表示没有数据要发送或者接收
- **urb** usb 请求块
- **return** 返回 0 表示正确,其他表示错误
usbh_ep_bulk_transfer
""""""""""""""""""""""""""""""""""""
``usbh_ep_bulk_transfer`` 对指定端点进行批量传输, **此函数为阻塞式传输****此函数对用户开放**
其中, `urb` 结构体信息如下:
.. code-block:: C
int usbh_ep_bulk_transfer(usbh_epinfo_t ep, uint8_t *buffer, uint32_t buflen, uint32_t timeout);
struct usbh_urb {
usbh_pipe_t pipe;
struct usb_setup_packet *setup;
uint8_t *transfer_buffer;
uint32_t transfer_buffer_length;
int transfer_flags;
uint32_t actual_length;
uint32_t timeout;
int errorcode;
usbh_complete_callback_t complete;
void *arg;
};
- **ep** 端点信息
- **buffer** 要发送或者读取的数据缓冲区
- **buflen** 要发送或者接收的长度,最大不得高于 16K
- **timeout** 超时时间,单位 ms
- **return** 大于等于0 表示实际发送或者接收的长度,小于 0 表示错误
- **pipe** 端点对应的 pipe 句柄
- **setup** setup 请求缓冲区端点0使用
- **transfer_buffer** 传输的数据缓冲区
- **transfer_buffer_length** 传输长度
- **transfer_flags** 传输时携带的 flag
- **actual_length** 实际传输长度
- **timeout** 传输超时时间,为 0 该函数则为非阻塞,可在中断中使用
- **errorcode** 错误码
- **complete** 传输完成回调函数
- **arg** 传输完成时携带的参数
其中小于 0 的错误码如下
`errorcode` 可以返回以下值
.. list-table::
:widths: 30 30
@@ -253,57 +241,21 @@ usbh_ep_bulk_transfer
* - ERROR CODE
- desc
* - ENOMEM
- 内存不足
* - ENODEV
- 设备未连接
* - EBUSY
- 当前数据发送或者接收还未完成
* - EAGAIN
- 主机一直收到 NAK 包
* - ETIMEDOUT
- 数据发送或者接收超时
* - EPERM
- 主机收到 STALL 包
- 主机收到 STALL 包或者 BABBLE
* - EIO
- 数据传输错误
* - EAGAIN
- 主机一直收到 NAK 包
* - EPIPE
- 数据溢出
* - ENXIO
- 设备断开,传输中止
usbh_ep_intr_transfer
""""""""""""""""""""""""""""""""""""
``usbh_ep_intr_transfer`` 同上。
usbh_ep_bulk_async_transfer
""""""""""""""""""""""""""""""""""""
``usbh_ep_bulk_async_transfer`` 对指定端点进行批量传输,传输完成将触发指定回调函数, **此函数为异步传输****此函数对用户开放**
.. code-block:: C
int usbh_ep_bulk_async_transfer(usbh_epinfo_t ep, uint8_t *buffer, uint32_t buflen, usbh_asynch_callback_t callback, void *arg);
- **ep** 端点信息
- **buffer** 要发送或者读取的数据缓冲区
- **buflen** 要发送或者接收的长度,最大不得高于 16K
- **callback** 传输完成回调函数, **该函数最终处于中断上下文**
- **arg** 用户自定义参数
- **return** 为 0 表示配置正常小于0 表示错误
usbh_ep_intr_async_transfer
""""""""""""""""""""""""""""""""""""
``usbh_ep_intr_async_transfer`` 同上。
usb_ep_cancel
""""""""""""""""""""""""""""""""""""
``usb_ep_cancel`` 中止当前端点传输, **此函数不对用户开放**
.. code-block:: C
int usb_ep_cancel(usbh_epinfo_t ep);
- **ep** 端点信息
- **return** 为 0 表示正确小于0 表示错误
* - ESHUTDOWN
- 设备断开,传输中止

View File

@@ -8,12 +8,13 @@ CherryUSB 使用指南
CherryUSB 是一个小而美的、可移植性高的、用于嵌入式系统的 USB 主从协议栈。同时 CherryUSB 具有以下优点:
- 比较全面的 class 驱动,并且 class 驱动全部模板化,方便学习和自主添加
- 树状化编程,方便理清 class 驱动与接口、端点的关系hub、port、class 之间的关系;代码层层递进,调用关系一目了然,方便理清 usb 枚举过程和 class 驱动加载
- 设备协议栈使用等价于 uart tx/rx dma 的使用,主机协议栈的使用等价于文件的使用
- 标准化的 porting 接口,同时面向 ip 化编程,相同 ip 无需重复编写驱动
- Api 少分类清晰dcd/hcd api、注册 api、命令回调 api
- 协议栈代码精简内存占用极小ip 驱动代码也做到精简,能够达到 usb 硬件理论带宽
- 代码精简,并且内存占用极小,而且还可进一步的裁剪
- 全面的 class 驱动,并且主从 class 驱动全部模板化,方便用户增加新的 class 驱动以及学习的时候查找规律
- 可供用户使用的 API 非常少,并且分类清晰。从机:初始化 + 注册、命令回调类、数据收发类;主机:初始化 + 查找类、数据收发类
- 树状化编程,代码层层递进,方便用户理清函数调用关系、枚举和 class 驱动加载过程
- 标准化的 porting 接口,相同 ip 无需重写驱动,并且 porting 驱动也进行了模板化,方便用户新增 porting。
- 主从收发接口的使用等价于 uart tx/rx dma 的使用,长度也没有限制
- 能够达到 USB 硬件理论带宽
其他相关链接: