kappanhang/streamcommon.go

185 lines
4.6 KiB
Go
Raw Normal View History

2020-10-17 23:53:33 +02:00
package main
import (
"bytes"
"encoding/binary"
2020-10-19 09:45:49 +02:00
"errors"
2020-10-17 23:53:33 +02:00
"fmt"
"net"
"time"
"github.com/nonoo/kappanhang/log"
)
const expectTimeoutDuration = time.Second
2020-10-18 11:01:53 +02:00
type streamCommon struct {
name string
conn *net.UDPConn
localSID uint32
remoteSID uint32
gotRemoteSID bool
readChan chan []byte
readerCloseNeededChan chan bool
readerCloseFinishedChan chan bool
2020-10-18 14:21:58 +02:00
pkt7 pkt7Type
2020-10-17 23:53:33 +02:00
}
func (s *streamCommon) send(d []byte) error {
if _, err := s.conn.Write(d); err != nil {
return err
2020-10-17 23:53:33 +02:00
}
return nil
2020-10-17 23:53:33 +02:00
}
func (s *streamCommon) read() ([]byte, error) {
2020-10-17 23:53:33 +02:00
b := make([]byte, 1500)
2020-10-18 11:01:53 +02:00
n, _, err := s.conn.ReadFromUDP(b)
return b[:n], err
2020-10-17 23:53:33 +02:00
}
2020-10-18 13:02:41 +02:00
func (s *streamCommon) reader() {
2020-10-18 10:53:16 +02:00
for {
r, err := s.read()
if err != nil {
reportError(err)
} else if s.pkt7.isPkt7(r) {
if err := s.pkt7.handle(s, r); err != nil {
reportError(err)
}
continue
2020-10-18 10:53:16 +02:00
}
select {
case s.readChan <- r:
case <-s.readerCloseNeededChan:
s.readerCloseFinishedChan <- true
return
}
2020-10-18 10:53:16 +02:00
}
}
func (s *streamCommon) tryReceivePacket(timeout time.Duration, packetLength, matchStartByte int, b []byte) []byte {
2020-10-17 23:53:33 +02:00
var r []byte
timer := time.NewTimer(timeout)
2020-10-17 23:53:33 +02:00
for {
select {
case r = <-s.readChan:
case <-timer.C:
return nil
}
if len(r) == packetLength && bytes.Equal(r[matchStartByte:len(b)+matchStartByte], b) {
2020-10-17 23:53:33 +02:00
break
}
}
return r
}
func (s *streamCommon) expect(packetLength int, b []byte) ([]byte, error) {
r := s.tryReceivePacket(expectTimeoutDuration, packetLength, 0, b)
if r == nil {
return nil, errors.New(s.name + "/expect timeout")
}
return r, nil
}
func (s *streamCommon) sendPkt3() error {
p := []byte{0x10, 0x00, 0x00, 0x00, 0x03, 0x00, 0x00, 0x00,
byte(s.localSID >> 24), byte(s.localSID >> 16), byte(s.localSID >> 8), byte(s.localSID),
byte(s.remoteSID >> 24), byte(s.remoteSID >> 16), byte(s.remoteSID >> 8), byte(s.remoteSID)}
if err := s.send(p); err != nil {
return err
}
if err := s.send(p); err != nil {
return err
}
return nil
}
func (s *streamCommon) waitForPkt4Answer() error {
log.Debug(s.name + "/expecting a pkt4 answer")
// Example answer from radio: 0x10, 0x00, 0x00, 0x00, 0x04, 0x00, 0x00, 0x00, 0x8c, 0x7d, 0x45, 0x7a, 0x1d, 0xf6, 0xe9, 0x0b
r, err := s.expect(16, []byte{0x10, 0x00, 0x00, 0x00, 0x04, 0x00, 0x00, 0x00})
if err != nil {
return err
}
s.remoteSID = binary.BigEndian.Uint32(r[8:12])
s.gotRemoteSID = true
return nil
}
func (s *streamCommon) sendPkt6() error {
p := []byte{0x10, 0x00, 0x00, 0x00, 0x06, 0x00, 0x01, 0x00,
byte(s.localSID >> 24), byte(s.localSID >> 16), byte(s.localSID >> 8), byte(s.localSID),
byte(s.remoteSID >> 24), byte(s.remoteSID >> 16), byte(s.remoteSID >> 8), byte(s.remoteSID)}
if err := s.send(p); err != nil {
return err
}
if err := s.send(p); err != nil {
return err
}
return nil
}
func (s *streamCommon) waitForPkt6Answer() error {
log.Debug(s.name + "/expecting pkt6 answer")
// Example answer from radio: 0x10, 0x00, 0x00, 0x00, 0x06, 0x00, 0x01, 0x00, 0xe8, 0xd0, 0x44, 0x50, 0xa0, 0x61, 0x39, 0xbe
_, err := s.expect(16, []byte{0x10, 0x00, 0x00, 0x00, 0x06, 0x00, 0x01, 0x00})
return err
}
func (s *streamCommon) sendDisconnect() error {
log.Print(s.name + "/disconnecting")
p := []byte{0x10, 0x00, 0x00, 0x00, 0x05, 0x00, 0x00, 0x00,
byte(s.localSID >> 24), byte(s.localSID >> 16), byte(s.localSID >> 8), byte(s.localSID),
byte(s.remoteSID >> 24), byte(s.remoteSID >> 16), byte(s.remoteSID >> 8), byte(s.remoteSID)}
if err := s.send(p); err != nil {
return err
}
if err := s.send(p); err != nil {
return err
}
return nil
}
func (s *streamCommon) init(name string, portNumber int) error {
2020-10-18 11:17:40 +02:00
s.name = name
2020-10-17 23:53:33 +02:00
hostPort := fmt.Sprint(connectAddress, ":", portNumber)
2020-10-18 11:17:40 +02:00
log.Print(s.name+"/connecting to ", hostPort)
2020-10-17 23:53:33 +02:00
raddr, err := net.ResolveUDPAddr("udp", hostPort)
if err != nil {
return err
2020-10-17 23:53:33 +02:00
}
2020-10-18 16:48:57 +02:00
2020-10-23 22:52:14 +02:00
s.conn, err = net.DialUDP("udp", &net.UDPAddr{Port: portNumber}, raddr)
2020-10-17 23:53:33 +02:00
if err != nil {
return err
2020-10-17 23:53:33 +02:00
}
// Constructing the local session ID by combining the local IP address and port.
2020-10-23 22:52:14 +02:00
// laddr := s.conn.LocalAddr().(*net.UDPAddr)
// s.localSID = binary.BigEndian.Uint32(laddr.IP[len(laddr.IP)-4:])<<16 | uint32(laddr.Port&0xffff)
s.localSID = 0x8aff0539
2020-10-18 13:02:41 +02:00
s.readChan = make(chan []byte)
s.readerCloseNeededChan = make(chan bool)
s.readerCloseFinishedChan = make(chan bool)
2020-10-18 13:02:41 +02:00
go s.reader()
return nil
}
func (s *streamCommon) deinit() {
s.pkt7.stopPeriodicSend()
if s.gotRemoteSID && s.conn != nil {
_ = s.sendDisconnect()
}
s.conn.Close()
if s.readerCloseNeededChan != nil {
s.readerCloseNeededChan <- true
<-s.readerCloseFinishedChan
}
2020-10-18 13:19:52 +02:00
}