1
0
mirror of https://github.com/kidoman/embd synced 2025-01-20 18:37:17 +01:00
embd/sensor/bmp085/bmp085.go

418 lines
8.1 KiB
Go
Raw Normal View History

2013-12-07 23:11:06 +05:30
// Package bmp085 allows interfacing with Bosch BMP085 barometric pressure sensor. This sensor
// has the ability to provided compensated temperature and pressure readings.
package bmp085
import (
"log"
"math"
"sync"
"time"
2014-03-03 00:51:23 +05:30
"github.com/kidoman/embd"
2013-12-07 23:11:06 +05:30
)
const (
address = 0x77
calAc1 = 0xAA
calAc2 = 0xAC
calAc3 = 0xAE
calAc4 = 0xB0
calAc5 = 0xB2
calAc6 = 0xB4
calB1 = 0xB6
calB2 = 0xB8
calMB = 0xBA
calMC = 0xBC
calMD = 0xBE
control = 0xF4
tempData = 0xF6
pressureData = 0xF6
readTempCmd = 0x2E
readPressureCmd = 0x34
tempReadDelay = 5 * time.Millisecond
p0 = 101325
pollDelay = 250
)
2014-03-23 15:47:58 +05:30
// BMP085 represents a Bosch BMP085 barometric sensor.
2014-03-03 00:51:23 +05:30
type BMP085 struct {
Bus embd.I2CBus
Poll int
Debug bool
2013-12-07 23:11:06 +05:30
oss uint
ac1, ac2, ac3 int16
ac4, ac5, ac6 uint16
b1, b2, mb, mc, md int16
b5 int32
calibrated bool
2014-03-03 00:51:23 +05:30
cmu sync.RWMutex
2013-12-07 23:11:06 +05:30
temps chan uint16
pressures chan int32
altitudes chan float64
quit chan struct{}
}
2014-03-23 15:47:58 +05:30
// New returns a handle to a BMP085 sensor.
2014-03-03 00:51:23 +05:30
func New(bus embd.I2CBus) *BMP085 {
return &BMP085{Bus: bus, Poll: pollDelay}
2013-12-07 23:11:06 +05:30
}
2014-03-03 00:51:23 +05:30
func (d *BMP085) calibrate() (err error) {
2013-12-07 23:11:06 +05:30
d.cmu.RLock()
if d.calibrated {
d.cmu.RUnlock()
return
}
d.cmu.RUnlock()
d.cmu.Lock()
defer d.cmu.Unlock()
readInt16 := func(reg byte) (value int16, err error) {
var v uint16
2014-03-03 00:51:23 +05:30
if v, err = d.Bus.ReadWordFromReg(address, reg); err != nil {
2013-12-07 23:11:06 +05:30
return
}
value = int16(v)
return
}
readUInt16 := func(reg byte) (value uint16, err error) {
var v uint16
2014-03-03 00:51:23 +05:30
if v, err = d.Bus.ReadWordFromReg(address, reg); err != nil {
2013-12-07 23:11:06 +05:30
return
}
value = uint16(v)
return
}
d.ac1, err = readInt16(calAc1)
if err != nil {
return
}
d.ac2, err = readInt16(calAc2)
if err != nil {
return
}
d.ac3, err = readInt16(calAc3)
if err != nil {
return
}
d.ac4, err = readUInt16(calAc4)
if err != nil {
return
}
d.ac5, err = readUInt16(calAc5)
if err != nil {
return
}
d.ac6, err = readUInt16(calAc6)
if err != nil {
return
}
d.b1, err = readInt16(calB1)
if err != nil {
return
}
d.b2, err = readInt16(calB2)
if err != nil {
return
}
d.mb, err = readInt16(calMB)
if err != nil {
return
}
d.mc, err = readInt16(calMC)
if err != nil {
return
}
d.md, err = readInt16(calMD)
if err != nil {
return
}
d.calibrated = true
2014-03-03 00:51:23 +05:30
if d.Debug {
2013-12-07 23:11:06 +05:30
log.Print("bmp085: calibration data retrieved")
log.Printf("bmp085: param AC1 = %v", d.ac1)
log.Printf("bmp085: param AC2 = %v", d.ac2)
log.Printf("bmp085: param AC3 = %v", d.ac3)
log.Printf("bmp085: param AC4 = %v", d.ac4)
log.Printf("bmp085: param AC5 = %v", d.ac5)
log.Printf("bmp085: param AC6 = %v", d.ac6)
log.Printf("bmp085: param B1 = %v", d.b1)
log.Printf("bmp085: param B2 = %v", d.b2)
log.Printf("bmp085: param MB = %v", d.mb)
log.Printf("bmp085: param MC = %v", d.mc)
log.Printf("bmp085: param MD = %v", d.md)
}
return
}
2014-03-03 00:51:23 +05:30
func (d *BMP085) readUncompensatedTemp() (temp uint16, err error) {
if err = d.Bus.WriteByteToReg(address, control, readTempCmd); err != nil {
2013-12-07 23:11:06 +05:30
return
}
time.Sleep(tempReadDelay)
2014-03-03 00:51:23 +05:30
if temp, err = d.Bus.ReadWordFromReg(address, tempData); err != nil {
2013-12-07 23:11:06 +05:30
return
}
return
}
2014-03-03 00:51:23 +05:30
func (d *BMP085) calcTemp(utemp uint16) uint16 {
2013-12-07 23:11:06 +05:30
x1 := ((int(utemp) - int(d.ac6)) * int(d.ac5)) >> 15
x2 := (int(d.mc) << 11) / (x1 + int(d.md))
d.cmu.Lock()
d.b5 = int32(x1 + x2)
d.cmu.Unlock()
return uint16((d.b5 + 8) >> 4)
}
2014-03-03 00:51:23 +05:30
func (d *BMP085) measureTemp() (temp uint16, err error) {
2013-12-07 23:11:06 +05:30
if err = d.calibrate(); err != nil {
return
}
var utemp uint16
if utemp, err = d.readUncompensatedTemp(); err != nil {
return
}
2014-03-03 00:51:23 +05:30
if d.Debug {
2013-12-07 23:11:06 +05:30
log.Printf("bcm085: uncompensated temp: %v", utemp)
}
temp = d.calcTemp(utemp)
2014-03-03 00:51:23 +05:30
if d.Debug {
2013-12-07 23:11:06 +05:30
log.Printf("bcm085: compensated temp %v", temp)
}
return
}
2013-12-09 02:49:39 +05:30
// Temperature returns the current temperature reading.
2014-03-03 00:51:23 +05:30
func (d *BMP085) Temperature() (temp float64, err error) {
2013-12-07 23:11:06 +05:30
select {
case t := <-d.temps:
temp = float64(t) / 10
return
default:
2014-03-03 00:51:23 +05:30
if d.Debug {
2013-12-07 23:11:06 +05:30
log.Print("bcm085: no temps available... measuring")
}
var t uint16
t, err = d.measureTemp()
if err != nil {
return
}
temp = float64(t) / 10
return
}
}
2014-03-03 00:51:23 +05:30
func (d *BMP085) readUncompensatedPressure() (pressure uint32, err error) {
if err = d.Bus.WriteByteToReg(address, control, byte(readPressureCmd+(d.oss<<6))); err != nil {
2013-12-07 23:11:06 +05:30
return
}
time.Sleep(time.Duration(2+(3<<d.oss)) * time.Millisecond)
data := make([]byte, 3)
2014-03-03 00:51:23 +05:30
if err = d.Bus.ReadFromReg(address, pressureData, data); err != nil {
2013-12-07 23:11:06 +05:30
return
}
pressure = ((uint32(data[0]) << 16) | (uint32(data[1]) << 8) | uint32(data[2])) >> (8 - d.oss)
return
}
2014-03-03 00:51:23 +05:30
func (d *BMP085) calcPressure(upressure uint32) (p int32) {
2013-12-07 23:11:06 +05:30
var x1, x2, x3 int32
l := func(s string, v interface{}) {
2014-03-03 00:51:23 +05:30
if d.Debug {
2013-12-07 23:11:06 +05:30
log.Printf("bcm085: %v = %v", s, v)
}
}
b6 := d.b5 - 4000
l("b6", b6)
// Calculate b3
x1 = (int32(d.b2) * int32(b6*b6) >> 12) >> 11
x2 = (int32(d.ac2) * b6) >> 11
x3 = x1 + x2
b3 := (((int32(d.ac1)*4 + x3) << d.oss) + 2) >> 2
l("x1", x1)
l("x2", x2)
l("x3", x3)
l("b3", b3)
// Calculate b4
x1 = (int32(d.ac3) * b6) >> 13
x2 = (int32(d.b1) * ((b6 * b6) >> 12)) >> 16
x3 = ((x1 + x2) + 2) >> 2
b4 := (uint32(d.ac4) * uint32(x3+32768)) >> 15
l("x1", x1)
l("x2", x2)
l("x3", x3)
l("b4", b4)
b7 := (uint32(upressure-uint32(b3)) * (50000 >> d.oss))
if b7 < 0x80000000 {
p = int32((b7 << 1) / b4)
} else {
p = int32((b7 / b4) << 1)
}
l("b7", b7)
l("p", p)
x1 = (p >> 8) * (p >> 8)
x1 = (x1 * 3038) >> 16
x2 = (-7357 * p) >> 16
p += (x1 + x2 + 3791) >> 4
l("x1", x1)
l("x2", x2)
l("x3", x3)
l("p", p)
return
}
2014-03-03 00:51:23 +05:30
func (d *BMP085) calcAltitude(pressure int32) float64 {
2013-12-07 23:11:06 +05:30
return 44330 * (1 - math.Pow(float64(pressure)/p0, 0.190295))
}
2014-03-03 00:51:23 +05:30
func (d *BMP085) measurePressureAndAltitude() (pressure int32, altitude float64, err error) {
2013-12-07 23:11:06 +05:30
if err = d.calibrate(); err != nil {
return
}
var upressure uint32
if upressure, err = d.readUncompensatedPressure(); err != nil {
return
}
2014-03-03 00:51:23 +05:30
if d.Debug {
2013-12-07 23:11:06 +05:30
log.Printf("bcm085: uncompensated pressure: %v", upressure)
}
pressure = d.calcPressure(upressure)
2014-03-03 00:51:23 +05:30
if d.Debug {
2013-12-07 23:11:06 +05:30
log.Printf("bcm085: compensated pressure %v", pressure)
}
altitude = d.calcAltitude(pressure)
2014-03-03 00:51:23 +05:30
if d.Debug {
2013-12-07 23:11:06 +05:30
log.Printf("bcm085: calculated altitude %v", altitude)
}
return
}
2013-12-09 02:49:39 +05:30
// Pressure returns the current pressure reading.
2014-03-03 00:51:23 +05:30
func (d *BMP085) Pressure() (pressure int, err error) {
2013-12-07 23:11:06 +05:30
if err = d.calibrate(); err != nil {
return
}
select {
case p := <-d.pressures:
pressure = int(p)
return
default:
2014-03-03 00:51:23 +05:30
if d.Debug {
2013-12-07 23:11:06 +05:30
log.Print("bcm085: no pressures available... measuring")
}
var p int32
p, _, err = d.measurePressureAndAltitude()
if err != nil {
return
}
pressure = int(p)
return
}
}
2013-12-09 02:49:39 +05:30
// Altitude returns the current altitude reading.
2014-03-03 00:51:23 +05:30
func (d *BMP085) Altitude() (altitude float64, err error) {
2013-12-07 23:11:06 +05:30
if err = d.calibrate(); err != nil {
return
}
select {
case altitude = <-d.altitudes:
return
default:
2014-03-03 00:51:23 +05:30
if d.Debug {
2013-12-07 23:11:06 +05:30
log.Print("bcm085: no altitudes available... measuring")
}
_, altitude, err = d.measurePressureAndAltitude()
if err != nil {
return
}
return
}
}
2013-12-09 02:49:39 +05:30
// Run starts the sensor data acquisition loop.
2014-03-03 00:51:23 +05:30
func (d *BMP085) Run() (err error) {
2013-12-07 23:11:06 +05:30
go func() {
d.quit = make(chan struct{})
2014-03-03 00:51:23 +05:30
timer := time.Tick(time.Duration(d.Poll) * time.Millisecond)
2013-12-07 23:11:06 +05:30
var temp uint16
var pressure int32
var altitude float64
for {
select {
case <-timer:
t, err := d.measureTemp()
if err == nil {
temp = t
}
if err == nil && d.temps == nil {
d.temps = make(chan uint16)
}
p, a, err := d.measurePressureAndAltitude()
if err == nil {
pressure = p
altitude = a
}
if err == nil && d.pressures == nil && d.altitudes == nil {
d.pressures = make(chan int32)
d.altitudes = make(chan float64)
}
case d.temps <- temp:
case d.pressures <- pressure:
case d.altitudes <- altitude:
case <-d.quit:
d.temps = nil
d.pressures = nil
d.altitudes = nil
return
}
}
}()
return
}
// Close.
2014-03-03 00:51:23 +05:30
func (d *BMP085) Close() {
2013-12-07 23:11:06 +05:30
if d.quit != nil {
d.quit <- struct{}{}
}
}