diff options
-rw-r--r-- | lib/bmp280.c | 185 | ||||
-rw-r--r-- | lib/bmp280.h | 107 | ||||
-rw-r--r-- | usb-bmp280/Makefile | 51 | ||||
-rw-r--r-- | usb-bmp280/usb-bmp280.c | 120 |
4 files changed, 463 insertions, 0 deletions
diff --git a/lib/bmp280.c b/lib/bmp280.c new file mode 100644 index 0000000..89cf4cf --- /dev/null +++ b/lib/bmp280.c @@ -0,0 +1,185 @@ +/* + * spreadspace avr utils + * + * + * Copyright (C) 2016 Bernhard Tittelbach <bernhard@tittelbach.org> + * appreciative nod to Adafruit for writing a great example, go buy their stuff + * + * This file is part of spreadspace avr utils. + * + * spreadspace avr utils is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * spreadspace avr utils is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with spreadspace avr utils. If not, see <http://www.gnu.org/licenses/>. + */ + +#include <stdio.h> +#include <avr/pgmspace.h> +#include <LUFA/Drivers/Peripheral/SPI.h> +#include "bmp280.h" + + +int32_t bmp280_t_fine = 0; + +void bmp280_cs(bmp280_sensor *sensor, uint8_t select) +{ + if (!select) + *(sensor->cs_port) |= _BV(sensor->cs_pin); + else + *(sensor->cs_port) &= ~ _BV(sensor->cs_pin); +} + +void bmp280_write8(uint8_t reg, uint8_t value) +{ + SPI_SendByte(reg & ~0x80); //bit 7 needs to be low for write mode + SPI_SendByte(value); +} + +uint8_t bmp280_read8(uint8_t reg) +{ + SPI_SendByte(reg | 0x80); //bit 7 high for read mode + return SPI_ReceiveByte(); +} + +uint16_t bmp280_read16(uint8_t reg) +{ + uint16_t value = 0; + SPI_SendByte(reg | 0x80); //bit 7 high for read mode + value |= ((uint16_t) SPI_ReceiveByte() & 0xFF) << 8; + value |= (uint16_t) SPI_ReceiveByte() & 0xFF; + return value; +} + +uint32_t bmp280_read24(uint8_t reg) +{ + uint32_t value = 0; + SPI_SendByte(reg | 0x80); //bit 7 high for read mode + value |= ((uint32_t) SPI_ReceiveByte() & 0xFF) << 16; + value |= ((uint32_t) SPI_ReceiveByte() & 0xFF) << 8; + value |= (uint32_t) SPI_ReceiveByte() & 0xFF; + return value; +} + +int16_t bmp280_readS16(uint8_t reg) +{ + return (int16_t) bmp280_read16(reg); +} + +uint16_t bmp280_read16_LE(uint8_t reg) +{ + uint16_t value = 0; + SPI_SendByte(reg | 0x80); //bit 7 high for read mode + value |= (uint16_t) SPI_ReceiveByte() & 0xFF; + value |= ((uint16_t) SPI_ReceiveByte() & 0xFF) << 8; + return value; +} + +int16_t bmp280_readS16_LE(uint8_t reg) +{ + return (int16_t)bmp280_read16_LE(reg); +} + +void bmp280_readCoefficients(bmp280_sensor *sensor) +{ + bmp280_cs(sensor, true); + sensor->dig_T1 = bmp280_read16_LE(BMP280_REGISTER_DIG_T1); + sensor->dig_T2 = bmp280_readS16_LE(BMP280_REGISTER_DIG_T2); + sensor->dig_T3 = bmp280_readS16_LE(BMP280_REGISTER_DIG_T3); + sensor->dig_P1 = bmp280_read16_LE(BMP280_REGISTER_DIG_P1); + sensor->dig_P2 = bmp280_readS16_LE(BMP280_REGISTER_DIG_P2); + sensor->dig_P3 = bmp280_readS16_LE(BMP280_REGISTER_DIG_P3); + sensor->dig_P4 = bmp280_readS16_LE(BMP280_REGISTER_DIG_P4); + sensor->dig_P5 = bmp280_readS16_LE(BMP280_REGISTER_DIG_P5); + sensor->dig_P6 = bmp280_readS16_LE(BMP280_REGISTER_DIG_P6); + sensor->dig_P7 = bmp280_readS16_LE(BMP280_REGISTER_DIG_P7); + sensor->dig_P8 = bmp280_readS16_LE(BMP280_REGISTER_DIG_P8); + sensor->dig_P9 = bmp280_readS16_LE(BMP280_REGISTER_DIG_P9); + bmp280_cs(sensor, false); +} + +//need to init SPI beforehand +void bmp280_init(bmp280_sensor *sensor, volatile uint8_t *cs_port, uint8_t cs_pin) +{ + memset(sensor,0,sizeof(bmp280_sensor)); + sensor->cs_port = cs_port; + sensor->cs_pin = cs_pin; + bmp280_readCoefficients(sensor); +} + +float bmp280_readTemp(bmp280_sensor *sensor) +{ + int32_t var1, var2; + + bmp280_cs(sensor, true); + int32_t adc_T = bmp280_read24(BMP280_REGISTER_TEMPDATA); + bmp280_cs(sensor, false); + adc_T >>= 4; + + var1 = ((((adc_T>>3) - ((int32_t)sensor->dig_T1 <<1))) * + ((int32_t)sensor->dig_T2)) >> 11; + + var2 = (((((adc_T>>4) - ((int32_t)sensor->dig_T1)) * + ((adc_T>>4) - ((int32_t)sensor->dig_T1))) >> 12) * + ((int32_t)sensor->dig_T3)) >> 14; + + bmp280_t_fine = var1 + var2; + + float T = (bmp280_t_fine * 5 + 128) >> 8; + return T/100; +} + +float bmp280_readPressure(bmp280_sensor *sensor) +{ + int64_t var1, var2, p; + + // Must be done first to get the bmp280_t_fine variable + bmp280_readTemp(sensor); + + bmp280_cs(sensor, true); + int32_t adc_P = bmp280_read24(BMP280_REGISTER_PRESSUREDATA); + bmp280_cs(sensor, false); + adc_P >>= 4; + + var1 = ((int64_t)bmp280_t_fine) - 128000; + var2 = var1 * var1 * (int64_t)sensor->dig_P6; + var2 = var2 + ((var1*(int64_t)sensor->dig_P5)<<17); + var2 = var2 + (((int64_t)sensor->dig_P4)<<35); + var1 = ((var1 * var1 * (int64_t)sensor->dig_P3)>>8) + + ((var1 * (int64_t)sensor->dig_P2)<<12); + var1 = (((((int64_t)1)<<47)+var1))*((int64_t)sensor->dig_P1)>>33; + + if (var1 == 0) + { + return 0; // avoid exception caused by division by zero + } + p = 1048576 - adc_P; + p = (((p<<31) - var2)*3125) / var1; + var1 = (((int64_t)sensor->dig_P9) * (p>>13) * (p>>13)) >> 25; + var2 = (((int64_t)sensor->dig_P8) * p) >> 19; + + p = ((p + var1 + var2) >> 8) + (((int64_t)sensor->dig_P7)<<4); + return (float)p/256; +} + +/*float bmp280_readAltitude(bmp280_sensor *sensor, float sealevelp) +{ + float altitude; + + float pressure = bmp280_readPressure(sensor); // in Si units for Pascal + pressure /= 100; + + altitude = 44330 * (1.0 - pow(pressure / seaLevelhPa, 0.1903)); + + return altitude; +}*/ + + +/// Helper Functions diff --git a/lib/bmp280.h b/lib/bmp280.h new file mode 100644 index 0000000..22472e3 --- /dev/null +++ b/lib/bmp280.h @@ -0,0 +1,107 @@ +/* + * spreadspace avr utils + * + * + * Copyright (C) 2016 Bernhard Tittelbach <bernhard@tittelbach.org> + * appreciative nod to Adafruit for writing a great example, go buy their stuff + * + * This file is part of spreadspace avr utils. + * + * spreadspace avr utils is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * spreadspace avr utils is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with spreadspace avr utils. If not, see <http://www.gnu.org/licenses/>. + */ + +#ifndef SPREADAVR_bmp280_h_INCLUDED +#define SPREADAVR_bmp280_h_INCLUDED + +#ifdef __cplusplus +extern "C" { +#endif + +#define BMP280_AVERAGE_SEA_LEVEL_PRESSURE 1013.25 +#define BMP280_LUFA_SPIO_OPTIONS SPI_SPEED_FCPU_DIV_16 | SPI_MODE_MASTER | SPI_ORDER_MSB_FIRST | SPI_SCK_LEAD_RISING | SPI_SAMPLE_LEADING + +/*========================================================================= + REGISTERS + -----------------------------------------------------------------------*/ + enum + { + BMP280_REGISTER_DIG_T1 = 0x88, + BMP280_REGISTER_DIG_T2 = 0x8A, + BMP280_REGISTER_DIG_T3 = 0x8C, + + BMP280_REGISTER_DIG_P1 = 0x8E, + BMP280_REGISTER_DIG_P2 = 0x90, + BMP280_REGISTER_DIG_P3 = 0x92, + BMP280_REGISTER_DIG_P4 = 0x94, + BMP280_REGISTER_DIG_P5 = 0x96, + BMP280_REGISTER_DIG_P6 = 0x98, + BMP280_REGISTER_DIG_P7 = 0x9A, + BMP280_REGISTER_DIG_P8 = 0x9C, + BMP280_REGISTER_DIG_P9 = 0x9E, + + BMP280_REGISTER_CHIPID = 0xD0, + BMP280_REGISTER_VERSION = 0xD1, + BMP280_REGISTER_SOFTRESET = 0xE0, + + BMP280_REGISTER_CAL26 = 0xE1, // R calibration stored in 0xE1-0xF0 + + BMP280_REGISTER_CONTROL = 0xF4, + BMP280_REGISTER_CONFIG = 0xF5, + BMP280_REGISTER_PRESSUREDATA = 0xF7, + BMP280_REGISTER_TEMPDATA = 0xFA, + }; + +/*=========================================================================*/ + + +/*========================================================================= + CALIBRATION DATA + -----------------------------------------------------------------------*/ + typedef struct + { + volatile uint8_t *cs_port; + uint8_t cs_pin; + uint16_t dig_T1; + int16_t dig_T2; + int16_t dig_T3; + + uint16_t dig_P1; + int16_t dig_P2; + int16_t dig_P3; + int16_t dig_P4; + int16_t dig_P5; + int16_t dig_P6; + int16_t dig_P7; + int16_t dig_P8; + int16_t dig_P9; + + uint8_t dig_H1; + int16_t dig_H2; + uint8_t dig_H3; + int16_t dig_H4; + int16_t dig_H5; + int8_t dig_H6; + } bmp280_sensor; +/*=========================================================================*/ + + +//make sure to configure the cs_pin as OUTPUT beforehand +//make sure to configure SPI beforehand +void bmp280_init(bmp280_sensor *sensor, volatile uint8_t *cs_port, uint8_t cs_pin); +float bmp280_readTemp(bmp280_sensor *sensor); +float bmp280_readPressure(bmp280_sensor *sensor); +float bmp280_readAltitude(bmp280_sensor *sensor, float sealevelp); + + +#endif
\ No newline at end of file diff --git a/usb-bmp280/Makefile b/usb-bmp280/Makefile new file mode 100644 index 0000000..5bfc049 --- /dev/null +++ b/usb-bmp280/Makefile @@ -0,0 +1,51 @@ +## +## spreadspace avr utils +## +## +## Copyright (C) 2013-2016 Christian Pointner <equinox@spreadspace.org> +## 2016 Bernhard Tittelbach <bernhard@tittelbach.org> +## +## This file is part of spreadspace avr utils. +## +## spreadspace avr utils is free software: you can redistribute it and/or modify +## it under the terms of the GNU General Public License as published by +## the Free Software Foundation, either version 3 of the License, or +## any later version. +## +## spreadspace avr utils is distributed in the hope that it will be useful, +## but WITHOUT ANY WARRANTY; without even the implied warranty of +## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +## GNU General Public License for more details. +## +## You should have received a copy of the GNU General Public License +## along with spreadspace avr utils. If not, see <http://www.gnu.org/licenses/>. +## + +NAME := usb-bmp280 +BOARD_TYPE := arduinoProMicro +OBJ := $(NAME).o +LIBS := util led lufa-descriptor-usbserial usbio bmp280 +EXTERNAL_LIBS := lufa +SPREADAVR_PATH := .. + +LUFA_PATH := $(SPREADAVR_PATH)/contrib/lufa-LUFA-151115 +LUFA_OPTS = -D USB_DEVICE_ONLY +LUFA_OPTS += -D DEVICE_STATE_AS_GPIOR=0 +LUFA_OPTS += -D ORDERED_EP_CONFIG +LUFA_OPTS += -D FIXED_CONTROL_ENDPOINT_SIZE=8 +LUFA_OPTS += -D FIXED_NUM_CONFIGURATIONS=1 +LUFA_OPTS += -D USE_FLASH_DESCRIPTORS +LUFA_OPTS += -D USE_STATIC_OPTIONS="(USB_DEVICE_OPT_FULLSPEED | USB_OPT_REG_ENABLED | USB_OPT_AUTO_PLL)" +LUFA_OPTS += -D INTERRUPT_CONTROL_ENDPOINT + +LUFA_OPTS += -D USB_MANUFACTURER="L\"xro\"" +LUFA_OPTS += -D USB_PRODUCT="L\"spreadspace usb-bmp280 example\"" + +LUFA_COMPONENTS := USB USBCLASS + +include $(SPREADAVR_PATH)/include.mk + +LDFLAGS += -Wl,-u,vfprintf -lprintf_flt -lm +AVRDUDE_PORT := $(shell $(SPREADAVR_PATH)/tools/detect_newest_acm) +#AVRDUDE_PORT := $(shell $(SPREADAVR_PATH)/tools/detect_lufa_cdc 1) +RESET_FUNC := $(SPREADAVR_PATH)/tools/reset_sparkfun diff --git a/usb-bmp280/usb-bmp280.c b/usb-bmp280/usb-bmp280.c new file mode 100644 index 0000000..0c19545 --- /dev/null +++ b/usb-bmp280/usb-bmp280.c @@ -0,0 +1,120 @@ +/* + * spreadspace avr utils - usb-bmp180 example + * + * + * Copyright (C) 2016 Bernhard Tittelbach <bernhard@tittelbach.org> + * basically this is refactored and enhanced code from: + * https://github.com/sparkfun/BMP180_Breakout_Arduino_Library + * Please buy the sparkfun people Beer when you see them!!!! + * + * This file is part of spreadspace avr utils. + * + * spreadspace avr utils is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * spreadspace avr utils is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with spreadspace avr utils. If not, see <http://www.gnu.org/licenses/>. + */ + +#include <avr/io.h> +#include <avr/wdt.h> +#include <avr/interrupt.h> +#include <util/delay.h> +#include <stdio.h> + +#include <LUFA/Drivers/Peripheral/SPI.h> + +#include "util.h" +#include "led.h" +#include "usbio.h" +#include "bmp280.h" + +#define ALTITUDE 353.0 // Altitude of Graz in Austria + +#define PIN_CS_S0 PD1 +#define REG_CS_S0 PIND +#define PIN_CS_S1 PD0 +#define REG_CS_S1 PIND + +#define OP_SETBIT |= +#define OP_CLEARBIT &= ~ +#define OP_CHECK & +#define PIN_SW(PORTDDRREG, PIN, OP) PORTDDRREG OP (1 << PIN) + +#define PINREG(x) x +#define DDRREG(x) *(&x+1) +#define PORTREG(x) *(&x+2) + +#define HIGHv OP_SETBIT +#define LOWv OP_CLEARBIT + +#define CS_SENSOR_0(LOWHIGH) (PIN_SW(PORTREG(REG_CS_S0),PIN_CS_S0,LOWHIGH)) +#define CS_SENSOR_1(LOWHIGH) (PIN_SW(PORTREG(REG_CS_S1),PIN_CS_S1,LOWHIGH)) +#define CS_SENSOR(x,LOWHIGH) CS_SENSOR_##x(LOWHIGH) + +void printCoeffs(bmp280_sensor *sensor) +{ + printf("Coeffients\r\n"); + printf("T1: %d, T2: %d, T3: %d\r\n", sensor->dig_T1, sensor->dig_T2, sensor->dig_T3); + printf("P1: %d, P2: %d, P3: %d, P4: %d, P5: %d, P6: %d, P7: %d, P8: %d, P9: %d\r\n",sensor->dig_P1,sensor->dig_P2,sensor->dig_P3,sensor->dig_P4,sensor->dig_P5,sensor->dig_P6,sensor->dig_P7,sensor->dig_P8,sensor->dig_P9); + printf("H1: %d, H2: %d, H3: %d, H4: %d, H5: %d, H6: %d\r\n",sensor->dig_H1,sensor->dig_H2,sensor->dig_H3 + ,sensor->dig_H4,sensor->dig_H5,sensor->dig_H6); +} + +int main(void) +{ + MCUSR &= ~(1 << WDRF); + wdt_disable(); + + cpu_init(); + led_init(); + usbio_init(); + sei(); + + bmp280_sensor sensor0; + bmp280_sensor sensor1; + + CS_SENSOR(0,HIGHv); + CS_SENSOR(1,HIGHv); + SPI_Init(BMP280_LUFA_SPIO_OPTIONS); + bmp280_init(&sensor0, &PORTREG(REG_CS_S0), PIN_CS_S0); + bmp280_init(&sensor1, &PORTREG(REG_CS_S1), PIN_CS_S1); + + printf("BMP280 initialized"); + + for(;;) { + + printf("\r\n== Sensor 0 ==\r\n"); + printCoeffs(&sensor0); + float temp = bmp280_readTemp(&sensor0); + float pressure = bmp280_readPressure(&sensor0); + printf("\r\nPressure: %.2f Pa @ %.2f degC\r\n", pressure, temp); + + printf("\r\n== Sensor 1 ==\r\n"); + printCoeffs(&sensor1); + temp = bmp280_readTemp(&sensor1); + pressure = bmp280_readPressure(&sensor1); + printf("\r\nPressure: %.2f Pa @ %.2f degC\r\n", pressure, temp); + + usbio_task(); + + int16_t BytesReceived = usbio_bytes_received(); + while(BytesReceived > 0) { + int ReceivedByte = fgetc(stdin); + if(ReceivedByte != EOF) { + if (ReceivedByte == '!') + reset2bootloader(); + } + BytesReceived--; + } + + _delay_ms(2000); // Pause for 5 seconds. + } +} |