robocar-arduino/arduino/arduino.go

275 lines
6.4 KiB
Go
Raw Normal View History

2019-12-01 21:35:19 +00:00
package arduino
import (
"bufio"
"github.com/cyrilix/robocar-protobuf/go/events"
mqtt "github.com/eclipse/paho.mqtt.golang"
"github.com/golang/protobuf/proto"
log "github.com/sirupsen/logrus"
2019-12-01 21:35:19 +00:00
"github.com/tarm/serial"
"io"
"regexp"
"strconv"
"strings"
"sync"
"time"
)
const (
MinPwmAngle = 960.0
MaxPwmAngle = 1980.0
MinPwmThrottle = 972.0
MaxPwmThrottle = 1954.0
)
var (
serialLineRegex = regexp.MustCompile(`(?P<timestamp>\d+),(?P<channel_1>\d+),(?P<channel_2>\d+),(?P<channel_3>\d+),(?P<channel_4>\d+),(?P<channel_5>\d+),(?P<channel_6>\d+),(?P<frequency>\d+),(?P<distance_cm>\d+)?`)
)
2019-12-18 22:21:52 +00:00
type Part struct {
client mqtt.Client
2020-01-27 18:21:26 +00:00
throttleTopic, steeringTopic, driveModeTopic, switchRecordTopic string
2019-12-01 21:35:19 +00:00
pubFrequency float64
serial io.Reader
mutex sync.Mutex
steering float32
throttle float32
2019-12-01 21:35:19 +00:00
ctrlRecord bool
driveMode events.DriveMode
2019-12-18 22:21:52 +00:00
debug bool
cancel chan interface{}
2019-12-01 21:35:19 +00:00
}
2020-01-27 18:21:26 +00:00
func NewPart(client mqtt.Client, name string, baud int, throttleTopic, steeringTopic, driveModeTopic, switchRecordTopic string, pubFrequency float64, debug bool) *Part {
2019-12-01 21:35:19 +00:00
c := &serial.Config{Name: name, Baud: baud}
s, err := serial.OpenPort(c)
if err != nil {
log.Panicf("unable to open serial port: %v", err)
}
return &Part{
client: client,
serial: s,
2020-01-27 18:21:26 +00:00
throttleTopic: throttleTopic,
steeringTopic: steeringTopic,
driveModeTopic:driveModeTopic,
switchRecordTopic:switchRecordTopic,
pubFrequency: pubFrequency,
driveMode: events.DriveMode_INVALID,
debug: debug,
cancel: make(chan interface{}),
}
2019-12-01 21:35:19 +00:00
}
2019-12-18 22:21:52 +00:00
func (a *Part) Start() error {
log.Printf("start arduino part")
2019-12-01 21:35:19 +00:00
go a.publishLoop()
for {
buff := bufio.NewReader(a.serial)
line, err := buff.ReadString('\n')
if err == io.EOF || line == "" {
log.Println("remote connection closed")
break
}
2019-12-18 22:21:52 +00:00
if !serialLineRegex.MatchString(line) {
2019-12-01 21:35:19 +00:00
log.Printf("invalid line: '%v'", line)
continue
}
values := strings.Split(strings.TrimSuffix(strings.TrimSuffix(line, "\n"), "\r"), ",")
a.updateValues(values)
}
2019-12-18 22:21:52 +00:00
return nil
2019-12-01 21:35:19 +00:00
}
2019-12-18 22:21:52 +00:00
func (a *Part) updateValues(values []string) {
2019-12-01 21:35:19 +00:00
a.mutex.Lock()
defer a.mutex.Unlock()
2019-12-18 22:21:52 +00:00
a.processChannel1(values[1])
a.processChannel2(values[2])
a.processChannel3(values[3])
a.processChannel4(values[4])
a.processChannel5(values[5])
a.processChannel6(values[6])
2019-12-01 21:35:19 +00:00
}
2019-12-18 22:21:52 +00:00
func (a *Part) Stop() {
log.Printf("stop ArduinoPart")
close(a.cancel)
2019-12-01 21:35:19 +00:00
switch s := a.serial.(type) {
case io.ReadCloser:
if err := s.Close(); err != nil {
log.Fatalf("unable to close serial port: %v", err)
}
}
}
2019-12-18 22:21:52 +00:00
func (a *Part) processChannel1(v string) {
if a.debug {
log.Printf("channel1: %v", v)
}
value, err := strconv.Atoi(v)
2019-12-01 21:35:19 +00:00
if err != nil {
log.Printf("invalid value for channel1, should be an int: %v", err)
}
if value < MinPwmAngle {
value = MinPwmAngle
} else if value > MaxPwmAngle {
value = MaxPwmAngle
}
a.steering = ((float32(value)-MinPwmAngle)/(MaxPwmAngle-MinPwmAngle))*2.0 - 1.0
2019-12-01 21:35:19 +00:00
}
2019-12-18 22:21:52 +00:00
func (a *Part) processChannel2(v string) {
if a.debug {
log.Printf("channel2: %v", v)
}
value, err := strconv.Atoi(v)
2019-12-01 21:35:19 +00:00
if err != nil {
log.Printf("invalid value for channel2, should be an int: %v", err)
}
if value < MinPwmThrottle {
value = MinPwmThrottle
} else if value > MaxPwmThrottle {
value = MaxPwmThrottle
}
a.throttle = ((float32(value)-MinPwmThrottle)/(MaxPwmThrottle-MinPwmThrottle))*2.0 - 1.0
2019-12-01 21:35:19 +00:00
}
2019-12-18 22:21:52 +00:00
func (a *Part) processChannel3(v string) {
if a.debug {
log.Printf("channel3: %v", v)
}
2019-12-01 21:35:19 +00:00
}
2019-12-18 22:21:52 +00:00
func (a *Part) processChannel4(v string) {
if a.debug {
log.Printf("channel4: %v", v)
}
2019-12-01 21:35:19 +00:00
}
2019-12-18 22:21:52 +00:00
func (a *Part) processChannel5(v string) {
if a.debug {
log.Printf("channel5: %v", v)
}
value, err := strconv.Atoi(v)
2019-12-01 21:35:19 +00:00
if err != nil {
log.Printf("invalid value for channel5, should be an int: %v", err)
}
if value < 1800 {
2019-12-18 22:21:52 +00:00
if !a.ctrlRecord {
2019-12-01 21:35:19 +00:00
log.Printf("Update channel 5 with value %v, record: %v", true, false)
a.ctrlRecord = true
}
} else {
if a.ctrlRecord {
log.Printf("Update channel 5 with value %v, record: %v", false, true)
a.ctrlRecord = false
}
}
}
2019-12-18 22:21:52 +00:00
func (a *Part) processChannel6(v string) {
if a.debug {
log.Printf("channel6: %v", v)
}
value, err := strconv.Atoi(v)
2019-12-01 21:35:19 +00:00
if err != nil {
log.Printf("invalid value for channel6, should be an int: %v", err)
return
}
if value > 1800 {
if a.driveMode != events.DriveMode_PILOT {
log.Printf("Update channel 6 with value %v, new user_mode: %v", value, events.DriveMode_PILOT)
a.driveMode = events.DriveMode_PILOT
2019-12-01 21:35:19 +00:00
}
} else {
if a.driveMode != events.DriveMode_USER {
log.Printf("Update channel 6 with value %v, new user_mode: %v", value, events.DriveMode_USER)
2019-12-01 21:35:19 +00:00
}
a.driveMode = events.DriveMode_USER
2019-12-01 21:35:19 +00:00
}
}
2019-12-18 22:21:52 +00:00
func (a *Part) publishLoop() {
ticker := time.NewTicker(time.Second / time.Duration(int(a.pubFrequency)))
2019-12-01 21:35:19 +00:00
for {
select {
case <-ticker.C:
2020-01-27 18:21:26 +00:00
a.publishValues()
case <-a.cancel:
ticker.Stop()
return
}
2019-12-01 21:35:19 +00:00
}
}
2020-01-27 18:21:26 +00:00
func (a *Part) publishValues() {
2019-12-01 21:35:19 +00:00
a.mutex.Lock()
defer a.mutex.Unlock()
2020-01-27 18:21:26 +00:00
a.publishThrottle()
a.publishSteering()
a.publishDriveMode()
a.publishSwitchRecord()
}
2020-01-27 18:21:26 +00:00
func (a *Part) publishThrottle() {
throttle := events.ThrottleMessage{
Throttle: a.throttle,
Confidence: 1.0,
}
throttleMessage, err := proto.Marshal(&throttle)
if err != nil {
log.Errorf("unable to marshal protobuf throttle message: %v", err)
return
}
2020-01-27 18:21:26 +00:00
log.Infof("throttle channel: %v", a.throttle)
publish(a.client, a.throttleTopic, &throttleMessage)
}
2020-01-27 18:21:26 +00:00
func (a *Part) publishSteering() {
steering := events.SteeringMessage{
Steering: a.steering,
Confidence: 1.0,
}
steeringMessage, err := proto.Marshal(&steering)
if err != nil {
log.Errorf("unable to marshal protobuf steering message: %v", err)
return
}
2020-01-27 18:21:26 +00:00
publish(a.client, a.steeringTopic, &steeringMessage)
}
2020-01-27 18:21:26 +00:00
func (a *Part) publishDriveMode() {
dm := events.DriveModeMessage{
DriveMode: a.driveMode,
}
driveModeMessage, err := proto.Marshal(&dm)
if err != nil {
log.Errorf("unable to marshal protobuf driveMode message: %v", err)
return
}
2020-01-27 18:21:26 +00:00
publish(a.client, a.driveModeTopic, &driveModeMessage)
}
2020-01-27 18:21:26 +00:00
func (a *Part) publishSwitchRecord() {
sr := events.SwitchRecordMessage{
Enabled: a.ctrlRecord,
}
switchRecordMessage, err := proto.Marshal(&sr)
if err != nil {
log.Errorf("unable to marshal protobuf SwitchRecord message: %v", err)
return
}
2020-01-27 18:21:26 +00:00
publish(a.client, a.switchRecordTopic, &switchRecordMessage)
}
var publish = func(client mqtt.Client, topic string, payload *[]byte) {
client.Publish(topic, 0, false, *payload)
2019-12-01 21:35:19 +00:00
}