mirror of
https://github.com/golang/go
synced 2024-11-19 22:54:39 -07:00
69275eef5e
This changes fixes two issues with regard to handling routing messages as follows: - Misparsing on platforms (such as FreeBSD) supporting multiple architectures in the same kernel (kern.supported_archs="amd64 i386") - Misparsing with unimplemented messages such as route, interface address state notifications To fix those issues, this change implements all the required socket address parsers, adds a processor architecture identifying function to FreeBSD and tests. Fixes #9707. Fixes #8203. Change-Id: I7ed7b4a0b6f10f54b29edc681a2f35603f2d8d45 Reviewed-on: https://go-review.googlesource.com/4330 Reviewed-by: Ian Lance Taylor <iant@golang.org>
63 lines
1.7 KiB
Go
63 lines
1.7 KiB
Go
// 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"
|
|
"syscall"
|
|
)
|
|
|
|
// interfaceMulticastAddrTable returns addresses for a specific
|
|
// interface.
|
|
func interfaceMulticastAddrTable(ifi *Interface) ([]Addr, error) {
|
|
tab, err := syscall.RouteRIB(syscall.NET_RT_IFLIST2, ifi.Index)
|
|
if err != nil {
|
|
return nil, os.NewSyscallError("route rib", err)
|
|
}
|
|
msgs, err := syscall.ParseRoutingMessage(tab)
|
|
if err != nil {
|
|
return nil, os.NewSyscallError("route message", err)
|
|
}
|
|
var ifmat []Addr
|
|
for _, m := range msgs {
|
|
switch m := m.(type) {
|
|
case *syscall.InterfaceMulticastAddrMessage:
|
|
if ifi.Index == int(m.Header.Index) {
|
|
ifma, err := newMulticastAddr(ifi, m)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if ifma != nil {
|
|
ifmat = append(ifmat, ifma)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return ifmat, nil
|
|
}
|
|
|
|
func newMulticastAddr(ifi *Interface, m *syscall.InterfaceMulticastAddrMessage) (*IPAddr, error) {
|
|
sas, err := syscall.ParseRoutingSockaddr(m)
|
|
if err != nil {
|
|
return nil, os.NewSyscallError("route sockaddr", err)
|
|
}
|
|
switch sa := sas[syscall.RTAX_IFA].(type) {
|
|
case *syscall.SockaddrInet4:
|
|
return &IPAddr{IP: IPv4(sa.Addr[0], sa.Addr[1], sa.Addr[2], sa.Addr[3])}, nil
|
|
case *syscall.SockaddrInet6:
|
|
ifma := IPAddr{IP: make(IP, IPv6len)}
|
|
copy(ifma.IP, sa.Addr[:])
|
|
// NOTE: KAME based IPv6 protcol stack usually embeds
|
|
// the interface index in the interface-local or
|
|
// link-local address as the kernel-internal form.
|
|
if ifma.IP.IsInterfaceLocalMulticast() || ifma.IP.IsLinkLocalMulticast() {
|
|
ifma.IP[2], ifma.IP[3] = 0, 0
|
|
}
|
|
return &ifma, nil
|
|
default:
|
|
return nil, nil
|
|
}
|
|
}
|