multicast.go 2.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108
  1. // Copyright (C) 2014 The Syncthing Authors.
  2. //
  3. // This Source Code Form is subject to the terms of the Mozilla Public
  4. // License, v. 2.0. If a copy of the MPL was not distributed with this file,
  5. // You can obtain one at http://mozilla.org/MPL/2.0/.
  6. package beacon
  7. import (
  8. "errors"
  9. "net"
  10. "golang.org/x/net/ipv6"
  11. )
  12. type Multicast struct {
  13. conn *ipv6.PacketConn
  14. addr *net.UDPAddr
  15. inbox chan []byte
  16. outbox chan recv
  17. intfs []net.Interface
  18. }
  19. func NewMulticast(addr string) (*Multicast, error) {
  20. gaddr, err := net.ResolveUDPAddr("udp6", addr)
  21. if err != nil {
  22. return nil, err
  23. }
  24. conn, err := net.ListenPacket("udp6", addr)
  25. if err != nil {
  26. return nil, err
  27. }
  28. intfs, err := net.Interfaces()
  29. if err != nil {
  30. return nil, err
  31. }
  32. p := ipv6.NewPacketConn(conn)
  33. joined := 0
  34. for _, intf := range intfs {
  35. err := p.JoinGroup(&intf, &net.UDPAddr{IP: gaddr.IP})
  36. if debug {
  37. if err != nil {
  38. l.Debugln("IPv6 join", intf.Name, "failed:", err)
  39. } else {
  40. l.Debugln("IPv6 join", intf.Name, "success")
  41. }
  42. }
  43. joined++
  44. }
  45. if joined == 0 {
  46. return nil, errors.New("no multicast interfaces available")
  47. }
  48. b := &Multicast{
  49. conn: p,
  50. addr: gaddr,
  51. inbox: make(chan []byte),
  52. outbox: make(chan recv, 16),
  53. intfs: intfs,
  54. }
  55. go genericReader(ipv6ReaderAdapter{b.conn}, b.outbox)
  56. go b.writer()
  57. return b, nil
  58. }
  59. func (b *Multicast) Send(data []byte) {
  60. b.inbox <- data
  61. }
  62. func (b *Multicast) Recv() ([]byte, net.Addr) {
  63. recv := <-b.outbox
  64. return recv.data, recv.src
  65. }
  66. func (b *Multicast) writer() {
  67. wcm := &ipv6.ControlMessage{
  68. HopLimit: 1,
  69. }
  70. for bs := range b.inbox {
  71. for _, intf := range b.intfs {
  72. wcm.IfIndex = intf.Index
  73. _, err := b.conn.WriteTo(bs, wcm, b.addr)
  74. if err != nil && debug {
  75. l.Debugln(err, "on write to", b.addr)
  76. } else if debug {
  77. l.Debugf("sent %d bytes to %v on %s", len(bs), b.addr, intf.Name)
  78. }
  79. }
  80. }
  81. }
  82. // This makes ReadFrom on an *ipv6.PacketConn behave like ReadFrom on a
  83. // net.PacketConn.
  84. type ipv6ReaderAdapter struct {
  85. c *ipv6.PacketConn
  86. }
  87. func (i ipv6ReaderAdapter) ReadFrom(bs []byte) (int, net.Addr, error) {
  88. n, _, src, err := i.c.ReadFrom(bs)
  89. return n, src, err
  90. }