STM32 HID 键盘

标签 stm32 stm32f4discovery stm32f0

我正在尝试将 STM32F0-disco 用作 Windows PC 的键盘。正在打印的字符有问题。

下面的代码等到板载按钮被按下,然后应该打印一次这三个字符。

    /* USER CODE BEGIN 2 */

USBDelay = USBD_HID_GetPollingInterval(&hUsbDeviceFS);
while (HAL_GPIO_ReadPin(B1_GPIO_Port, B1_Pin) == 0);
if (HAL_GPIO_ReadPin(B1_GPIO_Port, B1_Pin) == 1) {
    HAL_GPIO_TogglePin(LD3_GPIO_Port, LD3_Pin);


    buff[0] = 0x00;
    buff[2] = 0x04;
    buff[3] = 0x05;
    buff[4] = 0x06;
    USBD_HID_SendReport(&hUsbDeviceFS, buff, 8);
    HAL_Delay(USBDelay);

    buff[0] = 0x00;
    buff[2] = 0x00;
    buff[3] = 0x00;
    buff[4] = 0x00;

    USBD_HID_SendReport(&hUsbDeviceFS, buff, 8);
    HAL_GPIO_TogglePin(LD4_GPIO_Port, LD4_Pin);
    HAL_Delay(1000);
}
while (HAL_GPIO_ReadPin(B1_GPIO_Port, B1_Pin) == 1);

/* USER CODE END 2 */

我遇到的问题是缓冲区前两个关键位置中唯一的字符(这将是数组的第三个和第四个元素,因为元素一是修饰符而第二个元素是保留的)。此外,字符不是打印一次,而是连续打印而不停止。

根据我为测试目的添加的 LED,我要打印的字符在第二个 USB_SendReport 之前不会打印,它应该发送空白字符以停止打印。

我已对 usbd_hid.c 和 usbd_hid.h 以及时钟配置和启用 USB 等进行了必要的更改。我把我的键盘描述符放在下面。

    0x05, 0x01,                    // USAGE_PAGE (Generic Desktop)
0x09, 0x06,                    // USAGE (Keyboard)
0xa1, 0x01,                    // COLLECTION (Application)
0x05, 0x07,                    //   USAGE_PAGE (Keyboard)
0x19, 0xe0,                    //   USAGE_MINIMUM (Keyboard LeftControl)
0x29, 0xe7,                    //   USAGE_MAXIMUM (Keyboard Right GUI)
0x15, 0x00,                    //   LOGICAL_MINIMUM (0)
0x25, 0x01,                    //   LOGICAL_MAXIMUM (1)
0x75, 0x01,                    //   REPORT_SIZE (1)
0x95, 0x08,                    //   REPORT_COUNT (8)
0x81, 0x02,                    //   INPUT (Data,Var,Abs)
0x95, 0x01,                    //   REPORT_COUNT (1)
0x75, 0x08,                    //   REPORT_SIZE (8)
0x81, 0x03,                    //   INPUT (Cnst,Var,Abs)
0x95, 0x05,                    //   REPORT_COUNT (5)
0x75, 0x01,                    //   REPORT_SIZE (1)
0x05, 0x08,                    //   USAGE_PAGE (LEDs)
0x19, 0x01,                    //   USAGE_MINIMUM (Num Lock)
0x29, 0x05,                    //   USAGE_MAXIMUM (Kana)
0x91, 0x02,                    //   OUTPUT (Data,Var,Abs)
0x95, 0x01,                    //   REPORT_COUNT (1)
0x75, 0x03,                    //   REPORT_SIZE (3)
0x91, 0x03,                    //   OUTPUT (Cnst,Var,Abs)
0x95, 0x06,                    //   REPORT_COUNT (6)
0x75, 0x08,                    //   REPORT_SIZE (8)
0x15, 0x00,                    //   LOGICAL_MINIMUM (0)
0x25, 0x65,                    //   LOGICAL_MAXIMUM (101)
0x05, 0x07,                    //   USAGE_PAGE (Keyboard)
0x19, 0x00,                    //   USAGE_MINIMUM (Reserved (no event indicated))
0x29, 0x65,                    //   USAGE_MAXIMUM (Keyboard Application)
0x81, 0x00,                    //   INPUT (Data,Ary,Abs)
0xc0                           // END_COLLECTION

有什么想法吗?

最佳答案

USB HID 协议(protocol)的工作方式与您想象的不同。如果输入报告包含多个按下的键,主机(您的 Windows PC)会假定它们是同时按下的。所以它们可以按任何顺序插入。这也可能是第三个 key 未插入的原因。

因此您需要为每个键发送单独的输出报告。在最后一份报告之后,您需要发送另一个空报告(全为 0)以指示 key 已被释放(正如您已经在做的那样)。在同一键的两次按下之间也需要这样的额外报告。

连续插入 key 的原因表明没有收到空报告。尽管该函数使用术语发送 (USBD_HID_SendReport()),但 USB 协议(protocol)的工作方式使得主机需要获取输入报告。由于它是一个 USB 中断端点,因此这种情况只会经常发生。如果尚未提取报告,USBD_HID_SendReport() 不执行任何操作。它甚至不返回错误。

虽然查询轮询间隔 (USBD_HID_GetPollingInterval()) 起初看起来没问题,但它是 USB 端点描述符中声明的轮询间隔。但是,Windows 只支持几个间隔值并四舍五入到下一个支持的值。所以你必须增加值(value)。

关于STM32 HID 键盘,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/71847789/

相关文章:

c - FreeRTOS:osDelay 与 HAL_delay

使用stm32f407板控制无刷直流电机

struct - 读/写结构到闪存

embedded - STM32F4 在进行 DMA 传输 (RX) 时处理外设错误

c - STM32CubeIDE 在地址 "xxx"处中断,没有可用的调试信息,或在程序代码之外

c - 为 n 个 cpu 周期添加延迟/什么都不做的最佳方法

c - SPI写STM32很慢

c - STM32 SPI 发送

c - USART1中断IRQHandler如何接收字符串

stm32 - 校准 STM32 ADC (VREFINT)