首页 > 其他分享 >【ADXL373、ADXL372】超低功耗加速度计的驱动代码测试

【ADXL373、ADXL372】超低功耗加速度计的驱动代码测试

时间:2024-09-01 18:21:45浏览次数:15  
标签:int16 AccelData 低功耗 offset SPI 加速度计 ADXL372 OFFSET ADXL373

 

一、概述

        前言:基于对大G值加速度传感计的开发需求,我先后接触了ADXL375、ADXL373、ADXL372,其中ADXL375的示例代码比较丰富,另外两个相对较少,所以我后续就根据数据手册对ADXL373的驱动代码进行了编写(ADXL372的寄存器和ADXL373相似度极高),最终完成了对两种芯片的驱动编写。

        目的:本文将重点介绍ADXL373在SPI通信模式下的驱动代码结构以及编写过程。当然我最希望的还是读者可以避免单纯地复制粘贴别人的代码。(虽然我之前自己就是这样……)

二、ADXL373简要介绍

        对于汽车中,ADXL373的应用场景主要集中在会出现碰撞或者是某些可以导致大G值加速度的地方,也与脑震荡或者是脑创伤有关系。

官方资料网址:

ADXL373数据手册和产品信息 | Analog Devices

        我筛选出的一些主要特征: (都是有必要了解的)

        1、 ADXL373在200 mg/LSB比例因子时提供12位输出数据;

        2、ADXL373一款超低功耗、3轴、±400 g微机电系统(MEMS)加速度计,以2560 Hz输出数据速率(ODR)工作时功耗为19 μA;

        3、提供两种额外的较低功耗模式和中断驱动的唤醒特性,用于监控非活动期间的运动情况。在唤醒模式下,可以对加速度数据求均值以获取足够低的输出噪声,从而触发低g阈值。在即时导通模式下,ADXL373功耗为1.4 μA,同时连续监控冲击环境。当检测到冲击事件超过内部设定的阈值时,器件会切换到正常的工作模式,其速度非常快以便记录事件。

        我来简要分析一下,首先比例因子和多少位输出数据这是为了帮助我们进行数据换算的,在通过寄存器读到数据之后我们需要把寄存器内的数据转化为加速度(通常是以重力加速度g为单位);除了进行数据换算,还和我们的校准有非常大的关系,因为对于一个传感器而言如果数据不准确那么它无论有其他任何优点那么都是毫无意义的;除此之外,芯片的运行模式也是一个重点,根据手册可以知道ADXL373具备三种MODE,这在后文中会提到。

三、驱动的结构划分

        以我个人目前的理解,对于传感器的驱动开发而言分为四步:第一步,初始化启动;第二步,校准数据;第三步,读取数据;第四步,特色功能。

(一)通过SPI实现主机和传感器之间的沟通

        在这里默认已经掌握SPI的基础知识,以及完成了SPI基本库的搭建,因为这一部分几乎适用于所有会使用到SPI的地方,在本文中只涉及到与ADXL373开发的相关内容。

        我们需要先阅读datasheet里面关于SPI时序图解(SPI Timing Diagrams):

        通过上图我们可以确定SPI的MODE,以及对于寄存器的读与写操作具体怎么完成;

        其中:SPI_MODE = 0 ,传输字节时 MSB FIRST 最高位在前。第一个字节的最后一位为1时,表示进行SPI读Read;最后一位为0时表示SPI写Write;

        到这儿我们应该可以得出两个Function:unsigned int RegisterRead(uint8_t Address);和 void RegisterWrite(uint8_t Address, uint8_t data);


unsigned int readRegister(uint8_t thisRegister)
{
  uint8_t result = 0;   

  //ADXL373 COMMUNICATE BY SPI :|ADDRESS| = | A6~A0+RW |,………………

  thisRegister = thisRegister << 1;
  uint8_t dataToSend = thisRegister | RW_R;

  //digitalWrite(CS_PIN, LOW);
  digitalWrite(CS_PIN, LOW);  //Set the Chip Select pin low to start an SPI packet

  SPI.transfer(dataToSend);  // Tell device register to read from
  result = SPI.transfer(0x00); // Binary word read

  digitalWrite(CS_PIN, HIGH);  //Set CS high to close communcation

  return result;
}

void writeRegister(uint8_t thisRegister, uint8_t thisValue)
{
  //SPI :|ADDRESS| = | A6~A0+RW |,………………
 thisRegister = thisRegister << 1;

 uint8_t dataWrite = thisRegister & RW_W; // Combine the register address and WRITE command

 digitalWrite(CS_PIN,LOW); //Set CS pin low to signal SPI packet start

 SPI.transfer(dataWrite); // Transfer the register address, RW
 SPI.transfer(thisValue); // Transfer the value to write 

 digitalWrite(CS_PIN,HIGH); //Set the Chip Select pin high to signal the end of an SPI packet.

}

注意!这只是我应用在开发项目里面的源代码的节选复制,不具备直接复制使用的功能,而且我的代码是增加了MCP23017模块的,所以需要仔细斟酌,但是编写代码的思路是一致的。 

(二)初始化启动——运行模式

        初始化也就是我们见得非常多的 void XXX_init(void);函数,那么其实在我们自己编写驱动的时候往往可以给其附上参数:void  XXX_init(uint8_t Mode);可以根据自己的需要自由编排,怎么顺手怎么来。

        先搞清楚ADXL373具体有哪些MODE呢?具体模式又应该怎么配置呢?具体的得需要去仔细阅读数据手册,在手册里面详细记录了所有寄存器的信息以及配置要求。

        在这里我简要展示一下我写的 void init(void);其中“mcp”是因为我选用了MCP23017作为拓展IO进行SPI的片选,所以在初始化的时候调用的是MCP上的引脚。

void ADXL373_init(void)
{

  SPI.begin(); 
  SPI.setDataMode(SPI_MODE0); 
  SPI.setBitOrder(MSBFIRST);   

  mcp.pinMode(CS_ADXL373,OUTPUT); 

  // set the ADXL373 in the Measure Mode
  mcp.digitalWrite(CS_ADXL373, LOW); 

  adxl373.writeRegister(MEASR_CTRL, 0x04);     // Set Measurement Mode to 2560Hz bandwidth (0x04)
  adxl373.writeRegister(TIME_CTRL, 0x80);      // Set ODR to 5120 Hz
  
  adxl373.writeRegister(FIFO_CTL,0x02);
  adxl373.writeRegister(FIFO_SAMPLES,0xFF);  

  adxl373.writeRegister(REG_HPF, 0x00);


  adxl373.writeRegister(POWER_CTRL, 0x1F);     // Set full bandwidth measurement mode, HPF disabled (0x07

  byte ID_Check = adxl373.readRegister(0x02);       //√√√
  byte offset = adxl373.readRegister(0x20);

  mcp.digitalWrite(CS_ADXL373, HIGH);  
}

(三)获取数据 

        根据数据手册的内容,ADXL373的Data存储在六个连续的寄存器内:(黄色框里面是偏移寄存器,用来校准的

        那么在知道寄存器在的具体位置之后,只需要用SPI去读寄存器的内容就好了:(注意芯片是把每一个轴的数据都分成了两部分,一部分叫做高位H,另一部分叫做低位L,分别从这两个寄存器内读取数据之后,我们需要先把它们合并成一个12bit数,在对这个12bit的数进行正负号判定,因为它是以补码的形式储存的)

int16_t ADXL373_MCP23017::getValue(int axis)
{
  int16_t AccelData = 0;    //adxl373's data is 12bit
  int high, low; 

  if (axis == x_axis)
  {
    high = readRegister(XDATA_H);
    low = readRegister(XDATA_L);
  }
  else if (axis == y_axis)
  {
    high = readRegister(YDATA_H);
    low = readRegister(YDATA_L);
  }
  else if (axis == z_axis)
  {
    high = readRegister(ZDATA_H);
    low = readRegister(ZDATA_L);
  }

  /* according to the data_H and data_L, get the acceldata in 12bit */  
  AccelData = (high << 4) & 0x0FF0;
  low = low >>4;
  AccelData |= low;

  /* Make a sign judgment 12bit */
  if(AccelData & 0x800){
    AccelData = ~(AccelData);
    AccelData &= 0x0FFF;
    AccelData +=1;
    AccelData = -AccelData;
  }

  return AccelData;
}

(四)设置自动校准 

        对于ADXL373的校准主要是基于它的偏移曲线:

        这条曲线包含了三个轴的变化情况,千万不要觉得这几条曲线的总体趋势都差不多呀,有什么需要注意的呢?

        其实在我们最开始的启动过程中,大部分是水平校准也就是(加速度 X: 0g,Y: 0g,Z: 1g),所以我们假设芯片水平静止放置时,我们先把OFFSET的寄存器全部设置为0x00,然后读取一次数据(或者读取多次取平均值,目的都是为了知道在没有偏移量的情况下,我们传感器的测量情况)。

        假设是X: a, Y: b, Z: c 单位都是g,那么我们为了校准是不是就应该让每一个X轴的测量值都 - a ,每一个Y轴的测量值都 -b,每一个Z轴的测量值都 + (1- c)。而这个作用于每一个测量值,让它们都多同样的一截,或者都少同样的一截的这个量就是偏移量产生的作用。

        现在再回到偏移曲线图,横坐标代表OFFSET Register 内的值,纵坐标代表对应的偏移量(以LSB为单位),而我们在最开始的介绍中就提到 ADXL373的比例因子是200mg/LSB,把它带入到曲线内,我们可以得出纵坐标每一格对应的数值就是4g,0.2g*20=4g,所以OFFSET Register 内的值每+1,那么对应的数值就会加2g(注意我这里写的是横坐标加一)。

        为了实现自动校准的过程,我的思路是,先启动AXL373,再把OFFSET Register的值设置为0x00,然后再连续读取33个样本数据,算出平均值,在根据平均值设置OFFSET的偏移量:

void ADXL373_MCP23017::auto_Calibration(void)
{
  int n = 0;
  int16_t x373, y373, z373;

  //G_offset -> the flog ensure the process is started only once at a time
  if(G_offset){
    int16_t x, y, z;
    uint8_t X_OFFSET,Y_OFFSET,Z_OFFSET;

    /* first set the OFFSET into 0, 0, 0  */
    setOFFSET(0x00, 0x00, 0x00);
    
    /*Read 33 groups of data continuously*/
    while(n < offset_samples){
      x = getValue(x_axis);  //get the data of LSB--int16_t--12bit
      y = getValue(y_axis);
      z = getValue(z_axis);

      x373 += x;
      y373 += y;
      z373 += z;
      n++;   

      delay(10);   
    }

    /* Calculate their average value, and convert to offset value*/
    int16_t X = -(x373/offset_samples);   //turn to 0g
    int16_t Y = -(y373/offset_samples);   // .......0g
    int16_t Z = 5-(z373/offset_samples);  //........1g, 5LSB == 1g

    /* Use the X Y Z above to get the OFFSET */
    X_OFFSET = OFFSET_4bit(X, offsetCurve_X);
    Y_OFFSET = OFFSET_4bit(Y, offsetCurve_Y);
    Z_OFFSET = OFFSET_4bit(Z, offsetCurve_Z);

    /* reset the OFFSET register */
    setOFFSET(X_OFFSET,Y_OFFSET,Z_OFFSET);

    // Serial.println(OFFSET_4bit(X, offset_X));
    // Serial.println(OFFSET_4bit(Y, offset_Y));
    // Serial.println(OFFSET_4bit(Z, offset_Z));

    G_offset = false;
    
    delay(100);
  }
}

        其中OFFSET_4bit(X,offsetCurve_X);是我根据偏移曲线写的一个函数,它的作用就是输入一个X,也就是测得的未偏移的加速度大小,在对应的X的偏移量曲线下输出它的对应的OFFSETRegister_Value。在这里我没有找到合适的换算公式去完成换算,所以使用了数组去把曲线上对应的点描述出来,然后再把输入的X的值与数组的元素依次作比较,最后得出哪一个OFFSET_VALUE最理想。

/* Data reference points from the offset curve in the datasheet */
const int16_t offsetCurve_X[16] = {0,10,15,25,30,40,50,60,-60,-50,-40,-35,-30,-15,-10,-10};
const int16_t offsetCurve_Y[16] = {0,10,10,20,30,40,45,50,-60,-55,-50,-40,-30,-20,-20,-10};
const int16_t offsetCurve_Z[16] = {0,10,15,25,30,35,40,50,-60,-50,-40,-30,-25,-20,-15,-10};

uint8_t ADXL373_MCP23017::OFFSET_4bit(int16_t numberLSB, const int16_t *offset_array)
{
  uint8_t result = 0;
  if(numberLSB <= offset_array[8]){
    result = 0x08;
    return result;
  }
  if (numberLSB >= offset_array[7]){
    result = 0x07;
    return result;
  }
  for(uint8_t i = 0;i < 15; i++){
    if(numberLSB == offset_array[i]){
        result = i ;
        return result;
    }
    else if((numberLSB > offset_array[i])&&(numberLSB < offset_array[i+1])){
        result = i ;
        return result;      
    }      
  }
}

        OK,那么到这里的话也就完成了整个主体部分的讲解,由于我之前也没有找到直接适合Arduino的ADXL373的库,所以也给不了参考链接,但是主题思路已经展示完了。使用这些代码就可以实现基本的ADXL373获取数据。

四、ADXL373的实验测试

        对于ADXL373的测试我只用了跌落机进行操作,没有用冲击台,所以没办法得到很好的对比数据,只能通过跌落之后所测得的Z轴曲线来初步分析ADXL373的稳定性以及精准性。

        这里我展示其中一个,从一米的高度跌落,跌落至刚性材料的地面,测得的加速度曲线。请教了老师傅,这个加速度范围是合理的,峰值能达到60g左右。

      

五、总结

        对于我而言,ADXL373的驱动代码是我第一个真正意义上自己写的驱动,虽然是被迫的,因为没找到可以借用的库……但是也算是一种突破吧,也切实感受到自己写驱动可以带来的好处:如果习惯调用寄存器的话是可以自己很快编写出一个库的,而且代码可以做到尽可能的简洁,这有利于后续的开发与维护。

        挖坑:作为本篇文章的延续,计划把ADXL375、ADXL372以及它们的相关的实验测试数据给写一篇总结性的文章。

        如果您对我所介绍的内容有任何改进的建议也欢迎告诉我!如果本文对你有帮助的话,不妨点个赞。欢迎留言讨论问题,一起讨论问题、解决问题。

        

        

标签:int16,AccelData,低功耗,offset,SPI,加速度计,ADXL372,OFFSET,ADXL373
From: https://blog.csdn.net/m0_74001972/article/details/141760628

相关文章

  • 合宙低功耗4G模组Air780EQ——硬件设计手册02
    Air780EQ是一款基于移芯EC716E平台设计的LTECat1无线通信模组。支持FDD-LTE/TDD-LTE的4G远距离无线传输技术。另外,模组提供了USB/UART/I2C等通用接口满足IoT行业的各种应用诉求。本文将继续介绍合宙Air780EQ的硬件设计中的应用接口,射频接口,电气特性,结构尺寸和存储生产等内容。二......
  • 小尺寸BLE 5.2低功耗串口透传蓝牙模块 - ANS-BT103M
    ANS-BT103M是安朔科技自主开发的一款小尺寸BLE蓝牙5.2模块,它支持HID、GATT、ATT和其他配置文件,使用UART作为编程接口,用户可以使用AT命令通过UART读取或写入模块的配置,支持空中升级。支持蓝牙主从一体,一对多连接,透传速率可达60KB/s,支持定制开发。产品参数:模块型号      ......
  • ESP8684 系列芯片搭载 RISCV 32 位单核处理器的极低功耗 SoC 支持(2.4 GHz WiFi) 和 B
    ESP8684系列芯片搭载RISCV32位单核处理器的极低功耗SoC支持(2.4GHzWiFi)和Bluetooth5(LE)ESP8684系列芯片搭载RISCV32位单核处理器的极低功耗SoC支持IEEE802.11b/g/n(2.4GHzWiFi)和Bluetooth5(LE)在4×4mm的QFN封装中叠封1MB、2MB或4MBf......
  • 霍尔IC分享|高灵敏度、低压、低功耗的全极性霍尔传感器MST-MH251
    今天给大家分享的是全极性霍尔传感器,美加(MST)科技的霍尔IC。现在给大家分享一款低压、低功耗的全极性霍尔传感器MH251。MH251这款产品是COMS工艺,推挽输出,工作电压1.8到6V,功耗低至5uA,高灵敏度,ESD大于4KV,频率50HZ,工作温度-40℃到150℃;全极性(霍尔开关的丝印面与磁铁的N极S极都......
  • 超低功耗模式在合宙模组中的应用与配置
    随着物联网技术的飞速发展,设备功耗成为影响系统稳定性和续航能力的重要因素。合宙科技推出的多款模组,如Air780E&600E(EC618平台)、780EP系列(EC718平台)、以及780EL_780ET_700EL_700ET系列(EC716S平台),均支持超低功耗模式,以满足不同应用场景下的节能需求。本文将详细介绍两种主要的低......
  • STM32&低功耗与备用备份区域
    STM的备份备用区域其实就是两个区块:BKP和RTC。低功耗则其实是STM32四种模式中的三种耗能很低的模式。目录一:备用区域1.BKP2.RTC二:低功耗模式1.睡眠模式:2.停机模式:3.待机模式:一:备用区域1.BKPBKP就是一个备份寄存器,大小不是一定的。但基本单位都是16位。所谓的的备......
  • App Inventor 2 低功耗蓝牙 BlueToothLE 拓展中文文档(完整翻译加强版)
    低功耗蓝牙,也称为蓝牙LE或简称BLE,是一种类似于经典蓝牙的新通信协议,不同之处在于它旨在消耗更少的功耗和成本,同时保持同等的功能。因此,低功耗蓝牙是与耗电资源有限的物联网设备进行通信的首选。BluetoothLE扩展需要Android5.0或更高版本。BlueToothLE拓展中文文档入口......
  • Android低功耗子系统的投票机制以及触发进入系统休眠的过程
    从kernel角度看,系统是否进入休眠应该由内核来控制,因此Linux引入了wakeupsource以及autosleep机制关于wakeupsource的介绍,请参考:WakeupSource框架设计与实现关于autosleep机制,请参考:autosleep框架设计与实现在内核中,使用wakeupsource提供投票机制,让各个系统模块投票......
  • 【解决方案】华普微基于CMT2189D的低功耗广域网解决方案
    一、方案概述随着物联网的快速发展,对于无线通信的需求越来越高。传统的通信技术可能无法满足物联网设备的特殊要求,如低功耗、长距离覆盖和大规模连接。LPWAN技术应运而生,旨在为物联网设备提供低成本、低功耗的远距离通信解决方案。ZETA作为其中的一种LPWAN技术,专注于在UNB技......
  • ChirpLAN™--助力低功耗远距离物联网
      在物联网的世界里,无线通信模块无比重要!它能实现数据的传送,它能完成信息的收发,它能进行信息的传递、路由和控制。它覆盖延伸网,它连接接入网,它掌控核心网。它依托公众电信网,它依靠互联网,它借助行业专用通信网络。常见的自组网模块,如蓝牙、窗口、2.4G等等,您一定熟知!磐启微电子......