# PMW3901MB
**Repository Path**: libdriver/pmw3901mb
## Basic Information
- **Project Name**: PMW3901MB
- **Description**: 适用于通用MCU和Linux系统的PMW3901MB全功能驱动
- **Primary Language**: Unknown
- **License**: MIT
- **Default Branch**: main
- **Homepage**: None
- **GVP Project**: No
## Statistics
- **Stars**: 0
- **Forks**: 0
- **Created**: 2025-01-07
- **Last Updated**: 2025-11-02
## Categories & Tags
**Categories**: Uncategorized
**Tags**: SPI, 光流量
## README
[English](/README.md) | [ 简体中文](/README_zh-Hans.md) | [繁體中文](/README_zh-Hant.md) | [日本語](/README_ja.md) | [Deutsch](/README_de.md) | [한국어](/README_ko.md)
## LibDriver PMW3901MB
[](/misra/README.md) [](https://www.libdriver.com/docs/pmw3901mb/index.html) [](/LICENSE)
PMW3901MB 是 PixArt Imaging 最新的光学导航芯片,采用远场光学技术设计,可实现空中导航。 它采用 28 引脚板载芯片 (COB) 封装,提供 X-Y 运动信息,工作范围从 80 毫米到无穷远。 它最适合运动检测的远场应用。
LibDriver PMW3901MB 是LibDriver推出的PMW3901MB 的全功能驱动,该驱动提供原始数据帧读取,导航数据读取,中断读取等功能并且它符合MISRA标准。
### 目录
- [说明](#说明)
- [安装](#安装)
- [使用](#使用)
- [example basic](#example-basic)
- [example frame](#example-frame)
- [example interrupt](#example-interrupt)
- [文档](#文档)
- [贡献](#贡献)
- [版权](#版权)
- [联系我们](#联系我们)
### 说明
/src目录包含了LibDriver PMW3901MB的源文件。
/interface目录包含了LibDriver PMW3901MB与平台无关的SPI总线模板。
/test目录包含了LibDriver PMW3901MB驱动测试程序,该程序可以简单的测试芯片必要功能。
/example目录包含了LibDriver PMW3901MB编程范例。
/doc目录包含了LibDriver PMW3901MB离线文档。
/datasheet目录包含了PMW3901MB数据手册。
/project目录包含了常用Linux与单片机开发板的工程样例。所有工程均采用shell脚本作为调试方法,详细内容可参考每个工程里面的README.md。
/misra目录包含了LibDriver MISRA代码扫描结果。
### 安装
参考/interface目录下与平台无关的SPI总线模板,完成指定平台的SPI总线驱动。
将/src目录,您使用平台的接口驱动和您开发的驱动加入工程,如果您想要使用默认的范例驱动,可以将/example目录加入您的工程。
### 使用
您可以参考/example目录下的编程范例完成适合您的驱动,如果您想要使用默认的编程范例,以下是它们的使用方法。
#### example basic
```C
#include "driver_pmw3901mb_basic.h"
uint8_t res;
float height;
float delta_x;
float delta_y;
uint32_t i, times;
pmw3901mb_motion_t motion;
height = 1.0f;
times = 3;
res = pmw3901mb_basic_init();
if (res != 0)
{
return 1;
}
...
for (i = 0; i < times; i++)
{
read:
res = pmw3901mb_basic_read(height, &motion, (float *)&delta_x, (float *)&delta_y);
if (res != 0)
{
(void)pmw3901mb_basic_deinit();
return 1;
}
/* check the result */
if (motion.is_valid == 1)
{
/* print the result */
pmw3901mb_interface_debug_print("pmw3901mb: %d/%d.\n", i + 1, times);
pmw3901mb_interface_debug_print("pmw3901mb: delta_x: %0.3fcm delta_y: %0.3fcm.\n", delta_x, delta_y);
pmw3901mb_interface_debug_print("pmw3901mb: raw_average is 0x%02X.\n", motion.raw_average);
pmw3901mb_interface_debug_print("pmw3901mb: raw_max is 0x%02X.\n", motion.raw_max);
pmw3901mb_interface_debug_print("pmw3901mb: raw_min is 0x%02X.\n", motion.raw_min);
pmw3901mb_interface_debug_print("pmw3901mb: observation is 0x%02X.\n", motion.observation);
pmw3901mb_interface_debug_print("pmw3901mb: shutter is 0x%04X.\n", motion.shutter);
pmw3901mb_interface_debug_print("pmw3901mb: surface quality is 0x%04X.\n\n", motion.surface_quality);
}
else
{
pmw3901mb_interface_delay_ms(500);
goto read;
}
/* delay 1000 ms */
pmw3901mb_interface_delay_ms(1000);
...
}
...
(void)pmw3901mb_basic_deinit();
return 0;
```
#### example frame
```C
#include "driver_pmw3901mb_frame.h"
uint8_t res;
uint32_t k, times;
uint32_t i, j;
static uint8_t gs_frame[35][35];
times = 3;
res = pmw3901mb_frame_init();
if (res != 0)
{
return 1;
}
...
for (k = 0; k < times; k++)
{
res = pmw3901mb_frame_read(gs_frame);
if (res != 0)
{
(void)pmw3901mb_frame_deinit();
return 1;
}
pmw3901mb_interface_debug_print("pmw3901mb: %d/%d.\n", k + 1, times);
/* print frame */
for (i = 0; i < 35; i++)
{
for (j = 0; j < 35; j++)
{
pmw3901mb_interface_debug_print("0x%02X ", gs_frame[i][j]);
}
pmw3901mb_interface_debug_print("\n");
}
pmw3901mb_interface_debug_print("\n");
/* delay 1000 ms */
pmw3901mb_interface_delay_ms(1000);
...
}
...
(void)pmw3901mb_frame_deinit();
return 0;
```
#### example interrupt
```C
#include "driver_pmw3901mb_interrupt.h"
uint8_t (*g_gpio_irq)(float height_m) = NULL;
static uint8_t gs_flag;
uint8_t res;
uint32_t i, times;
static void a_callback(pmw3901mb_motion_t *motion, float delta_x, float delta_y)
{
/* check the result */
if (motion->is_valid == 1)
{
/* print the result */
pmw3901mb_interface_debug_print("pmw3901mb: find interrupt.\n");
pmw3901mb_interface_debug_print("pmw3901mb: delta_x: %0.3fcm delta_y: %0.3fcm.\n", delta_x, delta_y);
pmw3901mb_interface_debug_print("pmw3901mb: raw_average is 0x%02X.\n", motion->raw_average);
pmw3901mb_interface_debug_print("pmw3901mb: raw_max is 0x%02X.\n", motion->raw_max);
pmw3901mb_interface_debug_print("pmw3901mb: raw_min is 0x%02X.\n", motion->raw_min);
pmw3901mb_interface_debug_print("pmw3901mb: observation is 0x%02X.\n", motion->observation);
pmw3901mb_interface_debug_print("pmw3901mb: shutter is 0x%04X.\n", motion->shutter);
pmw3901mb_interface_debug_print("pmw3901mb: surface quality is 0x%04X.\n\n", motion->surface_quality);
/* set flag */
gs_flag = 1;
}
}
...
times = 3;
res = gpio_interrupt_init();
if (res != 0)
{
return 1;
}
g_gpio_irq = pmw3901mb_interrupt_irq_handler;
...
res = pmw3901mb_interrupt_init(_callback);
if (res != 0)
{
(void)pmw3901mb_interrupt_deinit();
return 1;
}
...
for (i = 0; i < times; i++)
{
pmw3901mb_interface_debug_print("pmw3901mb: %d/%d.\n", i + 1, times);
gs_flag = 0;
while (gs_flag == 0)
{
pmw3901mb_interface_delay_ms(10);
}
...
}
...
(void)pmw3901mb_interrupt_deinit();
(void)gpio_interrupt_deinit();
g_gpio_irq = NULL;
return 0;
```
### 文档
在线文档: [https://www.libdriver.com/docs/pmw3901mb/index.html](https://www.libdriver.com/docs/pmw3901mb/index.html)。
离线文档: /doc/html/index.html。
### 贡献
请参考CONTRIBUTING.md。
### 版权
版权 (c) 2015 - 现在 LibDriver 版权所有
MIT 许可证(MIT)
特此免费授予任何获得本软件副本和相关文档文件(下称“软件”)的人不受限制地处置该软件的权利,包括不受限制地使用、复制、修改、合并、发布、分发、转授许可和/或出售该软件副本,以及再授权被配发了本软件的人如上的权利,须在下列条件下:
上述版权声明和本许可声明应包含在该软件的所有副本或实质成分中。
本软件是“如此”提供的,没有任何形式的明示或暗示的保证,包括但不限于对适销性、特定用途的适用性和不侵权的保证。在任何情况下,作者或版权持有人都不对任何索赔、损害或其他责任负责,无论这些追责来自合同、侵权或其它行为中,还是产生于、源于或有关于本软件以及本软件的使用或其它处置。
### 联系我们
请联系lishifenging@outlook.com。