kappanhang/streamcommon.go
2020-10-18 15:40:25 +02:00

140 lines
3.5 KiB
Go

package main
import (
"bytes"
"crypto/rand"
"encoding/binary"
"fmt"
"net"
"time"
"github.com/nonoo/kappanhang/log"
)
type streamCommon struct {
name string
conn *net.UDPConn
localSID uint32
remoteSID uint32
readChan chan []byte
pkt7 pkt7Type
}
func (s *streamCommon) send(d []byte) {
_, err := s.conn.Write(d)
if err != nil {
log.Fatal(err)
}
}
func (s *streamCommon) read() ([]byte, error) {
err := s.conn.SetReadDeadline(time.Now().Add(time.Second))
if err != nil {
log.Fatal(err)
}
b := make([]byte, 1500)
n, _, err := s.conn.ReadFromUDP(b)
if err != nil {
log.Fatal(err)
}
return b[:n], err
}
func (s *streamCommon) reader() {
var errCount int
for {
r, err := s.read()
if err == nil {
if s.pkt7.isPkt7(r) {
s.pkt7.handle(s, r)
}
s.readChan <- r
} else {
errCount++
if errCount > 5 {
log.Fatal(s.name + "/timeout")
}
log.Error(s.name + "/stream break detected")
}
errCount = 0
}
}
func (s *streamCommon) expect(packetLength int, b []byte) []byte {
var r []byte
expectStart := time.Now()
for {
r = <-s.readChan
if len(r) == packetLength && bytes.Equal(r[:len(b)], b) {
break
}
if time.Since(expectStart) > time.Second {
log.Fatal(s.name + "/expect timeout")
}
}
return r
}
func (s *streamCommon) open(name string, portNumber int) {
s.name = name
hostPort := fmt.Sprint(connectAddress, ":", portNumber)
log.Print(s.name+"/connecting to ", hostPort)
raddr, err := net.ResolveUDPAddr("udp", hostPort)
if err != nil {
log.Fatal(err)
}
s.conn, err = net.DialUDP("udp", nil, raddr)
if err != nil {
log.Fatal(err)
}
// Constructing the local session ID by combining the local IP address and port.
laddr := s.conn.LocalAddr().(*net.UDPAddr)
s.localSID = binary.BigEndian.Uint32(laddr.IP[len(laddr.IP)-4:])<<16 | uint32(laddr.Port&0xffff)
log.Debugf(s.name+"/using session id %.8x", s.localSID)
_, err = rand.Read(s.pkt7.randIDByte[:])
if err != nil {
log.Fatal(err)
}
s.readChan = make(chan []byte)
go s.reader()
}
func (s *streamCommon) sendPkt3() {
s.send([]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)})
}
func (s *streamCommon) waitForPkt4Answer() {
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 := s.expect(16, []byte{0x10, 0x00, 0x00, 0x00, 0x04, 0x00, 0x00, 0x00})
s.remoteSID = binary.BigEndian.Uint32(r[8:12])
log.Debugf(s.name+"/got remote session id %.8x", s.remoteSID)
}
func (s *streamCommon) sendPkt6() {
s.send([]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)})
}
func (s *streamCommon) waitForPkt6Answer() {
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
s.expect(16, []byte{0x10, 0x00, 0x00, 0x00, 0x06, 0x00, 0x01, 0x00})
}
func (s *streamCommon) sendDisconnect() {
s.send([]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)})
}