mirror of
https://github.com/yggdrasil-network/yggdrasil-go.git
synced 2024-11-28 20:45:18 +00:00
Merge pull request #176 from cathugger/develop
Simpler flowlabel parsing; avoid using 0 flowlabel.
This commit is contained in:
commit
514de5434f
@ -4,7 +4,10 @@ package yggdrasil
|
|||||||
// It's responsible for keeping track of open sessions to other nodes
|
// It's responsible for keeping track of open sessions to other nodes
|
||||||
// The session information consists of crypto keys and coords
|
// The session information consists of crypto keys and coords
|
||||||
|
|
||||||
import "time"
|
import (
|
||||||
|
"bytes"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
// All the information we know about an active session.
|
// All the information we know about an active session.
|
||||||
// This includes coords, permanent and ephemeral keys, handles and nonces, various sorts of timing information for timeout and maintenance, and some metadata for the admin API.
|
// This includes coords, permanent and ephemeral keys, handles and nonces, various sorts of timing information for timeout and maintenance, and some metadata for the admin API.
|
||||||
@ -72,7 +75,10 @@ func (s *sessionInfo) update(p *sessionPing) bool {
|
|||||||
if p.MTU >= 1280 || p.MTU == 0 {
|
if p.MTU >= 1280 || p.MTU == 0 {
|
||||||
s.theirMTU = p.MTU
|
s.theirMTU = p.MTU
|
||||||
}
|
}
|
||||||
s.coords = append([]byte{}, p.Coords...)
|
if !bytes.Equal(s.coords, p.Coords) {
|
||||||
|
// allocate enough space for additional coords
|
||||||
|
s.coords = append(make([]byte, 0, len(p.Coords)+11), p.Coords...)
|
||||||
|
}
|
||||||
now := time.Now()
|
now := time.Now()
|
||||||
s.time = now
|
s.time = now
|
||||||
s.tstamp = p.Tstamp
|
s.tstamp = p.Tstamp
|
||||||
@ -423,30 +429,38 @@ func (sinfo *sessionInfo) doWorker() {
|
|||||||
func (sinfo *sessionInfo) doSend(bs []byte) {
|
func (sinfo *sessionInfo) doSend(bs []byte) {
|
||||||
defer util_putBytes(bs)
|
defer util_putBytes(bs)
|
||||||
if !sinfo.init {
|
if !sinfo.init {
|
||||||
|
// To prevent using empty session keys
|
||||||
return
|
return
|
||||||
} // To prevent using empty session keys
|
|
||||||
// Now we append something to the coords
|
|
||||||
// Specifically, we append a 0, and then arbitrary data
|
|
||||||
// The 0 ensures that the destination node switch forwards to the self peer (router)
|
|
||||||
// The rest is ignored, but it's still part as the coords, so it affects switch queues
|
|
||||||
// This helps separate traffic streams (coords, flowlabel) to be queued independently
|
|
||||||
var coords []byte
|
|
||||||
addUint64 := func(bs []byte) {
|
|
||||||
// Converts bytes to a uint64
|
|
||||||
// Converts that back to variable length bytes
|
|
||||||
// Appends it to coords
|
|
||||||
var u uint64
|
|
||||||
for _, b := range bs {
|
|
||||||
u <<= 8
|
|
||||||
u |= uint64(b)
|
|
||||||
}
|
|
||||||
coords = append(coords, wire_encode_uint64(u)...)
|
|
||||||
}
|
}
|
||||||
coords = append(coords, sinfo.coords...) // Start with the real coords
|
// code isn't multithreaded so appending to this is safe
|
||||||
coords = append(coords, 0) // Then target the local switchport
|
coords := sinfo.coords
|
||||||
flowlabel := append([]byte(nil), bs[1:4]...)
|
// Read IPv6 flowlabel field (20 bits).
|
||||||
flowlabel[0] &= 0x0f
|
// Assumes packet at least contains IPv6 header.
|
||||||
addUint64(flowlabel)
|
flowkey := uint64(bs[1]&0x0f)<<16 | uint64(bs[2])<<8 | uint64(bs[3])
|
||||||
|
// Check if the flowlabel was specified
|
||||||
|
if flowkey == 0 {
|
||||||
|
// Does the packet meet the minimum UDP packet size? (others are bigger)
|
||||||
|
if len(bs) >= 48 {
|
||||||
|
// Is the protocol TCP, UDP, SCTP?
|
||||||
|
if bs[6] == 0x06 || bs[6] == 0x11 || bs[6] == 0x84 {
|
||||||
|
// if flowlabel was unspecified (0), try to use known protocols' ports
|
||||||
|
// protokey: proto | sport | dport
|
||||||
|
flowkey = uint64(bs[6])<<32 /* proto */ |
|
||||||
|
uint64(bs[40])<<24 | uint64(bs[41])<<16 /* sport */ |
|
||||||
|
uint64(bs[42])<<8 | uint64(bs[43]) /* dport */
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// If we have a flowkey, either through the IPv6 flowlabel field or through
|
||||||
|
// known TCP/UDP/SCTP proto-sport-dport triplet, then append it to the coords.
|
||||||
|
// Appending extra coords after a 0 ensures that we still target the local router
|
||||||
|
// but lets us send extra data (which is otherwise ignored) to help separate
|
||||||
|
// traffic streams into independent queues
|
||||||
|
if flowkey != 0 {
|
||||||
|
coords = append(coords, 0) // First target the local switchport
|
||||||
|
coords = wire_put_uint64(flowkey, coords) // Then variable-length encoded flowkey
|
||||||
|
}
|
||||||
|
// Prepare the payload
|
||||||
payload, nonce := boxSeal(&sinfo.sharedSesKey, bs, &sinfo.myNonce)
|
payload, nonce := boxSeal(&sinfo.sharedSesKey, bs, &sinfo.myNonce)
|
||||||
defer util_putBytes(payload)
|
defer util_putBytes(payload)
|
||||||
p := wire_trafficPacket{
|
p := wire_trafficPacket{
|
||||||
|
@ -25,19 +25,15 @@ func wire_encode_uint64(elem uint64) []byte {
|
|||||||
|
|
||||||
// Encode uint64 using a variable length scheme.
|
// Encode uint64 using a variable length scheme.
|
||||||
// Similar to binary.Uvarint, but big-endian.
|
// Similar to binary.Uvarint, but big-endian.
|
||||||
func wire_put_uint64(elem uint64, out []byte) []byte {
|
func wire_put_uint64(e uint64, out []byte) []byte {
|
||||||
bs := make([]byte, 0, 10)
|
var b [10]byte
|
||||||
bs = append(bs, byte(elem&0x7f))
|
i := len(b) - 1
|
||||||
for e := elem >> 7; e > 0; e >>= 7 {
|
b[i] = byte(e & 0x7f)
|
||||||
bs = append(bs, byte(e|0x80))
|
for e >>= 7; e != 0; e >>= 7 {
|
||||||
|
i--
|
||||||
|
b[i] = byte(e | 0x80)
|
||||||
}
|
}
|
||||||
// Now reverse bytes, because we set them in the wrong order
|
return append(out, b[i:]...)
|
||||||
// TODO just put them in the right place the first time...
|
|
||||||
last := len(bs) - 1
|
|
||||||
for idx := 0; idx < len(bs)/2; idx++ {
|
|
||||||
bs[idx], bs[last-idx] = bs[last-idx], bs[idx]
|
|
||||||
}
|
|
||||||
return append(out, bs...)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Returns the length of a wire encoded uint64 of this value.
|
// Returns the length of a wire encoded uint64 of this value.
|
||||||
|
Loading…
Reference in New Issue
Block a user