概述
陀螺儀通??梢宰x取三個方向上的旋轉(zhuǎn),即繞X軸、Y軸和Z軸的旋轉(zhuǎn)。每個方向上的旋轉(zhuǎn)包括正向旋轉(zhuǎn)和反向旋轉(zhuǎn),因此一共有六個位置。這六個位置分別是:1.X軸正向旋轉(zhuǎn)、2.X軸反向旋轉(zhuǎn)、3.Y軸正向旋轉(zhuǎn)、4.Y軸反向旋轉(zhuǎn)、5.Z軸正向旋轉(zhuǎn)、6.Z軸反向旋轉(zhuǎn) 通過檢測陀螺儀在每個方向上的旋轉(zhuǎn),可以確定物體的旋轉(zhuǎn)姿態(tài)和方向,從而用于導(dǎo)航、飛行控制等應(yīng)用。
最近在弄ST和瑞薩RA的課程,需要樣片的可以加群申請:615061293 。
視頻教學(xué)
[https://www.bilibili.com/video/BV1jw41137Zo/
樣品申請
[https://www.wjx.top/vm/OhcKxJk.aspx#]
源碼下載
[https://download.csdn.net/download/qq_24312945/88719389]
生成STM32CUBEMX
用STM32CUBEMX生成例程,這里使用MCU為STM32WB55RG。 配置時鐘樹,配置時鐘為32M。
串口配置
查看原理圖,PB6和PB7設(shè)置為開發(fā)板的串口。
配置串口。
IIC配置
配置IIC為快速模式,速度為400k。
CS和SA0設(shè)置
串口重定向
打開魔術(shù)棒,勾選MicroLIB
在main.c中,添加頭文件,若不添加會出現(xiàn) identifier "FILE" is undefined報錯。
/* USER CODE BEGIN Includes */
#include "stdio.h"
/* USER CODE END Includes */
函數(shù)聲明和串口重定向:
/* USER CODE BEGIN PFP */
int fputc(int ch, FILE *f){
HAL_UART_Transmit(&huart1 , (uint8_t *)&ch, 1, 0xFFFF);
return ch;
}
/* USER CODE END PFP */
參考程序
[https://github.com/STMicroelectronics/lsm6dsv16x-pid/tree/main]
初始換管腳
由于需要向LSM6DSV16X_I2C_ADD_L寫入以及為IIC模式。
所以使能CS為高電平,配置為IIC模式。 配置SA0為高電平。
printf("123123123");
lsm6dsv16x_reset_t rst;
stmdev_ctx_t dev_ctx;
/* Initialize mems driver interface */
dev_ctx.write_reg = platform_write;
dev_ctx.read_reg = platform_read;
dev_ctx.handle = &SENSOR_BUS;
HAL_GPIO_WritePin(CS_GPIO_Port, CS_Pin, GPIO_PIN_SET);
HAL_GPIO_WritePin(SA0_GPIO_Port, SA0_Pin, GPIO_PIN_RESET);
獲取ID
可以向WHO_AM_I (0Fh)獲取固定值,判斷是否為0x70。
lsm6dsv16x_device_id_get為獲取函數(shù)。
對應(yīng)的獲取ID驅(qū)動程序,如下所示。
/* Wait sensor boot time */
platform_delay(BOOT_TIME);
/* Check device ID */
lsm6dsv16x_device_id_get(&dev_ctx, &whoamI);
printf("LSM6DSV16X_ID=0x%x,whoamI=0x%x",LSM6DSV16X_ID,whoamI);
if (whoamI != LSM6DSV16X_ID)
while (1);
復(fù)位操作
可以向CTRL3 (12h)的SW_RESET寄存器寫入1進(jìn)行復(fù)位。
lsm6dsv16x_reset_set為重置函數(shù)。
對應(yīng)的驅(qū)動程序,如下所示。
/* Restore default configuration */
lsm6dsv16x_reset_set(&dev_ctx, LSM6DSV16X_RESTORE_CTRL_REGS);
do {
lsm6dsv16x_reset_get(&dev_ctx, &rst);
} while (rst != LSM6DSV16X_READY);
BDU設(shè)置
在很多傳感器中,數(shù)據(jù)通常被存儲在輸出寄存器中,這些寄存器分為兩部分:MSB和LSB。這兩部分共同表示一個完整的數(shù)據(jù)值。例如,在一個加速度計中,MSB和LSB可能共同表示一個加速度的測量值。
連續(xù)更新模式(BDU = ‘0’):在默認(rèn)模式下,輸出寄存器的值會持續(xù)不斷地被更新。這意味著在你讀取MSB和LSB的時候,寄存器中的數(shù)據(jù)可能會因為新的測量數(shù)據(jù)而更新。這可能導(dǎo)致一個問題:當(dāng)你讀取MSB時,如果寄存器更新了,接下來讀取的LSB可能就是新的測量值的一部分,而不是與MSB相對應(yīng)的值。這樣,你得到的就是一個“拼湊”的數(shù)據(jù),它可能無法準(zhǔn)確代表任何實際的測量時刻。
塊數(shù)據(jù)更新(BDU)模式(BDU = ‘1’):當(dāng)激活BDU功能時,輸出寄存器中的內(nèi)容不會在讀取MSB和LSB之間更新。這就意味著一旦開始讀取數(shù)據(jù)(無論是先讀MSB還是LSB),寄存器中的那一組數(shù)據(jù)就被“鎖定”,直到兩部分都被讀取完畢。這樣可以確保你讀取的MSB和LSB是同一測量時刻的數(shù)據(jù),避免了讀取到代表不同采樣時刻的數(shù)據(jù)。
簡而言之,BDU位的作用是確保在讀取數(shù)據(jù)時,輸出寄存器的內(nèi)容保持穩(wěn)定,從而避免讀取到拼湊或錯誤的數(shù)據(jù)。這對于需要高精度和穩(wěn)定性的應(yīng)用尤為重要。
可以向CTRL3 (12h)的BDU寄存器寫入1進(jìn)行開啟。
對應(yīng)的驅(qū)動程序,如下所示。
/* Enable Block Data Update */
lsm6dsv16x_block_data_update_set(&dev_ctx, PROPERTY_ENABLE);
6D方向檢測功能配置
1.使能六維位置檢測功能,配置陀螺儀模塊的中斷引腳,使得當(dāng)六維傳感器檢測到某種運(yùn)動時,能夠通過中斷方式通知主控制器。
可以通過使能MD1_CFG (5Eh)中的INT1_6D開啟6D事件。
pin_int.sixd = PROPERTY_ENABLE;
lsm6dsv16x_pin_int1_route_set(&dev_ctx, &pin_int);
2.使能中斷,將中斷配置應(yīng)用到LSM6DSV16X模塊上。 FUNCTIONS_ENABLE (50h)可以開啟6D中斷。
在 LSM6DSV16X 傳感器的 TAP_CFG0 (56h) 寄存器中,LIR(Latched Interrupt Request)位用于控制中斷請求的鎖存。如果將 LIR 設(shè)為 0(默認(rèn)值),則表示中斷請求不鎖存;如果設(shè)為 1,則中斷請求被鎖存。鎖存中斷意味著一旦中斷被觸發(fā),它會保持其激活狀態(tài)直到被明確地通過軟件讀取或清除。這通常用于確保不會錯過任何中斷事件,特別是在多任務(wù)環(huán)境中,其中一個中斷可能會在處理另一個中斷時發(fā)生。
irq.enable = 1;
irq.lir = 1;
lsm6dsv16x_interrupt_enable_set(&dev_ctx, irq);
選擇濾波和角度閾值
濾波可以通過 TAP_CFG0 (56h) 設(shè)置低通濾波。
角度閾值可以通過 TAP_THS_6D (59h) 設(shè)置為60度,用于檢測姿態(tài)變化。
設(shè)置濾波和角度閾值可用以下函數(shù):
lsm6dsv16x_filt_sixd_feed_set(&dev_ctx, LSM6DSV16X_SIXD_FEED_LOW_PASS);
lsm6dsv16x_6d_threshold_set(&dev_ctx, LSM6DSV16X_DEG_60);
設(shè)置量程和速率
速率可以通過CTRL1 (10h)設(shè)置加速度速率。
設(shè)置加速度量程可以通過CTRL8 (17h)進(jìn)行設(shè)置。
設(shè)置加速度的量程和速率可以使用如下函數(shù)。
/* Set Output Data Rate.*/
lsm6dsv16x_xl_data_rate_set(&dev_ctx, LSM6DSV16X_ODR_AT_120Hz);
/* Set full scale */
lsm6dsv16x_xl_full_scale_set(&dev_ctx, LSM6DSV16X_2g);
獲取所有中斷源的狀態(tài)
判斷是否發(fā)生了六維傳感器事件,如果有發(fā)生,則將不同方向的事件標(biāo)志位按位組合起來,其中包括了傳感器的各種狀態(tài),比如 FIFO 狀態(tài)、自由落體、喚醒、六維方向、數(shù)據(jù)就緒等狀態(tài)。然后通過一系列的操作和寄存器讀寫,獲取嵌入式功能執(zhí)行狀態(tài)、傳感器中樞狀態(tài)等信息,并將這些信息也保存,最后將這些信息返回給主控。這些數(shù)據(jù)可以通過以下寄存器進(jìn)行設(shè)置。
ALL_INT_SRC (1Dh) 為所有中斷的源寄存器,D6D_IA變量包含了自由落體狀態(tài)的信息。
D6D_SRC (47h) 用于六維方向識別,該寄存器包含了用于檢測設(shè)備方向的相關(guān)信息。
每個位都對應(yīng)一個方向,用于表示設(shè)備是否在相應(yīng)的方向上運(yùn)動。例如,如果XH位和YL位都被置位,則表示設(shè)備正在以X軸正方向向上運(yùn)動,并且Y軸負(fù)方向朝下。通過讀取D6D_SRC寄存器的值,可以了解設(shè)備當(dāng)前的運(yùn)動方向,從而進(jìn)行相應(yīng)的處理。LSM6DSV16X傳感器中的方向檢測是基于重力加速度和線性加速度的,因此需要在使用之前進(jìn)行校準(zhǔn)和配置,以適應(yīng)不同的應(yīng)用場景。
通過 int32_t lsm6dsv16x_all_sources_get(stmdev_ctx_t *ctx, lsm6dsv16x_all_sources_t *val) 函數(shù)可以獲取所有中斷源的狀態(tài)。
lsm6dsv16x_all_sources_t status;
/* Read output only if new xl value is available */
lsm6dsv16x_all_sources_get(&dev_ctx, &status);
if (status.six_d) {
sixd_event_catched = (status.six_d < < 6) |
(status.six_d_zh < < 5) |
(status.six_d_zl < < 4) |
(status.six_d_yh < < 3) |
(status.six_d_yl < < 2) |
(status.six_d_xh < < 1) |
(status.six_d_xl < < 0);
}
發(fā)送相應(yīng)信息
檢查sixd_event_catched的第7位是否為1,如果第7位為1,表示檢測到了六維傳感器事件。
根據(jù)sixd_event_catched的具體數(shù)值,通過switch語句匹配不同的情況。每個情況對應(yīng)著不同的六維傳感器位置,并且針對每種情況都會生成相應(yīng)的消息并調(diào)用tx_com函數(shù)發(fā)送消息。
最后,在處理完六維傳感器事件后,將sixd_event_catched重置為0,以便下一次事件的檢測。
/* Infinite loop */
/* USER CODE BEGIN WHILE */
while (1)
{
/* wait forever (6D event handle in irq handler) */
lsm6dsv16x_all_sources_t status;
/* Read output only if new xl value is available */
lsm6dsv16x_all_sources_get(&dev_ctx, &status);
if (status.six_d) {
sixd_event_catched = (status.six_d < < 6) |
(status.six_d_zh < < 5) |
(status.six_d_zl < < 4) |
(status.six_d_yh < < 3) |
(status.six_d_yl < < 2) |
(status.six_d_xh < < 1) |
(status.six_d_xl < < 0);
}
if (sixd_event_catched & 0x40) {
switch (sixd_event_catched) {
case 0x48:
printf("6D Position A Y↑rn");
break;
case 0x41:
printf("6D Position B X↓rn");
break;
case 0x42:
printf("6D Position C X↑rn");
break;
case 0x44:
printf("6D Position D Y↓rn");
break;
case 0x60:
printf("6D Position E Z↑rn");
break;
case 0x50:
printf("6D Position F Z↓rn");
break;
}
sixd_event_catched = 0;
}
/* USER CODE END WHILE */
/* USER CODE BEGIN 3 */
}
/* USER CODE END 3 */
演示
審核編輯 黃宇
-
陀螺儀
+關(guān)注
關(guān)注
44文章
787瀏覽量
98814 -
stm32cubemx
+關(guān)注
關(guān)注
5文章
283瀏覽量
14867
發(fā)布評論請先 登錄
相關(guān)推薦
評論