multicastsockopt_test.go 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113
  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. "net"
  7. "os"
  8. "runtime"
  9. "testing"
  10. "code.google.com/p/go.net/internal/nettest"
  11. "code.google.com/p/go.net/ipv4"
  12. )
  13. var packetConnMulticastSocketOptionTests = []struct {
  14. net, proto, addr string
  15. gaddr net.Addr
  16. }{
  17. {"udp4", "", "224.0.0.0:0", &net.UDPAddr{IP: net.IPv4(224, 0, 0, 249)}}, // see RFC 4727
  18. {"ip4", ":icmp", "0.0.0.0", &net.IPAddr{IP: net.IPv4(224, 0, 0, 250)}}, // see RFC 4727
  19. }
  20. func TestPacketConnMulticastSocketOptions(t *testing.T) {
  21. switch runtime.GOOS {
  22. case "plan9":
  23. t.Skipf("not supported on %q", runtime.GOOS)
  24. }
  25. ifi := nettest.RoutedInterface("ip4", net.FlagUp|net.FlagMulticast|net.FlagLoopback)
  26. if ifi == nil {
  27. t.Skipf("not available on %q", runtime.GOOS)
  28. }
  29. for _, tt := range packetConnMulticastSocketOptionTests {
  30. if tt.net == "ip4" && os.Getuid() != 0 {
  31. t.Skip("must be root")
  32. }
  33. c, err := net.ListenPacket(tt.net+tt.proto, tt.addr)
  34. if err != nil {
  35. t.Fatalf("net.ListenPacket failed: %v", err)
  36. }
  37. defer c.Close()
  38. testMulticastSocketOptions(t, ipv4.NewPacketConn(c), ifi, tt.gaddr)
  39. }
  40. }
  41. func TestRawConnMulticastSocketOptions(t *testing.T) {
  42. switch runtime.GOOS {
  43. case "plan9":
  44. t.Skipf("not supported on %q", runtime.GOOS)
  45. }
  46. if os.Getuid() != 0 {
  47. t.Skip("must be root")
  48. }
  49. ifi := nettest.RoutedInterface("ip4", net.FlagUp|net.FlagMulticast|net.FlagLoopback)
  50. if ifi == nil {
  51. t.Skipf("not available on %q", runtime.GOOS)
  52. }
  53. c, err := net.ListenPacket("ip4:icmp", "0.0.0.0")
  54. if err != nil {
  55. t.Fatalf("net.ListenPacket failed: %v", err)
  56. }
  57. defer c.Close()
  58. r, err := ipv4.NewRawConn(c)
  59. if err != nil {
  60. t.Fatalf("ipv4.NewRawConn failed: %v", err)
  61. }
  62. testMulticastSocketOptions(t, r, ifi, &net.IPAddr{IP: net.IPv4(224, 0, 0, 250)}) /// see RFC 4727
  63. }
  64. type testIPv4MulticastConn interface {
  65. MulticastTTL() (int, error)
  66. SetMulticastTTL(ttl int) error
  67. MulticastLoopback() (bool, error)
  68. SetMulticastLoopback(bool) error
  69. JoinGroup(*net.Interface, net.Addr) error
  70. LeaveGroup(*net.Interface, net.Addr) error
  71. }
  72. func testMulticastSocketOptions(t *testing.T, c testIPv4MulticastConn, ifi *net.Interface, gaddr net.Addr) {
  73. const ttl = 255
  74. if err := c.SetMulticastTTL(ttl); err != nil {
  75. t.Fatalf("ipv4.PacketConn.SetMulticastTTL failed: %v", err)
  76. }
  77. if v, err := c.MulticastTTL(); err != nil {
  78. t.Fatalf("ipv4.PacketConn.MulticastTTL failed: %v", err)
  79. } else if v != ttl {
  80. t.Fatalf("got unexpected multicast TTL value %v; expected %v", v, ttl)
  81. }
  82. for _, toggle := range []bool{true, false} {
  83. if err := c.SetMulticastLoopback(toggle); err != nil {
  84. t.Fatalf("ipv4.PacketConn.SetMulticastLoopback failed: %v", err)
  85. }
  86. if v, err := c.MulticastLoopback(); err != nil {
  87. t.Fatalf("ipv4.PacketConn.MulticastLoopback failed: %v", err)
  88. } else if v != toggle {
  89. t.Fatalf("got unexpected multicast loopback %v; expected %v", v, toggle)
  90. }
  91. }
  92. if err := c.JoinGroup(ifi, gaddr); err != nil {
  93. t.Fatalf("ipv4.PacketConn.JoinGroup(%v, %v) failed: %v", ifi, gaddr, err)
  94. }
  95. if err := c.LeaveGroup(ifi, gaddr); err != nil {
  96. t.Fatalf("ipv4.PacketConn.LeaveGroup(%v, %v) failed: %v", ifi, gaddr, err)
  97. }
  98. }