1
0
mirror of https://github.com/kidoman/embd synced 2025-01-23 03:47:17 +01:00
embd/sensor/bh1750fvi/bh1750fvi.go

124 lines
2.6 KiB
Go
Raw Normal View History

// Package BH1750FVI allows interfacing with the BH1750FVI ambient light sensor through I2C protocol.
2013-12-21 01:52:36 +05:30
package bh1750fvi
2013-12-17 20:37:58 +05:30
import (
"log"
"sync"
"time"
2014-03-03 00:51:23 +05:30
"github.com/kidoman/embd"
2013-12-17 20:37:58 +05:30
)
//accuracy = sensorValue/actualValue] (min = 0.96, typ = 1.2, max = 1.44
2013-12-17 20:37:58 +05:30
const (
High = "H"
High2 = "H2"
2013-12-17 20:37:58 +05:30
measurementAcuuracy = 1.2
defReadReg = 0x00
2013-12-17 20:37:58 +05:30
sensorI2cAddr = 0x23
2013-12-17 20:37:58 +05:30
highResOpCode = 0x10
highResMode2OpCode = 0x11
2013-12-17 20:37:58 +05:30
pollDelay = 150
)
2014-03-03 00:51:23 +05:30
type BH1750FVI struct {
Bus embd.I2CBus
Poll int
2013-12-17 20:37:58 +05:30
2014-03-03 00:51:23 +05:30
mu sync.RWMutex
2013-12-17 20:37:58 +05:30
lightingReadings chan float64
quit chan bool
i2cAddr byte
operationCode byte
2013-12-17 20:37:58 +05:30
}
2014-03-03 00:51:23 +05:30
func New(mode string, bus embd.I2CBus) *BH1750FVI {
2013-12-17 20:37:58 +05:30
switch mode {
case High:
2014-03-03 00:51:23 +05:30
return &BH1750FVI{Bus: bus, i2cAddr: sensorI2cAddr, operationCode: highResOpCode, Poll: pollDelay}
case High2:
2014-03-03 00:51:23 +05:30
return &BH1750FVI{Bus: bus, i2cAddr: sensorI2cAddr, operationCode: highResMode2OpCode, Poll: pollDelay}
2013-12-17 20:37:58 +05:30
default:
2014-03-03 00:51:23 +05:30
return &BH1750FVI{Bus: bus, i2cAddr: sensorI2cAddr, operationCode: highResOpCode, Poll: pollDelay}
2013-12-17 20:37:58 +05:30
}
}
// NewHighMode returns a BH1750FVI inteface on high resolution mode (1lx resolution)
2014-03-03 00:51:23 +05:30
func NewHighMode(bus embd.I2CBus) *BH1750FVI {
return New(High, bus)
}
2013-12-17 20:37:58 +05:30
// NewHighMode returns a BH1750FVI inteface on high resolution mode2 (0.5lx resolution)
2014-03-03 00:51:23 +05:30
func NewHigh2Mode(bus embd.I2CBus) *BH1750FVI {
return New(High2, bus)
}
2013-12-17 20:37:58 +05:30
2014-03-03 00:51:23 +05:30
func (d *BH1750FVI) measureLighting() (lighting float64, err error) {
err = d.Bus.WriteByte(d.i2cAddr, d.operationCode)
2013-12-17 20:37:58 +05:30
if err != nil {
2013-12-21 01:52:36 +05:30
log.Print("bh1750fvi: Failed to initialize sensor")
2013-12-17 20:37:58 +05:30
return
}
time.Sleep(180 * time.Millisecond)
2013-12-17 20:37:58 +05:30
var reading uint16
2014-03-03 00:51:23 +05:30
if reading, err = d.Bus.ReadWordFromReg(d.i2cAddr, defReadReg); err != nil {
2013-12-17 20:37:58 +05:30
return
}
lighting = float64(int16(reading)) / measurementAcuuracy
2013-12-17 20:37:58 +05:30
return
}
// Lighting returns the ambient lighting in lx.
2014-03-03 00:51:23 +05:30
func (d *BH1750FVI) Lighting() (lighting float64, err error) {
2013-12-17 20:37:58 +05:30
select {
case lighting = <-d.lightingReadings:
2013-12-17 20:37:58 +05:30
return
default:
return d.measureLighting()
2013-12-17 20:37:58 +05:30
}
}
// Run starts continuous sensor data acquisition loop.
2014-03-03 00:51:23 +05:30
func (d *BH1750FVI) Run() (err error) {
2013-12-17 20:37:58 +05:30
go func() {
d.quit = make(chan bool)
2013-12-17 20:37:58 +05:30
2014-03-03 00:51:23 +05:30
timer := time.Tick(time.Duration(d.Poll) * time.Millisecond)
2013-12-17 20:37:58 +05:30
var lighting float64
for {
select {
case d.lightingReadings <- lighting:
2013-12-17 20:37:58 +05:30
case <-timer:
if l, err := d.measureLighting(); err == nil {
2013-12-17 20:37:58 +05:30
lighting = l
}
if err == nil && d.lightingReadings == nil {
d.lightingReadings = make(chan float64)
2013-12-17 20:37:58 +05:30
}
case <-d.quit:
d.lightingReadings = nil
2013-12-17 20:37:58 +05:30
return
}
}
}()
return
}
// Close.
2014-03-03 00:51:23 +05:30
func (d *BH1750FVI) Close() {
if d.quit != nil {
d.quit <- true
2013-12-17 20:37:58 +05:30
}
return
}