Skip to content

Instantly share code, notes, and snippets.

@xrl
Created March 28, 2014 16:46
Show Gist options
  • Star 3 You must be signed in to star a gist
  • Fork 2 You must be signed in to fork a gist
  • Save xrl/9837382 to your computer and use it in GitHub Desktop.
Save xrl/9837382 to your computer and use it in GitHub Desktop.
ADXL345 accelerometer programming code. Immediately raises interrupt when inactivity below THRESH_INACT is detected.
//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;
//ADXL345 Register Addresses
#define DEVID 0x00 //Device ID Register
#define THRESH_TAP 0x1D //Tap Threshold
#define OFSX 0x1E //X-axis offset
#define OFSY 0x1F //Y-axis offset
#define OFSZ 0x20 //Z-axis offset
#define DURATION 0x21 //Tap Duration
#define LATENT 0x22 //Tap latency
#define WINDOW 0x23 //Tap window
#define THRESH_ACT 0x24 //Activity Threshold
#define THRESH_INACT 0x25 //Inactivity Threshold
#define TIME_INACT 0x26 //Inactivity Time
#define ACT_INACT_CTL 0x27 //Axis enable control for activity and inactivity detection
#define THRESH_FF 0x28 //free-fall threshold
#define TIME_FF 0x29 //Free-Fall Time
#define TAP_AXES 0x2A //Axis control for tap/double tap
#define ACT_TAP_STATUS 0x2B //Source of tap/double tap
#define BW_RATE 0x2C //Data rate and power mode control
#define POWER_CTL 0x2D //Power Control Register
#define INT_ENABLE 0x2E //Interrupt Enable Control
#define INT_MAP 0x2F //Interrupt Mapping Control
#define INT_SOURCE 0x30 //Source of interrupts
#define DATA_FORMAT 0x31 //Data format control
#define DATAX0 0x32 //X-Axis Data 0
#define DATAX1 0x33 //X-Axis Data 1
#define DATAY0 0x34 //Y-Axis Data 0
#define DATAY1 0x35 //Y-Axis Data 1
#define DATAZ0 0x36 //Z-Axis Data 0
#define DATAZ1 0x37 //Z-Axis Data 1
#define FIFO_CTL 0x38 //FIFO control
#define FIFO_STATUS 0x39 //FIFO status
//This buffer will hold values read from the ADXL345 registers.
char values[10];
char output[20];
//These variables will be used to hold the x,y and z axis accelerometer values.
int x,y,z;
double xg, yg, zg;
char tapType=0;
char inactivityEvent=0;
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);
//Create an interrupt that will trigger when a tap is detected.
attachInterrupt(0, tap, RISING);
//Put the ADXL345 into +/- 4G range by writing the value 0x01 to the DATA_FORMAT register.
writeRegister(DATA_FORMAT, 0x01);
// Suggested to properly program accelerometer
writeRegister(INT_ENABLE, 0x00);
// Set a threshold of 3g
writeRegister(THRESH_INACT, 0x0F);
// Raise INACT event immeidately (otherwise it would be in 1 second...)
writeRegister(TIME_INACT,0);
// Consider all axes AND compare THRESH_INACT from 0 acceleration (otherwise it would be differential)
writeRegister(ACT_INACT_CTL, 0x07);
// Send all interrrupts to INT 1 (PIN 2)
writeRegister(INT_MAP,0);
// Enable interrupts for INACTIVITY only.
writeRegister(INT_ENABLE, 0x08);
//Put the ADXL345 into Measurement Mode by writing 0x08 to the POWER_CTL register.
writeRegister(POWER_CTL, 0x08); //Measurement mode
readRegister(INT_SOURCE, 1, values); //Clear the interrupts from the INT_SOURCE register.
}
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];
//Convert the accelerometer value to G's.
//With 10 bits measuring over a +/-4g range we can find how to convert by using the equation:
// Gs = Measurement Value * (G-range/(2^10)) or Gs = Measurement Value * (8/1024)
xg = x * 0.0078;
yg = y * 0.0078;
zg = z * 0.0078;
Serial.print(xg,DEC);
Serial.print(yg,DEC);
Serial.print(zg,DEC);
Serial.println();
if(inactivityEvent == 1){
Serial.println("inactivity event");
inactivityEvent = 0;
}
// detachInterrupt(0);
delay(100);
// attachInterrupt(0, tap, RISING);
}
//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);
}
void tap(void){
//Clear the interrupts on the ADXL345
if(values[0] & (1<<3)){
inactivityEvent = 1;
} else {
inactivityEvent = 0;
}
readRegister(INT_SOURCE, 1, values);
}
@RichardBronosky
Copy link

If you configure thresholds to trigger an interrupt pin, is that configuration retained through power cycles? I have a very simple application that I would like to avoid using a microcontroller in. Ideally, I'd configure the ADXL345 once with a microcontroller and never need it again. Is that doable? I can find no info on that in the datasheet, but I am terrible at skimming. http://www.analog.com/media/en/technical-documentation/data-sheets/ADXL345.pdf

@revathy1995
Copy link

how to do the above coding with raspberry pi?

@revathy1995
Copy link

how to do the coding part for adxl345 sensor with raspberry pi?

@S2017144000
Copy link

Brother, I am so happy with this code. It helps me. Can you add more code if the Arduino Uno (pin 7 and 8) is connected to GSM SIM900A module (Pin Tx and Rx). Buzzer is connect to arduino uno pin 9. I like if falls happens, the buzzer will set off and message of help will send to mobile phone. Its my project. Can you help me?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment