STM32-USB学习系列(五):USB复合设备的实现(MSC + HID)_stm32usb复合设备-程序员宅基地

技术标签: STM32  复合设备  MSC+HID  USB  # USB  

目录

一、整体步骤

二、USB的整体大致的初始化流程

三、USB复合设备的工程添加

四、USB复合设备文件与配置的修改

1、USB复合配置描述符的修改

2、USB复合设备类的函数实现

3、usbd_conf.c 与 usbd_conf.h的修改

(1)usbd_conf.c 中修改 USBD_LL_Init

(2)usbd_conf.h 中 主要修改的是接口数量

(3)usb_device.c 中的修改

五、注意事项

1、实现USB复合设备中遇到的一些问题

2、USB HID 中 USBD_HID_SendReport的修改


一、整体步骤

  1. 首先先使用STM32CubeMX生成 MSC 与 HID 的模版,然后使用其中一个模版进行工程整理
  2. 然后创建USB Composite的C文件,将MSC 与 HID 的Init、DeInit、DataIn、DataOut.....等函数进行整合,然后再编写MSC+HID的USB复合设备的配置描述符
  3. 修改usbd_conf.c 与 usbd_conf.h 中的内容,创建新端点的FIFO、最大接口数量等等

其中MSC与HID模版生成,请见前面文章:

STM32-USB学习系列(三):USB-MSC实现以SD卡为载体的U盘

STM32-USB学习系列(四):USB-HID模拟鼠标功能

二、USB的整体大致的初始化流程

三、USB复合设备的工程添加

建议:直接以MSC为模版,将HID类移放到模版中

四、USB复合设备文件与配置的修改

1、USB复合配置描述符的修改

其中USB的设备描述符不需要修改,使用默认生成的就可以。

配置描述符中需要修改的内容:

  • USB_DESC_TYPE_CONFIGURATION 配置返回的所有数据大小(自己计算新的配置描述符的整体长度
  • bNumInterfaces: 2 interface (配置描述符的接口由0x01该成0x02,因为有两个接口)
  • USBD_MSC_INTERFACE_NUM /* bInterfaceNumber: Number of Interface */                                      USBD_HID_INTERFACE_NUM  对应接口编号的修改
  • MSC与HID端点地址
uint8_t USBD_Composite_CfgFSDesc[USB_COMPOSITE_CONFIG_DESC_SIZ]  __ALIGN_END =
{
																		/* 配置描述符 */
  0x09,   /* bLength: Configuation Descriptor size */					/* 配置描述符的字节大小 */
  USB_DESC_TYPE_CONFIGURATION,   /* bDescriptorType: Configuration */	/* 配置描述符类型编号 0x02*/
  USB_COMPOSITE_CONFIG_DESC_SIZ,										/* 此配置返回的所有数据大小 (整个配置描述符长度加起来) */
  0x00,
  0x02,   /* bNumInterfaces: 2 interface */			/* 此配置所支持的接口数量: 2个接口(msc + hid) */
  0x01,  /* bConfigurationValue: */					/* Set_Configuration 命令所需要的参数值 */
  0x00,   /* iConfiguration: */						/* 描述该配置的字符串的索引值 */
  0xC0,   /* bmAttributes: */						/* 供电模式选择 (D7:总线供电 D6:自供电   D5:远程唤醒   D4..0:保留(复位为零))*/
  0x64,   /* MaxPower 200 mA */						/* 设备从总线提取的最大电流 ,每个单位为2mA(即: 50 = 100mA),USB2.0最大500mA*/
  /**************************************************** MSC ***************************************************/
  /********************  Mass Storage interface ********************/
  0x09,   /* bLength: Interface Descriptor size */			/* 接口字节数大小 */
  0x04,   /* bDescriptorType: */							/* 接口描述符的类型编号 */
  USBD_MSC_INTERFACE_NUM,   /* bInterfaceNumber: Number of Interface */		/* 接口的编号,第一个接口的编号为0 */
  0x00,   /* bAlternateSetting: Alternate setting */		/* 备用的接口描述符编号 */
  0x02,   /* bNumEndpoints*/								/* 该接口使用的端点数,不包括端点0 */
  0x08,   /* bInterfaceClass: MSC Class */					/* 该接口的类型 */
  0x06,   /* bInterfaceSubClass : SCSI transparent*/		/* 接口子类型 */
  0x50,   /* nInterfaceProtocol */							/* 接口遵循的协议 */
  0x05,          /* iInterface: */							/* 描述该接口的字符串索引值 */
  /********************  Mass Storage Endpoints ********************/
  0x07,   /*Endpoint descriptor length = 7*/				/* 端点描述符大小 */
  0x05,   /*Endpoint descriptor type */						/* 端点描述符的类型编号 */
  MSC_EPIN_ADDR,   /*Endpoint address (IN, address 1) */	/* USB设备的端点地址: 0x81 => 1000 00001(Bit7 表示方向 1/0 => In/Out ,Bit3~0表示端点号)*/
  0x02,   /*Bulk endpoint type */							/* 端点属性:(Bit1-0: 00控制,01同步,02批量,03中断) */
  LOBYTE(MSC_MAX_FS_PACKET),								/* 本端点接受或发送的最大信息包大小 */
  HIBYTE(MSC_MAX_FS_PACKET),
  0x00,   /*Polling interval in milliseconds */				/* 轮训数据传送端点的时间间隔,对于批量和控制的端点忽略,对于同步传送的端点必须为1,对于中断传送的端点,范围为1~255 */

  0x07,   /*Endpoint descriptor length = 7 */
  0x05,   /*Endpoint descriptor type */
  MSC_EPOUT_ADDR,   /*Endpoint address (OUT, address 1) */
  0x02,   /*Bulk endpoint type */
  LOBYTE(MSC_MAX_FS_PACKET),
  HIBYTE(MSC_MAX_FS_PACKET),
  0x00,   /*Polling interval in milliseconds*/
  /**************************************************** HID ***************************************************/
  /************** Descriptor of Joystick Mouse interface ****************/
  0x09,         /*bLength: Interface Descriptor size*/
  USB_DESC_TYPE_INTERFACE,/*bDescriptorType: Interface descriptor type*/
  USBD_HID_INTERFACE_NUM,			/*bInterfaceNumber: Number of Interface*/	/* 第二个接口的编号为1 */
  0x00,         /*bAlternateSetting: Alternate setting*/
  0x01,         /*bNumEndpoints*/
  0x03,         /*bInterfaceClass: HID*/
  0x01,         /*bInterfaceSubClass : 1=BOOT, 0=no boot*/
  0x02,         /*nInterfaceProtocol : 0=none, 1=keyboard, 2=mouse*/
  0,            /*iInterface: Index of string descriptor*/
  /******************** Descriptor of Joystick Mouse HID ********************/
  0x09,         /*bLength: HID Descriptor size*/
  HID_DESCRIPTOR_TYPE, /*bDescriptorType: HID*/
  0x11,         /*bcdHID: HID Class Spec release number*/
  0x01,
  0x00,         /*bCountryCode: Hardware target country*/
  0x01,         /*bNumDescriptors: Number of HID class descriptors to follow*/ /* 下级描述符的数目,只有1个HID报告描述符 */
  0x22,         /*bDescriptorType*/											   /* 下级描述符类型,HID报告类型 */
  HID_MOUSE_REPORT_DESC_SIZE,/*wItemLength: Total length of Report descriptor*/
  0x00,
  /******************** Descriptor of Mouse endpoint ********************/
  0x07,          /*bLength: Endpoint Descriptor size*/
  USB_DESC_TYPE_ENDPOINT, /*bDescriptorType:*/
  HID_EPIN_ADDR,     /*bEndpointAddress: Endpoint Address (IN)*/
  0x03,          /*bmAttributes: Interrupt endpoint*/
  HID_EPIN_SIZE, /*wMaxPacketSize: 4 Byte max */
  0x00,
  HID_FS_BINTERVAL,         /*bInterval: Polling Interval */
};

2、USB复合设备类的函数实现

uint8_t  USBD_Composite_Init (USBD_HandleTypeDef *pdev, uint8_t cfgidx)
{
	uint8_t res = 0;

	res = USBD_HID.Init(pdev, cfgidx);

	pdev->pUserData = &USBD_Storage_Interface_fops_FS;//MSC用户函数
	res = USBD_MSC.Init(pdev, cfgidx);


	return res;
}

uint8_t  USBD_Composite_DeInit (USBD_HandleTypeDef *pdev, uint8_t cfgidx)
{
	uint8_t res = 0;

	res +=  USBD_HID.DeInit(pdev,cfgidx);

	pdev->pUserData = &USBD_Storage_Interface_fops_FS;//MSC用户函数
	res +=  USBD_MSC.DeInit(pdev,cfgidx);

	return res;
}

uint8_t  USBD_Composite_Setup (USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req)
{
	printf("SetUp: req->wIndex: %d\r\n", req->wIndex);
	 switch(req->wIndex)
	 {
		 case USBD_MSC_INTERFACE_NUM:
			 pdev->pUserData = &USBD_Storage_Interface_fops_FS;
		 	 return (USBD_MSC.Setup(pdev, req));

		 case USBD_HID_INTERFACE_NUM:
		 	 return (USBD_HID.Setup(pdev, req));

		 default:
			 break;
	 }


	  return USBD_OK;
}

uint8_t  USBD_Composite_DataIn (USBD_HandleTypeDef *pdev, uint8_t epnum)
{
	switch(epnum)
	{
	case MSC_EP_NUM:
		pdev->pUserData = &USBD_Storage_Interface_fops_FS;
		return (USBD_MSC.DataIn(pdev, epnum));

	case HID_EP_NUM:
		return (USBD_HID.DataIn(pdev, epnum));

	default:
		break;
	}

	return USBD_OK;
}

uint8_t  USBD_Composite_DataOut (USBD_HandleTypeDef *pdev, uint8_t epnum)
{
	return (USBD_MSC.DataOut(pdev, epnum));
}

uint8_t  *USBD_Composite_GetFSCfgDesc (uint16_t *length)
{
	*length = sizeof (USBD_Composite_CfgFSDesc);
	return USBD_Composite_CfgFSDesc;
}

uint8_t  *USBD_Composite_GetDeviceQualifierDescriptor (uint16_t *length)
{
	*length = sizeof (USBD_Composite_DeviceQualifierDesc);
	return USBD_Composite_DeviceQualifierDesc;
}

3、usbd_conf.c 与 usbd_conf.h的修改

(1)usbd_conf.c 中修改 USBD_LL_Init

给新的端点配置FIFO通道:(其中USB-FS中的FIFO只有1.25KB大小,也就是设置不能超过0x140)

USBD_StatusTypeDef USBD_LL_Init(USBD_HandleTypeDef *pdev)
{
  /* Init USB Ip. */
  if (pdev->id == DEVICE_FS) {
  /* Link the driver to the stack. */
  hpcd_USB_OTG_FS.pData = pdev;
  pdev->pData = &hpcd_USB_OTG_FS;
  
  hpcd_USB_OTG_FS.Instance = USB_OTG_FS;
  hpcd_USB_OTG_FS.Init.dev_endpoints = 4;
  hpcd_USB_OTG_FS.Init.speed = PCD_SPEED_FULL;
  hpcd_USB_OTG_FS.Init.dma_enable = DISABLE;
  hpcd_USB_OTG_FS.Init.phy_itface = PCD_PHY_EMBEDDED;
  hpcd_USB_OTG_FS.Init.Sof_enable = DISABLE;
  hpcd_USB_OTG_FS.Init.low_power_enable = DISABLE;
  hpcd_USB_OTG_FS.Init.lpm_enable = DISABLE;
  hpcd_USB_OTG_FS.Init.vbus_sensing_enable = DISABLE;
  hpcd_USB_OTG_FS.Init.use_dedicated_ep1 = DISABLE;

  if (HAL_PCD_Init(&hpcd_USB_OTG_FS) != HAL_OK)
  {
    Error_Handler( );
  }

#if (USE_HAL_PCD_REGISTER_CALLBACKS == 1U)
  /* Register USB PCD CallBacks */
  HAL_PCD_RegisterCallback(&hpcd_USB_OTG_FS, HAL_PCD_SOF_CB_ID, PCD_SOFCallback);
  HAL_PCD_RegisterCallback(&hpcd_USB_OTG_FS, HAL_PCD_SETUPSTAGE_CB_ID, PCD_SetupStageCallback);
  HAL_PCD_RegisterCallback(&hpcd_USB_OTG_FS, HAL_PCD_RESET_CB_ID, PCD_ResetCallback);
  HAL_PCD_RegisterCallback(&hpcd_USB_OTG_FS, HAL_PCD_SUSPEND_CB_ID, PCD_SuspendCallback);
  HAL_PCD_RegisterCallback(&hpcd_USB_OTG_FS, HAL_PCD_RESUME_CB_ID, PCD_ResumeCallback);
  HAL_PCD_RegisterCallback(&hpcd_USB_OTG_FS, HAL_PCD_CONNECT_CB_ID, PCD_ConnectCallback);
  HAL_PCD_RegisterCallback(&hpcd_USB_OTG_FS, HAL_PCD_DISCONNECT_CB_ID, PCD_DisconnectCallback);

  HAL_PCD_RegisterDataOutStageCallback(&hpcd_USB_OTG_FS, PCD_DataOutStageCallback);
  HAL_PCD_RegisterDataInStageCallback(&hpcd_USB_OTG_FS, PCD_DataInStageCallback);
  HAL_PCD_RegisterIsoOutIncpltCallback(&hpcd_USB_OTG_FS, PCD_ISOOUTIncompleteCallback);
  HAL_PCD_RegisterIsoInIncpltCallback(&hpcd_USB_OTG_FS, PCD_ISOINIncompleteCallback);
#endif /* USE_HAL_PCD_REGISTER_CALLBACKS */
  /*所有EP共享的Rx FiFo + Tx FiFo 不能超过1.25kB 即0x140*/
  HAL_PCDEx_SetRxFiFo(&hpcd_USB_OTG_FS, 0x80);
  HAL_PCDEx_SetTxFiFo(&hpcd_USB_OTG_FS, 0, 0x40);
  HAL_PCDEx_SetTxFiFo(&hpcd_USB_OTG_FS, 1, 0x40);
  HAL_PCDEx_SetTxFiFo(&hpcd_USB_OTG_FS, 2, 0x40);

  }
  return USBD_OK;
}

(2)usbd_conf.h 中 主要修改的是接口数量

/*---------- -----------*/
#define USBD_MAX_NUM_INTERFACES     2U
/*---------- -----------*/
#define USBD_MAX_NUM_CONFIGURATION     1U
/*---------- -----------*/
#define USBD_MAX_STR_DESC_SIZ     512U
/*---------- -----------*/
#define USBD_DEBUG_LEVEL     0U
/*---------- -----------*/
#define USBD_LPM_ENABLED     0U
/*---------- -----------*/
#define USBD_SELF_POWERED     1U
/*---------- -----------*/
#define MSC_MEDIA_PACKET     512U

/****************************************/
/* #define for FS and HS identification */
#define DEVICE_FS 		0
#define DEVICE_HS 		1

(3)usb_device.c 中的修改

  • MX_USB_DEVICE_Init 中 ,USBD_RegisterClass() 中改成自己的USB的复合类

  if (USBD_Init(&hUsbDeviceFS, &FS_Desc, DEVICE_FS) != USBD_OK)
  {
    Error_Handler();
  }
  if (USBD_RegisterClass(&hUsbDeviceFS, &USBD_COMPOSITE) != USBD_OK)
  {
    Error_Handler();
  }
  if (USBD_Start(&hUsbDeviceFS) != USBD_OK)
  {
    Error_Handler();
  }

五、注意事项

1、实现USB复合设备中遇到的一些问题

  • 首先,使用USB-MSC 读写SD卡的过程中,如果SD卡没有使用中断,则需要确保USB属于高优先级。否则在读写SD过程中,被其他中断打断,会导致掉盘。如果使用了SD的中断或者读写DMA中断,其中优先级: SD > SD DMA > USB
  • 其次,在HID 的发送报文函数,需要做一些修改。不然会产生USB复合设备只能使用一种功能的现象

2、USB HID 中 USBD_HID_SendReport的修改

主要是将 HID 状态的判断给去掉了。当USB已经配置好后,直接通过对应的端点地址发送HID的报文就行了!

uint8_t USBD_HID_SendReport     (USBD_HandleTypeDef  *pdev,
                                 uint8_t *report,
                                 uint16_t len)
{
  USBD_HID_HandleTypeDef     *hhid = (USBD_HID_HandleTypeDef*)pdev->pClassData;

  if (pdev->dev_state == USBD_STATE_CONFIGURED )
  {
      USBD_LL_Transmit (pdev,
                        HID_EPIN_ADDR,
                        report,
                        len);
  }
  return USBD_OK;
}

 

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/laifengyuan1/article/details/108243522

智能推荐

python中文显示不出来_解决Python词云库wordcloud不显示中文的问题-程序员宅基地

文章浏览阅读2.6k次。解决Python词云库wordcloud不显示中文的问题2018-11-25背景:wordcloud是基于Python开发的词云生成库,功能强大使用简单。github地址:https://github.com/amueller/word_cloudwordcloud默认是不支持显示中文的,中文会被显示成方框。安装:安装命令:pip install wordcloud解决:经过测试发现不支持显示中文..._词云python代码无法输出文字

台式计算机cpu允许温度,玩游戏cpu温度多少正常(台式电脑夏季CPU一般温度多少)...-程序员宅基地

文章浏览阅读1.1w次。随着炎热夏季的到来,当玩游戏正爽的时候,电脑突然死机了,自动关机了,是不是有想给主机一脚的冲动呢?这个很大的原因是因为CPU温度过高导致的。很多新手玩家可能都有一个疑虑,cpu温度多少以下正常?有些说是60,有些说是70,到底多高CPU温度不会死机呢?首先我们先看看如何查看CPU的温度。下载鲁大师并安装,运行鲁大师软件,即可进入软件界面,并点击温度管理,即可看到电脑各个硬件的温度。鲁大师一般情况下..._台式机玩游戏温度多少正常

小白自学Python日记 Day2-打印打印打印!_puthon打印任务收获-程序员宅基地

文章浏览阅读243次。Day2-打印打印打印!我终于更新了!(哭腔)一、 最简单的打印最最简单的打印语句: print(“打印内容”)注意:python是全英的,符号记得是半角下面是我写的例子:然后进入power shell ,注意:你需要使用cd来进入你保存的例子的文件夹,保存时名字应该取为xxx.py我终于知道为什么文件夹取名都建议取英文了,因为进入的时候是真的很麻烦!如果你没有进入正确的文件夹..._puthon打印任务收获

Docker安装:Errors during downloading metadata for repository ‘appstream‘:_"cenerrors during download metadata for repository-程序员宅基地

文章浏览阅读1k次。centos8问题参考CentOS 8 EOL如何切换源? - 云服务器 ECS - 阿里云_"cenerrors during download metadata for repository \"appstream"

尚硅谷_谷粒学苑-微服务+全栈在线教育实战项目之旅_基于微服务的在线教育平台尚硅谷-程序员宅基地

文章浏览阅读2.7k次,点赞3次,收藏11次。SpringBoot+Maven+MabatisPlusmaven在新建springboot项目引入RELEASE版本出错maven在新建springboot项目引入RELEASE版本出错maven详解maven就是通过pom.xml中的配置,就能够从仓库获取到想要的jar包。仓库分为:本地仓库、第三方仓库(私服)、中央仓库springframework.boot:spring-boot-starter-parent:2.2.1.RELEASE’ not found若出现jar包下载不了只有两_基于微服务的在线教育平台尚硅谷

随便推点

网络学习第六天(路由器、VLAN)_路由和vlan-程序员宅基地

文章浏览阅读316次。路由的概念路由器它称之为网关设备。路由器就是用于连接不同网络的设备路由器是位于OSI模型的第三层。路由器通过路由决定数据的转发。网关的背景:当时每家计算机厂商,用于交换数据的通信程序(协议)和数据描述格式各不相同。因此,就把用于相互转换这些协议和格式的计算机称为网关。路由器与三层交换器的对比路由协议对比路由器的作用:1.路由寻址2.实现不同网络之间相连的功能3.通过路由决定数据的转发,转发策略称为 路由选择。VLAN相关技术什么是VLAN?中文名称叫:虚拟局域网。虚_路由和vlan

设置div背景颜色透明度,内部元素不透明_div设置透明度,里面的内容不透明-程序员宅基地

文章浏览阅读2.8w次,点赞6次,收藏22次。设置div背景颜色透明度,内部元素不透明:.demo{  background-color:rgba(255,255,255,0.15) } 错误方式:.demo{ background-color:#5CACEE;opacity:0.75;} 这样会导致div里面的元素内容和背景颜色一起变透明只针对谷歌浏览器的测试_div设置透明度,里面的内容不透明

Discuz!代码大全-程序员宅基地

文章浏览阅读563次。1.[ u]文字:在文字的位置可以任意加入您需要的字符,显示为下划线效果。2.[ align=center]文字:在文字的位置可以任意加入您需要的字符,center位置center表示居中,left表示居左,right表示居右。5.[ color=red]文字:输入您的颜色代码,在标签的中间插入文字可以实现文字颜色改变。6.[ SIZE=数字]文字:输入您的字体大小,在标签的中间插入文..._discuzcode 大全

iOS NSTimer定时器-程序员宅基地

文章浏览阅读2.6k次。iOS中定时器有三种,分别是NSTimer、CADisplayLink、dispatch_source,下面就分别对这三种计时器进行说明。一、NSTimerNSTimer这种定时器用的比较多,但是特别需要注意释放问题,如果处理不好很容易引起循环引用问题,造成内存泄漏。1.1 NSTimer的创建NSTimer有两种创建方法。方法一:这种方法虽然创建了NSTimer,但是定时器却没有起作用。这种方式创建的NSTimer,需要加入到NSRunLoop中,有NSRunLoop的驱动才会让定时器跑起来。_ios nstimer

Linux常用命令_ls-lmore-程序员宅基地

文章浏览阅读4.8k次,点赞17次,收藏51次。Linux的命令有几百个,对程序员来说,常用的并不多,考虑各位是初学者,先学习本章节前15个命令就可以了,其它的命令以后用到的时候再学习。1、开机 物理机服务器,按下电源开关,就像windows开机一样。 在VMware中点击“开启此虚拟机”。2、登录 启动完成后,输入用户名和密码,一般情况下,不要用root用户..._ls-lmore

MySQL基础命令_mysql -u user-程序员宅基地

文章浏览阅读4.1k次。1.登录MYSQL系统命令打开DOS命令框shengfen,以管理员的身份运行命令1:mysql -u usernae -p password命令2:mysql -u username -p password -h 需要连接的mysql主机名(localhost本地主机名)或是mysql的ip地址(默认为:127.0.0.1)-P 端口号(默认:3306端口)使用其中任意一个就OK,输入命令后DOS命令框得到mysql>就说明已经进入了mysql系统2. 查看mysql当中的._mysql -u user

推荐文章

热门文章

相关标签