2020-10-17 23:53:33 +02:00
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"bytes"
|
2020-10-18 12:50:09 +02:00
|
|
|
"crypto/rand"
|
|
|
|
|
"encoding/binary"
|
2020-10-17 23:53:33 +02:00
|
|
|
"fmt"
|
|
|
|
|
"net"
|
|
|
|
|
"time"
|
|
|
|
|
|
|
|
|
|
"github.com/nonoo/kappanhang/log"
|
|
|
|
|
)
|
|
|
|
|
|
2020-10-18 18:34:22 +02:00
|
|
|
const expectTimeoutDuration = time.Second
|
|
|
|
|
|
2020-10-18 11:01:53 +02:00
|
|
|
type streamCommon struct {
|
2020-10-18 18:34:22 +02:00
|
|
|
name string
|
|
|
|
|
conn *net.UDPConn
|
|
|
|
|
localSID uint32
|
|
|
|
|
remoteSID uint32
|
|
|
|
|
gotRemoteSID bool
|
|
|
|
|
readChan chan []byte
|
2020-10-18 12:50:09 +02:00
|
|
|
|
2020-10-18 14:21:58 +02:00
|
|
|
pkt7 pkt7Type
|
2020-10-17 23:53:33 +02:00
|
|
|
}
|
|
|
|
|
|
2020-10-18 11:01:53 +02:00
|
|
|
func (s *streamCommon) send(d []byte) {
|
|
|
|
|
_, err := s.conn.Write(d)
|
2020-10-17 23:53:33 +02:00
|
|
|
if err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-18 18:34:22 +02:00
|
|
|
func (s *streamCommon) read() []byte {
|
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)
|
2020-10-17 23:53:33 +02:00
|
|
|
if err != nil {
|
2020-10-18 18:34:22 +02:00
|
|
|
// Ignoring timeout errors.
|
|
|
|
|
if err, ok := err.(net.Error); ok && !err.Timeout() {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2020-10-17 23:53:33 +02:00
|
|
|
}
|
2020-10-18 18:34:22 +02:00
|
|
|
return b[:n]
|
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 {
|
2020-10-18 18:34:22 +02:00
|
|
|
r := s.read()
|
|
|
|
|
if s.pkt7.isPkt7(r) {
|
|
|
|
|
s.pkt7.handle(s, r)
|
2020-10-18 10:53:16 +02:00
|
|
|
}
|
2020-10-18 18:34:22 +02:00
|
|
|
|
|
|
|
|
s.readChan <- r
|
2020-10-18 10:53:16 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-18 18:34:22 +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
|
|
|
|
|
expectStart := time.Now()
|
|
|
|
|
for {
|
2020-10-18 18:34:22 +02:00
|
|
|
err := s.conn.SetReadDeadline(time.Now().Add(timeout - time.Since(expectStart)))
|
|
|
|
|
if err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-18 13:02:41 +02:00
|
|
|
r = <-s.readChan
|
2020-10-18 18:34:22 +02:00
|
|
|
|
|
|
|
|
err = s.conn.SetReadDeadline(time.Time{})
|
|
|
|
|
if err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if len(r) == packetLength && bytes.Equal(r[matchStartByte:len(b)+matchStartByte], b) {
|
2020-10-17 23:53:33 +02:00
|
|
|
break
|
|
|
|
|
}
|
2020-10-18 18:34:22 +02:00
|
|
|
if time.Since(expectStart) > timeout {
|
|
|
|
|
return nil
|
2020-10-17 23:53:33 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return r
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-18 18:34:22 +02:00
|
|
|
func (s *streamCommon) expect(packetLength int, b []byte) []byte {
|
|
|
|
|
r := s.tryReceivePacket(expectTimeoutDuration, packetLength, 0, b)
|
|
|
|
|
if r == nil {
|
|
|
|
|
log.Fatal(s.name + "/expect timeout")
|
|
|
|
|
}
|
|
|
|
|
return r
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-18 11:17:40 +02:00
|
|
|
func (s *streamCommon) open(name string, portNumber int) {
|
|
|
|
|
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 {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2020-10-18 16:48:57 +02:00
|
|
|
|
|
|
|
|
// Use the same local and remote port. The radio does not handle different ports well.
|
|
|
|
|
l := net.UDPAddr{
|
|
|
|
|
Port: portNumber,
|
|
|
|
|
}
|
|
|
|
|
s.conn, err = net.DialUDP("udp", &l, raddr)
|
2020-10-17 23:53:33 +02:00
|
|
|
if err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-18 12:50:09 +02:00
|
|
|
// 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)
|
2020-10-18 11:17:40 +02:00
|
|
|
log.Debugf(s.name+"/using session id %.8x", s.localSID)
|
2020-10-18 12:50:09 +02:00
|
|
|
|
|
|
|
|
_, err = rand.Read(s.pkt7.randIDByte[:])
|
|
|
|
|
if err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2020-10-18 13:02:41 +02:00
|
|
|
|
|
|
|
|
s.readChan = make(chan []byte)
|
|
|
|
|
go s.reader()
|
2020-10-18 18:34:22 +02:00
|
|
|
|
|
|
|
|
if r := s.pkt7.tryReceive(300*time.Millisecond, s); s.pkt7.isPkt7(r) {
|
|
|
|
|
s.remoteSID = binary.BigEndian.Uint32(r[8:12])
|
|
|
|
|
s.gotRemoteSID = true
|
|
|
|
|
log.Print(s.name + "/closing running stream")
|
|
|
|
|
s.sendDisconnect()
|
|
|
|
|
time.Sleep(time.Second)
|
|
|
|
|
s.gotRemoteSID = false
|
|
|
|
|
}
|
2020-10-18 12:50:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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)})
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-18 13:52:38 +02:00
|
|
|
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])
|
2020-10-18 18:34:22 +02:00
|
|
|
s.gotRemoteSID = true
|
2020-10-18 13:52:38 +02:00
|
|
|
|
|
|
|
|
log.Debugf(s.name+"/got remote session id %.8x", s.remoteSID)
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-18 12:50:09 +02:00
|
|
|
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)})
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-18 13:52:38 +02:00
|
|
|
func (s *streamCommon) waitForPkt6Answer() {
|
2020-10-18 14:21:58 +02:00
|
|
|
log.Debug(s.name + "/expecting pkt6 answer")
|
2020-10-18 13:52:38 +02:00
|
|
|
// 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})
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-18 13:19:52 +02:00
|
|
|
func (s *streamCommon) sendDisconnect() {
|
2020-10-18 18:34:22 +02:00
|
|
|
if !s.gotRemoteSID {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-18 13:19:52 +02:00
|
|
|
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)})
|
|
|
|
|
}
|