Skip to content

Instantly share code, notes, and snippets.

@shoffing
Created February 19, 2014 01:40
Show Gist options
  • Star 2 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save shoffing/9084561 to your computer and use it in GitHub Desktop.
Save shoffing/9084561 to your computer and use it in GitHub Desktop.
Source code for working with the L3G4200D using Arduino and Processing
#include <Wire.h>
#define CTRL_REG1 0x20
#define CTRL_REG2 0x21
#define CTRL_REG3 0x22
#define CTRL_REG4 0x23
#define CTRL_REG5 0x24
// I2C address of the L3G4200D.
// Use I2C scanner to find this value!
int L3G4200D_Address = 0x69;
// Can fine-tune this if you need to
float DPS_MULT = 0.0000085;
// Delta angles (raw input from gyro)
int x = 0;
int y = 0;
int z = 0;
// Actual angles
float angX = 0;
float angY = 0;
float angZ = 0;
// Previous angles for calculation
float p_angX = 0;
float p_angY = 0;
float p_angZ = 0;
// Calibration values
int gyroLowX = 0;
int gyroLowY = 0;
int gyroLowZ = 0;
int gyroHighX = 0;
int gyroHighY = 0;
int gyroHighZ = 0;
// Used for calculating delta time
unsigned long pastMicros = 0;
void setup()
{
Wire.begin();
Serial.begin(9600);
Serial.println("Starting up L3G4200D");
setupL3G4200D(250); // Configure L3G4200 - 250, 500 or 2000 deg/sec
delay(1000); // wait for the sensor to be ready
calibrate();
attachInterrupt(0, updateAngle, RISING);
pastMicros = micros();
}
void loop()
{
getGyroValues();
// Calculate delta time
float dt;
if(micros() > pastMicros) //micros() overflows every ~70 minutes
dt = (float) (micros()-pastMicros)/1000000.0;
else
dt = (float) ((4294967295-pastMicros)+micros())/1000000.0;
// Calculate angles
if(x >= gyroHighX || x <= gyroLowX) {
angX += ((p_angX + (x * DPS_MULT))/2) * dt;
p_angX = x * DPS_MULT;
} else {
p_angX = 0;
}
if(y >= gyroHighY || y <= gyroLowY) {
angY += ((p_angY + (y * DPS_MULT))/2) * dt;
p_angY = y * DPS_MULT;
} else {
p_angY = 0;
}
if(z >= gyroHighZ || z <= gyroLowZ) {
angZ += ((p_angZ + (z * DPS_MULT))/2) * dt;
p_angZ = z * DPS_MULT;
} else {
p_angZ = 0;
}
pastMicros = micros();
sendJson();
delay(10);
}
void updateAngle()
{
getGyroValues();
}
void calibrate()
{
Serial.println("Calibrating gyro, don't move!");
for(int i = 0; i < 200; i++) {
getGyroValues();
if(x > gyroHighX)
gyroHighX = x;
else if(x < gyroLowX)
gyroLowX = x;
if(y > gyroHighY)
gyroHighY = y;
else if(y < gyroLowY)
gyroLowY = y;
if(z > gyroHighZ)
gyroHighZ = z;
else if(z < gyroLowZ)
gyroLowZ = z;
delay(10);
}
Serial.println("Calibration complete.");
}
// Print angles to Serial (for use in Processing, for example)
void sendJson() {
char json[40];
sprintf(json, "{\"x\":%d,\"y\":%d,\"z\":%d}", (int)(angX*1000), (int)(angY*1000), (int)(angZ*1000));
Serial.println(json);
}
void getGyroValues() {
byte xMSB = readRegister(L3G4200D_Address, 0x29);
byte xLSB = readRegister(L3G4200D_Address, 0x28);
x = ((xMSB << 8) | xLSB);
byte yMSB = readRegister(L3G4200D_Address, 0x2B);
byte yLSB = readRegister(L3G4200D_Address, 0x2A);
y = ((yMSB << 8) | yLSB);
byte zMSB = readRegister(L3G4200D_Address, 0x2D);
byte zLSB = readRegister(L3G4200D_Address, 0x2C);
z = ((zMSB << 8) | zLSB);
}
int setupL3G4200D(int scale) {
//From Jim Lindblom of Sparkfun's code
// Enable x, y, z and turn off power down:
writeRegister(L3G4200D_Address, CTRL_REG1, 0b00001111);
// If you'd like to adjust/use the HPF, you can edit the line below to configure CTRL_REG2:
writeRegister(L3G4200D_Address, CTRL_REG2, 0b00000000);
// Configure CTRL_REG3 to generate data ready interrupt on INT2
// No interrupts used on INT1, if you'd like to configure INT1
// or INT2 otherwise, consult the datasheet:
writeRegister(L3G4200D_Address, CTRL_REG3, 0b00001000);
// CTRL_REG4 controls the full-scale range, among other things:
if(scale == 250) {
writeRegister(L3G4200D_Address, CTRL_REG4, 0b00000000);
} else if(scale == 500) {
writeRegister(L3G4200D_Address, CTRL_REG4, 0b00010000);
} else {
writeRegister(L3G4200D_Address, CTRL_REG4, 0b00110000);
}
// CTRL_REG5 controls high-pass filtering of outputs, use it
// if you'd like:
writeRegister(L3G4200D_Address, CTRL_REG5, 0b00000000);
}
void writeRegister(int deviceAddress, byte address, byte val)
{
Wire.beginTransmission(deviceAddress); // start transmission to device
Wire.write(address); // send register address
Wire.write(val); // send value to write
Wire.endTransmission(); // end transmission
}
int readRegister(int deviceAddress, byte address)
{
int v;
Wire.beginTransmission(deviceAddress);
Wire.write(address); // register to read
Wire.endTransmission();
Wire.requestFrom(deviceAddress, 1); // read a byte
while(!Wire.available()) {
// waiting
}
v = Wire.read();
return v;
}
import org.json.JSONObject;
import processing.serial.*;
Serial myPort;
boolean ready = false;
int x = 0;
int y = 0;
int z = 0;
void setup()
{
size(1080, 1080, P3D);
println(Serial.list());
myPort = new Serial(this, Serial.list()[1], 9600);
myPort.bufferUntil(10);
}
void draw()
{
lights();
background(0);
if(ready) {
fill(255);
textSize(16);
text("X: " + x, 10, 20);
text("Y: " + y, 10, 40);
text("Z: " + z, 10, 60);
pushMatrix();
fill(0, 255, 0);
translate(width/2, height/2, 0);
rotateX(radians(-x));
rotateY(radians(z));
rotateZ(radians(y));
box(300);
popMatrix();
} else {
fill(255);
textSize(64);
text("Calibrating, don't move...", 10, 80);
}
}
void serialEvent(Serial port)
{
String str = port.readString();
str = str.substring(0, str.length() - 1);
if(str.charAt(0) == '{') {
ready = true;
JSONObject json = new JSONObject(str);
x = json.getInt("x");
y = json.getInt("y");
z = json.getInt("z");
} else {
ready = false;
println(str);
}
}
@hungking
Copy link

can you instruct detail more ? i need now and then. please !

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