mirror of
				https://github.com/tailscale/tailscale.git
				synced 2025-10-25 10:09:17 +00:00 
			
		
		
		
	 a93dc6cdb1
			
		
	
	a93dc6cdb1
	
	
	
		
			
			This commit adds a batchingConn interface, and renames batchingUDPConn to linuxBatchingConn. tryUpgradeToBatchingConn() may return a platform- specific implementation of batchingConn. So far only a Linux implementation of this interface exists, but this refactor is being done in anticipation of a Windows implementation. Updates tailscale/corp#21874 Signed-off-by: Jordan Whited <jordan@tailscale.com>
		
			
				
	
	
		
			245 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			245 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright (c) Tailscale Inc & AUTHORS
 | |
| // SPDX-License-Identifier: BSD-3-Clause
 | |
| 
 | |
| package magicsock
 | |
| 
 | |
| import (
 | |
| 	"encoding/binary"
 | |
| 	"net"
 | |
| 	"testing"
 | |
| 
 | |
| 	"golang.org/x/net/ipv6"
 | |
| )
 | |
| 
 | |
| func setGSOSize(control *[]byte, gsoSize uint16) {
 | |
| 	*control = (*control)[:cap(*control)]
 | |
| 	binary.LittleEndian.PutUint16(*control, gsoSize)
 | |
| }
 | |
| 
 | |
| func getGSOSize(control []byte) (int, error) {
 | |
| 	if len(control) < 2 {
 | |
| 		return 0, nil
 | |
| 	}
 | |
| 	return int(binary.LittleEndian.Uint16(control)), nil
 | |
| }
 | |
| 
 | |
| func Test_linuxBatchingConn_splitCoalescedMessages(t *testing.T) {
 | |
| 	c := &linuxBatchingConn{
 | |
| 		setGSOSizeInControl:   setGSOSize,
 | |
| 		getGSOSizeFromControl: getGSOSize,
 | |
| 	}
 | |
| 
 | |
| 	newMsg := func(n, gso int) ipv6.Message {
 | |
| 		msg := ipv6.Message{
 | |
| 			Buffers: [][]byte{make([]byte, 1024)},
 | |
| 			N:       n,
 | |
| 			OOB:     make([]byte, 2),
 | |
| 		}
 | |
| 		binary.LittleEndian.PutUint16(msg.OOB, uint16(gso))
 | |
| 		if gso > 0 {
 | |
| 			msg.NN = 2
 | |
| 		}
 | |
| 		return msg
 | |
| 	}
 | |
| 
 | |
| 	cases := []struct {
 | |
| 		name        string
 | |
| 		msgs        []ipv6.Message
 | |
| 		firstMsgAt  int
 | |
| 		wantNumEval int
 | |
| 		wantMsgLens []int
 | |
| 		wantErr     bool
 | |
| 	}{
 | |
| 		{
 | |
| 			name: "second last split last empty",
 | |
| 			msgs: []ipv6.Message{
 | |
| 				newMsg(0, 0),
 | |
| 				newMsg(0, 0),
 | |
| 				newMsg(3, 1),
 | |
| 				newMsg(0, 0),
 | |
| 			},
 | |
| 			firstMsgAt:  2,
 | |
| 			wantNumEval: 3,
 | |
| 			wantMsgLens: []int{1, 1, 1, 0},
 | |
| 			wantErr:     false,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "second last no split last empty",
 | |
| 			msgs: []ipv6.Message{
 | |
| 				newMsg(0, 0),
 | |
| 				newMsg(0, 0),
 | |
| 				newMsg(1, 0),
 | |
| 				newMsg(0, 0),
 | |
| 			},
 | |
| 			firstMsgAt:  2,
 | |
| 			wantNumEval: 1,
 | |
| 			wantMsgLens: []int{1, 0, 0, 0},
 | |
| 			wantErr:     false,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "second last no split last no split",
 | |
| 			msgs: []ipv6.Message{
 | |
| 				newMsg(0, 0),
 | |
| 				newMsg(0, 0),
 | |
| 				newMsg(1, 0),
 | |
| 				newMsg(1, 0),
 | |
| 			},
 | |
| 			firstMsgAt:  2,
 | |
| 			wantNumEval: 2,
 | |
| 			wantMsgLens: []int{1, 1, 0, 0},
 | |
| 			wantErr:     false,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "second last no split last split",
 | |
| 			msgs: []ipv6.Message{
 | |
| 				newMsg(0, 0),
 | |
| 				newMsg(0, 0),
 | |
| 				newMsg(1, 0),
 | |
| 				newMsg(3, 1),
 | |
| 			},
 | |
| 			firstMsgAt:  2,
 | |
| 			wantNumEval: 4,
 | |
| 			wantMsgLens: []int{1, 1, 1, 1},
 | |
| 			wantErr:     false,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "second last split last split",
 | |
| 			msgs: []ipv6.Message{
 | |
| 				newMsg(0, 0),
 | |
| 				newMsg(0, 0),
 | |
| 				newMsg(2, 1),
 | |
| 				newMsg(2, 1),
 | |
| 			},
 | |
| 			firstMsgAt:  2,
 | |
| 			wantNumEval: 4,
 | |
| 			wantMsgLens: []int{1, 1, 1, 1},
 | |
| 			wantErr:     false,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "second last no split last split overflow",
 | |
| 			msgs: []ipv6.Message{
 | |
| 				newMsg(0, 0),
 | |
| 				newMsg(0, 0),
 | |
| 				newMsg(1, 0),
 | |
| 				newMsg(4, 1),
 | |
| 			},
 | |
| 			firstMsgAt:  2,
 | |
| 			wantNumEval: 4,
 | |
| 			wantMsgLens: []int{1, 1, 1, 1},
 | |
| 			wantErr:     true,
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	for _, tt := range cases {
 | |
| 		t.Run(tt.name, func(t *testing.T) {
 | |
| 			got, err := c.splitCoalescedMessages(tt.msgs, 2)
 | |
| 			if err != nil && !tt.wantErr {
 | |
| 				t.Fatalf("err: %v", err)
 | |
| 			}
 | |
| 			if got != tt.wantNumEval {
 | |
| 				t.Fatalf("got to eval: %d want: %d", got, tt.wantNumEval)
 | |
| 			}
 | |
| 			for i, msg := range tt.msgs {
 | |
| 				if msg.N != tt.wantMsgLens[i] {
 | |
| 					t.Fatalf("msg[%d].N: %d want: %d", i, msg.N, tt.wantMsgLens[i])
 | |
| 				}
 | |
| 			}
 | |
| 		})
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func Test_linuxBatchingConn_coalesceMessages(t *testing.T) {
 | |
| 	c := &linuxBatchingConn{
 | |
| 		setGSOSizeInControl:   setGSOSize,
 | |
| 		getGSOSizeFromControl: getGSOSize,
 | |
| 	}
 | |
| 
 | |
| 	cases := []struct {
 | |
| 		name     string
 | |
| 		buffs    [][]byte
 | |
| 		wantLens []int
 | |
| 		wantGSO  []int
 | |
| 	}{
 | |
| 		{
 | |
| 			name: "one message no coalesce",
 | |
| 			buffs: [][]byte{
 | |
| 				make([]byte, 1, 1),
 | |
| 			},
 | |
| 			wantLens: []int{1},
 | |
| 			wantGSO:  []int{0},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "two messages equal len coalesce",
 | |
| 			buffs: [][]byte{
 | |
| 				make([]byte, 1, 2),
 | |
| 				make([]byte, 1, 1),
 | |
| 			},
 | |
| 			wantLens: []int{2},
 | |
| 			wantGSO:  []int{1},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "two messages unequal len coalesce",
 | |
| 			buffs: [][]byte{
 | |
| 				make([]byte, 2, 3),
 | |
| 				make([]byte, 1, 1),
 | |
| 			},
 | |
| 			wantLens: []int{3},
 | |
| 			wantGSO:  []int{2},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "three messages second unequal len coalesce",
 | |
| 			buffs: [][]byte{
 | |
| 				make([]byte, 2, 3),
 | |
| 				make([]byte, 1, 1),
 | |
| 				make([]byte, 2, 2),
 | |
| 			},
 | |
| 			wantLens: []int{3, 2},
 | |
| 			wantGSO:  []int{2, 0},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "three messages limited cap coalesce",
 | |
| 			buffs: [][]byte{
 | |
| 				make([]byte, 2, 4),
 | |
| 				make([]byte, 2, 2),
 | |
| 				make([]byte, 2, 2),
 | |
| 			},
 | |
| 			wantLens: []int{4, 2},
 | |
| 			wantGSO:  []int{2, 0},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	for _, tt := range cases {
 | |
| 		t.Run(tt.name, func(t *testing.T) {
 | |
| 			addr := &net.UDPAddr{
 | |
| 				IP:   net.ParseIP("127.0.0.1"),
 | |
| 				Port: 1,
 | |
| 			}
 | |
| 			msgs := make([]ipv6.Message, len(tt.buffs))
 | |
| 			for i := range msgs {
 | |
| 				msgs[i].Buffers = make([][]byte, 1)
 | |
| 				msgs[i].OOB = make([]byte, 0, 2)
 | |
| 			}
 | |
| 			got := c.coalesceMessages(addr, tt.buffs, msgs)
 | |
| 			if got != len(tt.wantLens) {
 | |
| 				t.Fatalf("got len %d want: %d", got, len(tt.wantLens))
 | |
| 			}
 | |
| 			for i := range got {
 | |
| 				if msgs[i].Addr != addr {
 | |
| 					t.Errorf("msgs[%d].Addr != passed addr", i)
 | |
| 				}
 | |
| 				gotLen := len(msgs[i].Buffers[0])
 | |
| 				if gotLen != tt.wantLens[i] {
 | |
| 					t.Errorf("len(msgs[%d].Buffers[0]) %d != %d", i, gotLen, tt.wantLens[i])
 | |
| 				}
 | |
| 				gotGSO, err := getGSOSize(msgs[i].OOB)
 | |
| 				if err != nil {
 | |
| 					t.Fatalf("msgs[%d] getGSOSize err: %v", i, err)
 | |
| 				}
 | |
| 				if gotGSO != tt.wantGSO[i] {
 | |
| 					t.Errorf("msgs[%d] gsoSize %d != %d", i, gotGSO, tt.wantGSO[i])
 | |
| 				}
 | |
| 			}
 | |
| 		})
 | |
| 	}
 | |
| }
 |