readwrite_test.go 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  1. // Copyright 2012 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. package ipv4_test
  5. import (
  6. "bytes"
  7. "net"
  8. "runtime"
  9. "sync"
  10. "testing"
  11. "golang.org/x/net/internal/nettest"
  12. "golang.org/x/net/ipv4"
  13. )
  14. func benchmarkUDPListener() (net.PacketConn, net.Addr, error) {
  15. c, err := net.ListenPacket("udp4", "127.0.0.1:0")
  16. if err != nil {
  17. return nil, nil, err
  18. }
  19. dst, err := net.ResolveUDPAddr("udp4", c.LocalAddr().String())
  20. if err != nil {
  21. c.Close()
  22. return nil, nil, err
  23. }
  24. return c, dst, nil
  25. }
  26. func BenchmarkReadWriteNetUDP(b *testing.B) {
  27. c, dst, err := benchmarkUDPListener()
  28. if err != nil {
  29. b.Fatal(err)
  30. }
  31. defer c.Close()
  32. wb, rb := []byte("HELLO-R-U-THERE"), make([]byte, 128)
  33. b.ResetTimer()
  34. for i := 0; i < b.N; i++ {
  35. benchmarkReadWriteNetUDP(b, c, wb, rb, dst)
  36. }
  37. }
  38. func benchmarkReadWriteNetUDP(b *testing.B, c net.PacketConn, wb, rb []byte, dst net.Addr) {
  39. if _, err := c.WriteTo(wb, dst); err != nil {
  40. b.Fatal(err)
  41. }
  42. if _, _, err := c.ReadFrom(rb); err != nil {
  43. b.Fatal(err)
  44. }
  45. }
  46. func BenchmarkReadWriteIPv4UDP(b *testing.B) {
  47. c, dst, err := benchmarkUDPListener()
  48. if err != nil {
  49. b.Fatal(err)
  50. }
  51. defer c.Close()
  52. p := ipv4.NewPacketConn(c)
  53. defer p.Close()
  54. cf := ipv4.FlagTTL | ipv4.FlagInterface
  55. if err := p.SetControlMessage(cf, true); err != nil {
  56. b.Fatal(err)
  57. }
  58. ifi := nettest.RoutedInterface("ip4", net.FlagUp|net.FlagLoopback)
  59. wb, rb := []byte("HELLO-R-U-THERE"), make([]byte, 128)
  60. b.ResetTimer()
  61. for i := 0; i < b.N; i++ {
  62. benchmarkReadWriteIPv4UDP(b, p, wb, rb, dst, ifi)
  63. }
  64. }
  65. func benchmarkReadWriteIPv4UDP(b *testing.B, p *ipv4.PacketConn, wb, rb []byte, dst net.Addr, ifi *net.Interface) {
  66. cm := ipv4.ControlMessage{TTL: 1}
  67. if ifi != nil {
  68. cm.IfIndex = ifi.Index
  69. }
  70. if n, err := p.WriteTo(wb, &cm, dst); err != nil {
  71. b.Fatal(err)
  72. } else if n != len(wb) {
  73. b.Fatalf("got %v; want %v", n, len(wb))
  74. }
  75. if _, _, _, err := p.ReadFrom(rb); err != nil {
  76. b.Fatal(err)
  77. }
  78. }
  79. func TestPacketConnConcurrentReadWriteUnicastUDP(t *testing.T) {
  80. switch runtime.GOOS {
  81. case "nacl", "plan9", "solaris", "windows":
  82. t.Skipf("not supported on %s", runtime.GOOS)
  83. }
  84. c, err := net.ListenPacket("udp4", "127.0.0.1:0")
  85. if err != nil {
  86. t.Fatal(err)
  87. }
  88. defer c.Close()
  89. p := ipv4.NewPacketConn(c)
  90. defer p.Close()
  91. dst, err := net.ResolveUDPAddr("udp4", c.LocalAddr().String())
  92. if err != nil {
  93. t.Fatal(err)
  94. }
  95. ifi := nettest.RoutedInterface("ip4", net.FlagUp|net.FlagLoopback)
  96. cf := ipv4.FlagTTL | ipv4.FlagSrc | ipv4.FlagDst | ipv4.FlagInterface
  97. wb := []byte("HELLO-R-U-THERE")
  98. if err := p.SetControlMessage(cf, true); err != nil { // probe before test
  99. if nettest.ProtocolNotSupported(err) {
  100. t.Skipf("not supported on %s", runtime.GOOS)
  101. }
  102. t.Fatal(err)
  103. }
  104. var wg sync.WaitGroup
  105. reader := func() {
  106. defer wg.Done()
  107. rb := make([]byte, 128)
  108. if n, cm, _, err := p.ReadFrom(rb); err != nil {
  109. t.Error(err)
  110. return
  111. } else if !bytes.Equal(rb[:n], wb) {
  112. t.Errorf("got %v; want %v", rb[:n], wb)
  113. return
  114. } else {
  115. t.Logf("rcvd cmsg: %v", cm)
  116. }
  117. }
  118. writer := func(toggle bool) {
  119. defer wg.Done()
  120. cm := ipv4.ControlMessage{
  121. Src: net.IPv4(127, 0, 0, 1),
  122. }
  123. if ifi != nil {
  124. cm.IfIndex = ifi.Index
  125. }
  126. if err := p.SetControlMessage(cf, toggle); err != nil {
  127. t.Error(err)
  128. return
  129. }
  130. if n, err := p.WriteTo(wb, &cm, dst); err != nil {
  131. t.Error(err)
  132. return
  133. } else if n != len(wb) {
  134. t.Errorf("short write: %v", n)
  135. return
  136. }
  137. }
  138. const N = 10
  139. wg.Add(N)
  140. for i := 0; i < N; i++ {
  141. go reader()
  142. }
  143. wg.Add(2 * N)
  144. for i := 0; i < 2*N; i++ {
  145. go writer(i%2 != 0)
  146. }
  147. wg.Add(N)
  148. for i := 0; i < N; i++ {
  149. go reader()
  150. }
  151. wg.Wait()
  152. }