ADXL345II

■關於 ADXL345:(摘錄於 ADI 官方網頁)

ADXL345是一款小巧纖薄的低功耗三軸加速度計,可以對高達±16 g的加速度進行高分辨率(13位)測量。數字輸出數據為16位二進制補碼格式,可通過SPI(3線或4線)或者I2C 數字接口訪問。

ADXL345非常適合移動設備應用。它可以在傾斜檢測應用中測量靜態重力加速度,還可以測量運動或衝擊導致的動態加速度。它具有高分辨率(4 mg/LSB),能夠測量約0.25°的傾角變化。使用ADXL345等數字輸出加速度計時,無需進行模數轉換,從而可以節省系統成本和電路板面積。此外,ADXL345內置多種功能。活動/非活動檢測、單擊/雙擊檢測以及自由落體檢測均在內部完成,無需主機處理器執行任何計算。內置32級FIFO存儲緩衝器可以減輕主機處理器的負擔,起到簡化算法和省電的作用。利用內置的活動/非活動檢測功能,將ADXL345用作“運動開關”(無活動時關閉整個系統,檢測到活動時才開啟),系統可以實現進一步省電。

 

■關於接線:圖左側為 SPI 四線式接線方法(SPI 三線式接線則省略SDO不接),圖右側為 I2C 接線方法(Rp=10K Ohm)。

CN0133-03-1024

補充:引腳名稱有很多叫法(或標示)但他們都是相同。

  • SDA/SDI/SDIO
  • SDO/ALT ADDRESS
  • SCL/SCLK

■關於程式:(以下提供 Arduino Code)

程式一:使用 SPI 連接(四線式)(摘錄於:https://www.sparkfun.com/tutorials/240)

//Add the SPI library so we can communicate with the ADXL345 sensor
#include <SPI.h>

//Assign the Chip Select signal to pin 10.
int CS=10;

//This is a list of some of the registers available on the ADXL345.
//To learn more about these and the rest of the registers on the ADXL345, read the datasheet!
char POWER_CTL = 0x2D;	//Power Control Register
char DATA_FORMAT = 0x31;
char DATAX0 = 0x32;	//X-Axis Data 0
char DATAX1 = 0x33;	//X-Axis Data 1
char DATAY0 = 0x34;	//Y-Axis Data 0
char DATAY1 = 0x35;	//Y-Axis Data 1
char DATAZ0 = 0x36;	//Z-Axis Data 0
char DATAZ1 = 0x37;	//Z-Axis Data 1

//This buffer will hold values read from the ADXL345 registers.
char values[10];
//These variables will be used to hold the x,y and z axis accelerometer values.
int x,y,z;

void setup(){ 
  //Initiate an SPI communication instance.
  SPI.begin();
  //Configure the SPI connection for the ADXL345.
  SPI.setDataMode(SPI_MODE3);
  //Create a serial connection to display the data on the terminal.
  Serial.begin(9600);
  
  //Set up the Chip Select pin to be an output from the Arduino.
  pinMode(CS, OUTPUT);
  //Before communication starts, the Chip Select pin needs to be set high.
  digitalWrite(CS, HIGH);
  
  //Put the ADXL345 into +/- 4G range by writing the value 0x01 to the DATA_FORMAT register.
  writeRegister(DATA_FORMAT, 0x01);
  //Put the ADXL345 into Measurement Mode by writing 0x08 to the POWER_CTL register.
  writeRegister(POWER_CTL, 0x08);  //Measurement mode  
}

void loop(){
  //Reading 6 bytes of data starting at register DATAX0 will retrieve the x,y and z acceleration values from the ADXL345.
  //The results of the read operation will get stored to the values[] buffer.
  readRegister(DATAX0, 6, values);

  //The ADXL345 gives 10-bit acceleration values, but they are stored as bytes (8-bits). To get the full value, two bytes must be combined for each axis.
  //The X value is stored in values[0] and values[1].
  x = ((int)values[1]<<8)|(int)values[0];
  //The Y value is stored in values[2] and values[3].
  y = ((int)values[3]<<8)|(int)values[2];
  //The Z value is stored in values[4] and values[5].
  z = ((int)values[5]<<8)|(int)values[4];
  
  //Print the results to the terminal.
  Serial.print(x, DEC);
  Serial.print(',');
  Serial.print(y, DEC);
  Serial.print(',');
  Serial.println(z, DEC);      
  delay(10); 
}

//This function will write a value to a register on the ADXL345.
//Parameters:
//  char registerAddress - The register to write a value to
//  char value - The value to be written to the specified register.
void writeRegister(char registerAddress, char value){
  //Set Chip Select pin low to signal the beginning of an SPI packet.
  digitalWrite(CS, LOW);
  //Transfer the register address over SPI.
  SPI.transfer(registerAddress);
  //Transfer the desired register value over SPI.
  SPI.transfer(value);
  //Set the Chip Select pin high to signal the end of an SPI packet.
  digitalWrite(CS, HIGH);
}

//This function will read a certain number of registers starting from a specified address and store their values in a buffer.
//Parameters:
//  char registerAddress - The register addresse to start the read sequence from.
//  int numBytes - The number of registers that should be read.
//  char * values - A pointer to a buffer where the results of the operation should be stored.
void readRegister(char registerAddress, int numBytes, char * values){
  //Since we're performing a read operation, the most significant bit of the register address should be set.
  char address = 0x80 | registerAddress;
  //If we're doing a multi-byte read, bit 6 needs to be set as well.
  if(numBytes > 1)address = address | 0x40;
  
  //Set the Chip select pin low to start an SPI packet.
  digitalWrite(CS, LOW);
  //Transfer the starting register address that needs to be read.
  SPI.transfer(address);
  //Continue to read registers until we've read the number specified, storing the results to the input buffer.
  for(int i=0; i<numBytes; i++){
    values[i] = SPI.transfer(0x00);
  }
  //Set the Chips Select pin high to end the SPI packet.
  digitalWrite(CS, HIGH);
}

程式二:使用 I2C 連接

#include <Wire.h>

int I2C_Address = 0xA7 >> 1; 

int X0, X1, Y0, Y1, Z1, Z0;
float X,Y,Z;

void setup(){
    Serial.begin(9600);
    Serial.println("Start : \n");
    
    Wire.begin(); 
    setReg(0x2D, 0xA);
}

void loop(){
    X0 = getData(0x32);
    X1 = getData(0x33);
    X = ((X1 << 8)  + X0) / 256.0;
 
    Y0 = getData(0x34);
    Y1 = getData(0x35);
    Y = ((Y1 << 8)  + Y0) / 256.0;
 
    Z0 = getData(0x36);
    Z1 = getData(0x37);
    Z = ((Z1 << 8)  + Z0) / 256.0;
    
    Serial.print("X= ");
    Serial.print(X);
    Serial.print("    Y= ");
    Serial.print(Y);
    Serial.print("    Z= ");
    Serial.println(Z);
    
    delay(500);
}

/* setReg(reg,data): */
void setReg(int reg, int data){
    Wire.beginTransmission(I2C_Address);
    Wire.write(reg); 
    Wire.write(data);
    Wire.endTransmission();
}

/* getData(reg): */
int getData(int reg){
    Wire.beginTransmission(I2C_Address);
    Wire.write(reg);
    Wire.endTransmission();
    
    Wire.requestFrom(I2C_Address,1);
    
    if(Wire.available()<=1){
        return Wire.read();
    }
}

Comments are closed.

Post Navigation