1
0
mirror of https://github.com/kidoman/embd synced 2025-01-22 03:17:18 +01:00
embd/sensor/lsm303/lsm303.go

165 lines
3.2 KiB
Go
Raw Normal View History

2013-12-07 23:11:06 +05:30
// Package lsm303 allows interfacing with the LSM303 magnetometer.
package lsm303
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 (
magAddress = 0x1E
magConfigRegA = 0x00
2013-12-09 02:49:39 +05:30
MagHz75 = 0x00 // ODR = 0.75 Hz
Mag1Hz5 = 0x04 // ODR = 1.5 Hz
Mag3Hz = 0x08 // ODR = 3 Hz
Mag7Hz5 = 0x0C // ODR = 7.5 Hz
Mag15Hz = 0x10 // ODR = 15 Hz
Mag30Hz = 0x14 // ODR = 30 Hz
Mag75Hz = 0x18 // ODR = 75 Hz
MagNormal = 0x00 // Normal mode
MagPositiveBias = 0x01 // Positive bias mode
MagNegativeBias = 0x02 // Negative bias mode
MagCRADefault = Mag15Hz | MagNormal // 15 Hz and normal mode is the default
2013-12-07 23:11:06 +05:30
magModeReg = 0x02
2013-12-09 02:49:39 +05:30
MagContinuous = 0x00 // Continuous conversion mode
2013-12-09 02:51:15 +05:30
MagSingle = 0x01 // Single conversion mode
2013-12-09 02:49:39 +05:30
MagSleep = 0x03 // Sleep mode
2013-12-07 23:11:06 +05:30
2013-12-09 02:49:39 +05:30
MagMRDefault = MagContinuous // Continuous conversion is the default
2013-12-07 23:11:06 +05:30
magDataSignal = 0x02
magData = 0x03
pollDelay = 250
)
2014-03-03 00:51:23 +05:30
type LSM303 struct {
Bus embd.I2CBus
Poll int
2013-12-07 23:11:06 +05:30
initialized bool
2014-03-03 00:51:23 +05:30
mu sync.RWMutex
2013-12-07 23:11:06 +05:30
headings chan float64
quit chan struct{}
2014-03-03 00:51:23 +05:30
Debug bool
2013-12-07 23:11:06 +05:30
}
// New creates a new LSM303 interface. The bus variable controls
// the I2C bus used to communicate with the device.
2014-03-03 00:51:23 +05:30
func New(bus embd.I2CBus) *LSM303 {
return &LSM303{Bus: bus, Poll: pollDelay}
2013-12-07 23:11:06 +05:30
}
// Initialize the device
2014-03-03 00:51:23 +05:30
func (d *LSM303) setup() error {
2013-12-07 23:11:06 +05:30
d.mu.RLock()
if d.initialized {
d.mu.RUnlock()
2014-03-03 00:51:23 +05:30
return nil
2013-12-07 23:11:06 +05:30
}
d.mu.RUnlock()
d.mu.Lock()
defer d.mu.Unlock()
2014-03-03 00:51:23 +05:30
if err := d.Bus.WriteByteToReg(magAddress, magConfigRegA, MagCRADefault); err != nil {
return err
2013-12-07 23:11:06 +05:30
}
2014-03-03 00:51:23 +05:30
if err := d.Bus.WriteByteToReg(magAddress, magModeReg, MagMRDefault); err != nil {
return err
2013-12-07 23:11:06 +05:30
}
d.initialized = true
2014-03-03 00:51:23 +05:30
return nil
2013-12-07 23:11:06 +05:30
}
2014-03-03 00:51:23 +05:30
func (d *LSM303) measureHeading() (float64, error) {
if err := d.setup(); err != nil {
return 0, err
2013-12-07 23:11:06 +05:30
}
2014-03-03 00:51:23 +05:30
if _, err := d.Bus.ReadByteFromReg(magAddress, magDataSignal); err != nil {
return 0, err
2013-12-07 23:11:06 +05:30
}
data := make([]byte, 6)
2014-03-03 00:51:23 +05:30
if err := d.Bus.ReadFromReg(magAddress, magData, data); err != nil {
return 0, err
2013-12-07 23:11:06 +05:30
}
x := int16(data[0])<<8 | int16(data[1])
y := int16(data[2])<<8 | int16(data[3])
2014-03-03 00:51:23 +05:30
heading := math.Atan2(float64(y), float64(x)) / math.Pi * 180
2013-12-07 23:11:06 +05:30
if heading < 0 {
heading += 360
}
2014-03-03 00:51:23 +05:30
return heading, nil
2013-12-07 23:11:06 +05:30
}
2013-12-09 02:49:39 +05:30
// Heading returns the current heading [0, 360).
2014-03-03 00:51:23 +05:30
func (d *LSM303) Heading() (float64, error) {
2013-12-07 23:11:06 +05:30
select {
2014-03-03 00:51:23 +05:30
case heading := <-d.headings:
return heading, nil
2013-12-07 23:11:06 +05:30
default:
2014-03-03 00:51:23 +05:30
if d.Debug {
2013-12-07 23:11:06 +05:30
log.Print("lsm303: no headings available... measuring")
}
return d.measureHeading()
}
}
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 *LSM303) 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 heading float64
for {
select {
case <-timer:
h, err := d.measureHeading()
if err == nil {
heading = h
}
if err == nil && d.headings == nil {
d.headings = make(chan float64)
}
case d.headings <- heading:
case <-d.quit:
d.headings = nil
return
}
}
}()
return
}
// Close the sensor data acquisition loop and put the LSM303 into sleep mode.
2014-03-03 00:51:23 +05:30
func (d *LSM303) Close() error {
2013-12-07 23:11:06 +05:30
if d.quit != nil {
d.quit <- struct{}{}
}
2014-03-03 00:51:23 +05:30
return d.Bus.WriteByteToReg(magAddress, magModeReg, MagSleep)
2013-12-07 23:11:06 +05:30
}