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

智能推荐

MySQL数据库入侵及防御方法-程序员宅基地

文章浏览阅读521次。来自:http://blog.51cto.com/simeon/1981572作者介绍陈小兵,高级工程师,具有丰富的信息系统项目经验及18年以上网络安全经验,现主要从事网络安全及数据库技术研究工作。《黑客攻防及实战案例解析》《Web渗透及实战案例解析》《安全之路-Web渗透及实战案例解析第二版》《黑客攻防实战加密与解密》《网络攻防实战研究:漏洞利用与提权》作者,在国内多本学术期..._mysql 5.0.16入侵

SQL Server SSMS历史版本下载地址-程序员宅基地

文章浏览阅读135次。https://learn.microsoft.com/zh-cn/sql/ssms/release-notes-ssms?view=sql-server-ver16#previous-ssms-releases_sql server历史版本哪儿下

【狂神JAVA】MyBatis笔记_jdk1.7的mybatis-程序员宅基地

文章浏览阅读2.5k次。简介自学的【狂神JAVA】MyBatis分享自写源码和笔记,希望对大家有帮助本人配置jdk13.0.2 (jdk1.7以上均可)Maven 3.6.3MySQL 5.7.23 (mysql5.6以上均可)1. 配置官网文档: https://mybatis.org/mybatis-3/zh/getting-started.htmlpom.xml<?xml version="1.0" encoding="UTF-8"?><project xmlns="http://_jdk1.7的mybatis

学习笔记---分布式调度之xxlJob调度中心的启动源码解析_xxl 调度失败:执行器地址为空-程序员宅基地

文章浏览阅读913次。调度中心的代码启动源码是从:XxlJobAdminConfig 入口;直接进入: xxlJobScheduler.init();第一个: initI18n() 处理国际化;第二个:JobRegistryMonitorHelper.getInstance().start(); 创建启动后台线程来维护在线的执行器组下的机器列表,从上篇学习笔记—分布式调度之xxlJob执行器的启动源码解析可以..._xxl 调度失败:执行器地址为空

RS485/RS232串口通信实现源码_485代码-程序员宅基地

文章浏览阅读1.3w次,点赞3次,收藏72次。之前贴出了代码,但是源码已经找不到了;鉴于很多同学私信想要参考,找时间重新写了一个工程一、参考代码1.不方便下载的同学可以参考贴出来的源代码链接:RS485二、基本知识1.RS485通信讲解:读30001、30002两个寄存器,假设从机地址为1上位机(主机)发送下行报文:01 03 00 03 00 02 34 0B从机地址功能码寄存器起始地址读取寄存器个数CRC校验010300 0300 0285 ca010300 0400 0285 ca上_485代码

李开复揭密微软成功之道 寄语中国软件业(4)_在微软许多人都像我一样主动从事发现人才、跟踪人才和吸引人才的工作....-程序员宅基地

文章浏览阅读1k次。http://www.sina.com.cn 2005年04月07日 11:19 新浪科技  文/李开复  人才:微软的立业之本  微软公司把重视人才的管理理念视为公司的核心财富。在信息时代里,人才的价值尤为重要。在工业时代里,一个优秀技工和一个普通技工的效率差异可能是30%,但在信息时代里,一个高级程序员和一个普通程序员的效率差异可能高达10倍以上。 ad1= "打造校_在微软许多人都像我一样主动从事发现人才、跟踪人才和吸引人才的工作....

随便推点

数据结构实验5《基于哈夫曼树的数据压缩》_基于哈夫曼树的数据压缩算法c语言-程序员宅基地

文章浏览阅读2k次,点赞4次,收藏25次。(visual studio 2019可运行)输入及输出要求见《数据结构C语言(第二版)》严蔚敏版【本文仅用于啥都看不懂还想交作业选手】#include<iostream>#include<map>#include<string>#include<stdio.h>#include<memory.h>using namespace std;typedef struct{ char c; int weight; in_基于哈夫曼树的数据压缩算法c语言

Teams Bot App 代码解析_adaptivecards.declare<datainterface>(rawlearncard)-程序员宅基地

文章浏览阅读1w次。Teams Bot App 代码解析_adaptivecards.declare(rawlearncard).render(this.likecountobj)

Unity UGUI(三)RawImage(原始图像)_unity原始图像-程序员宅基地

文章浏览阅读2.5k次。RawImage(Script)Texture 纹理 要显示的图片,注意:图片类型可以是任何类型 Color 颜色 图片的主颜色 Material 材质 渲染材质 Raycast Target 光线投射目标 是否可接收射线碰撞事件检测 UV Rect UV矩形 显示效果:X、Y属性用于控制纹理左右..._unity原始图像

SpringBoot与分布式事务组件-程序员宅基地

文章浏览阅读2k次。随着互联网应用的复杂性增加,越来越多的公司选择使用微服务架构模式进行应用开发,将单体应用拆分成多个小型服务,每个服务部署在不同的服务器上。同时,为了提升系统的可用性、容错性和可扩展性,需要考虑分布式事务问题。本文将介绍 Spring Boot 在分布式事务中的一些实现方案,并给出相关原理。

小程序基础入门(黑马学习笔记)_黑马微信小程序笔记-程序员宅基地

文章浏览阅读2.8k次,点赞12次,收藏90次。权当学习笔记吧_黑马微信小程序笔记

SpringBoot的旅游网站的设计与实现 - 源码免费(私信领取)

采用Spring Boot框架进行后端开发,结合前端技术(如Vue.js、React等)进行页面设计,数据库采用MySQL进行数据存储,确保系统的稳定性和性能。本项目旨在设计并实现一个基于Spring Boot的旅游网站,为用户提供便捷的旅游信息查询、预订服务,以及旅游资讯分享功能,提升用户旅游体验。通过市场调研和用户需求分析,了解用户对旅游网站的需求和偏好,明确系统的功能和特点,确保系统能够满足用户的旅游需求。进行全面的系统测试,包括功能测试、性能测试、安全性测试和用户体验测试,确保系统的质量和可靠性。

推荐文章

热门文章

相关标签