• stm32之HAL库操作PAJ75620


    一、模块简介

            手势模块PAJ7620主要利用IIC或SPI协议来实现数据的传输,本实验用的模块是以IIC来进行信息传输。支持电压从2.8v到3.6v, 正常可以选择3.3v。检测的距离从5到15cm, 可以检测9种手势,包括

    • 右:编码为 0x01
    • 左:编码为 0x02
    • 上:编码为 0x04
    • 下:编码为 0x08
    • 前:编码为 0x10
    • 后:编码为 0x20
    • 顺时针:编码为 0x40
    • 逆时针:编码为 0x80
    • 挥动:编码为 0x0100

    1.1、PAJ7620 框图

            红色框是实验模块的引脚,无SPI,其中SDA,SCL 是IIC协议线,INT是手势结果输出引脚,输出结果时会触发低电平,所以可以利用轮询的方式检测该引脚或者利用外部中断的方式。

    1.2、模块原理图

    1.3、IIC 协议

            这里不过多描述,详细可参考51 iic  读写格式如下:

            PAJ7620的Slave ID 是0x73, 在使用时要左移一位,因为IIC中的第一个字节包括SlaveID + (R/W),R/W 确定读或写。

    1.4、寄存器

            PAJ7620寄存器也挺多,都是八位,但是每个寄存器功能比较单一。有两个主要的寄存器,BANK0和BANK1, 向0xEF 地址写0或写1分别选中BANK0或BANK1。每个BANK下又有好多个8位寄存器。比如:

    BANK0下包括手势中断的寄存器

    BANK1下包括模式选择寄存器,手势使用0x00

    二、代码

    根据上述的IIC读写格式,可以对应以下读写代码

    1. uint8_t iic_write_7620(uint8_t devAddress, uint8_t regAddress, uint8_t *data, uint16_t length) {
    2. uint16_t cnt;
    3. iic_start();
    4. iic_send_byte(devAddress << 1 | 0);
    5. if(iic_wait_ack() == NACK){
    6. iic_stop();
    7. return 1;
    8. }
    9. iic_send_byte(regAddress);
    10. if(iic_wait_ack() == NACK) {
    11. iic_stop();
    12. return 2;
    13. }
    14. for(cnt = 0; cnt < length; cnt++){
    15. iic_send_byte(data[cnt]);
    16. if(iic_wait_ack() == NACK) {
    17. iic_stop();
    18. return 3;
    19. }
    20. }
    21. iic_stop();
    22. return ACK;
    23. }
    24. uint8_t iic_read_7620(uint8_t devAddress, uint8_t regAddress, uint8_t *data, uint16_t length) {
    25. uint16_t cnt;
    26. iic_start();
    27. iic_send_byte(devAddress << 1 | 0);
    28. if(iic_wait_ack() == NACK) {
    29. iic_stop();
    30. return 1;
    31. }
    32. iic_send_byte(regAddress);
    33. if(iic_wait_ack() == NACK) {
    34. iic_stop();
    35. return 2;
    36. }
    37. iic_start();
    38. iic_send_byte(devAddress << 1 | 1);
    39. if(iic_wait_ack() == NACK) {
    40. iic_stop();
    41. return 3;
    42. }
    43. for(cnt = 0; cnt < length; cnt++){
    44. data[cnt] = iic_read_byte();
    45. if(cnt == length - 1) {
    46. iic_send_ack(NACK);
    47. } else {
    48. iic_send_ack(ACK);
    49. }
    50. }
    51. iic_stop();
    52. return 0;
    53. }

    2.1、获取手势的代码

    轮询

    1. while (1)
    2. {
    3. /* USER CODE END WHILE */
    4. /* USER CODE BEGIN 3 */
    5. if(paj7620_getInterrupt())
    6. {
    7. paj7620_action();
    8. }
    9. HAL_Delay(20);
    10. }

    中断(记得配置成下降沿触发)

    1. void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin) {
    2. paj7620_action();
    3. }

    手势处理代码

    1. void paj7620_action(void) {
    2. uint16_t gCode = 0;
    3. gCode = paj7620_get_gesture();
    4. printf("%d====》", gCode);
    5. switch (gCode) {
    6. case 0x01: // up
    7. paj7620_up();
    8. break;
    9. case 0x02: // down
    10. paj7620_down();
    11. break;
    12. case 0x04: // left
    13. paj7620_left();
    14. break;
    15. case 0x08: // right
    16. paj7620_right();
    17. break;
    18. case 0x10: // push
    19. paj7620_push();
    20. break;
    21. case 0x20: // pop
    22. paj7620_pop();
    23. break;
    24. case 0x40: // rotate right
    25. paj7620_rotate_right();
    26. break;
    27. case 0x80: // rotate left
    28. paj7620_rotate_left();
    29. break;
    30. case 0x100:// wave
    31. paj7620_wave();
    32. break;
    33. case 0x00: // nothing
    34. paj7620_nothing();
    35. break;
    36. default:
    37. paj7620_error();
    38. }
    39. }

    初始化代码

    1. uint8_t paj_init(void) {
    2. uint8_t data = 0, rtn = 0;
    3. uint16_t cnt = 0;
    4. delay_us(700); //Wait 700us for PAJ7620U2 to stabilize
    5. paj7620_sekect_bank(PAJ7620_BANK0);
    6. // 读0x00地址,正常会返回0x20
    7. rtn = iic_read_7620(PAJ7620_ADDRESS, PAJ7620_ADDR_PART_ID_0, &data, 1);
    8. if(rtn) {
    9. return rtn;
    10. }
    11. if(data != PAJ7620_PART_ID_0)
    12. {
    13. return 0xff;
    14. }
    15. // 这一步文档上没有找到说明
    16. rtn = iic_read_7620(PAJ7620_ADDRESS, PAJ7620_ADDR_PART_ID_1, &data, 1);
    17. if(rtn)
    18. {
    19. return rtn;
    20. }
    21. if(data != PAJ7620_PART_ID_1)
    22. {
    23. return 0xfe;
    24. }
    25. for (cnt = 0; cnt < INIT_REG_ARRAY_SIZE; cnt++)
    26. {
    27. rtn = iic_write_7620(PAJ7620_ADDRESS, initRegisterArray[cnt][0], &initRegisterArray[cnt][1], 1);
    28. if(rtn)
    29. {
    30. return rtn;
    31. }
    32. }
    33. paj7620_sekect_bank(PAJ7620_BANK1); //gesture flage reg in Bank1
    34. // 这一步文档上没有找到说明
    35. //data = 0xB7; // far mode 120 fps
    36. //data = 0x12; // near mode 240 fps
    37. //iic_write_7620(PAJ7620_ADDRESS, 0x65, &data, 1);
    38. paj7620_sekect_bank(PAJ7620_BANK0); //gesture flage reg in Bank0
    39. return 0;
    40. }

    三、效果

    手势小车

    手势目前用在了小车上,简单操作了小车的左右转向,前后进等。 

    完整版代码评论区留言。

  • 相关阅读:
    无人驾驶压路的应用之研华工控机ARK-1123助力
    测试一下Pinia,Vuex 要出局了?
    大数据专业毕业后职业前景如何?
    python深度学习入门-从零构建CNN和RNN
    RabbitMQ第一个实操小案例
    JAVA面经整理(8)
    JVM之jps虚拟机进程状态工具
    某果的一个小参数分析
    MapReduce Shuffle源码解读
    springboot+knife4j初体验
  • 原文地址:https://blog.csdn.net/TSC1235/article/details/133527572