概述
本文檔詳細(xì)介紹了如何使用匿名助手的上位機(jī)實(shí)現(xiàn)加速度計(jì)和陀螺儀數(shù)據(jù)的可視化顯示。內(nèi)容涵蓋了加速度計(jì)和陀螺儀的工作原理、上位機(jī)通信協(xié)議、數(shù)據(jù)處理流程以及具體的代碼實(shí)現(xiàn)。通過(guò)本文檔,讀者可以了解如何通過(guò)串口通訊協(xié)議將傳感器數(shù)據(jù)發(fā)送到上位機(jī),并進(jìn)行實(shí)時(shí)的曲線顯示和數(shù)據(jù)分析。
最近在弄ST和瑞薩RA的課程,需要樣片的可以加群申請(qǐng):615061293 。
視頻教學(xué)
[https://www.bilibili.com/video/BV1HU411S7Az/]
樣品申請(qǐng)
[https://www.wjx.top/vm/OhcKxJk.aspx#]
源碼下載
[https://download.csdn.net/download/qq_24312945/89590386]
硬件準(zhǔn)備
首先需要準(zhǔn)備一個(gè)開發(fā)板,這里我準(zhǔn)備的是自己繪制的開發(fā)板,需要的可以進(jìn)行申請(qǐng)。
主控為STM32H503CB,陀螺儀為L(zhǎng)SM6DSOW,磁力計(jì)為L(zhǎng)IS2MDL。
參考坐標(biāo)系
加速度計(jì)工作方式
假設(shè)立方體在外太空,那里的一切都是失重的,球會(huì)簡(jiǎn)單地漂浮在立方體的中心。
現(xiàn)在,假設(shè)每面墻代表一個(gè)特定的軸。
如果我們突然以 1g 的加速度向左移動(dòng)盒子(單個(gè) G 力 1g 相當(dāng)于重力加速度 9.8 m/s 2),球無(wú)疑會(huì)撞到墻壁 X。如果我們測(cè)量球?qū)Ρ谑┘拥牧,我們可以得到沿X軸的輸出值為1g。
讓我們看看當(dāng)我們把那個(gè)立方體放在地球上時(shí)會(huì)發(fā)生什么。球?qū)⒑?jiǎn)單地落在墻 Z 上,施加 1g 的力,如下圖所示:
在這種情況下,盒子沒有移動(dòng),但我們?nèi)匀辉?Z 軸上得到 1g 的讀數(shù)。這是因?yàn)橹亓Γ▽?shí)際上是加速度的一種形式)以 1g 的力向下拉球。
雖然此模型并不完全代表真實(shí)世界的加速度計(jì)傳感器是如何構(gòu)建的,但它通常有助于理解為什么加速度計(jì)的輸出信號(hào)通常以 ±g 為單位指定,或者為什么加速度計(jì)在靜止時(shí)在 z 軸上讀數(shù)為 1g,或者您可以在不同方向上獲得什么樣的加速度計(jì)讀數(shù)。
上位機(jī)通訊
這里使用的是匿名助手的上位機(jī)
https://gitee.com/anotc/AnoAssistant
有專門的通訊協(xié)議
串口通訊協(xié)議格式如下所示,需要注意傳輸為小端模式傳輸。
對(duì)應(yīng)的源地址和目標(biāo)地址分別為0xFD和0xFE。
我們只需要上報(bào)加速度和陀螺儀數(shù)據(jù),所以功能碼為0x01,數(shù)據(jù)長(zhǎng)度為0x0D,需要主要為小端模式傳輸。
加速度演示
實(shí)測(cè)移動(dòng)模塊分別為X、Y、Z軸向下,可以看見數(shù)值基本上為1000mg。
加速度曲線顯示
設(shè)置數(shù)據(jù)列表->添加波形,可以添加加速度或者角速度曲線
可以通過(guò)右下角設(shè)置具體顯示。
陀螺儀工作方式
加速度計(jì)測(cè)量線性加速度,而陀螺儀測(cè)量角旋轉(zhuǎn)。為此,他們測(cè)量了科里奧利效應(yīng)產(chǎn)生的力。
陀螺儀是一種運(yùn)動(dòng)傳感器,能夠感測(cè)物體在一軸或多軸上的旋轉(zhuǎn)角速度。它能夠精確地感測(cè)自由空間中復(fù)雜的移動(dòng)動(dòng)作,因此成為追蹤物體移動(dòng)方位和旋轉(zhuǎn)動(dòng)作的必要設(shè)備。與加速計(jì)和電子羅盤不同,陀螺儀不需要依賴外部力量(如重力或磁場(chǎng)),可以自主地發(fā)揮其功能。因此,從理論上講,只使用陀螺儀就可以完成姿態(tài)導(dǎo)航的任務(wù)。
陀螺儀的每個(gè)通道檢測(cè)一個(gè)軸的旋轉(zhuǎn)。也就是說(shuō)陀螺儀通過(guò)測(cè)量自身的旋轉(zhuǎn)狀態(tài),判斷出設(shè)備當(dāng)前運(yùn)動(dòng)狀態(tài),是向前、向后、向上、向下、向左還是向右呢,是加速(角速度)還是減速(角速度)呢,都可以實(shí)現(xiàn),但是要判斷出設(shè)備的方位(東西南北),陀螺儀就沒有辦法。
MEMS陀螺儀主要利用科里奧利力(旋轉(zhuǎn)物體在有徑向運(yùn)動(dòng)時(shí)所受到的切向力)原理,公開的微機(jī)械陀螺儀均采用振動(dòng)物體傳感角速度的概念,利用振動(dòng)來(lái)誘導(dǎo)和探測(cè)科里奧利力。
MEMS陀螺儀的核心是一個(gè)微加工機(jī)械單元,在設(shè)計(jì)上按照一個(gè)音叉機(jī)制共振運(yùn)動(dòng),通過(guò)科里奧利力原理把角速率轉(zhuǎn)換成一個(gè)特定感測(cè)結(jié)構(gòu)的位移。
兩個(gè)相同的質(zhì)量塊以方向相反的做水平震蕩。當(dāng)外部施加一個(gè)角速率,就會(huì)出現(xiàn)一個(gè)科氏力,力的方向垂直于質(zhì)量運(yùn)動(dòng)方向,如垂直方向箭頭所示。產(chǎn)生的科氏力使感測(cè)質(zhì)量發(fā)生位移,位移大小與所施加的角速率大小成正比,科氏力引起的電容變化即可計(jì)算出角速率大小。
科里奧利效應(yīng)指出,當(dāng)質(zhì)量 (m) 以速度 (v) 沿特定方向移動(dòng)并施加外部角速率 (Ω)(紅色箭頭)時(shí),科里奧利效應(yīng)會(huì)產(chǎn)生一個(gè)力(黃色箭頭),導(dǎo)致質(zhì)量垂直移動(dòng)。該位移的值與應(yīng)用的角速率直接相關(guān)。
上報(bào)源碼
/* USER CODE BEGIN 2 */
printf("HELLO!n");
HAL_GPIO_WritePin(CS1_GPIO_Port, CS1_Pin, GPIO_PIN_SET);
HAL_GPIO_WritePin(SA0_GPIO_Port, SA0_Pin, GPIO_PIN_RESET);
HAL_GPIO_WritePin(CS2_GPIO_Port, CS2_Pin, GPIO_PIN_SET);
HAL_Delay(100);
stmdev_ctx_t dev_ctx;
/* Initialize mems driver interface */
dev_ctx.write_reg = platform_write;
dev_ctx.read_reg = platform_read;
dev_ctx.mdelay = platform_delay;
dev_ctx.handle = &SENSOR_BUS;
/* Init test platform */
// platform_init();
/* Wait sensor boot time */
platform_delay(BOOT_TIME);
/* Check device ID */
lsm6dso_device_id_get(&dev_ctx, &whoamI);
printf("LSM6DSO_ID=0x%x,whoamI=0x%x",LSM6DSO_ID,whoamI);
if (whoamI != LSM6DSO_ID)
while (1);
/* Restore default configuration */
lsm6dso_reset_set(&dev_ctx, PROPERTY_ENABLE);
do {
lsm6dso_reset_get(&dev_ctx, &rst);
} while (rst);
/* Disable I3C interface */
lsm6dso_i3c_disable_set(&dev_ctx, LSM6DSO_I3C_DISABLE);
/* Enable Block Data Update */
lsm6dso_block_data_update_set(&dev_ctx, PROPERTY_ENABLE);
/* Set Output Data Rate */
lsm6dso_xl_data_rate_set(&dev_ctx, LSM6DSO_XL_ODR_52Hz);
lsm6dso_gy_data_rate_set(&dev_ctx, LSM6DSO_GY_ODR_52Hz);
/* Set full scale */
lsm6dso_xl_full_scale_set(&dev_ctx, LSM6DSO_2g);
lsm6dso_gy_full_scale_set(&dev_ctx, LSM6DSO_2000dps);
/* Configure filtering chain(No aux interface)
* Accelerometer - LPF1 + LPF2 path
*/
lsm6dso_xl_hp_path_on_out_set(&dev_ctx, LSM6DSO_LP_ODR_DIV_100);
lsm6dso_xl_filter_lp2_set(&dev_ctx, PROPERTY_ENABLE);
int16_t acc_int16[3] ={0,0,0};
int16_t gyr_int16[3] ={0,0,0};
float acc[3] = {0};
float gyr[3] = {0};
uint8_t data[21]={0};
data[0]=0xAB;//幀頭
data[1]=0xFD;//源地址
data[2]=0xFE;//目標(biāo)地址
data[3]=0x01;//功能碼ID
data[4]=0x0D;//數(shù)據(jù)長(zhǎng)度LEN
data[5]=0x00;//數(shù)據(jù)長(zhǎng)度LEN 13
uint8_t sumcheck = 0;
uint8_t addcheck = 0;
/* USER CODE END 2 */
/* Infinite loop */
/* USER CODE BEGIN WHILE */
while (1)
{
uint8_t reg;
/* Read output only if new xl value is available */
lsm6dso_xl_flag_data_ready_get(&dev_ctx, ®);
if (reg) {
/* Read acceleration field data */
memset(data_raw_acceleration, 0x00, 3 * sizeof(int16_t));
lsm6dso_acceleration_raw_get(&dev_ctx, data_raw_acceleration);
acceleration_mg[0] =
lsm6dso_from_fs2_to_mg(data_raw_acceleration[0]);
acceleration_mg[1] =
lsm6dso_from_fs2_to_mg(data_raw_acceleration[1]);
acceleration_mg[2] =
lsm6dso_from_fs2_to_mg(data_raw_acceleration[2]);
// printf( "Acceleration [mg]:%4.2ft%4.2ft%4.2frn",
// acceleration_mg[0], acceleration_mg[1], acceleration_mg[2]);
//匿名上位機(jī)
acc_int16[0]=(int16_t)(acceleration_mg[0]);
acc_int16[1]=(int16_t)(acceleration_mg[1]);
acc_int16[2]=(int16_t)(acceleration_mg[2]);
}
lsm6dso_gy_flag_data_ready_get(&dev_ctx, ®);
if (reg) {
/* Read angular rate field data */
memset(data_raw_angular_rate, 0x00, 3 * sizeof(int16_t));
lsm6dso_angular_rate_raw_get(&dev_ctx, data_raw_angular_rate);
angular_rate_mdps[0] =
lsm6dso_from_fs2000_to_mdps(data_raw_angular_rate[0]);
angular_rate_mdps[1] =
lsm6dso_from_fs2000_to_mdps(data_raw_angular_rate[1]);
angular_rate_mdps[2] =
lsm6dso_from_fs2000_to_mdps(data_raw_angular_rate[2]);
// printf("Angular rate [mdps]:%4.2ft%4.2ft%4.2frn",
// angular_rate_mdps[0], angular_rate_mdps[1], angular_rate_mdps[2]);
gyr_int16[0]=(int16_t)(angular_rate_mdps[0]/1000);
gyr_int16[1]=(int16_t)(angular_rate_mdps[1]/1000);
gyr_int16[2]=(int16_t)(angular_rate_mdps[2]/1000);
}
// lsm6dso_temp_flag_data_ready_get(&dev_ctx, ®);
// if (reg) {
// /* Read temperature data */
// memset(&data_raw_temperature, 0x00, sizeof(int16_t));
// lsm6dso_temperature_raw_get(&dev_ctx, &data_raw_temperature);
// temperature_degC =
// lsm6dso_from_lsb_to_celsius(data_raw_temperature);
// printf("Temperature [degC]:%6.2frn", temperature_degC);
// }
data[7]=acc_int16[0] >>8;//ACC_X
data[6]=acc_int16[0];
data[9]=acc_int16[1] >>8;//ACC_Y
data[8]=acc_int16[1];
data[11]=acc_int16[2] >>8;//ACC_Z
data[10]=acc_int16[2];
data[13]=gyr_int16[0] >>8;//GYR_X
data[12]=gyr_int16[0];
data[15]=gyr_int16[1] >>8;//GYR_Y
data[14]=gyr_int16[1];
data[17]=gyr_int16[2] >>8;//GYR_Z
data[16]=gyr_int16[2];
data[18]=0;
sumcheck = 0;
addcheck = 0;
for(uint16_t i=0; i < 19; i++)
{
sumcheck += data[i]; //從幀頭開始,對(duì)每一字節(jié)進(jìn)行求和,直到 DATA 區(qū)結(jié)束
addcheck += sumcheck; //每一字節(jié)的求和操作,進(jìn)行一次 sumcheck 的累加
}
data[19]=sumcheck;
data[20]=addcheck;
HAL_UART_Transmit(&huart1 , (uint8_t *)&data, 21, 0xFFFF);
HAL_Delay(10);
/* USER CODE END WHILE */
/* USER CODE BEGIN 3 */
}
/* USER CODE END 3 */
演示
旋轉(zhuǎn)X軸的加速度數(shù)據(jù)。
旋轉(zhuǎn)Y軸的加速度數(shù)據(jù)。
旋轉(zhuǎn)Z軸的加速度數(shù)據(jù)。
旋轉(zhuǎn)X軸的角速度數(shù)據(jù)。
旋轉(zhuǎn)Y軸的角速度數(shù)據(jù)。
旋轉(zhuǎn)Z軸的角速度數(shù)據(jù)。
審核編輯 黃宇
-
陀螺儀
+關(guān)注
關(guān)注
44文章
776瀏覽量
98486 -
上位機(jī)
+關(guān)注
關(guān)注
27文章
930瀏覽量
54694
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
評(píng)論