Files
dmxconnect/hardware/FTDIPeripheral.go

216 lines
8.6 KiB
Go
Raw Normal View History

2024-12-15 13:45:46 +01:00
package hardware
import (
"bufio"
"context"
2024-12-15 13:45:46 +01:00
_ "embed"
"fmt"
"io"
2024-12-29 13:09:46 +01:00
"github.com/rs/zerolog/log"
2024-12-15 13:45:46 +01:00
"os"
"os/exec"
)
const (
activateCommandString = 0x01
deactivateCommandString = 0x02
setCommandString = 0x03
)
// FTDIPeripheral contains the data of an FTDI peripheral
type FTDIPeripheral struct {
name string // The name of the peripheral
serialNumber string // The S/N of the FTDI peripheral
location int // The location of the peripheral
universesNumber int // The number of DMX universes handled by this peripheral
programName string // The temp file name of the executable
dmxSender *exec.Cmd // The command to pilot the DMX sender program
stdin io.WriteCloser // For writing in the DMX sender
stdout io.ReadCloser // For reading from the DMX sender
stderr io.ReadCloser // For reading the errors
disconnectChan chan struct{} // Channel to cancel the connection
errorsChan chan error // Channel to get the errors
}
//go:embed third-party/ftdi/dmxSender.exe
var dmxSender []byte
2024-12-15 13:45:46 +01:00
// NewFTDIPeripheral creates a new FTDI peripheral
2024-12-23 17:22:37 +01:00
func NewFTDIPeripheral(name string, serialNumber string, location int) (*FTDIPeripheral, error) {
2024-12-29 13:09:46 +01:00
log.Info().Str("file", "FTDIPeripheral").Str("name", name).Str("s/n", serialNumber).Int("location", location).Msg("FTDI peripheral created")
2024-12-15 13:45:46 +01:00
// Create a temporary file
tempFile, err := os.Create(fmt.Sprintf("dmxSender-%s.exe", serialNumber))
2024-12-15 13:45:46 +01:00
if err != nil {
return nil, err
}
2024-12-29 13:09:46 +01:00
log.Trace().Str("file", "FTDIPeripheral").Str("s/n", serialNumber).Msg("FTDI sender temp created")
2024-12-15 13:45:46 +01:00
// Write the embedded executable to the temp file
if _, err := tempFile.Write(dmxSender); err != nil {
return nil, err
}
2024-12-29 13:09:46 +01:00
log.Trace().Str("file", "FTDIPeripheral").Str("s/n", serialNumber).Msg("FTDI sender written")
2024-12-15 13:45:46 +01:00
tempFile.Close()
return &FTDIPeripheral{
name: name,
dmxSender: nil,
2024-12-15 13:45:46 +01:00
programName: tempFile.Name(),
serialNumber: serialNumber,
location: location,
universesNumber: 1,
disconnectChan: make(chan struct{}),
errorsChan: make(chan error, 1),
}, nil
}
// Connect connects the FTDI peripheral
func (p *FTDIPeripheral) Connect(ctx context.Context) error {
2024-12-15 13:45:46 +01:00
// Connect if no connection is already running
2024-12-29 13:09:46 +01:00
log.Trace().Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("connecting FTDI peripheral...")
2024-12-15 13:45:46 +01:00
// Check if the connection has already been established
if p.dmxSender != nil {
log.Warn().Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("dmxSender already initialized")
return nil
}
2024-12-15 13:45:46 +01:00
// Initialize the exec.Command for running the process
p.dmxSender = exec.Command(p.programName, fmt.Sprintf("%d", p.location))
log.Trace().Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("dmxSender instance created")
2024-12-15 13:45:46 +01:00
// Create the pipes for stdin, stdout, and stderr asynchronously without blocking
var err error
if p.stdout, err = p.dmxSender.StdoutPipe(); err != nil {
log.Err(err).Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("unable to create stdout pipe")
return fmt.Errorf("unable to create stdout pipe: %v", err)
}
if p.stdin, err = p.dmxSender.StdinPipe(); err != nil {
log.Err(err).Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("unable to create stdin pipe")
return fmt.Errorf("unable to create stdin pipe: %v", err)
}
if p.stderr, err = p.dmxSender.StderrPipe(); err != nil {
log.Err(err).Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("unable to create stderr pipe")
return fmt.Errorf("unable to create stderr pipe: %v", err)
}
2024-12-15 13:45:46 +01:00
// Launch a goroutine to read stderr asynchronously
go func() {
scanner := bufio.NewScanner(p.stderr)
for scanner.Scan() {
// Process each line read from stderr
log.Err(fmt.Errorf(scanner.Text())).Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("error detected in dmx sender")
}
if err := scanner.Err(); err != nil {
log.Err(err).Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("error reading from stderr")
}
}()
// Launch the command asynchronously in another goroutine
go func() {
// Run the command, respecting the context cancellation
err := p.dmxSender.Run()
select {
case <-ctx.Done():
// If the context is canceled, handle it gracefully
log.Warn().Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("dmxSender was canceled by context")
return
default:
// Handle command exit normally
2024-12-15 13:45:46 +01:00
if err != nil {
log.Err(err).Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("error while execution of dmx sender")
2024-12-15 13:45:46 +01:00
if exitError, ok := err.(*exec.ExitError); ok {
2024-12-29 13:09:46 +01:00
log.Warn().Str("file", "FTDIPeripheral").Int("exitCode", exitError.ExitCode()).Str("s/n", p.serialNumber).Msg("dmx sender exited with code")
2024-12-15 13:45:46 +01:00
}
} else {
2024-12-29 13:09:46 +01:00
log.Debug().Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("dmx sender exited successfully")
2024-12-15 13:45:46 +01:00
}
}
}()
log.Debug().Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("dmxSender process started successfully")
2024-12-15 13:45:46 +01:00
return nil
}
// Disconnect disconnects the FTDI peripheral
func (p *FTDIPeripheral) Disconnect(ctx context.Context) error {
2024-12-29 13:09:46 +01:00
log.Trace().Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("disconnecting FTDI peripheral...")
2024-12-15 13:45:46 +01:00
if p.dmxSender != nil {
2024-12-29 13:09:46 +01:00
log.Debug().Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("dmxsender is defined for this FTDI")
2024-12-15 13:45:46 +01:00
_, err := io.WriteString(p.stdin, string([]byte{0x04, 0x00, 0x00, 0x00}))
if err != nil {
2024-12-29 13:09:46 +01:00
log.Err(err).Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("unable to write command to sender")
return fmt.Errorf("unable to disconnect: %v", err)
2024-12-15 13:45:46 +01:00
}
p.stdin.Close()
p.stdout.Close()
p.dmxSender = nil
err = os.Remove(p.programName)
if err != nil {
log.Warn().Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Str("senderPath", p.programName).Msg("unable to delete the dmx sender temporary file")
2024-12-29 13:09:46 +01:00
return fmt.Errorf("unable to delete the temporary file: %v", err)
2024-12-15 13:45:46 +01:00
}
return nil
}
2024-12-29 13:09:46 +01:00
log.Warn().Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("error while disconnecting: not connected")
return fmt.Errorf("unable to disconnect: not connected")
2024-12-15 13:45:46 +01:00
}
// Activate activates the FTDI peripheral
func (p *FTDIPeripheral) Activate(ctx context.Context) error {
2024-12-15 13:45:46 +01:00
if p.dmxSender != nil {
2024-12-29 13:09:46 +01:00
log.Debug().Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("dmxsender is defined for this FTDI")
2024-12-15 13:45:46 +01:00
_, err := io.WriteString(p.stdin, string([]byte{0x01, 0x00, 0x00, 0x00}))
if err != nil {
2024-12-29 13:09:46 +01:00
log.Err(err).Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("unable to write command to sender")
return fmt.Errorf("unable to activate: %v", err)
2024-12-15 13:45:46 +01:00
}
return nil
}
2024-12-29 13:09:46 +01:00
log.Warn().Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("error while activating: not connected")
return fmt.Errorf("unable to activate: not connected")
2024-12-15 13:45:46 +01:00
}
// Deactivate deactivates the FTDI peripheral
func (p *FTDIPeripheral) Deactivate(ctx context.Context) error {
2024-12-15 13:45:46 +01:00
if p.dmxSender != nil {
2024-12-29 13:09:46 +01:00
log.Debug().Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("dmxsender is defined for this FTDI")
2024-12-15 13:45:46 +01:00
_, err := io.WriteString(p.stdin, string([]byte{0x02, 0x00, 0x00, 0x00}))
if err != nil {
2024-12-29 13:09:46 +01:00
log.Err(err).Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("unable to write command to sender")
return fmt.Errorf("unable to deactivate: %v", err)
2024-12-15 13:45:46 +01:00
}
return nil
}
2024-12-29 13:09:46 +01:00
log.Warn().Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("error while deactivating: not connected")
return fmt.Errorf("unable to deactivate: not connected")
2024-12-15 13:45:46 +01:00
}
// SetDeviceProperty sends a command to the specified device
func (p *FTDIPeripheral) SetDeviceProperty(ctx context.Context, uint32, channelNumber uint32, channelValue byte) error {
2024-12-15 13:45:46 +01:00
if p.dmxSender != nil {
2024-12-29 13:09:46 +01:00
log.Debug().Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("dmxsender is defined for this FTDI")
2024-12-15 13:45:46 +01:00
commandString := []byte{0x03, 0x01, 0x00, 0xff, 0x03, 0x02, 0x00, channelValue}
_, err := io.WriteString(p.stdin, string(commandString))
if err != nil {
2024-12-29 13:09:46 +01:00
log.Err(err).Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("unable to write command to sender")
return fmt.Errorf("unable to set device property: %v", err)
2024-12-15 13:45:46 +01:00
}
return nil
}
2024-12-29 13:09:46 +01:00
log.Warn().Str("file", "FTDIPeripheral").Str("s/n", p.serialNumber).Msg("error while setting device property: not connected")
return fmt.Errorf("unable to set device property: not connected")
2024-12-15 13:45:46 +01:00
}
// GetInfo gets all the peripheral information
func (p *FTDIPeripheral) GetInfo() PeripheralInfo {
return PeripheralInfo{
2024-12-20 17:18:57 +01:00
Name: p.name,
SerialNumber: p.serialNumber,
ProtocolName: "FTDI",
2024-12-15 13:45:46 +01:00
}
}