Work in progress, add a linkInterfaceMsgIO interface type and make stream implement it, this will be used by link

This commit is contained in:
Arceliar 2019-01-21 21:27:52 -06:00
parent c8e1be0f73
commit 5a4d6481dd
4 changed files with 110 additions and 6 deletions

View File

@ -45,7 +45,7 @@ func (l *awdl) create(fromAWDL chan []byte, toAWDL chan []byte, name string) (*a
inPacket := func(packet []byte) { inPacket := func(packet []byte) {
intf.link.fromlink <- packet intf.link.fromlink <- packet
} }
intf.stream.init(inPacket) intf.stream.init(nil, inPacket) // FIXME nil = ReadWriteCloser
go intf.handler() go intf.handler()
l.mutex.Lock() l.mutex.Lock()
l.interfaces[name] = &intf l.interfaces[name] = &intf

View File

@ -17,6 +17,15 @@ type link struct {
interfaces map[string]*linkInterface interfaces map[string]*linkInterface
} }
type linkInterfaceMsgIO interface {
readMsg() ([]byte, error)
writeMsg([]byte) (int, error)
close() error
// These are temporary workarounds to stream semantics
_sendMetaBytes([]byte) error
_recvMetaBytes() ([]byte, error)
}
type linkInterface struct { type linkInterface struct {
name string name string
link *link link *link

View File

@ -3,21 +3,115 @@ package yggdrasil
import ( import (
"errors" "errors"
"fmt" "fmt"
"io"
"github.com/yggdrasil-network/yggdrasil-go/src/util" "github.com/yggdrasil-network/yggdrasil-go/src/util"
) )
// Test that this matches the interface we expect
var _ = linkInterfaceMsgIO(&stream{})
type stream struct { type stream struct {
inputBuffer []byte rwc io.ReadWriteCloser
inputBuffer []byte // Incoming packet stream
didFirstSend bool // Used for metadata exchange
didFirstRecv bool // Used for metadata exchange
// TODO remove the rest, it shouldn't matter in the long run
handlePacket func([]byte) handlePacket func([]byte)
} }
func (s *stream) close() error {
return s.rwc.Close()
}
const streamMsgSize = 2048 + 65535 const streamMsgSize = 2048 + 65535
var streamMsg = [...]byte{0xde, 0xad, 0xb1, 0x75} // "dead bits" var streamMsg = [...]byte{0xde, 0xad, 0xb1, 0x75} // "dead bits"
func (s *stream) init(in func([]byte)) { func (s *stream) init(rwc io.ReadWriteCloser, in func([]byte)) {
// TODO have this also do the metadata handshake and create the peer struct
s.rwc = rwc
s.handlePacket = in s.handlePacket = in
// TODO call something to do the metadata exchange
}
// writeMsg writes a message with stream padding, and is *not* thread safe.
func (s *stream) writeMsg(bs []byte) (int, error) {
buf := util.GetBytes()
defer util.PutBytes(buf)
buf = append(buf, streamMsg[:]...)
buf = append(buf, wire_encode_uint64(uint64(len(bs)))...)
padLen := len(buf)
buf = append(buf, bs...)
var bn int
for bn < len(buf) {
n, err := s.rwc.Write(buf[bn:])
bn += n
if err != nil {
l := bn - padLen
if l < 0 {
l = 0
}
return l, err
}
}
return len(bs), nil
}
// readMsg reads a message from the stream, accounting for stream padding, and is *not* thread safe.
func (s *stream) readMsg() ([]byte, error) {
for {
buf := s.inputBuffer
msg, ok, err := stream_chopMsg(&buf)
switch {
case err != nil:
// Something in the stream format is corrupt
return nil, fmt.Errorf("message error: %v", err)
case ok:
// Copy the packet into bs, shift the buffer, and return
msg = append(util.GetBytes(), msg...)
s.inputBuffer = append(s.inputBuffer[:0], buf...)
return msg, nil
default:
// Wait for the underlying reader to return enough info for us to proceed
frag := make([]byte, 2*streamMsgSize)
n, err := s.rwc.Read(frag)
if n > 0 {
s.inputBuffer = append(s.inputBuffer, frag[:n]...)
} else if err != nil {
return nil, err
}
}
}
}
// Writes metadata bytes without stream padding, meant to be temporary
func (s *stream) _sendMetaBytes(metaBytes []byte) error {
var written int
for written < len(metaBytes) {
n, err := s.rwc.Write(metaBytes)
written += n
if err != nil {
return err
}
}
return nil
}
// Reads metadata bytes without stream padding, meant to be temporary
func (s *stream) _recvMetaBytes() ([]byte, error) {
var meta version_metadata
frag := meta.encode()
metaBytes := make([]byte, 0, len(frag))
for len(metaBytes) < len(frag) {
n, err := s.rwc.Read(frag)
if err != nil {
return nil, err
}
metaBytes = append(metaBytes, frag[:n]...)
}
return metaBytes, nil
} }
// This reads from the channel into a []byte buffer for incoming messages. It // This reads from the channel into a []byte buffer for incoming messages. It
@ -28,7 +122,8 @@ func (s *stream) init(in func([]byte)) {
func (s *stream) handleInput(bs []byte) error { func (s *stream) handleInput(bs []byte) error {
if len(bs) > 0 { if len(bs) > 0 {
s.inputBuffer = append(s.inputBuffer, bs...) s.inputBuffer = append(s.inputBuffer, bs...)
msg, ok, err2 := stream_chopMsg(&s.inputBuffer) buf := s.inputBuffer
msg, ok, err2 := stream_chopMsg(&buf)
if err2 != nil { if err2 != nil {
return fmt.Errorf("message error: %v", err2) return fmt.Errorf("message error: %v", err2)
} }
@ -37,7 +132,7 @@ func (s *stream) handleInput(bs []byte) error {
return nil return nil
} }
newMsg := append(util.GetBytes(), msg...) newMsg := append(util.GetBytes(), msg...)
s.inputBuffer = append(s.inputBuffer[:0], s.inputBuffer...) s.inputBuffer = append(s.inputBuffer[:0], buf...)
s.handlePacket(newMsg) s.handlePacket(newMsg)
util.Yield() // Make sure we give up control to the scheduler util.Yield() // Make sure we give up control to the scheduler
} }

View File

@ -440,7 +440,7 @@ func (iface *tcpInterface) handler(sock net.Conn, incoming bool) {
themAddrString := net.IP(themAddr[:]).String() themAddrString := net.IP(themAddr[:]).String()
themString := fmt.Sprintf("%s@%s", themAddrString, them) themString := fmt.Sprintf("%s@%s", themAddrString, them)
iface.core.log.Printf("Connected: %s, source: %s", themString, us) iface.core.log.Printf("Connected: %s, source: %s", themString, us)
iface.stream.init(p.handlePacket) iface.stream.init(sock, p.handlePacket)
bs := make([]byte, 2*streamMsgSize) bs := make([]byte, 2*streamMsgSize)
var n int var n int
for { for {