fusb.md 14.1 KB
Newer Older
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 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535

# FUSB 驱动程序

## 1. 概述

USB是通用串行总线(Universal Serial Bus)的缩写, USB是一种简易、双向、快速、同步、即插即用(Plug and Play,PnP)且支持热插拔功能的串行接口。USB设备现在已经非常普及,如U盘、鼠标、键盘等。USB协议曾经出现过多种版本,如USB1.0、USB1.1、USB2.0、USB3.0等

USB总线体系中, USB主机(Host)是系统的主人, 负责USB通信过程中数据的控制和处理, USB设备(Device)是系统的从机,用于实现特定的功能,如常用的U盘、移动硬盘、鼠标、键盘、游戏手柄等,在USB传输过程中,USB主机处于主导地位,由USB主机发起数据和命令的传输,USB设备被动响应USB主机发来的请求命令,USB总线中的主要角色包括USB主机、根Hub、USB Hub和USB设备等

USB主机和USB设备间的通信是通过管道(Pipe)进行的,管道是指在USB主机端的一组缓冲区,用于管道中数据的收发;在USB设备端,有一个特定端点(Endpoint,ENDP)与管道对接

USB可靠传输(具有反馈机制)的最小单位是事务,事务就是利用令牌包、数据包和握手包实现一个带有错误反馈机制的通信,使USB传输更加安全可靠,按照令牌包的类型可以分为,Setup事务、IN事务、OUT事务等。基于事务,USB协议定义了传输(Transfer)用于完成一组具有特定目的的事务,包括控制传输、中断传输、批量传输和同步传输

## 2. 功能

- 驱动相关的源文件如下,
- drivers/usb
```
.
├── Kconfig
├── fusb.c
├── fusb.h
├── fusb_debug.c
├── fusb_def.h
├── fusb_dev.c
├── fusb_g.c
├── fusb_generic_hub.c
├── fusb_generic_hub.h
├── fusb_hid.c
├── fusb_hid.h
├── fusb_hub.c
├── fusb_hub.h
├── fusb_msc.c
├── fusb_msc.h
├── fusb_private.h
├── fusb_sinit.c
```

- 关于 [XHCI](./fxhci.md)

## 3. 配置方法

参考以下步骤完成 FUSB 硬件配置,

- 1. FUSB驱动支持 E2000,在 E2000 上完成测试

参考以下步骤完成 FUSB 软件配置,

- 1. 选择USB控制器类型为 XHCI
- 2. 配置 FUSB 的内存池空间,推荐在1Mb以上
- 3. 注册USB设备驱动
- 4. 调用USB控制器驱动接口启动FUSB实例
- 5. 调用USB控制器驱动接口轮询FUSB,更新USB设备状态
- 6. 调用USB设备驱动操作USB设备

## 4 应用示例

- USB主机操作和USB设备发现
### [fxhci_host](../../../baremetal/example/peripheral/usb/fxhci_host)

- USB大容量存储器读写
### [fusb_fatfs](../../../baremetal/example/storage/fusb_fatfs)

## 5. API参考

### 5.1. 用户数据结构

#### 5.1.1 USB端点

```c
typedef struct 
{
	FUsbDev *dev; /* device instance of this endpoint */
	int endpoint; /* endpoint address ep0 = 0, epn = n */
	FUsbDirection direction; /* type or direction of ep */
	int toggle; /* ep state for some device to toggle */
	int maxpacketsize; /* max packet size for ep transfer */
	FUsbEpType type; /* transfer type of ep, control, bulk or so on */
	int interval; /* expressed as binary logarithm of the number
			 		of microframes (i.e. t = 125us * 2^interval) */
} FUsbEndpoint; /* encapsulates a single endpoint of an USB device */
```

#### 5.1.2 USB设备

```c
typedef struct _FUsbDev
{
	FUsbHc *controller; /* Hc instance where device attached */
	FUsbEndpoint endpoints[FUSB_MAX_EP_NUM]; /* all Ep instance of device */
	int num_endp; /* num of Ep in use */
	FUsbDevAddr address; /* USB address */
	FUsbDevClass class;	/* USB device class, e.g hid */
	int hub; /* hub where device is attached to */
	int port; /* port where device is attached */
	FUsbSpeed speed; /* speed type of device */
	void *data; /* private data for specific type of device */
	FUsbDeviceDescriptor *descriptor; /* device descriptor ever get from device hw */
	FUsbConfigurationDescriptor *configuration; /* configure descriptor followed with interface descriptor ever get from device hw */
	FUsbConfigParser  config_parser; /* parser for configure descriptor */
	FUsbStringParser  string_parser; /* parser for string descriptor */
	void (*init)(FUsbDev *dev); /* device init function of specific device type for register */
	void (*destroy) (FUsbDev *dev); /* device deinit function of specific device type for register */
	void (*poll) (FUsbDev *dev); /* device poll function of specific device type for register */
} FUsbDev; /* encapsulates a single USB device */
```

#### 5.1.3 USB主机

```c
typedef struct _FUsbHc
{
	FUsbHc *next; /* next Hc instance in the list */
	uintptr reg_base; /* base address of Hc register */
	FUsb *usb; /* instance of USB system */
	FUsbHcType type; /* type of Hc, e.g XHCI */
	FUsbDev *devices[FUSB_MAX_DEV_NUM];	/* dev 0 is root hub, 127 is last addressable */

	/* start():     Resume operation. */
	void (*start) (FUsbHc *controller);
	/* stop():      Stop operation but keep controller initialized. */
	void (*stop) (FUsbHc *controller);
	/* reset():     Perform a controller reset. The controller needs to
	                be (re)initialized afterwards to work (again). */
	void (*reset) (FUsbHc *controller);
	/* init():      Initialize a (previously reset) controller
	                to a working state. */
	void (*init) (FUsbHc *controller);
	/* shutdown():  Stop operation, detach host controller and shutdown
	                this driver instance. After calling shutdown() any
					other usage of this hci_t* is invalid. */
	void (*shutdown) (FUsbHc *controller);

	FUsbTransCode (*bulk) (FUsbEndpoint *ep, int size, u8 *data, int finalize);
	FUsbTransCode (*control) (FUsbDev *dev, FUsbDirection pid, int dr_length,
							  void *devreq, int data_length, u8 *data);
	void* (*create_intr_queue) (FUsbEndpoint *ep, int reqsize, int reqcount, int reqtiming);
	void (*destroy_intr_queue) (FUsbEndpoint *ep, void *queue);
	u8* (*poll_intr_queue) (void *queue);
	void *instance; /* instance to specific Hc implementation, e.g XHCI */

	/* set_address():		Tell the USB device its address (xHCI
					controllers want to do this by
					themselves). Also, allocate the FUsbDev
					structure, initialize enpoint 0
					(including MPS) and return it. */
	FUsbDev *(*set_address) (FUsbHc *controller, FUsbSpeed speed,
				  			 int hubport, int hubaddr);
	/* finish_device_config():	Another hook for xHCI, returns 0 on success. */
	int (*finish_device_config) (FUsbDev *dev);
	/* destroy_device(): Finally, destroy all structures that
						 were allocated during set_address()
						 and finish_device_config(). */
	void (*destroy_device) (FUsbHc *controller, int devaddr);
} FUsbHc; /* encapsulates a single USB host */
```

#### 5.1.4 USB系统配置

```c
typedef struct
{
	void *(*malloc_align)(size_t size, size_t align);
	void (*free)(void *mem);
} FUsbMemAllocator; /* memory allocator used in USB system */

typedef struct
{
    u32 instance_id; /* id for this USB system */
    uintptr base_addr; /* base addr of Hc register, set as 0 for pci-usb */
	u32 irq_num;
	u32 irq_priority;
	FUsbMemAllocator allocator; /* memory allocator to support dynamic memory */
} FUsbConfig; /* configure data of the USB system */
```

#### 5.1.4 USB系统

```c
typedef struct _FUsb
{
    FUsbConfig config; /* configuration of USB system */
    void *pcie_instance; /* NULL if unused */
	void *pcie_info[FUSB_MAX_CTRL_NUM]; /* NULL if unused */
	FUsbHc *hc; /* first hc, there might have multiple hc in pcie-mode */
	/* hook to set init function for specific device type */
	FUsbDevInitFunc dev_init[FUSB_MAX_DEV_TYPE_NUM];
	u32 dev_init_num; /* number of init function in used */
	u32 is_ready; /* indicator of system okay */
} FUsb; /* instance of the USB system */
```

### 5.2  错误码定义


- 模块错误码编号 `0x1110000`

- [0x0] FUSB_SUCCESS : success
- [0x1110000] FUSB_ERR_WAIT_TIMEOUT : wait for status timeout
- [0x1110001] FUSB_ERR_INVALID_PARA : invalid input parameters
- [0x1110002] FUSB_ERR_NOT_SUPPORT : parameters or feature not supported
- [0x1110003] FUSB_ERR_NON_INSTANCE : cannot find instance
- [0x1110004] FUSB_ERR_INVALID_DATA : invalid input data
- [0x1110005] FUSB_ERR_DESC_PARSE_ERR : failed to parse descriptor
- [0x1110006] FUSB_ERR_ALLOCATE_FAIL : failed to allocate memory from memory pool
- [0x1110007] FUSB_ERR_TRANS_FAIL : failed to transfer data


- 传输过程完成错误码

- [0] FUSB_CC_ZERO_BYTES : failed, transfer zero bytes
- [1] FUSB_CC_SUCCESS : transfer success with bytes unkonwn

### 5.3. 用户API接口

#### FUsbLookupConfig

```c
const FUsbConfig *FUsbLookupConfig(u32 instance_id)
```

Note:

- 获取USB的默认配置

Input:

- {u32} instance_id USB实例号

Return:

- {const FUsbConfig *} USB默认配置

#### FUsbCfgInitialize

```c
FError FUsbCfgInitialize(FUsb *instance, const FUsbConfig *input_config)
```

Note:

- 初始化USB实例
- 在PCIE模式下,USB Hc实例在PCIE总线发现控制器后创建

Input:

- {FUsb} *instance, USB实例
- {const FUsbConfig} *input_config, USB输入配置

Return:

- {FError} 初始化错误码

#### FUsbDeInitialize

```c
void FUsbDeInitialize(FUsb *instance);
```

Note:

- 去初始化USB实例

Input:

- {FUsb} *instance, USB实例

Return:

-

#### FUsbPoll

```c
void FUsbPoll(FUsb *instance)
```

Note:

- 轮询USB控制器连接的所有设备, 更新设备拓扑

Input:

- {FUsb} *instance, USB实例

Return:

-

#### FUsbExit

```c
void FUsbExit(FUsb *instance)
```

Note:

- 关闭所有的USB控制器,移除所有连接的设备

Input:

- {FUsb} *instance, USB实例

Return:

-


#### FUsbAssignDevInitFunc

```c
FError FUsbAssignDevInitFunc(FUsb *instance, const FUsbDevIndex *index, FUsbDevInitHandler handler)
```

Note:

- 指定特定USB设备的初始化函数,供创建USB设备实例时使用

Input:

- {FUsb} *instance, USB实例
- {FUsbDevIndex} *index, 特定USB设备的索引
- {FUsbDevInitHandler} handler, 特定USB设备的初始化函数

Return:

- {FError} 处理返回错误码

#### FUsbGetAllDevEntries

```c
size_t FUsbGetAllDevEntries(FUsbHc *controller, FUsbDev *devs[], size_t max_dev_num)
```

Note:

- 获取USB控制器上连接的所有USB设备实例

Input:

- {FUsbHc} *controller, USB控制器实例
- {FUsbDev} *devs, 放置USB设备实例的缓冲区
- {size_t} max_dev_num, 最多可以获取的USB设备实例数目

Return:

- {size_t} 实际获取的USB设备实例数目

#### FUsbSetFeature

```c
FUsbTransCode FUsbSetFeature(FUsbDev *dev, int endp, int feature, int rtype)
```

Note:

- 标准USB主机请求,使能设备/接口/端点的某个特性

Input:

- {FUsbDev} *dev, USB设备实例
- {int} endp, 设备号(0x00)/接口号/端点号
- {int} feature, 待使能的特性
- {int} rtype, 请求类型,由FUsbGenerateReqType生成

Return:

- {FUsbTransCode} 控制传输的返回值,小于0表示失败,大于0表示成功传输的字节数目

#### FUsbGetStatus

```c
FUsbTransCode FUsbGetStatus(FUsbDev *dev, int intf, int rtype, int len, void *data)
```

Note:

- 标准USB主机请求,获取设备/接口/端点的状态

Input:

- {FUsbDev} *dev, USB设备实例
- {int} intf,设备号(0x00)/接口号/端点号
- {int} rtype, 请求类型,由FUsbGenerateReqType生成
- {int} len, Data Stage的数据长度
- {void} *data, Data Stage的数据缓冲区

Return:

- {FUsbTransCode} 控制传输的返回值,小于0表示失败,大于0表示成功传输的字节数目

#### FUsbGetDescriptor

```c
FUsbTransCode FUsbGetDescriptor(FUsbDev *dev, int rtype, FUsbDescriptorType desc_type, int desc_idx, void *data, size_t len)
```

Note:

- 标准USB主机请求,获取指定描述符

Input:

- {FUsbDev} *dev, USB设备实例
- {int} rtype, 请求类型,由FUsbGenerateReqType生成
- {FUsbDescriptorType} desc_type, 描述符类型
- {int} desc_idx, 描述符索引
- {void} *data, Data Stage的数据缓冲区
- {size_t} len, Data Stage的数据长度

Return:

- {FUsbTransCode} 控制传输的返回值,小于0表示失败,大于0表示成功传输的字节数目

#### FUsbGetStringDescriptor

```c
FUsbTransCode FUsbGetStringDescriptor(FUsbDev *dev, int rtype, FUsbDescriptorType desc_type, int desc_idx, int lang_id, void *data, size_t len)
```

Note:

- USB主机请求,获取字符串描述符

Input:

- {FUsbDev} *dev, USB设备实例
- {int} rtype, 请求类型,由FUsbGenerateReqType生成
- {int} desc_type, 描述符类型
- {int} desc_idx, 描述符索引
- {int} lang_id, 语言类型
- {void} *data, Data Stage的数据缓冲区
- {size_t} len, Data Stage的数据长度

Return:

- {int} 控制传输的返回值,小于0表示失败,大于0表示成功传输的字节数目

#### FUsbSetConfiguration

```c
FUsbTransCode FUsbSetConfiguration(FUsbDev *dev)
```

Note:

- 标准USB主机请求,设置配置值

Input:

- {FUsbDev} *dev, USB设备实例

Return:

- {FUsbTransCode} 控制传输的返回值,小于0表示失败,大于0表示成功传输的字节数目

#### FUsbClearFeature

```c
FUsbTransCode FUsbClearFeature(FUsbDev *dev, int endp, int feature, int rtype)
```

Note:

- 标准USB主机请求,去使能设备/接口/端点的某个特性

Input:

- {FUsbDev} *dev, USB设备实例
- {int} endp, 设备号(0x00)/接口号/端点号
- {int} feature,待去除的特性
- {int} rtype, 请求类型,由FUsbGenerateReqType生成

Return:

- {FUsbTransCode} 控制传输的返回值,小于0表示失败,大于0表示成功传输的字节数目

#### FUsbDumpAllDescriptors

```c
void FUsbDumpAllDescriptors(FUsbDev *dev)
```

Note:

- 打印USB设备的描述符信息(设备描述符,配置描述符和接口描述符)

Input:

- {FUsbDev} *dev, USB设备实例,已完成初始化

Return:

-

#### FUsbDetachDev

```c
void FUsbDetachDev(FUsbHc *controller, int devno)
```

Note:

- 从USB主机移除指定USB设备(USB设备驱动使用)

Input:

- {FUsbHc} *controller, USB控制器实例
- {int} devno, USB设备地址

Return:

-


#### FUsbAttachDev

```c
FUsbDevAddr FUsbAttachDev(FUsbHc *controller, int hubaddress, int port, FUsbSpeed speed)
```

Note:

- 向USB主机添加USB设备(USB设备驱动使用)

Input:

- {FUsbHc} *controller, USB控制器实例
- {int} hubaddress, Hub地址
- {int} port, 连接的Port
- {FUsbSpeed} speed, USB设备的设置速度类型

Return:

- {FUsbDevAddr} 分配的USB设备地址