2018-08-13 17:03:47 +02:00
|
|
|
package i2c
|
|
|
|
|
|
|
|
import (
|
|
|
|
"strconv"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"gobot.io/x/gobot"
|
|
|
|
)
|
|
|
|
|
|
|
|
const grovePiAddress = 0x04
|
|
|
|
|
|
|
|
// Commands format
|
|
|
|
const (
|
|
|
|
CommandReadDigital = 1
|
|
|
|
CommandWriteDigital = 2
|
|
|
|
CommandReadAnalog = 3
|
|
|
|
CommandWriteAnalog = 4
|
2018-08-14 00:21:46 +02:00
|
|
|
CommandPinMode = 5
|
2018-08-13 17:03:47 +02:00
|
|
|
CommandReadDHT = 40
|
|
|
|
)
|
|
|
|
|
|
|
|
// GrovePiDriver is a driver for the GrovePi for I²C bus interface.
|
|
|
|
type GrovePiDriver struct {
|
2018-08-14 00:21:46 +02:00
|
|
|
name string
|
|
|
|
digitalPins map[int]string
|
|
|
|
analogPins map[int]string
|
|
|
|
connector Connector
|
|
|
|
connection Connection
|
2018-08-13 17:03:47 +02:00
|
|
|
Config
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewGrovePiDriver creates a new driver with specified i2c interface
|
|
|
|
// Params:
|
|
|
|
// conn Connector - the Adaptor to use with this Driver
|
|
|
|
//
|
|
|
|
// Optional params:
|
|
|
|
// i2c.WithBus(int): bus to use with this driver
|
|
|
|
// i2c.WithAddress(int): address to use with this driver
|
|
|
|
//
|
|
|
|
func NewGrovePiDriver(a Connector, options ...func(Config)) *GrovePiDriver {
|
|
|
|
d := &GrovePiDriver{
|
2018-08-14 00:21:46 +02:00
|
|
|
name: gobot.DefaultName("GrovePi"),
|
|
|
|
digitalPins: make(map[int]string),
|
|
|
|
analogPins: make(map[int]string),
|
|
|
|
connector: a,
|
|
|
|
Config: NewConfig(),
|
2018-08-13 17:03:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, option := range options {
|
|
|
|
option(d)
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: add commands for API
|
|
|
|
return d
|
|
|
|
}
|
|
|
|
|
|
|
|
// Name returns the Name for the Driver
|
|
|
|
func (d *GrovePiDriver) Name() string { return d.name }
|
|
|
|
|
|
|
|
// SetName sets the Name for the Driver
|
|
|
|
func (d *GrovePiDriver) SetName(n string) { d.name = n }
|
|
|
|
|
|
|
|
// Connection returns the connection for the Driver
|
|
|
|
func (d *GrovePiDriver) Connection() gobot.Connection { return d.connector.(gobot.Connection) }
|
|
|
|
|
|
|
|
// Start initialized the GrovePi
|
|
|
|
func (d *GrovePiDriver) Start() (err error) {
|
|
|
|
bus := d.GetBusOrDefault(d.connector.GetDefaultBus())
|
|
|
|
address := d.GetAddressOrDefault(grovePiAddress)
|
|
|
|
|
|
|
|
d.connection, err = d.connector.GetConnection(address, bus)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Halt returns true if devices is halted successfully
|
|
|
|
func (d *GrovePiDriver) Halt() (err error) { return }
|
|
|
|
|
|
|
|
// AnalogRead returns value from analog pin implementing the AnalogReader interface.
|
|
|
|
func (d *GrovePiDriver) AnalogRead(pin string) (value int, err error) {
|
|
|
|
// TODO: strip off the leading "A"
|
|
|
|
|
2018-08-14 00:21:46 +02:00
|
|
|
var pinNum int
|
|
|
|
pinNum, err = strconv.Atoi(pin)
|
2018-08-13 17:03:47 +02:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
2018-08-14 00:21:46 +02:00
|
|
|
|
|
|
|
if dir, ok := d.analogPins[pinNum]; !ok || dir != "input" {
|
|
|
|
d.PinMode(byte(pinNum), "input")
|
|
|
|
d.analogPins[pinNum] = "input"
|
|
|
|
}
|
|
|
|
|
|
|
|
value, err = d.ReadAnalog(byte(pinNum))
|
2018-08-13 17:03:47 +02:00
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// ReadAnalog reads analog value from the GrovePi
|
|
|
|
func (d *GrovePiDriver) ReadAnalog(pin byte) (int, error) {
|
|
|
|
b := []byte{1, CommandReadAnalog, pin, 0, 0}
|
|
|
|
_, err := d.connection.Write(b)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
time.Sleep(100 * time.Millisecond)
|
|
|
|
|
|
|
|
d.connection.Write([]byte{1})
|
|
|
|
d.connection.ReadByte()
|
|
|
|
|
|
|
|
data := make([]byte, 4)
|
|
|
|
d.connection.Write([]byte{1})
|
|
|
|
_, err = d.connection.Read(data)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
v1 := int(data[1])
|
|
|
|
v2 := int(data[2])
|
|
|
|
return ((v1 * 256) + v2), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ReadDigital reads digitally to the GrovePi
|
2018-08-14 00:21:46 +02:00
|
|
|
func (d *GrovePiDriver) ReadDigital(pin byte) (val int, err error) {
|
|
|
|
buf := []byte{1, CommandReadDigital, pin, 0, 0}
|
|
|
|
_, err = d.connection.Write(buf)
|
2018-08-13 17:03:47 +02:00
|
|
|
if err != nil {
|
2018-08-14 00:21:46 +02:00
|
|
|
return
|
2018-08-13 17:03:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
time.Sleep(100 * time.Millisecond)
|
|
|
|
|
|
|
|
d.connection.Write([]byte{1})
|
2018-08-14 00:21:46 +02:00
|
|
|
v, err := d.connection.ReadByte()
|
2018-08-13 17:03:47 +02:00
|
|
|
if err != nil {
|
2018-08-14 00:21:46 +02:00
|
|
|
return
|
2018-08-13 17:03:47 +02:00
|
|
|
}
|
|
|
|
|
2018-08-14 00:21:46 +02:00
|
|
|
return int(v), err
|
|
|
|
}
|
|
|
|
|
|
|
|
// DigitalRead performs a read on a digital pin.
|
|
|
|
func (d *GrovePiDriver) DigitalRead(pin string) (val int, err error) {
|
|
|
|
// TODO: strip off the leading "D"
|
|
|
|
var pinNum int
|
|
|
|
pinNum, err = strconv.Atoi(pin)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if dir, ok := d.digitalPins[pinNum]; !ok || dir != "input" {
|
|
|
|
d.PinMode(byte(pinNum), "input")
|
|
|
|
d.digitalPins[pinNum] = "input"
|
|
|
|
}
|
|
|
|
|
|
|
|
val, err = d.ReadDigital(byte(pinNum))
|
|
|
|
|
|
|
|
return
|
2018-08-13 17:03:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// WriteDigital writes digitally to the GrovePi
|
|
|
|
func (d *GrovePiDriver) WriteDigital(pin byte, val byte) error {
|
|
|
|
buf := []byte{1, CommandWriteDigital, pin, val, 0}
|
|
|
|
_, err := d.connection.Write(buf)
|
|
|
|
time.Sleep(100 * time.Millisecond)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// DigitalWrite writes a value to a specific digital pin implementing the DigitalWriter interface.
|
|
|
|
func (d *GrovePiDriver) DigitalWrite(pin string, val byte) (err error) {
|
|
|
|
// TODO: strip off the leading "D"
|
2018-08-14 00:21:46 +02:00
|
|
|
var pinNum int
|
|
|
|
pinNum, err = strconv.Atoi(pin)
|
2018-08-13 17:03:47 +02:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
2018-08-14 00:21:46 +02:00
|
|
|
|
|
|
|
if dir, ok := d.digitalPins[pinNum]; !ok || dir != "output" {
|
|
|
|
d.PinMode(byte(pinNum), "output")
|
|
|
|
d.digitalPins[pinNum] = "output"
|
|
|
|
}
|
|
|
|
|
|
|
|
err = d.WriteDigital(byte(pinNum), val)
|
2018-08-13 17:03:47 +02:00
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// WriteAnalog writes analog to the GrovePi
|
|
|
|
func (d *GrovePiDriver) WriteAnalog(pin byte, val byte) error {
|
|
|
|
buf := []byte{1, CommandWriteAnalog, pin, val, 0}
|
|
|
|
_, err := d.connection.Write(buf)
|
|
|
|
time.Sleep(100 * time.Millisecond)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2018-08-14 00:21:46 +02:00
|
|
|
// PinMode sets the pin mode to input or output.
|
|
|
|
func (d *GrovePiDriver) PinMode(pin byte, mode string) error {
|
|
|
|
var b []byte
|
|
|
|
if mode == "output" {
|
|
|
|
b = []byte{1, CommandPinMode, pin, 1, 0}
|
|
|
|
} else {
|
|
|
|
b = []byte{1, CommandPinMode, pin, 0, 0}
|
|
|
|
}
|
|
|
|
_, err := d.connection.Write(b)
|
|
|
|
time.Sleep(100 * time.Millisecond)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-08-13 17:03:47 +02:00
|
|
|
// ReadDHT returns temperature and humidity from DHT sensor
|
|
|
|
func (d *GrovePiDriver) ReadDHT(pin byte, size int) ([]byte, error) {
|
|
|
|
cmd := []byte{1, CommandReadDHT, pin, 0, 0}
|
|
|
|
|
|
|
|
// prepare and read raw data
|
|
|
|
_, err := d.connection.Write(cmd)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
time.Sleep(600 * time.Millisecond)
|
|
|
|
d.connection.Write([]byte{1})
|
|
|
|
d.connection.ReadByte()
|
|
|
|
time.Sleep(100 * time.Millisecond)
|
|
|
|
|
|
|
|
data := make([]byte, size)
|
|
|
|
d.connection.Write([]byte{1})
|
|
|
|
_, err = d.connection.Read(data)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return data, err
|
|
|
|
}
|