Mercurial > templog
changeset 111:cdb26addf4f2 tilt
Add bma180 code, untested
author | Matt Johnston <matt@ucc.asn.au> |
---|---|
date | Wed, 19 Sep 2012 22:40:49 +0800 (2012-09-19) |
parents | 5c8404549cc0 |
children | 5234ccc416e8 |
files | Makefile bma180.c bma180.h bma180_internal.h |
diffstat | 4 files changed, 212 insertions(+), 1 deletions(-) [+] |
line wrap: on
line diff
--- a/Makefile Fri Sep 07 22:59:26 2012 +0800 +++ b/Makefile Wed Sep 19 22:40:49 2012 +0800 @@ -22,7 +22,7 @@ CLOCK = 2000000 PROGRAMMER = #-c stk500v2 -P avrdoper PROGRAMMER = -c stk500 -P ~/dev/stk500 -p $(PROGDEVICE) -B 2 -SOURCE_1WIRE = onewire.c simple_ds18b20.c crc8.c +SOURCE_1WIRE = onewire.c simple_ds18b20.c crc8.c bma180.c SOURCE = main.c SOURCE += $(SOURCE_1WIRE) LIBS = -lm
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/bma180.c Wed Sep 19 22:40:49 2012 +0800 @@ -0,0 +1,146 @@ +#include <avr/io.h> +#include <util/delay.h> + +#include "bma180.h" +#include "bma180_internal.h" + +static void +bma180_init_SPI(void) +{ + //make SPI master + //SCL idle high, sample data on rising edge + //Fosc/4 is SPI frequency + //enable SPI + SPCR = _BV(MSTR) | _BV(SPE) | _BV(CPHA) | _BV(CPOL); +} + +static char +bma180_rxdata(void) +{ + //while((SPSR&0x80) == 0x80); + SPDR = 0x55; + while((SPSR&0x80) == 0x00); + + return SPDR; +} + +static void +bma180_txdata(char data) +{ + //while((SPSR&0x80) == 0x80); + SPDR = data; + while((SPSR&0x80) == 0x00); +} + +static void +bma180_write(uint8_t address, char data) +{ + //write any data byte to any single address + //adds a 0 to the MSB of the address byte (WRITE mode) + + address &= 0x7F; + + //printf("\nWriting 0x%x to 0x%x\n", data, address); + + BMA180_CS_PORT &= ~_BV(BMA180_CS_PIN); + _delay_ms(1); + bma180_txdata(address); + _delay_ms(1); + bma180_txdata(data); + _delay_ms(1); + BMA180_CS_PORT |= _BV(BMA180_CS_PIN); +} + +static char +bma180_read(uint8_t address) +{ + //returns the contents of any 1 byte register from any address + //sets the MSB for every address byte (READ mode) + + char byte; + + address |= 0x80; + + BMA180_CS_PORT &= ~_BV(BMA180_CS_PIN); + bma180_txdata(address); + byte = bma180_rxdata(); + BMA180_CS_PORT |= _BV(BMA180_CS_PIN); + + return byte; +} + +static int +bma180_set_config(uint8_t range, uint8_t bw) +{ + char temp, temp1; + + // if connected correctly, ID register should be 3 + if(bma180_read(ID) != 3) + return -1; + + + // Set ee_w bit + temp = bma180_read(CTRLREG0); + temp |= 0x10; + bma180_write(CTRLREG0, temp); // Have to set ee_w to write any other registers + + // Set BW + temp = bma180_read(BWTCS); + temp1 = bw; + temp1 = temp1<<4; + temp &= (~BWMASK); + temp |= temp1; + bma180_write(BWTCS, temp); // Keep tcs<3:0> in BWTCS, but write new BW + + // Set Range + temp = bma180_read(OLSB1); + temp1 = range; + temp1 = (temp1<<RANGESHIFT); + temp &= (~RANGEMASK); + temp |= temp1; + bma180_write(OLSB1, temp); //Write new range data, keep other bits the same + + return 0; +} + +void +bma180_init(uint8_t range, uint8_t bw) +{ + bma180_init_SPI(); + + // CS output + // XXX checkme, sample code has the opposite? + BMA180_CS_DDR |= _BV(BMA180_CS_PIN); + + bma180_set_config(range, bw); + +} + +static uint16_t +bma_get_reading_n(uint8_t reg_lsb, uint8_t reg_msb) +{ + uint16_t ret; + uint8_t val = 0; + while (val != 1) + { + val = bma180_read(reg_lsb) & 0x01; + } + val = bma180_read(reg_msb); + ret = val << 8; + + val = bma180_read(reg_lsb); + ret |= val; + // low bits aren't the reading + ret >>= 2; + return ret; +} + +void +bma180_get_reading(uint16_t *x, uint16_t *y, uint16_t *z, uint8_t *temperature) +{ + *x = bma_get_reading_n(ACCXLSB, ACCXMSB); + *y = bma_get_reading_n(ACCYLSB, ACCYMSB); + *z = bma_get_reading_n(ACCZLSB, ACCZMSB); + + *temperature = bma180_read(TEMPERATURE); +}
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/bma180.h Wed Sep 19 22:40:49 2012 +0800 @@ -0,0 +1,14 @@ +#ifndef BMA180_H_ +#define BMA180_H_ + +#include <stdint.h> + +// hardware config +#define BMA180_CS_PORT PORTB +#define BMA180_CS_DDR DDRB +#define BMA180_CS_PIN 1 + +void bma180_get_reading(uint16_t *x, uint16_t *y, uint16_t *z, uint8_t *temperature); +void bma180_init(uint8_t range, uint8_t bw); + +#endif /* BMA180_H_ */
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/bma180_internal.h Wed Sep 19 22:40:49 2012 +0800 @@ -0,0 +1,51 @@ +/* + * Copied from Sparkfun example code: + * + * bma180.h + * Created on: Mar 19, 2010 + */ + +#ifndef BMA180_INTERNAL_H_ +#define BMA180_INTERNAL_H_ + +//Address defines for BMA180// +//====================// +//ID and Version Registers +#define ID 0x00 +#define Version 0x01 +#define ACCXLSB 0x02 +#define ACCXMSB 0x03 +#define ACCYLSB 0x04 +#define ACCYMSB 0x05 +#define ACCZLSB 0x06 +#define ACCZMSB 0x07 +#define TEMPERATURE 0x08 +#define STATREG1 0x09 +#define STATREG2 0x0A +#define STATREG3 0x0B +#define STATREG4 0x0C +#define CTRLREG0 0x0D +#define CTRLREG1 0x0E +#define CTRLREG2 0x0F + +#define BWTCS 0x20 +#define CTRLREG3 0x21 + +#define HILOWNFO 0x25 +#define LOWDUR 0x26 + +#define LOWTH 0x29 + +#define tco_y 0x2F +#define tco_z 0x30 + +#define OLSB1 0x35 + +//====================// +//Range setting +#define RANGESHIFT 1 +#define RANGEMASK 0x0E +#define BWMASK 0xF0 +#define BWSHIFT 4 + +#endif /* BMA180_INTERNAL_H_ */