|
@@ -3,7 +3,6 @@ package remdeshid
|
|
|
import (
|
|
|
"fmt"
|
|
|
"log"
|
|
|
- "time"
|
|
|
|
|
|
"github.com/tarm/serial"
|
|
|
)
|
|
@@ -24,7 +23,9 @@ type Controller struct {
|
|
|
Config *Config
|
|
|
serialPort *serial.Port
|
|
|
serialRunning bool
|
|
|
- stopChan chan bool
|
|
|
+ readStopChan chan bool
|
|
|
+ writeStopChan chan bool
|
|
|
+ writeQueue chan []byte
|
|
|
}
|
|
|
|
|
|
func NewHIDController(config *Config) *Controller {
|
|
@@ -50,31 +51,58 @@ func (c *Controller) Connect() error {
|
|
|
}
|
|
|
|
|
|
c.serialPort = port
|
|
|
-
|
|
|
+ c.readStopChan = make(chan bool)
|
|
|
//Start reading from the serial port
|
|
|
go func() {
|
|
|
buf := make([]byte, 128)
|
|
|
for {
|
|
|
- n, err := port.Read(buf)
|
|
|
- if err != nil {
|
|
|
- if c.Config.OnReadError != nil {
|
|
|
- c.Config.OnReadError(err)
|
|
|
- } else {
|
|
|
- log.Println(err.Error())
|
|
|
- }
|
|
|
+ select {
|
|
|
+ case <-c.readStopChan:
|
|
|
return
|
|
|
- }
|
|
|
- if n > 0 {
|
|
|
- if c.Config.OnDataHandler != nil {
|
|
|
- c.Config.OnDataHandler(buf[:n], nil)
|
|
|
- } else {
|
|
|
- fmt.Print("Received bytes: ")
|
|
|
- for i := 0; i < n; i++ {
|
|
|
- fmt.Printf("0x%02X ", buf[i])
|
|
|
+ default:
|
|
|
+ n, err := port.Read(buf)
|
|
|
+ if err != nil {
|
|
|
+ if c.Config.OnReadError != nil {
|
|
|
+ c.Config.OnReadError(err)
|
|
|
+ } else {
|
|
|
+ log.Println(err.Error())
|
|
|
+ }
|
|
|
+ return
|
|
|
+ }
|
|
|
+ if n > 0 {
|
|
|
+ if c.Config.OnDataHandler != nil {
|
|
|
+ c.Config.OnDataHandler(buf[:n], nil)
|
|
|
+ } else {
|
|
|
+ fmt.Print("Received bytes: ")
|
|
|
+ for i := 0; i < n; i++ {
|
|
|
+ fmt.Printf("0x%02X ", buf[i])
|
|
|
+ }
|
|
|
+ fmt.Println()
|
|
|
}
|
|
|
- fmt.Println()
|
|
|
}
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }()
|
|
|
|
|
|
+ //Create a loop to write to the serial port
|
|
|
+ c.writeStopChan = make(chan bool)
|
|
|
+ c.writeQueue = make(chan []byte, 10)
|
|
|
+ c.serialRunning = true
|
|
|
+ go func() {
|
|
|
+ for {
|
|
|
+ select {
|
|
|
+ case data := <-c.writeQueue:
|
|
|
+ _, err := port.Write(data)
|
|
|
+ if err != nil {
|
|
|
+ if c.Config.OnWriteError != nil {
|
|
|
+ c.Config.OnWriteError(err)
|
|
|
+ } else {
|
|
|
+ log.Println(err.Error())
|
|
|
+ }
|
|
|
+ }
|
|
|
+ case <-c.writeStopChan:
|
|
|
+ c.serialRunning = false
|
|
|
+ return
|
|
|
}
|
|
|
}
|
|
|
}()
|
|
@@ -100,20 +128,21 @@ func (c *Controller) Connect() error {
|
|
|
}
|
|
|
|
|
|
func (c *Controller) Send(data []byte) error {
|
|
|
- n, err := c.serialPort.Write(data)
|
|
|
- if err != nil {
|
|
|
- return err
|
|
|
+ if !c.serialRunning {
|
|
|
+ return fmt.Errorf("serial port is not running")
|
|
|
}
|
|
|
- if n != len(data) {
|
|
|
- return fmt.Errorf("Failed to write all bytes to serial port")
|
|
|
- }
|
|
|
- time.Sleep(10 * time.Millisecond) //Minimum delay between writes
|
|
|
+ c.writeQueue <- data
|
|
|
return nil
|
|
|
}
|
|
|
|
|
|
func (c *Controller) Close() {
|
|
|
- if c.stopChan != nil {
|
|
|
- c.stopChan <- true
|
|
|
+ if c.readStopChan != nil {
|
|
|
+ c.readStopChan <- true
|
|
|
+ }
|
|
|
+
|
|
|
+ if c.writeStopChan != nil {
|
|
|
+ c.writeStopChan <- true
|
|
|
}
|
|
|
+
|
|
|
c.serialPort.Close()
|
|
|
}
|