STM32+HAL库驱动ADXL345传感器(SPI协议)

ADXL345传感器简介

在这里插入图片描述
ADXL345 是 ADI 公司推出的基于 iMEMS 技术的 3 轴、数字输出加速度传感器。该加速度传感器的特点有:

a. 分辨率高。最高 13 位分辨率。

b. 量程可变。具有+/-2g, +/-4g, +/-8g, +/-16g 可变的测量范围。

c. 灵敏度高。最高达 3.9mg/LSB,能测量不到 1.0°的倾斜角度变化。

d. 功耗低。 40~145uA 的超低功耗,待机模式只有 0.1uA。

e. 尺寸小。整个 IC 尺寸只有 3mm5mm1mm, LGA 封装。

ADXL 支持标准的 I2C 或 SPI 数字接口,自带 32 级 FIFO 存储,并且内部有多种运动状态检测和灵活的中断方式等特性。
引脚如下所示:
在这里插入图片描述

实物

使用的是STM32F103ZET6单片机,使用了SPI2
主要接线为:CS接PB6;SCL接PB13;SDA接PB15;SDO接PB14。
在这里插入图片描述

STM32CubeMX配置

使用的是STM32F103ZET6单片机,使用SPI2,通过串口进行数据显示
时钟配置如下:
在这里插入图片描述

SPI配置

在这里插入图片描述

使用硬件SPI,根据ADXL345的SPI时序图,选择SPI的模式3,片选使用软件片选
在这里插入图片描述

片选引脚配置

片选使用的是PB6引脚
在这里插入图片描述

串口配置

在这里插入图片描述

特别注意(重点部分)

1.模块默认使用IIC通信,如果需要使用四线SPI通信,需要将模块的R4电阻取掉,如果不取,就抓不到时序。原因如下图所示:
在这里插入图片描述
模块将芯片的SDO引脚通过R4电阻直接连接到GND,如果不把R4电阻取掉,接线上去读时序是读不到信号的。
在这里插入图片描述
2.对于SPI采集,需要注意一下,有的MCU会把SPI1的SCK引脚用作调试引脚,导致使用逻辑分析仪抓取波形的时候,SCK引脚一直处于拉高状态,解决方案:A.将SCK引脚重映射去其他引脚;B.换一组SPI

核心代码

ADXL345.C的代码


#include "ADXL345.h"
#include "spi.h"
#include "stdio.h"
/*
 * 作者:PrairieOne
 * csdn:PrairieOne
 * 邮箱:prairieone1024@163.com
 * 嵌入式技术交流群:738655377
 */
#define SET_SPI_CS_H	HAL_GPIO_WritePin(SPI1_CS_GPIO_Port,SPI1_CS_Pin,GPIO_PIN_SET);
#define SET_SPI_CS_L	HAL_GPIO_WritePin(SPI1_CS_GPIO_Port,SPI1_CS_Pin,GPIO_PIN_RESET);

//写字节
void ADXL345_Write(uint8_t addr, uint8_t value)
{
	addr &= 0x3F;
	SET_SPI_CS_L;
	HAL_SPI_Transmit(&hspi2, &addr, 1, 10);
	HAL_SPI_Transmit(&hspi2, &value, 1, 10);
	SET_SPI_CS_H;
}
//读字节
void ADXL345_Rread(uint8_t addr, uint8_t *value)
{
	addr &= 0x3F;	
	addr |= (0x80);
	SET_SPI_CS_L;
	HAL_SPI_Transmit(&hspi2, &addr, 1, 10);
	HAL_SPI_Receive(&hspi2, value, 1, 10);
	SET_SPI_CS_H;
}

//读ID
uint8_t Get_Adxl345_ID(void)
{
	uint8_t DEVICEID = 0x00;
	uint8_t result = 0;
	ADXL345_Rread(DEVICEID, &result);

	return result;
}
//初始化
void ADXL345_Init(void)
{
	while (Get_Adxl345_ID() != 0xE5)//e5
	{
		printf("ADXL345 Init Fail:%x\r\n", Get_Adxl345_ID());
		HAL_Delay(1000);
	}

	ADXL345_Write(INT_ENABLE, 0x00);
	ADXL345_Write(DATA_FORMAT, 0x0B);
	ADXL345_Write(BW_RATE, 0x1A);
	ADXL345_Write(POWER_CTL, 0x08);
	ADXL345_Write(INT_ENABLE, 0x14);

	printf("ADXL345 Init Success\r\n");
}
//读取X轴数据
void ADXL345_XTest(void)
{
	short x;
	uint8_t addrxl = 0x32;
	uint8_t addrxh = 0x33;
	uint8_t xl, xh;

	ADXL345_Rread(addrxl, &xl);
	ADXL345_Rread(addrxh, &xh);

	x = (short)(((uint16_t)xh << 8) + xl);

	printf("X:%d\r\n", x);
}

//读取Y轴数据
void ADXL345_YTest(void)
{
	short Y;
	uint8_t addrYl = 0x34;
	uint8_t addrYh = 0x35;
	uint8_t Yl, Yh;

	ADXL345_Rread(addrYl, &Yl);
	ADXL345_Rread(addrYh, &Yh);

	Y = (short)(((uint16_t)Yh << 8) + Yl);

	printf("Y:%d\r\n", Y);
}
//读取Z轴数据
void ADXL345_ZTest(void)
{
	short Z;
	uint8_t addrZl = 0x34;
	uint8_t addrZh = 0x35;
	uint8_t Zl, Zh;

	ADXL345_Rread(addrZl, &Zl);
	ADXL345_Rread(addrZh, &Zh);

	Z = (short)(((uint16_t)Zh << 8) + Zl);

	printf("Z:%d\r\n", Z);
}


ADXL345.H的代码

#ifndef ADXL345_H_
#define ADXL345_H_

#include "main.h"
/*
 * 作者:PrairieOne
 * csdn:PrairieOne
 * 邮箱:prairieone1024@163.com
 * 嵌入式技术交流群:826251093
 */
 
/******************ADXL345寄存器命令定义**********************/
#define DEVICE_ID           0X00        //获取器件ID,0XE5
#define THRESH_TAP          0X1D        //敲击阀值
#define OFSX                0X1E	//x轴调整偏移值
#define OFSY                0X1F
#define OFSZ                0X20
#define DUR                 0X21
#define Latent              0X22
#define Window              0X23
#define THRESH_ACK          0X24
#define THRESH_INACT        0X25
#define TIME_INACT          0X26
#define ACT_INACT_CTL       0X27
#define THRESH_FF           0X28
#define TIME_FF             0X29
#define TAP_AXES            0X2A
#define ACT_TAP_STATUS      0X2B
#define BW_RATE             0X2C
#define POWER_CTL           0X2D
 
#define INT_ENABLE          0X2E
#define INT_MAP             0X2F
#define INT_SOURCE          0X30
#define DATA_FORMAT        0X31
#define DATA_X0            0X32
#define DATA_X1            0X33
#define DATA_Y0            0X34
#define DATA_Y1            0X35
#define DATA_Z0            0X36
#define DATA_Z1            0X37
#define FIFO_CTL            0X38
#define FIFO_STATUS         0X39
 
#define I_M_DEVID      ((uint8_t)0xE5) //器件ID=0XE5
 
//初始化
void ADXL345_Init(void);
//读取X轴数据
void ADXL345_XTest(void);
//读取Y轴数据
void ADXL345_YTest(void);
//读取Z轴数据
void ADXL345_ZTest(void);
#endif /* ADXL345_H_ */


main的代码

/*
 * 作者:PrairieOne
 * csdn:PrairieOne
 * 邮箱:prairieone1024@163.com
 * 嵌入式技术交流群:826251093
 */
int main(void)
{
  /* USER CODE BEGIN 1 */

  /* USER CODE END 1 */

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_USART1_UART_Init();
  MX_SPI2_Init();
  /* USER CODE BEGIN 2 */
	HAL_UART_Receive_IT(&huart1, (uint8_t *)&aRxBuffer, 1);
  /* USER CODE END 2 */
	ADXL345_Init();		
  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {          
		//读取X轴数据
		ADXL345_XTest();
		//读取Y轴数据
		ADXL345_YTest();
		//读取Z轴数据
		ADXL345_ZTest();
		HAL_Delay(1000);
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */
}

效果展示

初始化模块的时序图:回读E5代码正确
在这里插入图片描述
在这里插入图片描述
遇事不决,可问春风!

### STM32 加速度传感器使用教程 #### 硬件连接 对于STM32与加速度传感器之间的硬件连接,通常依赖于I2C或SPI接口来完成通信。当选用I2C协议时,需将SCL(串行时钟线)和SDA(串行数据线)分别对接到对应的引脚上,并确保电源和接地线路正确无误地接入[^1]。 ```c // I2C初始化配置示例 (假设使用HAL) void MX_I2C1_Init(void){ hi2c1.Instance = I2C1; hi2c1.Init.Timing = 0x20909CEC; // 设置定时参数以适应具体应用场景下的速率需求 hi2c1.Init.OwnAddress1 = 0; hi2c1.Init.AddressingMode = I2C_ADDRESSINGMODE_7BIT; hi2c1.Init.DualAddressMode = I2C_DUALADDRESS_DISABLE; hi2c1.Init.OwnAddress2 = 0; hi2c1.Init.GeneralCallMode = I2C_GENERALCALL_DISABLE; hi2c1.Init.NoStretchMode = I2C_NOSTRETCH_DISABLE; if(HAL_I2C_Init(&hi2c1)!= HAL_OK){ Error_Handler(); } } ``` #### 驱动代码 为了能够有效地操作加速度计并获取其测量得到的数值,在编写驱动程序的过程中应当先定义好相应的寄存器地址以及命令字节等基本信息。接着通过调用`HAL_I2C_Master_Transmit()`函数向设备发送指令请求读取特定寄存器中的内容,再利用`HAL_I2C_Master_Receive()`接收返回的数据包。 ```c #define LSM6DS3_ACC_GYRO_WHO_AM_I_REG 0X0F #define LSM6DS3_ACC_GYRO_OUT_X_L_A 0X28 uint8_t buffer[6]; int16_t acc_x,acc_y,acc_z; if(HAL_I2C_Mem_Read(&hi2c1,LSM6DS3_ADDR<<1,LSM6DS3_ACC_GYRO_OUT_X_L_A,I2C_MEMADD_SIZE_8BIT, &buffer[0],6,100)==HAL_OK){ acc_x=((int16_t)(buffer[1]<<8)|buffer[0]); acc_y=((int16_t)(buffer[3]<<8)|buffer[2]); acc_z=((int16_t)(buffer[5]<<8)|buffer[4]); printf("Accel X:%d\r\n",acc_x); printf("Accel Y:%d\r\n",acc_y); printf("Accel Z:%d\r\n",acc_z); }else{ Error_Handler(); } ``` #### 示例项目 创建一个简单的工程文件夹结构,其中包含了源码目录(src),头文件目录(inclue),资源文件夹(Resources)以及其他必要的组成部分。在这个基础上构建起整个项目的框架之后就可以着手实现具体的逻辑功能了。比如设置中断服务例程ISR用来周期性的触发采样动作从而获得连续不断的运动状态变化情况报告给主机端显示出来或者进一步做其他处理分析工作.
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

电子阿星

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值