You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 

299 lines
6.9 KiB

  1. // Copyright 2017 The Go Authors. All rights reserved.
  2. // Use of this source code is governed by a BSD-style
  3. // license that can be found in the LICENSE file.
  4. // +build darwin dragonfly freebsd linux netbsd openbsd solaris windows
  5. package socket_test
  6. import (
  7. "bytes"
  8. "fmt"
  9. "net"
  10. "runtime"
  11. "syscall"
  12. "testing"
  13. "golang.org/x/net/internal/nettest"
  14. "golang.org/x/net/internal/socket"
  15. )
  16. func TestSocket(t *testing.T) {
  17. t.Run("Option", func(t *testing.T) {
  18. testSocketOption(t, &socket.Option{Level: syscall.SOL_SOCKET, Name: syscall.SO_RCVBUF, Len: 4})
  19. })
  20. }
  21. func testSocketOption(t *testing.T, so *socket.Option) {
  22. c, err := nettest.NewLocalPacketListener("udp")
  23. if err != nil {
  24. t.Skipf("not supported on %s/%s: %v", runtime.GOOS, runtime.GOARCH, err)
  25. }
  26. defer c.Close()
  27. cc, err := socket.NewConn(c.(net.Conn))
  28. if err != nil {
  29. t.Fatal(err)
  30. }
  31. const N = 2048
  32. if err := so.SetInt(cc, N); err != nil {
  33. t.Fatal(err)
  34. }
  35. n, err := so.GetInt(cc)
  36. if err != nil {
  37. t.Fatal(err)
  38. }
  39. if n < N {
  40. t.Fatalf("got %d; want greater than or equal to %d", n, N)
  41. }
  42. }
  43. type mockControl struct {
  44. Level int
  45. Type int
  46. Data []byte
  47. }
  48. func TestControlMessage(t *testing.T) {
  49. switch runtime.GOOS {
  50. case "windows":
  51. t.Skipf("not supported on %s", runtime.GOOS)
  52. }
  53. for _, tt := range []struct {
  54. cs []mockControl
  55. }{
  56. {
  57. []mockControl{
  58. {Level: 1, Type: 1},
  59. },
  60. },
  61. {
  62. []mockControl{
  63. {Level: 2, Type: 2, Data: []byte{0xfe}},
  64. },
  65. },
  66. {
  67. []mockControl{
  68. {Level: 3, Type: 3, Data: []byte{0xfe, 0xff, 0xff, 0xfe}},
  69. },
  70. },
  71. {
  72. []mockControl{
  73. {Level: 4, Type: 4, Data: []byte{0xfe, 0xff, 0xff, 0xfe, 0xfe, 0xff, 0xff, 0xfe}},
  74. },
  75. },
  76. {
  77. []mockControl{
  78. {Level: 4, Type: 4, Data: []byte{0xfe, 0xff, 0xff, 0xfe, 0xfe, 0xff, 0xff, 0xfe}},
  79. {Level: 2, Type: 2, Data: []byte{0xfe}},
  80. },
  81. },
  82. } {
  83. var w []byte
  84. var tailPadLen int
  85. mm := socket.NewControlMessage([]int{0})
  86. for i, c := range tt.cs {
  87. m := socket.NewControlMessage([]int{len(c.Data)})
  88. l := len(m) - len(mm)
  89. if i == len(tt.cs)-1 && l > len(c.Data) {
  90. tailPadLen = l - len(c.Data)
  91. }
  92. w = append(w, m...)
  93. }
  94. var err error
  95. ww := make([]byte, len(w))
  96. copy(ww, w)
  97. m := socket.ControlMessage(ww)
  98. for _, c := range tt.cs {
  99. if err = m.MarshalHeader(c.Level, c.Type, len(c.Data)); err != nil {
  100. t.Fatalf("(%v).MarshalHeader() = %v", tt.cs, err)
  101. }
  102. copy(m.Data(len(c.Data)), c.Data)
  103. m = m.Next(len(c.Data))
  104. }
  105. m = socket.ControlMessage(w)
  106. for _, c := range tt.cs {
  107. m, err = m.Marshal(c.Level, c.Type, c.Data)
  108. if err != nil {
  109. t.Fatalf("(%v).Marshal() = %v", tt.cs, err)
  110. }
  111. }
  112. if !bytes.Equal(ww, w) {
  113. t.Fatalf("got %#v; want %#v", ww, w)
  114. }
  115. ws := [][]byte{w}
  116. if tailPadLen > 0 {
  117. // Test a message with no tail padding.
  118. nopad := w[:len(w)-tailPadLen]
  119. ws = append(ws, [][]byte{nopad}...)
  120. }
  121. for _, w := range ws {
  122. ms, err := socket.ControlMessage(w).Parse()
  123. if err != nil {
  124. t.Fatalf("(%v).Parse() = %v", tt.cs, err)
  125. }
  126. for i, m := range ms {
  127. lvl, typ, dataLen, err := m.ParseHeader()
  128. if err != nil {
  129. t.Fatalf("(%v).ParseHeader() = %v", tt.cs, err)
  130. }
  131. if lvl != tt.cs[i].Level || typ != tt.cs[i].Type || dataLen != len(tt.cs[i].Data) {
  132. 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))
  133. }
  134. }
  135. }
  136. }
  137. }
  138. func TestUDP(t *testing.T) {
  139. switch runtime.GOOS {
  140. case "windows":
  141. t.Skipf("not supported on %s", runtime.GOOS)
  142. }
  143. c, err := nettest.NewLocalPacketListener("udp")
  144. if err != nil {
  145. t.Skipf("not supported on %s/%s: %v", runtime.GOOS, runtime.GOARCH, err)
  146. }
  147. defer c.Close()
  148. cc, err := socket.NewConn(c.(net.Conn))
  149. if err != nil {
  150. t.Fatal(err)
  151. }
  152. t.Run("Message", func(t *testing.T) {
  153. data := []byte("HELLO-R-U-THERE")
  154. wm := socket.Message{
  155. Buffers: bytes.SplitAfter(data, []byte("-")),
  156. Addr: c.LocalAddr(),
  157. }
  158. if err := cc.SendMsg(&wm, 0); err != nil {
  159. t.Fatal(err)
  160. }
  161. b := make([]byte, 32)
  162. rm := socket.Message{
  163. Buffers: [][]byte{b[:1], b[1:3], b[3:7], b[7:11], b[11:]},
  164. }
  165. if err := cc.RecvMsg(&rm, 0); err != nil {
  166. t.Fatal(err)
  167. }
  168. if !bytes.Equal(b[:rm.N], data) {
  169. t.Fatalf("got %#v; want %#v", b[:rm.N], data)
  170. }
  171. })
  172. switch runtime.GOOS {
  173. case "android", "linux":
  174. t.Run("Messages", func(t *testing.T) {
  175. data := []byte("HELLO-R-U-THERE")
  176. wmbs := bytes.SplitAfter(data, []byte("-"))
  177. wms := []socket.Message{
  178. {Buffers: wmbs[:1], Addr: c.LocalAddr()},
  179. {Buffers: wmbs[1:], Addr: c.LocalAddr()},
  180. }
  181. n, err := cc.SendMsgs(wms, 0)
  182. if err != nil {
  183. t.Fatal(err)
  184. }
  185. if n != len(wms) {
  186. t.Fatalf("got %d; want %d", n, len(wms))
  187. }
  188. b := make([]byte, 32)
  189. rmbs := [][][]byte{{b[:len(wmbs[0])]}, {b[len(wmbs[0]):]}}
  190. rms := []socket.Message{
  191. {Buffers: rmbs[0]},
  192. {Buffers: rmbs[1]},
  193. }
  194. n, err = cc.RecvMsgs(rms, 0)
  195. if err != nil {
  196. t.Fatal(err)
  197. }
  198. if n != len(rms) {
  199. t.Fatalf("got %d; want %d", n, len(rms))
  200. }
  201. nn := 0
  202. for i := 0; i < n; i++ {
  203. nn += rms[i].N
  204. }
  205. if !bytes.Equal(b[:nn], data) {
  206. t.Fatalf("got %#v; want %#v", b[:nn], data)
  207. }
  208. })
  209. }
  210. // The behavior of transmission for zero byte paylaod depends
  211. // on each platform implementation. Some may transmit only
  212. // protocol header and options, other may transmit nothing.
  213. // We test only that SendMsg and SendMsgs will not crash with
  214. // empty buffers.
  215. wm := socket.Message{
  216. Buffers: [][]byte{{}},
  217. Addr: c.LocalAddr(),
  218. }
  219. cc.SendMsg(&wm, 0)
  220. wms := []socket.Message{
  221. {Buffers: [][]byte{{}}, Addr: c.LocalAddr()},
  222. }
  223. cc.SendMsgs(wms, 0)
  224. }
  225. func BenchmarkUDP(b *testing.B) {
  226. c, err := nettest.NewLocalPacketListener("udp")
  227. if err != nil {
  228. b.Skipf("not supported on %s/%s: %v", runtime.GOOS, runtime.GOARCH, err)
  229. }
  230. defer c.Close()
  231. cc, err := socket.NewConn(c.(net.Conn))
  232. if err != nil {
  233. b.Fatal(err)
  234. }
  235. data := []byte("HELLO-R-U-THERE")
  236. wm := socket.Message{
  237. Buffers: [][]byte{data},
  238. Addr: c.LocalAddr(),
  239. }
  240. rm := socket.Message{
  241. Buffers: [][]byte{make([]byte, 128)},
  242. OOB: make([]byte, 128),
  243. }
  244. for M := 1; M <= 1<<9; M = M << 1 {
  245. b.Run(fmt.Sprintf("Iter-%d", M), func(b *testing.B) {
  246. for i := 0; i < b.N; i++ {
  247. for j := 0; j < M; j++ {
  248. if err := cc.SendMsg(&wm, 0); err != nil {
  249. b.Fatal(err)
  250. }
  251. if err := cc.RecvMsg(&rm, 0); err != nil {
  252. b.Fatal(err)
  253. }
  254. }
  255. }
  256. })
  257. switch runtime.GOOS {
  258. case "android", "linux":
  259. wms := make([]socket.Message, M)
  260. for i := range wms {
  261. wms[i].Buffers = [][]byte{data}
  262. wms[i].Addr = c.LocalAddr()
  263. }
  264. rms := make([]socket.Message, M)
  265. for i := range rms {
  266. rms[i].Buffers = [][]byte{make([]byte, 128)}
  267. rms[i].OOB = make([]byte, 128)
  268. }
  269. b.Run(fmt.Sprintf("Batch-%d", M), func(b *testing.B) {
  270. for i := 0; i < b.N; i++ {
  271. if _, err := cc.SendMsgs(wms, 0); err != nil {
  272. b.Fatal(err)
  273. }
  274. if _, err := cc.RecvMsgs(rms, 0); err != nil {
  275. b.Fatal(err)
  276. }
  277. }
  278. })
  279. }
  280. }
  281. }