HaaS EDU场景式应用整体介绍
手机上的“重力感应”想必大家都不陌生,基于重力感应可以实现飞车、辅助瞄准等各种体感非常强的交互功能。所谓的“重力感应”,其实就可以使用“加速度计及陀螺仪传感器”来实现。
本节实验中,我们将会基于 MPU-6050 实现一个根据开发板姿态滚动的小球。当我们倾斜 HaaS EDU K1 时,会看见小球根据我们的倾斜方向发生相对应的偏移。
开发用电脑一台
HAAS EDU K1 开发板一块 (开发板内置 MPU-6050 传感器)
USB2TypeC 数据线一根
"陀螺仪小球"功能已经包含在edu_demo应用中,并且包含在发布版本中。
固件版本:V1.0.0
git clone https://gitee.com/alios-things/AliOS-Things.git -b dev_3.1.0_haas
cd AliOS-Things/application/example/edu_demo/k1_apps/gyroscope
进入代码的顶层目录如AliOS-Things进行编译。直接编译application/example/目录下的edu_demo应用。
两种方法进行编译
命令行方式
aos make distclean
aos make edu_demo@haaseduk1 -c config
aos make
AliOS Studio IDE方式
见开发环境章节
MPU-60X0是世界上第一款集成 6 轴MotionTracking设备。它集成了3轴MEMS陀螺仪,3轴MEMS加速度计,以及一个可扩展的数字运动处理器 DMP( DigitalMotion Processor),可用I2C接口连接一个第三方的数字传感器,比如磁力计。
MPU-6050具有三个用于将陀螺仪输出数字化的16位模数转换器(ADC)和三个用于将加速度计输出数字化的16位ADC。 为了精确跟踪快速和慢速运动,这些部件具有用户可编程的陀螺仪满量程范围,范围为±250,±500,±1000和±2000°/ sec(dps),以及用户可编程的加速度计满量程范围 范围为±2g,±4g,±8g和±16g。使用400kHz的I2C与设备的所有寄存器进行通信。其他功能包括嵌入式温度传感器和片上振荡器,在整个工作温度范围内误差±1%。
更多硬件规格请参考 DataSheet。
陀螺仪由1850年法国物理学家莱昂·傅科在研究地球自传中获得灵感而发明出来的,类似像是把一个高速旋转的陀螺放到一个万向支架上,靠陀螺的方向来计算角速度,和现在小巧的芯片造型大相径庭。
早期的机械陀螺仪[1] |
目前常见的电子陀螺仪 |
那么如何将这么庞大的机械设备,塞进小小的芯片当中呢?以目前广泛使用的MPU-6050为例,它属于传感MEMS分支。传感MEMS技术是指用微电子微机械加工出来的、用敏感元件如电容、压电、压阻、热电耦、谐振、隧道电流等来感受转换电信号的器件和系统。感兴趣的同学可以来这里学习亚德诺半导体的公开课 —— MEMS传感器2:加速,旋转——陀螺仪工作原理。
我们使用的MPU-6050是一款经典的MEMS陀螺仪,即硅微机电陀螺仪。MEMS(Micro-Electro-Mechanical System)是指集机械元素、微型传感器、微型执行器以及信号处理和控制电路、接口电路、通信和电源于一体的完整微型机电系统。绝大多数的MEMS陀螺仪依赖于相互正交的振动和转动引起的交变科里奥利力。在MPU-6050内部,存在一质量块,当器件上电后,会触发质量块以固定频率横向运动。当器件遭受外力具备加速度时,就会触发质量块的纵向运动,从而改变四周梳齿之间的距离,改变输出的电容,进而通过ADC将模拟信号转换为数字信号,输出给外部[2]。
在原理图中我们可以看出,器件使用I2C通讯接口。并且,器件支持使用过INT引脚,当数据到来时可以在该引脚上触发中断。需要注意的是,AD0引脚决定了器件地址的第 0 bit。当AD0连接高电平,即AD0 = 1,此时器件地址为0x69。
由DataSheet可知,MPU-6050采用的通讯方式为I2C。默认7bit设备地址:0x69 (DataSheet P33 9.2)
在 AliOS Things 中,开发者只需要关心器件的设备地址即可,因为只要知道了设备地址,读写地址也可以计算出,AliOS Things 会自动处理这些计算。如果我们需要为了 MPU-6050 初始化I2C接口,那么对应的代码为:
// components\peripherals\sensor\drv\drv_mag_honeywell_qmc5883l.c
static i2c_dev_t i2c_dev;
i2c_dev.port = 1; // HaaS EDK 默认使用I2C 1
i2c_dev.config.address_width = I2C_HAL_ADDRESS_WIDTH_7BIT;
i2c_dev.config.freq = I2C_BUS_BIT_RATES_400K;
i2c_dev.config.mode = I2C_MODE_MASTER;
i2c_dev.config.dev_addr = MPU_ADDR; // MPU_ADDR 0X69
hal_i2c_init(&i2c_dev);
一般,使用I2C通讯的器件,都是通过读写寄存器的方式来完成对设备的读取和配置,因此了解寄存器的分布就非常重要。由于 MPU-6050 的寄存器数量较多,建议读者们查阅 Regsiter Map 文档来获取这些信息。我们进列出部分较为关键的寄存器。
0x3B-0x40 三轴加速度寄存器 每轴数据2Byte
0x41-0x42 温度寄存器 2Byte
0x43-0x48 三轴陀螺仪寄存器 每轴数据2Byte
uint8_t MPU_Init(void)
器件初始化。
// components/peripherals/sensor/drv/drv_acc_gyro_inv_mpu6050.c
uint8_t MPU_Init(void)
{
uint8_t device_id = 0;
// 初始化I2C
i2c_dev.port = 1;
i2c_dev.config.address_width = I2C_HAL_ADDRESS_WIDTH_7BIT;
i2c_dev.config.freq = I2C_BUS_BIT_RATES_100K;
i2c_dev.config.mode = I2C_MODE_MASTER;
i2c_dev.config.dev_addr = MPU_ADDR;
hal_i2c_init(&i2c_dev);
// 复位MPU6050
MPU_Write_Byte(MPU_PWR_MGMT1_REG, 0X80);
aos_msleep(100);
// 唤醒MPU6050
MPU_Write_Byte(MPU_PWR_MGMT1_REG, 0X00);
// 设置陀螺仪传感器量程,±2000dps
MPU_Set_Gyro_Fsr(3);
// 设置加速度传感器量程,±2g
MPU_Set_Accel_Fsr(0);
// 关闭所有中断
MPU_Write_Byte(MPU_INT_EN_REG, 0X00);
// I2C主模式关闭
MPU_Write_Byte(MPU_USER_CTRL_REG, 0X00);
// 关闭FIFO
MPU_Write_Byte(MPU_FIFO_EN_REG, 0X00);
// 读取器件ID
device_id = MPU_Read_Byte(MPU_DEVICE_ID_REG);
// 如果读取的器件ID与预期一致
if (device_id == MPU_DEV_ID)
{
LOGI("APP", "MPU init OK\n");
//设置CLKSEL,PLL X轴为参考
MPU_Write_Byte(MPU_PWR_MGMT1_REG, 0X01);
//加速度与陀螺仪都工作
MPU_Write_Byte(MPU_PWR_MGMT2_REG, 0X00);
//设置采样率为50Hz
MPU_Set_Rate(50);
}
else
{
LOGI("APP", "MPU init Error -- %x\n", device_id);
return 1;
}
return 0;
}
void MPU_Get_Gyroscope(short *gx, short *gy, short *gz)
读取三轴陀螺仪数据。由Register Map得知,只需要从GYRO_XOUTH向后依次读出6个寄存器内容即可。
// components/peripherals/sensor/drv/drv_acc_gyro_inv_mpu6050.c
void MPU_Get_Gyroscope(short *gx, short *gy, short *gz)
{
uint8_t buf[6];
MPU_Read_Len(MPU_GYRO_XOUTH_REG, 6, buf);
*gx = ((u16)buf[0] << 8) | buf[1];
*gy = ((u16)buf[2] << 8) | buf[3];
*gz = ((u16)buf[4] << 8) | buf[5];
}
void MPU_Get_Accelerometer(short *ax, short *ay, short *az)
读取三轴加速度数据。由Register Map得知,只需要从ACCEL_XOUTH向后依次读出6个寄存器内容即可。
// components/peripherals/sensor/drv/drv_acc_gyro_inv_mpu6050.c
void MPU_Get_Accelerometer(short *ax, short *ay, short *az)
{
uint8_t buf[6];
MPU_Read_Len(MPU_ACCEL_XOUTH_REG, 6, buf);
*ax = ((u16)buf[0] << 8) | buf[1];
*ay = ((u16)buf[2] << 8) | buf[3];
*az = ((u16)buf[4] << 8) | buf[5];
}
本实验的应用较为简单,只需要读出加速度数据,并显示在屏幕上即可。OLED的相关接口中已经给出了丰富的绘图函数。具体实现如下。
void gyroscope_task(void)
{
while (1)
{
// 清除屏幕memory
OLED_Clear();
// 获取三轴加速度信息
MPU_Get_Accelerometer(&r_ax, &r_ay, &r_az);
// 画出固定的圆形边框
OLED_DrawCircle(66, 32, 10, 1, 1);
// 画出填充的圆
OLED_FillCircle(66 - r_ax / 250, 32 + r_ay / 500, 8, 1);
// 将屏幕memory显示出来
OLED_Refresh_GRAM();
// 暂停20ms
aos_msleep(20);
}
}
本实验展示的仅是六轴传感器的一个非常简单的应用。随着类似传感器的体积越来越小,精度越来越高,它们也被应用在各种消费类电子产品,如穿戴设备、手机上。
使用它们可以进行一些非常有趣的应用,例如,手环中经常会使用到的计步算法,运动状态检测算法,都是基于其中的六轴传感器数据。近年来,还有很多学术界的工作,使用手环中的六轴传感器来实现空中写字的识别。
除此之外,它也广泛应用于辅助定位、飞行设备的姿态检测,摄像机云台的水平保持等等。期待读者们能够发掘出更多有价值的使用场景。
[1] Gyroscope invented by Léon Foucault in 1852. Replica built by Dumoulin-Froment for the Exposition universelle in 1867. National Conservatory of Arts and Crafts museum, Paris.
By Stéphane Magnenat - Own work by uploader, subject public domain, Public Domain, https://commons.wikimedia.org/w/index.php?curid=4302903
[2] 图片来自 https://www.analog.com/cn/education/education-library/videos/5996766351001.html
如需更多技术支持,可加入钉钉开发者群,或者关注微信公众号
更多技术与解决方案介绍,请访问阿里云AIoT首页https://iot.aliyun.com/