完善资料让更多小伙伴认识你,还能领取20积分哦, 立即完善>
USB是英文Universal Serial Bus(通用串行总线的缩写),是一种快速、灵活的总线接口。USB对所有的USB外设提供了单一的易于使用的标准的连接类型,简化了USB外设的设计。 |
|
相关推荐
4个回答
|
|
USB接口支持热插拔,计算机系统可以动态地检测外设的插拔,并且动态地加载驱动程序。另外,USB还具有很灵活的扩展性,一个USB端口接上一个USB HUB(集线器)就可以扩展多个USB端口。USB广泛的应用于PC机中的人机接口、音频、存储等功能,随着嵌入式系统在各个领域上广泛应用,USB也在嵌入式系统中被广泛的使用。
一、USB简介 USB是在1994年底由英特尔、康柏、IBM、Microsoft等多家公司联合提出。ISB1.1是较为普遍的USB规范,其高速方式的传输速率是12Mbps,低速模式的传输速率为1.5Mbps(b是bit的意思,12Mbps=1.5MB/s),USB2.0规范是由USB1.1规范演变而来,它的传输速率达到了480Mbps,足以满足大多数外设的速率要求。USB2.0中的EHCI(增强主机控制器接口)定义了一个与USB1.1相兼容的结构。它可以用USB2.0的驱动程序驱动USB1.1的设备,也就是说所有支持USB1.1的设备都可以直接在USB2.0的接口上使用而不必担心兼容性问题。USB采用四线电缆,其中两根是用来传送数据的串行通道,另两根为下游设备提供电源,对于任何已经成功连接且相互识别的外设,将以对方设备均能够支持的最高速率传输数据。USB总线会根据外设情况在所兼容的传输模式中自动地由高速向低速动态转换且匹配锁定在合适的速率。USB系统采用级联星型拓扑,该拓扑由三个基本部分组成:主机,集线器和功能设备。 主机包含有主控制器和根集线器,控制着USB总线上的数据和控制信息的流动,每个USB系统只能有一个根集线器,它连接在主控制器上。 集线器是USB结构中的特定成分,它可以对原有的USB端口在数量上进行扩展以获得更多的USB端口(集线器只能扩展出更多的USB端口,而不能扩展出更多的带宽),设备通过端口连接到USB总线上,同时检测连接在总线上的设备,并为这些设备提供电源管理,负责总线的故障检测和恢复。集线可为总线提供能源,亦可为自身提供能源。 功能设备通过端口与总线连接。 本文章以M1052平台为例,主要讲解在AWorks平台中如何创建一个自定义的USB设备。 二、AWorks工程配置 在AWorks工程中,用户可以通过aw_prj_params.h配置文件进行设备添加与删除,aw_prj_params.h在工程目录user_config中。 在aw_prj_params.h文件中,有效定义设备的使能宏,才表示一个有效的硬件设备,在本文章中我们需要使能USB Device,所以在aw_prj_params.h中添加AW_DEV_IMX1050_USBD设备宏。 #define AW_DEV_IMX1050_USBD /**< rief USB Device */ 在aw_prj_params.h中添加AW_DEV_IMX1050_USBD设备宏之后,在aw_prj_param_auto_cfg.h文件中会自动添加AW_COM_USBD的定义,AW_COM_USBD代表的是USB Device协议栈组件。此处要注意AW_DEV_IMX1050_USBD与AW_DEV_IMX1050_USBH1有冲突(一个USB控制器不能同时出现主机设备模式),两个只能使能一个。AW_DRV_IMX1050_USBD表示加载USB Device的驱动: /** * ame 内部USB Device控制器 * @{ */ #ifdef AW_DEV_IMX1050_USBD #ifndef AW_COM_USBD #define AW_COM_USBD #endif #define AW_DRV_IMX1050_USBD #endif |
|
|
|
三、AWorks中USB Device的硬件配置
在AWorks中所有设备集中由AWbus_lite进行管理,在使用一个硬件设备前,必须将其添加到系统硬件设备列表(在awbus_lite_hwconf_usrcfg.c中定义),各个硬件设备的类型为struct awbl_devhcf。USB设备相关硬件配置可以查阅工程目录下的user_configawbl_hwconf_usrcfgawbl_hwconf_imx1050_u***d.h文件,USB设备模式的设备信息详见程序清单2.1。 程序清单2.1 USB设备模式的设备信息类型定义 1 aw_local aw_const struct awbl_imx1050_u***d_info __g_imx1050_u***d_info = { 2 { 3 "/dev/u***d", /**< 控制器名字 */ 4 AWBL_USBD_CFG_SELF_POWERED /**< 控制器配置信息*/ 5 }, 6 7 IMX1050_USB1_BASE_ADDR, /**< 寄存器基地址 */ 8 IMX1050_USBPHY1_BASE_ADDR, /**< USB PHY寄存器基地址 */ 9 INUM_USB_OTG1, /**< USB 中断号*/ 10 5, /**< 处理u***中断业务的任务优先级 */ 11 32, 12 __imx1050_u***d_plfm_init /**< 平台相关初始化:初始化时钟 */ 13 }; 14 其中"/dev/u***d" 是USB Device外设的设备名字,在实际的应用中需要通过该名字来使用USB Device AWBL_USBD_CFG_SELF_POWERED 表示本设备为自行供电设备。__imx1050_u***d_plfm_init函数是用来初始化与USB Device平台相关的信息,在这里是初始化USB Device时钟,详见程序清单2.2。 程序清单2.2 USB设备模式的平台初始化函数 1 aw_local void __imx1050_u***d_plfm_init (void) { 2 /* 配置USB Device时钟 */ 3 u***d_clk_init(); 4 }; 四、与USB Device操作函数相关重要结构体介绍 USB设备类回调函数列表,这个结构体用于提供USB枚举以及USB通信过程中的事件回调接口,方便用户接入应用代码: /** rief USB设备类回调函数列表 */ struct aw_u***d_cb { /** rief 控制器链接到设备 */ int (*pfn_link) (struct aw_u***d *p_obj); /** rief 控制器和设备断开链接 */ int (*pfn_unlink) (struct aw_u***d *p_obj); /** rief 接收到SETUP包 */ int (*pfn_ctrl_setup) (struct aw_u***d *p_obj, struct aw_u***_ctrlreq *p_setup, void *p_buf, int buf_len); /** rief 设置或清除配置 */ int (*pfn_config_set) (struct aw_u***d *p_obj, struct aw_u***d_cfg *p_cfg, bool_t set); /** rief 总线挂起 */ void (*pfn_suspend) (struct aw_u***d *p_obj); /** rief 总线恢复 */ void (*pfn_resume) (struct aw_u***d *p_obj); /** rief 总线断开连接 */ void (*pfn_disconnect) (struct aw_u***d *p_obj); }; USB设备信息结构体,此结构体用于配置USB设备的基础信息(具体信息详见结构体定义),当设备连接到PC端后,可以在PC端查看到这些信息: /** rief USBD信息结构定义 */ struct aw_u***d_info { uint8_t clss; /**< rief 类代码 */ uint8_t sub_clss; /**< rief 子类代码 */ uint8_t protocol; /**< rief 协议代码 */ uint8_t mps0; /**< rief 端点0包最大尺寸 */ uint16_t bcd; /**< rief 设备版本号 */ uint16_t vid; /**< rief 厂商ID */ uint16_t pid; /**< rief 产品ID */ const char *manufacturer; /**< rief 制造商字符串描述 */ const char *product; /**< rief 产品字符串描述 */ const char *serial; /**< rief 序列号字符串描述 */ }; USB设备接口信息结构体,这个结构体是用来描述当前USB设备接口的信息: /** rief USBD接口信息 */ struct aw_u***d_fun_info { uint8_t clss; /**< rief 类代码 */ uint8_t sub_clss; /**< rief 子类代码 */ uint8_t protocol; /**< rief 协议代码 */ /** rief 替代设置回调 */ int (*pfn_alt_set) (struct aw_u***d_fun *p_fun, bool_t set); /** rief setup控制传输回调 */ int (*pfn_ctrl_setup) (struct aw_u***d_fun *p_fun, struct aw_u***_ctrlreq *p_setup, void *p_buf, int buf_len); }; 五、USB Device操作接口 Aworks提供了一系列的USB Device操作相关的标准接口,包括初始化一个USB设备、USB功能初始化、启动USB设备和USB数据传输等,用户在使用Aworks开发USB Device的应用程序时,使用上述标准接口就能方便快捷的开发出源代码。详见表4.1 表4.1 USB设备模式操作接口函数 函数原型 功能简介 int aw_u***d_init (struct aw_u***d *p_obj, const struct aw_u***d_info *p_info, const struct aw_u***d_cb *p_cb, const char *controller); 初始化USB设备对象 int aw_u***d_fun_init (struct aw_u***d_fun *p_fun, const char *name, const struct aw_u***d_fun_info *p_info); USB device初始化接口功能 int aw_u***d_fun_add (struct aw_u***d *p_obj, uint8_t value, struct aw_u***d_fun *p_fun); 添加接口到指定配置 int aw_u***d_pipe_create (struct aw_u***d_fun *p_fun, struct aw_u***d_pipe *p_pipe, const struct aw_u***d_pipe_info*p_info); 创建并添加一个管道到指定接口 int aw_u***d_pipe_reset (struct aw_u***d *p_obj, struct aw_u***d_pipe *p_pipe); USB管道复位 int aw_u***d_trans_sync (struct aw_u***d *p_obj, struct aw_u***d_pipe *p_pipe, void *p_buf, size_t len, int flag, int timeout); USB数据阻塞传输 int aw_u***d_trans_async (struct aw_u***d *p_obj, struct aw_u***d_trans *p_trans); USB数据非阻塞传输 int aw_u***d_start (struct aw_u***d *p_obj); 启动USB设备 int aw_u***d_stop (struct aw_u***d *p_obj); 停止USB设备 l struct aw_u***d 为USB Device对象 l struct aw_u***d_fun 为功能接口对象 l struct aw_u***d_pipe 为端点管道对象 |
|
|
|
六、应用案例代码详解
这里基于M1052平台,展示如何创建一个自定义的USB设备。本例子会初始化一个USB设备,为USB设备创建一个OUT管道和一个IN管道用于接收和发送数据(IN和OUT的概念是相对于USB主机而言),并在这个USB设备添加接口功能。通过USB设备与PC机的连接,实现与PC的通讯,当PC机上位机软件发送数据时,USB设备通过OUT管道接收数据,并通过IN管道把收到的数据发送回PC端上位机。这里只对部分关键的代码进行讲解,完整的代码请自行下载查看,本例程所在目录位于examplesperipheralcommonu*** demo_u***d_vendor.c 定义数据接收缓冲区的大小和等待超时时间: 11 #define __BUF_SIZE 512 /** 数据缓冲区大小*/ 12 #define __WAIT_TIMEOUT 5000 /** 等待超时时间*/ 定义USB Device的信息,当USB Device与PC端连接时,PC端会对其进行枚举从而获取USB Device的设备信息,,这些信息都可以在PC端进行查看: 14 /** rief 定义全局u***设备实例和u***设备信息结构体 */ 15 static struct aw_u***d __g_u***d_obj; 16 const static struct aw_u***d_info __g_u***d_info = { 17 0, /**< rief 类代码 */ 18 0, /**< rief 子类代码 */ 19 0, /**< rief 协议代码 */ 20 64, /**< rief 端点0包最大尺寸 */ 21 0x0100, /**< rief 设备版本号 */ 22 0x3068, /**< rief 厂商ID */ 23 0x0003, /**< rief 产品ID */ 24 "ZLG", /**< rief 制造商字符串描述 */ 25 "AWorks", /**< rief 产品字符串描述 */ 26 "20160824" /**< rief 序列号字符串描述 */ 27 }; USB Device接口信息结构体: 48 /** rief 接口信息 */ 49 static const struct aw_u***d_fun_info __g_info = { 50 AW_USB_CLASS_VENDOR_SPEC, /**< rief 类代码-厂商自定义设备 */ 51 0x00, /**< rief 子类代码 */ 52 0x00, /**< rief 协议代码 */ 53 __custom_alt_set, /**< rief 替代设置回调 */ 54 NULL /**< rief 控制传输回调 */ 55 }; USB Device例程入口函数,这里主要是初始化USB Device对象,并为这个USB Device对象添加自定义功能,最后通过调用aw_u***d_start函数启动配置好的USB Device。 174 /** 175 * rief 自定义USB设备 demo 176 * eturn 无 177 */ 178 void demo_u***d_vendor_entry (void) 179 { 180 int ret; 181 182 /* 183 * USB device 初始化 184 * "/dev/u***d" 在awbl_hwconf_xxxxxx_u***d.h 中定义 185 */ 186 /* 初始化一个USB设备对象*/ 187 ret = aw_u***d_init(&__g_u***d_obj, 188 &__g_u***d_info, 189 NULL, 190 "/dev/u***d"); 191 if (ret != AW_OK) { 192 AW_ERRF(("__g_u***d_obj init failed: %d ", ret)); 193 } 194 /* 对USB设备添加自定义功能*/ 195 __demo_u***d_vendor(); 196 /* 启动USB设备*/ 197 ret = aw_u***d_start(&__g_u***d_obj); 198 if (ret != AW_OK) { 199 AW_ERRF(("__g_u***d_obj start failed: %d ", ret)); 200 } 201 202 return ; 203 } 创建USB Device自定义功能,在本例程中,调用aw_u***d_pipe_create函数为USB Devie的自定义功能创建两个管道,一个OUT管道和一个IN管道用于接收和发送数据,然后把这个功能添加进上文中初始化好的USB Device对象,最后创建一个任务用于接收数据并把数据发送回去。 126 /** 127 * rief 自定义类例程入口 128 */ 129 aw_local int __demo_u***d_vendor (void) 130 { 131 int ret; 132 133 /* initialize function */ 134 /* 初始化一个USB设备功能*/ 135 ret = aw_u***d_fun_init(&__g_ufun, 136 "AWORKS-USB", 137 &__g_info); 138 if (ret != AW_OK) { 139 return ret; 140 } 141 142 /* 创建一个IN管道 */ 143 ret = aw_u***d_pipe_create(&__g_ufun, &__g_in, &__g_in_info); 144 if (ret != AW_OK) { 145 return ret; 146 } 147 148 /* 创建一个OUT管道 */ 149 ret = aw_u***d_pipe_create(&__g_ufun, &__g_out, &__g_out_info); 150 if (ret != AW_OK) { 151 return ret; 152 } 153 154 /* 把功能添加到创建的USB设备实例中*/ 155 ret = aw_u***d_fun_add(&__g_u***d_obj, 0, &__g_ufun); 156 if (ret != AW_OK) { 157 return ret; 158 } 159 160 AW_SEMB_INIT(__g_semb, 0, AW_SEM_Q_PRIORITY); 161 /* 创建一个USB设备任务*/ 162 AW_TASK_INIT(__g_task, 163 "CUSTOM-USB", 164 10, 165 2048, 166 __custom_task, 167 &__g_ufun); 168 169 AW_TASK_STARTUP(__g_task); 170 171 return AW_OK; 172 } USB Device接收发送任务,当USB Device连接到PC端后,PC端检测到USB Device的设备信息和设置功能接口后,就会调用USB Device接口信息的回调函数__custom_alt_set,__custom_alt_set会发送信号量,通知USB Device接收发送任务USB Device已连接到PC端,然后进入等待数据接收状态,当收到数据后会取消阻塞状态,然后把收到的数据通过IN管道再发送出去: 74 /* 回调函数*/ 75 static int __custom_alt_set (struct aw_u***d_fun *p_fun, 76 bool_t set) 77 { 78 if (set) { 79 AW_SEMB_GIVE(__g_semb); 80 } else { 81 aw_u***d_pipe_reset(p_fun->p_obj, &__g_in); 82 aw_u***d_pipe_reset(p_fun->p_obj, &__g_out); 83 } 84 return AW_OK; 85 } 87 static void __custom_task (void *arg) 88 { 89 int ret; 90 void *p_buf; 91 92 p_buf = aw_u***_mem_alloc(__BUF_SIZE); 93 if (p_buf == NULL){ 94 aw_kprintf("aw_u***_mem_alloc error!"); 95 return ; 96 } 97 98 AW_FOREVER { 99 /* 等待USB设备连接*/ 100 while (!aw_fun_valid(&__g_ufun)) { 101 AW_INFOF(("custom u*** wait for connect. ")); 102 AW_SEMB_TAKE(__g_semb, AW_SEM_WAIT_FOREVER); 103 } 104 105 memset(p_buf,0,__BUF_SIZE); 106 /* 等待接收数据*/ 107 ret = aw_u***d_trans_sync(__g_ufun.p_obj, 108 &__g_out, 109 p_buf, 110 __BUF_SIZE, 111 0, 112 AW_SEM_WAIT_FOREVER); 113 if (ret > 0) { 114 aw_kprintf("[recv]:%s ", (char *)p_buf); 115 } 116 117 /* 把收到的数据发送回去 */ 118 ret = aw_u***d_trans_sync(__g_ufun.p_obj, 119 &__g_in, 120 p_buf, 121 ret, 122 0, 123 __WAIT_TIMEOUT); 124 } 125 } |
|
|
|
七、应用案例演示
把完整的例程拷贝到工程的user_code目录,并把USB Device入口函数demo_u***d_vendor_entry添加到main.c文件中: 1 #include "aworks.h" 2 #include "aw_vdebug.h" 3 #include "aw_led.h" 4 #include "aw_delay.h" 5 6 #define LED 0 7 extern void demo_u***d_vendor_entry (void); 8 9 int aw_main() 10 { 11 aw_kprintf(" Application Start. "); 12 13 demo_u***d_vendor_entry(); 14 15 while (1) { 16 aw_led_toggle(LED); 17 aw_mdelay(500); 18 } 19 20 return 0; 21 } 编译程序并把程序烧写进M1052开发板后,通过USB线连接开发板上的USB HOST1 和PC机,这时PC即会检测到有USB设备插入,在PC机上安装对应的驱动后,PC机识别到我们的自定义USB设备。 打开PC端串口打印: PC端上检测到USB Device的插入,此时因为PC端没有相应的驱动,所以检测到是未知设备: 在PC机上安装相应的驱动后,PC机识别到我们的自定义USB设备: 打开专用的USB通讯软件,选择要操作的端点,因为在本例程中创建了两个管道,一个IN管道一个OUT管道,所以在软件中会看到本USB设备有两个端点,选择后会出现两个窗口,一个是接收一个是发送。 在发送窗口输入想要发送的数据,点击发送,接收窗口就会收到响应的数据。 |
|
|
|
只有小组成员才能发言,加入小组>>
692 浏览 0 评论
1106 浏览 1 评论
2473 浏览 5 评论
2808 浏览 9 评论
移植了freeRTOS到STMf103之后显示没有定义的原因?
2646 浏览 6 评论
使用eim外接fpga可是端口一点反应都没有有没有大哥指点一下啊
653浏览 9评论
649浏览 7评论
请教大神怎样去解决iMX6Q在linux3.0.35内核上做AP失败的问题呢
781浏览 6评论
628浏览 5评论
668浏览 5评论
小黑屋| 手机版| Archiver| 电子发烧友 ( 湘ICP备2023018690号 )
GMT+8, 2024-11-22 02:29 , Processed in 0.784403 second(s), Total 82, Slave 62 queries .
Powered by 电子发烧友网
© 2015 bbs.elecfans.com
关注我们的微信
下载发烧友APP
电子发烧友观察
版权所有 © 湖南华秋数字科技有限公司
电子发烧友 (电路图) 湘公网安备 43011202000918 号 电信与信息服务业务经营许可证:合字B2-20210191 工商网监 湘ICP备2023018690号