1
0
mirror of https://github.com/hybridgroup/gobot.git synced 2025-05-02 22:17:12 +08:00

383 lines
8.1 KiB
Go
Raw Normal View History

package edison
import (
"fmt"
"io"
"os"
"strconv"
"github.com/hybridgroup/gobot"
"github.com/hybridgroup/gobot/sysfs"
)
2014-11-07 18:37:57 -08:00
func writeFile(path string, data []byte) (i int, err error) {
file, err := sysfs.OpenFile(path, os.O_WRONLY, 0644)
defer file.Close()
if err != nil {
return
2014-11-04 12:21:18 -08:00
}
2014-11-07 18:37:57 -08:00
return file.Write(data)
2014-11-04 12:21:18 -08:00
}
2014-11-07 18:37:57 -08:00
func readFile(path string) ([]byte, error) {
file, err := sysfs.OpenFile(path, os.O_RDONLY, 0644)
defer file.Close()
if err != nil {
return make([]byte, 0), err
}
buf := make([]byte, 200)
var i = 0
i, err = file.Read(buf)
if i == 0 {
return buf, err
}
return buf[:i], err
2014-11-04 12:21:18 -08:00
}
type mux struct {
pin int
2014-10-30 13:41:27 -07:00
value int
}
type sysfsPin struct {
pin int
resistor int
levelShifter int
pwmPin int
mux []mux
}
type EdisonAdaptor struct {
gobot.Adaptor
2014-11-03 19:02:33 -08:00
tristate sysfs.DigitalPin
digitalPins map[int]sysfs.DigitalPin
pwmPins map[int]*pwmPin
i2cDevice io.ReadWriteCloser
connect func(e *EdisonAdaptor)
}
var sysfsPinMap = map[string]sysfsPin{
"0": sysfsPin{
pin: 130,
resistor: 216,
levelShifter: 248,
pwmPin: -1,
mux: []mux{},
},
"1": sysfsPin{
pin: 131,
resistor: 217,
levelShifter: 249,
pwmPin: -1,
mux: []mux{},
},
"2": sysfsPin{
pin: 128,
resistor: 218,
levelShifter: 250,
pwmPin: -1,
mux: []mux{},
},
"3": sysfsPin{
pin: 12,
resistor: 219,
levelShifter: 251,
pwmPin: 0,
mux: []mux{},
},
"4": sysfsPin{
pin: 129,
resistor: 220,
levelShifter: 252,
pwmPin: -1,
mux: []mux{},
},
"5": sysfsPin{
pin: 13,
resistor: 221,
levelShifter: 253,
pwmPin: 1,
mux: []mux{},
},
"6": sysfsPin{
pin: 182,
resistor: 222,
levelShifter: 254,
pwmPin: 2,
mux: []mux{},
},
"7": sysfsPin{
pin: 48,
resistor: 223,
levelShifter: 255,
pwmPin: -1,
mux: []mux{},
},
"8": sysfsPin{
pin: 49,
resistor: 224,
levelShifter: 256,
pwmPin: -1,
mux: []mux{},
},
"9": sysfsPin{
pin: 183,
resistor: 225,
levelShifter: 257,
pwmPin: 3,
mux: []mux{},
},
"10": sysfsPin{
pin: 41,
resistor: 226,
levelShifter: 258,
pwmPin: 4,
mux: []mux{
mux{263, sysfs.HIGH},
mux{240, sysfs.LOW},
},
},
"11": sysfsPin{
pin: 43,
resistor: 227,
levelShifter: 259,
pwmPin: 5,
mux: []mux{
mux{262, sysfs.HIGH},
mux{241, sysfs.LOW},
},
},
"12": sysfsPin{
pin: 42,
resistor: 228,
levelShifter: 260,
pwmPin: -1,
mux: []mux{
mux{242, sysfs.LOW},
},
},
"13": sysfsPin{
pin: 40,
resistor: 229,
levelShifter: 261,
pwmPin: -1,
mux: []mux{
mux{243, sysfs.LOW},
},
},
}
// changePinMode writes pin mode to current_pinmux file
func changePinMode(pin, mode string) {
2014-11-07 18:37:57 -08:00
_, err := writeFile(
"/sys/kernel/debug/gpio_debug/gpio"+pin+"/current_pinmux",
2014-11-07 18:37:57 -08:00
[]byte("mode"+mode),
)
if err != nil {
panic(err)
}
}
// NewEditionAdaptor creates a EdisonAdaptor with specified name and
// creates connect function
func NewEdisonAdaptor(name string) *EdisonAdaptor {
return &EdisonAdaptor{
Adaptor: *gobot.NewAdaptor(
name,
"EdisonAdaptor",
),
connect: func(e *EdisonAdaptor) {
2014-10-30 15:26:31 -07:00
e.tristate = sysfs.NewDigitalPin(214)
2014-10-30 13:41:27 -07:00
e.tristate.Export()
2014-11-03 19:02:33 -08:00
e.tristate.Direction(sysfs.OUT)
e.tristate.Write(sysfs.LOW)
for _, i := range []int{263, 262} {
2014-10-30 15:26:31 -07:00
io := sysfs.NewDigitalPin(i)
2014-10-30 13:41:27 -07:00
io.Export()
2014-11-03 19:02:33 -08:00
io.Direction(sysfs.OUT)
io.Write(sysfs.HIGH)
io.Unexport()
}
for _, i := range []int{240, 241, 242, 243} {
2014-10-30 15:26:31 -07:00
io := sysfs.NewDigitalPin(i)
2014-10-30 13:41:27 -07:00
io.Export()
2014-11-03 19:02:33 -08:00
io.Direction(sysfs.OUT)
io.Write(sysfs.LOW)
io.Unexport()
}
for _, i := range []string{"111", "115", "114", "109"} {
changePinMode(i, "1")
}
for _, i := range []string{"131", "129", "40"} {
changePinMode(i, "0")
}
e.tristate.Write(sysfs.HIGH)
},
}
}
// Connect starts conection with board and creates
// digitalPins and pwmPins adaptor maps
func (e *EdisonAdaptor) Connect() bool {
2014-11-03 19:02:33 -08:00
e.digitalPins = make(map[int]sysfs.DigitalPin)
e.pwmPins = make(map[int]*pwmPin)
e.connect(e)
return true
}
// Finalize closes connection to board and pins
func (e *EdisonAdaptor) Finalize() bool {
e.tristate.Unexport()
for _, pin := range e.digitalPins {
if pin != nil {
pin.Unexport()
}
}
for _, pin := range e.pwmPins {
if pin != nil {
pin.close()
}
}
if e.i2cDevice != nil {
e.i2cDevice.Close()
}
return true
}
// digitalPin returns matched digitalPin for specified values
2014-11-03 19:02:33 -08:00
func (e *EdisonAdaptor) digitalPin(pin string, dir string) sysfs.DigitalPin {
i := sysfsPinMap[pin]
if e.digitalPins[i.pin] == nil {
2014-10-30 15:26:31 -07:00
e.digitalPins[i.pin] = sysfs.NewDigitalPin(i.pin)
2014-10-30 13:41:27 -07:00
e.digitalPins[i.pin].Export()
2014-10-30 15:26:31 -07:00
e.digitalPins[i.resistor] = sysfs.NewDigitalPin(i.resistor)
2014-10-30 13:41:27 -07:00
e.digitalPins[i.resistor].Export()
2014-10-30 15:26:31 -07:00
e.digitalPins[i.levelShifter] = sysfs.NewDigitalPin(i.levelShifter)
2014-10-30 13:41:27 -07:00
e.digitalPins[i.levelShifter].Export()
if len(i.mux) > 0 {
for _, mux := range i.mux {
2014-10-30 15:26:31 -07:00
e.digitalPins[mux.pin] = sysfs.NewDigitalPin(mux.pin)
2014-10-30 13:41:27 -07:00
e.digitalPins[mux.pin].Export()
2014-11-03 19:02:33 -08:00
e.digitalPins[mux.pin].Direction(sysfs.OUT)
e.digitalPins[mux.pin].Write(mux.value)
}
}
}
2014-11-03 19:02:33 -08:00
if dir == "in" {
e.digitalPins[i.pin].Direction(sysfs.IN)
e.digitalPins[i.resistor].Direction(sysfs.OUT)
e.digitalPins[i.resistor].Write(sysfs.LOW)
2014-11-03 19:02:33 -08:00
e.digitalPins[i.levelShifter].Direction(sysfs.OUT)
e.digitalPins[i.levelShifter].Write(sysfs.LOW)
2014-11-03 19:02:33 -08:00
} else if dir == "out" {
e.digitalPins[i.pin].Direction(sysfs.OUT)
e.digitalPins[i.resistor].Direction(sysfs.IN)
e.digitalPins[i.levelShifter].Direction(sysfs.OUT)
e.digitalPins[i.levelShifter].Write(sysfs.HIGH)
}
return e.digitalPins[i.pin]
}
// DigitalRead reads digital value from pin
2014-10-30 13:41:27 -07:00
func (e *EdisonAdaptor) DigitalRead(pin string) (i int) {
i, _ = e.digitalPin(pin, "in").Read()
return
}
// DigitalWrite writes digital value to specified pin
func (e *EdisonAdaptor) DigitalWrite(pin string, val byte) {
2014-10-30 13:41:27 -07:00
e.digitalPin(pin, "out").Write(int(val))
}
// PwmWrite writes scaled pwm value to specified pin
func (e *EdisonAdaptor) PwmWrite(pin string, val byte) {
sysPin := sysfsPinMap[pin]
if sysPin.pwmPin != -1 {
if e.pwmPins[sysPin.pwmPin] == nil {
e.DigitalWrite(pin, 1)
changePinMode(strconv.Itoa(int(sysPin.pin)), "1")
e.pwmPins[sysPin.pwmPin] = newPwmPin(sysPin.pwmPin)
}
period, err := strconv.Atoi(e.pwmPins[sysPin.pwmPin].period())
if err != nil {
panic(err)
}
duty := gobot.FromScale(float64(val), 0, 255.0)
e.pwmPins[sysPin.pwmPin].writeDuty(strconv.Itoa(int(float64(period) * duty)))
} else {
fmt.Println("Not a PWM pin")
}
}
2014-11-05 18:45:58 -08:00
// AnalogWrite Not Implemented
func (e *EdisonAdaptor) AnalogWrite(string, byte) {}
// InitServo Not Implemented
func (e *EdisonAdaptor) InitServo() {}
// ServoWrite Not Implemented
func (e *EdisonAdaptor) ServoWrite(string, byte) {}
// AnalogRead returns value from analog reading of specified pin
func (e *EdisonAdaptor) AnalogRead(pin string) int {
2014-11-04 12:21:18 -08:00
buf, err := readFile(
"/sys/bus/iio/devices/iio:device1/in_voltage" + pin + "_raw",
)
if err != nil {
panic(err)
}
val, err := strconv.Atoi(string(buf[0 : len(buf)-1]))
if err != nil {
panic(err)
}
return val
}
// I2cStart initializes i2c device for addresss
func (e *EdisonAdaptor) I2cStart(address byte) {
e.tristate.Write(sysfs.LOW)
for _, i := range []int{14, 165, 212, 213} {
2014-10-30 15:26:31 -07:00
io := sysfs.NewDigitalPin(i)
2014-10-30 13:41:27 -07:00
io.Export()
2014-11-03 19:02:33 -08:00
io.Direction(sysfs.IN)
io.Unexport()
}
for _, i := range []int{236, 237, 204, 205} {
2014-10-30 15:26:31 -07:00
io := sysfs.NewDigitalPin(i)
2014-10-30 13:41:27 -07:00
io.Export()
2014-11-03 19:02:33 -08:00
io.Direction(sysfs.OUT)
io.Write(sysfs.LOW)
io.Unexport()
}
for _, i := range []string{"28", "27"} {
changePinMode(i, "1")
}
e.tristate.Write(sysfs.HIGH)
2014-11-07 18:37:57 -08:00
e.i2cDevice, _ = sysfs.NewI2cDevice("/dev/i2c-6", address)
}
// I2cWrite writes data to i2cDevice
func (e *EdisonAdaptor) I2cWrite(data []byte) {
e.i2cDevice.Write(data)
}
// I2cRead reads data from i2cDevice
func (e *EdisonAdaptor) I2cRead(size uint) []byte {
b := make([]byte, size)
e.i2cDevice.Read(b)
return b
}