94 lines
1.9 KiB
Go
94 lines
1.9 KiB
Go
// Copyright 2016 The Go Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
package ipv4_test
|
|
|
|
import (
|
|
"net"
|
|
"runtime"
|
|
"testing"
|
|
"time"
|
|
|
|
"golang.org/x/net/bpf"
|
|
"golang.org/x/net/ipv4"
|
|
)
|
|
|
|
func TestBPF(t *testing.T) {
|
|
if runtime.GOOS != "linux" {
|
|
t.Skipf("not supported on %s", runtime.GOOS)
|
|
}
|
|
|
|
l, err := net.ListenPacket("udp4", "127.0.0.1:0")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer l.Close()
|
|
|
|
p := ipv4.NewPacketConn(l)
|
|
|
|
// This filter accepts UDP packets whose first payload byte is
|
|
// even.
|
|
prog, err := bpf.Assemble([]bpf.Instruction{
|
|
// Load the first byte of the payload (skipping UDP header).
|
|
bpf.LoadAbsolute{Off: 8, Size: 1},
|
|
// Select LSB of the byte.
|
|
bpf.ALUOpConstant{Op: bpf.ALUOpAnd, Val: 1},
|
|
// Byte is even?
|
|
bpf.JumpIf{Cond: bpf.JumpEqual, Val: 0, SkipFalse: 1},
|
|
// Accept.
|
|
bpf.RetConstant{Val: 4096},
|
|
// Ignore.
|
|
bpf.RetConstant{Val: 0},
|
|
})
|
|
if err != nil {
|
|
t.Fatalf("compiling BPF: %s", err)
|
|
}
|
|
|
|
if err = p.SetBPF(prog); err != nil {
|
|
t.Fatalf("attaching filter to Conn: %s", err)
|
|
}
|
|
|
|
s, err := net.Dial("udp4", l.LocalAddr().String())
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer s.Close()
|
|
go func() {
|
|
for i := byte(0); i < 10; i++ {
|
|
s.Write([]byte{i})
|
|
}
|
|
}()
|
|
|
|
l.SetDeadline(time.Now().Add(2 * time.Second))
|
|
seen := make([]bool, 5)
|
|
for {
|
|
var b [512]byte
|
|
n, _, err := l.ReadFrom(b[:])
|
|
if err != nil {
|
|
t.Fatalf("reading from listener: %s", err)
|
|
}
|
|
if n != 1 {
|
|
t.Fatalf("unexpected packet length, want 1, got %d", n)
|
|
}
|
|
if b[0] >= 10 {
|
|
t.Fatalf("unexpected byte, want 0-9, got %d", b[0])
|
|
}
|
|
if b[0]%2 != 0 {
|
|
t.Fatalf("got odd byte %d, wanted only even bytes", b[0])
|
|
}
|
|
seen[b[0]/2] = true
|
|
|
|
seenAll := true
|
|
for _, v := range seen {
|
|
if !v {
|
|
seenAll = false
|
|
break
|
|
}
|
|
}
|
|
if seenAll {
|
|
break
|
|
}
|
|
}
|
|
}
|