ADXL345传感器

1. ADXL345传感器简介

ADXL345是ADI公司推出的基于iMEMS技术的3轴、数字输出加速度传感器。该传感器有最高13位的分辨率、具有±2/4/8/16g可变的测量范围、能测量不到1.0°的倾斜角度变化等特点。ADXL345支持标准的I2C或SPI数字接口,自带32级FIFO存储,并且内部有多种状态检测和灵活的中断方式等特性,ADXL345的检测轴如下图示:

STM32CubeMX系列|ADXL345传感器_ADXL345

当ADXL345沿检测轴正向加速时,它对正加速度进行检测。在检测重力时需要注意当检测轴的方向与重力的方向相反时检测到的是正加速度。下图列出了ADXL345在不同摆放方式时的输出对重力的影响:

STM32CubeMX系列|ADXL345传感器_ADXL345_02

ADXL345支持SPI和I2C两种通讯方式,本例程采用的是I2C方式连接,官方推荐的I2C连接电路如下图示:从图中可以看出ADXL345的连接比较简单,外围器件只需要2个电容。若SDO/ALTADDRESS接地,则ADXL345的地址为0x53(不含最低位);若SDO/ALTADDRESS接高,则ADXL345的地址为0x1D(不含最低位);

STM32CubeMX系列|ADXL345传感器_单片机_03

ADXL345的初始化步骤为:上电 --> 等待1.1ms --> 初始化命令序列 --> 结束,ADXL345正常工作;初始化序列最简单的只需要配置3个寄存器(如下图示),发送下图中的序列给ADXL345后,ADXL345即开始正常工作

STM32CubeMX系列|ADXL345传感器_ADXL345_04

ADXL345寄存器地址映射表:

STM32CubeMX系列|ADXL345传感器_STM32CubeMX_05

2. 硬件设计

D1指示灯用来提示系统运行状态,K_UP按键用来自动校准,TFTLCD模块用来显示传感器检测的三个方向加速度值和角度值



  • D1/D2指示灯
  • K_UP按键
  • USART1
  • TFTLCD模块
  • ADXL345


从电路图中可以看到ADXL345芯片的ADDR地址线接在3.3V上,所以ADXL345的器件地址是:0x1D(不包含最低位),因此写入为:0x3A,读取为:0x3B

STM32CubeMX系列|ADXL345传感器_STM32CubeMX_06

3. 软件设计

3.1 STM32CubeMX设置

  • RCC设置外接HSE,时钟设置为72M
  • PC0/PC1设置为GPIO推挽输出模式、上拉、高速、默认输出电平为高电平
  • PA0设置为GPIO输入模式、下拉模式
  • PA8设置为GPIO输入模式、下拉模式
  • USART1选择为异步通讯方式,波特率设置为115200Bits/s,传输数据长度为8Bit,无奇偶校验,1位停止位
  • 激活I2C2,选择标准传输模式,选择7位寻址地址,其余默认设置

STM32CubeMX系列|ADXL345传感器_STM32CubeMX_07


  • 激活FSMC,详细请参考​​TFTLCD显示​​章节的设置
  • 输入工程名,选择工程路径(不要有中文),选择MDK-ARM V5;勾选Generated periphera initialization as a pair of ‘.c/.h’ files per IP ;点击GENERATE CODE,生成工程代码

3.2 MDK-ARM编程

  • 添加按键驱动文件key.c和key.h,参考​​按键输入​​例程
  • 添加TFTLCD驱动文件tftlcd.c 和tftlcd.h,参考​​TFTLCD显示​​例程
  • 添加ADXL345芯片驱动文件adxl345.c和adxl345.h

/*ADXL345初始化函数:成功返回0,失败返回1*/
uint8_t ADXL345_Init(void){
uint8_t id,val;
HAL_I2C_Mem_Read(&hi2c2,ADXL_READ,DEVICE_ID,I2C_MEMADD_SIZE_8BIT,&id,1,0xff);
if(id ==0XE5){ //读器件ID,ADXL345的器件ID为0XE5
val = 0x2B; //低电平中断输出,13位全分辨率,输出数据右对齐,16g量程
HAL_I2C_Mem_Write(&hi2c2,ADXL_WRITE,DATA_FORMAT,I2C_MEMADD_SIZE_8BIT,&val,1,0xFF);
val = 0x0A; //数据输出速度为100Hz
HAL_I2C_Mem_Write(&hi2c2,ADXL_WRITE,BW_RATE,I2C_MEMADD_SIZE_8BIT,&val,1,0xFF);
val = 0x28; //链接使能,测量模式
HAL_I2C_Mem_Write(&hi2c2,ADXL_WRITE,POWER_CTL,I2C_MEMADD_SIZE_8BIT,&val,1,0xFF);
val = 0x00; //不使用中断
HAL_I2C_Mem_Write(&hi2c2,ADXL_WRITE,INT_ENABLE,I2C_MEMADD_SIZE_8BIT,&val,1,0xFF);
HAL_I2C_Mem_Write(&hi2c2,ADXL_WRITE,OFSX,I2C_MEMADD_SIZE_8BIT,&val,1,0xFF);
HAL_I2C_Mem_Write(&hi2c2,ADXL_WRITE,OFSY,I2C_MEMADD_SIZE_8BIT,&val,1,0xFF);
HAL_I2C_Mem_Write(&hi2c2,ADXL_WRITE,OFSZ,I2C_MEMADD_SIZE_8BIT,&val,1,0xFF);
return 0;
}
return 1;
}
/*读取ADXL345三个轴的数据*/
void ADXL345_RD_XYZ(short *x,short *y,short *z){
uint8_t buf[6];
HAL_I2C_Mem_Read(&hi2c2,ADXL_READ,DATA_X0,I2C_MEMADD_SIZE_8BIT,&buf[0],1,0xFF);
HAL_I2C_Mem_Read(&hi2c2,ADXL_READ,DATA_X1,I2C_MEMADD_SIZE_8BIT,&buf[1],1,0xFF);
HAL_I2C_Mem_Read(&hi2c2,ADXL_READ,DATA_Y0,I2C_MEMADD_SIZE_8BIT,&buf[2],1,0xFF);
HAL_I2C_Mem_Read(&hi2c2,ADXL_READ,DATA_Y1,I2C_MEMADD_SIZE_8BIT,&buf[3],1,0xFF);
HAL_I2C_Mem_Read(&hi2c2,ADXL_READ,DATA_Z0,I2C_MEMADD_SIZE_8BIT,&buf[4],1,0xFF);
HAL_I2C_Mem_Read(&hi2c2,ADXL_READ,DATA_Z1,I2C_MEMADD_SIZE_8BIT,&buf[5],1,0xFF);
*x=(short)(((uint16_t)buf[1]<<8)+buf[0]); //DATA_X1为高位有效字节
*y=(short)(((uint16_t)buf[3]<<8)+buf[2]); //DATA_Y1为高位有效字节
*z=(short)(((uint16_t)buf[5]<<8)+buf[4]); //DATA_Z1为高位有效字节
}
/*读取ADXL345的数据并做滤波处理,读times次再取平均值*/
void ADXL345_Read_Average(short *x,short *y,short *z,uint8_t times){
uint8_t i;
short tx,ty,tz;
*x=0; *y=0; *z=0;
if(times){
for(i=0;i<times;i++){
ADXL345_RD_XYZ(&tx,&ty,&tz);
*x+=tx; *y+=ty; *z+=tz;
HAL_Delay(5);
}
*x/=times; *y/=times; *z/=times;
}
}
/*ADXL345自动校准函数*/
void ADXL345_AUTO_Adjust(char *xval,char *yval,char *zval){
short tx,ty,tz;
uint8_t i, val;
short offx=0,offy=0,offz=0;
val = 0x00; //先进入休眠模式
HAL_I2C_Mem_Write(&hi2c2,ADXL_WRITE,POWER_CTL,I2C_MEMADD_SIZE_8BIT,&val,1,0xFF);
HAL_Delay(100);
val = 0x2B; //低电平中断输出,13位全分辨率,输出数据右对齐,16g量程
HAL_I2C_Mem_Write(&hi2c2,ADXL_WRITE,DATA_FORMAT,I2C_MEMADD_SIZE_8BIT,&val,1,0xFF);
val = 0x0A; //数据输出速度为100Hz
HAL_I2C_Mem_Write(&hi2c2,ADXL_WRITE,BW_RATE,I2C_MEMADD_SIZE_8BIT,&val,1,0xFF);
val = 0x28; //链接使能,测量模式
HAL_I2C_Mem_Write(&hi2c2,ADXL_WRITE,POWER_CTL,I2C_MEMADD_SIZE_8BIT,&val,1,0xFF);
val = 0x00; //不使用中断
HAL_I2C_Mem_Write(&hi2c2,ADXL_WRITE,INT_ENABLE,I2C_MEMADD_SIZE_8BIT,&val,1,0xFF);
HAL_I2C_Mem_Write(&hi2c2,ADXL_WRITE,OFSX,I2C_MEMADD_SIZE_8BIT,&val,1,0xFF);
HAL_I2C_Mem_Write(&hi2c2,ADXL_WRITE,OFSY,I2C_MEMADD_SIZE_8BIT,&val,1,0xFF);
HAL_I2C_Mem_Write(&hi2c2,ADXL_WRITE,OFSZ,I2C_MEMADD_SIZE_8BIT,&val,1,0xFF);
HAL_Delay(12);
for(i=0;i<10;i++){
ADXL345_Read_Average(&tx,&ty,&tz,10);
offx+=tx; offy+=ty; offz+=tz;
}
offx/=10; offy/=10; offz/=10;
*xval=-offx/4;
*yval=-offy/4;
*zval=-(offz-256)/4;
HAL_I2C_Mem_Write(&hi2c2,ADXL_WRITE,OFSX,I2C_MEMADD_SIZE_8BIT,(uint8_t *)xval,1,0xFF);
HAL_I2C_Mem_Write(&hi2c2,ADXL_WRITE,OFSY,I2C_MEMADD_SIZE_8BIT,(uint8_t *)yval,1,0xFF);
HAL_I2C_Mem_Write(&hi2c2,ADXL_WRITE,OFSZ,I2C_MEMADD_SIZE_8BIT,(uint8_t *)zval,1,0xFF);
}
/*计算ADXL345角度,x/y/表示各方向上的加速度分量,dir表示要获得的角度*/
short ADXL345_Get_Angle(float x,float y,float z,uint8_t dir){
float temp;
float res=0; //弧度值
switch(dir){
case 0: //0表示与Z轴的角度
temp=sqrt((x*x+y*y))/z;
res=atan(temp);
break;
case 1: //1表示与X轴的角度
temp=x/sqrt((y*y+z*z));
res=atan(temp);
break;
case 2: //2表示与Y轴的角度
temp=y/sqrt((x*x+z*z));
res=atan(temp);
break;
}
return res*180/3.14; //返回角度值
}
/*屏幕显示数字函数:x/y表示LCD显示的坐标位置*/
void ADXL_Show_num(uint16_t x,uint16_t y,short num,uint8_t mode){
uint8_t valbuf[3];
FRONT_COLOR=RED;

if(mode==0){ //mode为0,表示显示加速度值
if(num<0){ //num表示要显示的数据
num=-num;
LCD_ShowString(x,y,tftlcd_data.width,tftlcd_data.height,16,(uint8_t *)"-");
}
else{
LCD_ShowString(x,y,tftlcd_data.width,tftlcd_data.height,16,(uint8_t *)" ");
}
valbuf[0]=num/100+0x30;
valbuf[1]=num%100/10+0x30;
valbuf[2]=num%100%10+0x30;
LCD_ShowString(x+10,y,tftlcd_data.width,tftlcd_data.height,16,valbuf);
}
else{ //mode为1,表示显示角度值
if(num<0){
num=-num;
LCD_ShowString(x,y,tftlcd_data.width,tftlcd_data.height,16,(uint8_t *)"-");
}
else{
LCD_ShowString(x,y,tftlcd_data.width,tftlcd_data.height,16,(uint8_t *)" ");
}

valbuf[0]=num/10+0x30;
valbuf[1]='.';
valbuf[2]=num%10+0x30;
LCD_ShowString(x+10,y,tftlcd_data.width,tftlcd_data.height,16,valbuf);
}
}
/*数据处理函数*/
void data_pros(){
short x,y,z;
short xang,yang,zang;
uint8_t key;
ADXL345_Read_Average(&x,&y,&z,10);
ADXL_Show_num(60,120,x,0);
ADXL_Show_num(60,140,y,0);
ADXL_Show_num(60,160,z,0);
xang=ADXL345_Get_Angle(x,y,z,1);
yang=ADXL345_Get_Angle(x,y,z,2);
zang=ADXL345_Get_Angle(x,y,z,0);
ADXL_Show_num(60,180,xang,1);
ADXL_Show_num(60,200,yang,1);
ADXL_Show_num(60,220,zang,1);

key=KEY_Scan(0);
if(key==KEY_UP_PRES){ //按下KEY_UP启动校准
HAL_GPIO_WritePin(GPIOC, GPIO_PIN_1, GPIO_PIN_RESET); //LED2亮表示正在校准
ADXL345_AUTO_Adjust((char*)&x,(char*)&y,(char*)&z);
HAL_GPIO_WritePin(GPIOC, GPIO_PIN_1, GPIO_PIN_SET); //LED2灭表示正校准完成
}
}
  • 在main.c文件下编写ADXL345测试代码
int main(void){
HAL_Init();
SystemClock_Config();
MX_GPIO_Init();
MX_FSMC_Init();
MX_I2C2_Init();
MX_USART1_UART_Init();
/* USER CODE BEGIN 2 */
TFTLCD_Init();
FRONT_COLOR=BLACK;
LCD_ShowString(10,10,tftlcd_data.width,tftlcd_data.height,16,(uint8_t *)"ANDYXI STM32");
LCD_ShowString(10,30,tftlcd_data.width,tftlcd_data.height,16,(uint8_t *)"ADXL345 Test");
LCD_ShowString(10,90,tftlcd_data.width,tftlcd_data.height,16,(uint8_t *)"K_UP:ADXL345 Adjust");
LCD_ShowString(10,120,tftlcd_data.width,tftlcd_data.height,16,(uint8_t *)"X Val:");
LCD_ShowString(10,140,tftlcd_data.width,tftlcd_data.height,16,(uint8_t *)"Y Val:");
LCD_ShowString(10,160,tftlcd_data.width,tftlcd_data.height,16,(uint8_t *)"Z Val:");
LCD_ShowString(10,180,tftlcd_data.width,tftlcd_data.height,16,(uint8_t *)"X Ang:");
LCD_ShowString(10,200,tftlcd_data.width,tftlcd_data.height,16,(uint8_t *)"Y Ang:");
LCD_ShowString(10,220,tftlcd_data.width,tftlcd_data.height,16,(uint8_t *)"Z Ang:");

FRONT_COLOR=GREEN;
while(ADXL345_Init()){
printf("ADXL345 Error!\r\n");
LCD_ShowString(10,50,tftlcd_data.width,tftlcd_data.height,16,(uint8_t *)"ADXL345 Checked failed!");
HAL_Delay(200);
}
printf("ADXL345 OK!\r\n");
LCD_ShowString(10,50,tftlcd_data.width,tftlcd_data.height,16,(uint8_t *)"ADXL345 Checked Success!");
/* USER CODE END 2 */
while (1){
data_pros();
HAL_GPIO_TogglePin(GPIOC,GPIO_PIN_0);
HAL_Delay(200);
}
}

4. 下载验证

编译无误下载到开发板后,可以看到D1指示灯不断闪烁,触摸屏显示三个轴的加速度值和角度值

STM32CubeMX系列|ADXL345传感器_单片机_08


关注我的公众号,在公众号里发如下消息,即可获取相应的工程源代码:

玩转STM32CubeMX | ADXL345传感器

STM32CubeMX系列|ADXL345传感器_STM32CubeMX_09