// Copyright 2011 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 net import ( "os" "runtime" "testing" ) var listenMulticastUDPTests = []struct { net string gaddr *UDPAddr flags Flags ipv6 bool }{ // cf. RFC 4727: Experimental Values in IPv4, IPv6, ICMPv4, ICMPv6, UDP, and TCP Headers {"udp", &UDPAddr{IPv4(224, 0, 0, 254), 12345}, FlagUp | FlagLoopback, false}, {"udp4", &UDPAddr{IPv4(224, 0, 0, 254), 12345}, FlagUp | FlagLoopback, false}, {"udp", &UDPAddr{ParseIP("ff0e::114"), 12345}, FlagUp | FlagLoopback, true}, {"udp6", &UDPAddr{ParseIP("ff01::114"), 12345}, FlagUp | FlagLoopback, true}, {"udp6", &UDPAddr{ParseIP("ff02::114"), 12345}, FlagUp | FlagLoopback, true}, {"udp6", &UDPAddr{ParseIP("ff04::114"), 12345}, FlagUp | FlagLoopback, true}, {"udp6", &UDPAddr{ParseIP("ff05::114"), 12345}, FlagUp | FlagLoopback, true}, {"udp6", &UDPAddr{ParseIP("ff08::114"), 12345}, FlagUp | FlagLoopback, true}, {"udp6", &UDPAddr{ParseIP("ff0e::114"), 12345}, FlagUp | FlagLoopback, true}, } func TestListenMulticastUDP(t *testing.T) { switch runtime.GOOS { case "netbsd", "openbsd", "plan9", "windows": return case "linux": if runtime.GOARCH == "arm" || runtime.GOARCH == "alpha" { return } } for _, tt := range listenMulticastUDPTests { if tt.ipv6 && (!supportsIPv6 || os.Getuid() != 0) { continue } ift, err := Interfaces() if err != nil { t.Fatalf("Interfaces failed: %v", err) } var ifi *Interface for _, x := range ift { if x.Flags&tt.flags == tt.flags { ifi = &x break } } if ifi == nil { t.Logf("an appropriate multicast interface not found") return } c, err := ListenMulticastUDP(tt.net, ifi, tt.gaddr) if err != nil { t.Fatalf("ListenMulticastUDP failed: %v", err) } defer c.Close() // test to listen concurrently across multiple listeners if !tt.ipv6 { testIPv4MulticastSocketOptions(t, c.fd, ifi) } else { testIPv6MulticastSocketOptions(t, c.fd, ifi) } ifmat, err := ifi.MulticastAddrs() if err != nil { t.Fatalf("MulticastAddrs failed: %v", err) } var found bool for _, ifma := range ifmat { if ifma.(*IPAddr).IP.Equal(tt.gaddr.IP) { found = true break } } if !found { t.Fatalf("%q not found in RIB", tt.gaddr.String()) } } } func TestSimpleListenMulticastUDP(t *testing.T) { switch runtime.GOOS { case "plan9": return } for _, tt := range listenMulticastUDPTests { if tt.ipv6 { continue } tt.flags = FlagUp | FlagMulticast ift, err := Interfaces() if err != nil { t.Fatalf("Interfaces failed: %v", err) } var ifi *Interface for _, x := range ift { if x.Flags&tt.flags == tt.flags { ifi = &x break } } if ifi == nil { t.Logf("an appropriate multicast interface not found") return } c, err := ListenMulticastUDP(tt.net, ifi, tt.gaddr) if err != nil { t.Fatalf("ListenMulticastUDP failed: %v", err) } c.Close() } } func testIPv4MulticastSocketOptions(t *testing.T, fd *netFD, ifi *Interface) { ifmc, err := ipv4MulticastInterface(fd) if err != nil { t.Fatalf("ipv4MulticastInterface failed: %v", err) } t.Logf("IPv4 multicast interface: %v", ifmc) err = setIPv4MulticastInterface(fd, ifi) if err != nil { t.Fatalf("setIPv4MulticastInterface failed: %v", err) } ttl, err := ipv4MulticastTTL(fd) if err != nil { t.Fatalf("ipv4MulticastTTL failed: %v", err) } t.Logf("IPv4 multicast TTL: %v", ttl) err = setIPv4MulticastTTL(fd, 1) if err != nil { t.Fatalf("setIPv4MulticastTTL failed: %v", err) } loop, err := ipv4MulticastLoopback(fd) if err != nil { t.Fatalf("ipv4MulticastLoopback failed: %v", err) } t.Logf("IPv4 multicast loopback: %v", loop) err = setIPv4MulticastLoopback(fd, false) if err != nil { t.Fatalf("setIPv4MulticastLoopback failed: %v", err) } } func testIPv6MulticastSocketOptions(t *testing.T, fd *netFD, ifi *Interface) { ifmc, err := ipv6MulticastInterface(fd) if err != nil { t.Fatalf("ipv6MulticastInterface failed: %v", err) } t.Logf("IPv6 multicast interface: %v", ifmc) err = setIPv6MulticastInterface(fd, ifi) if err != nil { t.Fatalf("setIPv6MulticastInterface failed: %v", err) } hoplim, err := ipv6MulticastHopLimit(fd) if err != nil { t.Fatalf("ipv6MulticastHopLimit failed: %v", err) } t.Logf("IPv6 multicast hop limit: %v", hoplim) err = setIPv6MulticastHopLimit(fd, 1) if err != nil { t.Fatalf("setIPv6MulticastHopLimit failed: %v", err) } loop, err := ipv6MulticastLoopback(fd) if err != nil { t.Fatalf("ipv6MulticastLoopback failed: %v", err) } t.Logf("IPv6 multicast loopback: %v", loop) err = setIPv6MulticastLoopback(fd, false) if err != nil { t.Fatalf("setIPv6MulticastLoopback failed: %v", err) } }