OSDN Git Service

Hulk did something
[bytom/vapor.git] / vendor / golang.org / x / net / internal / socket / socket_go1_9_test.go
diff --git a/vendor/golang.org/x/net/internal/socket/socket_go1_9_test.go b/vendor/golang.org/x/net/internal/socket/socket_go1_9_test.go
new file mode 100644 (file)
index 0000000..109fed7
--- /dev/null
@@ -0,0 +1,256 @@
+// Copyright 2017 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.
+
+// +build go1.9
+// +build darwin dragonfly freebsd linux netbsd openbsd solaris
+
+package socket_test
+
+import (
+       "bytes"
+       "fmt"
+       "net"
+       "runtime"
+       "testing"
+
+       "golang.org/x/net/internal/nettest"
+       "golang.org/x/net/internal/socket"
+)
+
+type mockControl struct {
+       Level int
+       Type  int
+       Data  []byte
+}
+
+func TestControlMessage(t *testing.T) {
+       for _, tt := range []struct {
+               cs []mockControl
+       }{
+               {
+                       []mockControl{
+                               {Level: 1, Type: 1},
+                       },
+               },
+               {
+                       []mockControl{
+                               {Level: 2, Type: 2, Data: []byte{0xfe}},
+                       },
+               },
+               {
+                       []mockControl{
+                               {Level: 3, Type: 3, Data: []byte{0xfe, 0xff, 0xff, 0xfe}},
+                       },
+               },
+               {
+                       []mockControl{
+                               {Level: 4, Type: 4, Data: []byte{0xfe, 0xff, 0xff, 0xfe, 0xfe, 0xff, 0xff, 0xfe}},
+                       },
+               },
+               {
+                       []mockControl{
+                               {Level: 4, Type: 4, Data: []byte{0xfe, 0xff, 0xff, 0xfe, 0xfe, 0xff, 0xff, 0xfe}},
+                               {Level: 2, Type: 2, Data: []byte{0xfe}},
+                       },
+               },
+       } {
+               var w []byte
+               var tailPadLen int
+               mm := socket.NewControlMessage([]int{0})
+               for i, c := range tt.cs {
+                       m := socket.NewControlMessage([]int{len(c.Data)})
+                       l := len(m) - len(mm)
+                       if i == len(tt.cs)-1 && l > len(c.Data) {
+                               tailPadLen = l - len(c.Data)
+                       }
+                       w = append(w, m...)
+               }
+
+               var err error
+               ww := make([]byte, len(w))
+               copy(ww, w)
+               m := socket.ControlMessage(ww)
+               for _, c := range tt.cs {
+                       if err = m.MarshalHeader(c.Level, c.Type, len(c.Data)); err != nil {
+                               t.Fatalf("(%v).MarshalHeader() = %v", tt.cs, err)
+                       }
+                       copy(m.Data(len(c.Data)), c.Data)
+                       m = m.Next(len(c.Data))
+               }
+               m = socket.ControlMessage(w)
+               for _, c := range tt.cs {
+                       m, err = m.Marshal(c.Level, c.Type, c.Data)
+                       if err != nil {
+                               t.Fatalf("(%v).Marshal() = %v", tt.cs, err)
+                       }
+               }
+               if !bytes.Equal(ww, w) {
+                       t.Fatalf("got %#v; want %#v", ww, w)
+               }
+
+               ws := [][]byte{w}
+               if tailPadLen > 0 {
+                       // Test a message with no tail padding.
+                       nopad := w[:len(w)-tailPadLen]
+                       ws = append(ws, [][]byte{nopad}...)
+               }
+               for _, w := range ws {
+                       ms, err := socket.ControlMessage(w).Parse()
+                       if err != nil {
+                               t.Fatalf("(%v).Parse() = %v", tt.cs, err)
+                       }
+                       for i, m := range ms {
+                               lvl, typ, dataLen, err := m.ParseHeader()
+                               if err != nil {
+                                       t.Fatalf("(%v).ParseHeader() = %v", tt.cs, err)
+                               }
+                               if lvl != tt.cs[i].Level || typ != tt.cs[i].Type || dataLen != len(tt.cs[i].Data) {
+                                       t.Fatalf("%v: got %d, %d, %d; want %d, %d, %d", tt.cs[i], lvl, typ, dataLen, tt.cs[i].Level, tt.cs[i].Type, len(tt.cs[i].Data))
+                               }
+                       }
+               }
+       }
+}
+
+func TestUDP(t *testing.T) {
+       c, err := nettest.NewLocalPacketListener("udp")
+       if err != nil {
+               t.Skipf("not supported on %s/%s: %v", runtime.GOOS, runtime.GOARCH, err)
+       }
+       defer c.Close()
+
+       t.Run("Message", func(t *testing.T) {
+               testUDPMessage(t, c.(net.Conn))
+       })
+       switch runtime.GOOS {
+       case "linux":
+               t.Run("Messages", func(t *testing.T) {
+                       testUDPMessages(t, c.(net.Conn))
+               })
+       }
+}
+
+func testUDPMessage(t *testing.T, c net.Conn) {
+       cc, err := socket.NewConn(c)
+       if err != nil {
+               t.Fatal(err)
+       }
+       data := []byte("HELLO-R-U-THERE")
+       wm := socket.Message{
+               Buffers: bytes.SplitAfter(data, []byte("-")),
+               Addr:    c.LocalAddr(),
+       }
+       if err := cc.SendMsg(&wm, 0); err != nil {
+               t.Fatal(err)
+       }
+       b := make([]byte, 32)
+       rm := socket.Message{
+               Buffers: [][]byte{b[:1], b[1:3], b[3:7], b[7:11], b[11:]},
+       }
+       if err := cc.RecvMsg(&rm, 0); err != nil {
+               t.Fatal(err)
+       }
+       if !bytes.Equal(b[:rm.N], data) {
+               t.Fatalf("got %#v; want %#v", b[:rm.N], data)
+       }
+}
+
+func testUDPMessages(t *testing.T, c net.Conn) {
+       cc, err := socket.NewConn(c)
+       if err != nil {
+               t.Fatal(err)
+       }
+       data := []byte("HELLO-R-U-THERE")
+       wmbs := bytes.SplitAfter(data, []byte("-"))
+       wms := []socket.Message{
+               {Buffers: wmbs[:1], Addr: c.LocalAddr()},
+               {Buffers: wmbs[1:], Addr: c.LocalAddr()},
+       }
+       n, err := cc.SendMsgs(wms, 0)
+       if err != nil {
+               t.Fatal(err)
+       }
+       if n != len(wms) {
+               t.Fatalf("got %d; want %d", n, len(wms))
+       }
+       b := make([]byte, 32)
+       rmbs := [][][]byte{{b[:len(wmbs[0])]}, {b[len(wmbs[0]):]}}
+       rms := []socket.Message{
+               {Buffers: rmbs[0]},
+               {Buffers: rmbs[1]},
+       }
+       n, err = cc.RecvMsgs(rms, 0)
+       if err != nil {
+               t.Fatal(err)
+       }
+       if n != len(rms) {
+               t.Fatalf("got %d; want %d", n, len(rms))
+       }
+       nn := 0
+       for i := 0; i < n; i++ {
+               nn += rms[i].N
+       }
+       if !bytes.Equal(b[:nn], data) {
+               t.Fatalf("got %#v; want %#v", b[:nn], data)
+       }
+}
+
+func BenchmarkUDP(b *testing.B) {
+       c, err := nettest.NewLocalPacketListener("udp")
+       if err != nil {
+               b.Skipf("not supported on %s/%s: %v", runtime.GOOS, runtime.GOARCH, err)
+       }
+       defer c.Close()
+       cc, err := socket.NewConn(c.(net.Conn))
+       if err != nil {
+               b.Fatal(err)
+       }
+       data := []byte("HELLO-R-U-THERE")
+       wm := socket.Message{
+               Buffers: [][]byte{data},
+               Addr:    c.LocalAddr(),
+       }
+       rm := socket.Message{
+               Buffers: [][]byte{make([]byte, 128)},
+               OOB:     make([]byte, 128),
+       }
+
+       for M := 1; M <= 1<<9; M = M << 1 {
+               b.Run(fmt.Sprintf("Iter-%d", M), func(b *testing.B) {
+                       for i := 0; i < b.N; i++ {
+                               for j := 0; j < M; j++ {
+                                       if err := cc.SendMsg(&wm, 0); err != nil {
+                                               b.Fatal(err)
+                                       }
+                                       if err := cc.RecvMsg(&rm, 0); err != nil {
+                                               b.Fatal(err)
+                                       }
+                               }
+                       }
+               })
+               switch runtime.GOOS {
+               case "linux":
+                       wms := make([]socket.Message, M)
+                       for i := range wms {
+                               wms[i].Buffers = [][]byte{data}
+                               wms[i].Addr = c.LocalAddr()
+                       }
+                       rms := make([]socket.Message, M)
+                       for i := range rms {
+                               rms[i].Buffers = [][]byte{make([]byte, 128)}
+                               rms[i].OOB = make([]byte, 128)
+                       }
+                       b.Run(fmt.Sprintf("Batch-%d", M), func(b *testing.B) {
+                               for i := 0; i < b.N; i++ {
+                                       if _, err := cc.SendMsgs(wms, 0); err != nil {
+                                               b.Fatal(err)
+                                       }
+                                       if _, err := cc.RecvMsgs(rms, 0); err != nil {
+                                               b.Fatal(err)
+                                       }
+                               }
+                       })
+               }
+       }
+}