2015/07/01

How to calibrate ADXL345 Triple Axis Accelerometer

I recently tested 5 PCs of ADXL345 breakout boards that I bought on Taobao. Sadly, the readouts are so inconsistent and the z-axis readouts are too big to be corrected using the chip's own offset registers located at 0x1E (OSFX), 0x1F (OSFY), and 0x20 (OSFZ).

Below are the readouts of all 5 modules taken with the modules facing down.



Before doing calibration, the first thing is to decide the orientation for taking the measurements. After experimenting with different orientations and observing the readouts, I decided to use the below orientations.


Below is the wiring between ADXL345 and Arduino Uno. Pin 2 of Arduino Uno is connected to GND. When the switch is closed, the program running on Arduino Uno will enter into Calibration Mode. When the switch is open, the program is in Normal Mode.


Program Code

#include <Wire.h>

byte DEVICE_ADDRESS = 0x53; //This address is fixed for the board we use because pin 12 is fixed to GND

byte DATA_FORMAT = 0x31;   //Data Format & Measurement Range Register
byte POWER_CTRL = 0x2D;    //Power Control Register
byte INT_ENABLE = 0x2E;    //Enable Data Ready Interrupt

byte OFSTX = 0x1E;         //X_CALIB
byte OFSTY = 0x1F;         //Y_CALIB
byte OFSTZ = 0x20;         //Z_CALIB

byte DATAX0 = 0x32;        //X-Axis Data 0
byte DATAX1 = 0x33;        //X-Axis Data 1
byte DATAY0 = 0x34;        //Y-Axis Data 0
byte DATAY1 = 0x35;        //Y-Axis Data 1
byte DATAZ0 = 0x36;        //Z-Axis Data 0
byte DATAZ1 = 0x37;        //Z-Axis Data 1

byte values[6];

int x,y,z;                 //For storing the raw value
double xg, yg, zg;         //For storing the G value

float x_bound[3] = {-9999, -9999, -9999};            //Upper, Lower, Middle
float y_bound[3] = {-9999, -9999, -9999};            //Upper, Lower, Middle
float z_bound[3] = {-9999, -9999, -9999};            //Upper, Lower, Middle

int calibrate = -1;
int CAL_DELAY = 400;      //Calibration Delay - waiting time for user to make response

void setup() {
  Wire.begin();    //Initiate the Wire library and join the I2C bus as a master. This is called only once.
  Serial.begin(9600);

  //Minimum initialization sequance according to the application note.
  writeRegister(DEVICE_ADDRESS, DATA_FORMAT, 0x0B); //Put the ADXL345 into Measurement Mode with full resolution (13-bit) and +/-16G range.
  writeRegister(DEVICE_ADDRESS, POWER_CTRL, 0x08); //Start Measurement.
  writeRegister(DEVICE_ADDRESS, INT_ENABLE, 0x80); //Enable Data Ready Interrupt.

  pinMode(2, INPUT_PULLUP);     //Pin2 (LOW) -> Calibration Mode, Pin2(HIGH, DEFAULT) -> Normal Mode
}

void loop() {

  int calibration_mode = digitalRead(2);

  if (calibration_mode == HIGH) {
//    Serial.println("Normal Mode");
    normalMode();
 
    //Converting the raw accelerometer values to g's.

    xg = x * 0.0039;
    yg = y * 0.0039;
    zg = z * 0.0039;

    //Display the results

    Serial.print(x, DEC);
    Serial.print(", ");
    Serial.print(y, DEC);
    Serial.print(", ");
    Serial.print(z, DEC);
    Serial.print(" | ");

    Serial.print(x_bound[0]);
    Serial.print(", ");
    Serial.print(x_bound[1]);
    Serial.print(", ");
    Serial.print(x_bound[2]);
    Serial.print(" | ");  
 
    Serial.print(y_bound[0]);
    Serial.print(", ");
    Serial.print(y_bound[1]);
    Serial.print(", ");
    Serial.print(y_bound[2]);
    Serial.print(" | ");  

    Serial.print(z_bound[0]);
    Serial.print(", ");
    Serial.print(z_bound[1]);
    Serial.print(", ");
    Serial.print(z_bound[2]);
    Serial.println();
 
/*
    Serial.print((float)xg, 2);
    Serial.print("'g, ");
    Serial.print((float)yg, 2);
    Serial.print("g, ");
    Serial.print((float)zg, 2);
    Serial.println("g");
*/
  }

  if (calibration_mode == LOW) {
    Serial.println("Calibration Mode");
    if ((x_bound[0] == -9999) || (y_bound[0] == -9999) || (z_bound[0] == -9999)) {
        Serial.println("Please release pin 2 from logic LOW position");
        while (digitalRead(2) == LOW) {
        }    
        //delay(CAL_DELAY);
        calibrationMode();
   }
   else {
     Serial.println("Module already calibrated. Do you want to calibrate again? (0 = No, 1 = Yes) ");
     while (calibrate == -1) {
       calibrate = Serial.read();
     }
     if (calibrate == '1') {    
       calibrationMode();
       calibrate = -1;
     }
     else {
       calibrate = -1;
     }
   }
  }
}

void calibrationMode() {

//Get y-axis boundries
//----------------------------------------------------------------------------
  Serial.println("Place module to position 1 before count down reaches 0");
  for (int i = 25; i >= 0; i--) {
    Serial.println(i);
    delay(CAL_DELAY);
  }
  Serial.println();
  for (int count = 1; count < 100; count++) {
    normalMode();
    y_bound[0] = y_bound[0] + y;
  }
  y_bound[0] = y_bound[0] / 100;
  Serial.print("Y Low Bound = ");
  Serial.println(y_bound[0]);

  Serial.println();
  Serial.println("Place module to position 2 before count down reaches 0");
  for (int i = 25; i >= 0; i--) {
    Serial.println(i);
    delay(CAL_DELAY);
  }
  for (int count = 1; count < 100; count++) {
    normalMode();
    y_bound[1] = y_bound[1] + y;
  }
  y_bound[1] = y_bound[1] / 100;
  Serial.print("Y High Bound = ");
  Serial.println(y_bound[1]);
  y_bound[2] = (y_bound[0] + y_bound[1]) / 2;
  Serial.print("Y Mid Point = ");
  Serial.println(y_bound[2]);

//Get x-axis boundries
//----------------------------------------------------------------------------
  Serial.println("Place module to position 3 before count down reaches 0");
  for (int i = 25; i >= 0; i--) {
    Serial.println(i);
    delay(CAL_DELAY);
  }
  Serial.println();
  for (int count = 1; count < 100; count++) {
    normalMode();
    x_bound[0] = x_bound[0] + x;
  }
  x_bound[0] = x_bound[0] / 100;
  Serial.print("X Low Bound = ");
  Serial.println(x_bound[0]);

  Serial.println();
  Serial.println("Place module to position 4 before count down reaches 0");
  for (int i = 25; i >= 0; i--) {
    Serial.println(i);
    delay(CAL_DELAY);
  }
  for (int count = 1; count < 100; count++) {
    normalMode();
    x_bound[1] = x_bound[1] + x;
  }
  x_bound[1] = x_bound[1] / 100;
  Serial.print("X High Bound = ");
  Serial.println(x_bound[1]);
  x_bound[2] = (x_bound[0] + x_bound[1]) / 2;
  Serial.print("X Mid Point = ");
  Serial.println(x_bound[2]);

//Get z-axis boundries
//----------------------------------------------------------------------------
  Serial.println("Place module to position 5 before count down reaches 0");
  for (int i = 25; i >= 0; i--) {
    Serial.println(i);
    delay(CAL_DELAY);
  }
  Serial.println();
  for (int count = 1; count < 100; count++) {
    normalMode();
    z_bound[0] = z_bound[0] + z;
  }
  z_bound[0] = z_bound[0] / 100;
  Serial.print("Z Low Bound = ");
  Serial.println(z_bound[0]);

  Serial.println();
  Serial.println("Place module to position 6 before count down reaches 0");
  for (int i = 25; i >= 0; i--) {
    Serial.println(i);
    delay(CAL_DELAY);
  }
  for (int count = 1; count < 100; count++) {
    normalMode();
    z_bound[1] = z_bound[1] + z;
  }
  z_bound[1] = z_bound[1] / 100;
  Serial.print("Z High Bound = ");
  Serial.println(z_bound[1]);
  z_bound[2] = (z_bound[0] + z_bound[1]) / 2;
  Serial.print("Z Mid Point = ");
  Serial.println(z_bound[2]);

}

void normalMode() {

  readRegister(DEVICE_ADDRESS, DATAX0, 6, values);

  //The ADXL345 gives 10-bit or 13-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];
 
  delay(15);
}

void writeRegister(byte device, byte registerAddress, byte value) {
  Wire.beginTransmission(device);    //Start transmission to device
  Wire.write(registerAddress);       //Specify the address of the register to be written to
  Wire.write(value);                 //Send the value to be writen to the register
  Wire.endTransmission();            //End transmission
}

void readRegister(byte device, byte registerAddress, int numBytes, byte *values) {

  byte address = registerAddress;

  Wire.beginTransmission(device);
  Wire.write(address);
  Wire.endTransmission();

  Wire.beginTransmission(device);    
  Wire.requestFrom(device, numBytes);  //Request 6 bytes from device

  int i = 0;
  while (Wire.available() && i < numBytes) {        //Device may send less than requested
    values[i] = Wire.read();                        //Receive a byte from device and put it into the buffer
    i++;
 }
 
  Wire.endTransmission();
}

Note,
When using the above program, please refer to the 2nd pic of this post for how to position the module.

Output

Below is the screen output.

Here is the same output with added notes.


The Low and High of each axis are the average of 100 sample values. The Mid of each axis is the average of the Low and High of the axis.

With the available of the Low, High, and Mid readouts of each axis, it's now possible to map the readouts to other systems (i.e. degree, etc.).

Things to do
1. Determine the direction of rotation (tilt forward / backward / left / right),
2. Map and show the raw readouts to degree.