blob: 957975c265549211e3a6c11bbd7b19bc730f79c5 [file] [log] [blame]
// Copyright 2016 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 (
"errors"
"internal/itoa"
"os"
)
// If the ifindex is zero, interfaceTable returns mappings of all
// network interfaces. Otherwise it returns a mapping of a specific
// interface.
func interfaceTable(ifindex int) ([]Interface, error) {
if ifindex == 0 {
n, err := interfaceCount()
if err != nil {
return nil, err
}
ifcs := make([]Interface, n)
for i := range ifcs {
ifc, err := readInterface(i)
if err != nil {
return nil, err
}
ifcs[i] = *ifc
}
return ifcs, nil
}
ifc, err := readInterface(ifindex - 1)
if err != nil {
return nil, err
}
return []Interface{*ifc}, nil
}
func readInterface(i int) (*Interface, error) {
ifc := &Interface{
Index: i + 1, // Offset the index by one to suit the contract
Name: netdir + "/ipifc/" + itoa.Itoa(i), // Name is the full path to the interface path in plan9
}
ifcstat := ifc.Name + "/status"
ifcstatf, err := open(ifcstat)
if err != nil {
return nil, err
}
defer ifcstatf.close()
line, ok := ifcstatf.readLine()
if !ok {
return nil, errors.New("invalid interface status file: " + ifcstat)
}
fields := getFields(line)
if len(fields) < 4 {
return nil, errors.New("invalid interface status file: " + ifcstat)
}
device := fields[1]
mtustr := fields[3]
mtu, _, ok := dtoi(mtustr)
if !ok {
return nil, errors.New("invalid status file of interface: " + ifcstat)
}
ifc.MTU = mtu
// Not a loopback device ("/dev/null") or packet interface (e.g. "pkt2")
if stringsHasPrefix(device, netdir+"/") {
deviceaddrf, err := open(device + "/addr")
if err != nil {
return nil, err
}
defer deviceaddrf.close()
line, ok = deviceaddrf.readLine()
if !ok {
return nil, errors.New("invalid address file for interface: " + device + "/addr")
}
if len(line) > 0 && len(line)%2 == 0 {
ifc.HardwareAddr = make([]byte, len(line)/2)
var ok bool
for i := range ifc.HardwareAddr {
j := (i + 1) * 2
ifc.HardwareAddr[i], ok = xtoi2(line[i*2:j], 0)
if !ok {
ifc.HardwareAddr = ifc.HardwareAddr[:i]
break
}
}
}
ifc.Flags = FlagUp | FlagBroadcast | FlagMulticast
} else {
ifc.Flags = FlagUp | FlagMulticast | FlagLoopback
}
return ifc, nil
}
func interfaceCount() (int, error) {
d, err := os.Open(netdir + "/ipifc")
if err != nil {
return -1, err
}
defer d.Close()
names, err := d.Readdirnames(0)
if err != nil {
return -1, err
}
// Assumes that numbered files in ipifc are strictly
// the incrementing numbered directories for the
// interfaces
c := 0
for _, name := range names {
if _, _, ok := dtoi(name); !ok {
continue
}
c++
}
return c, nil
}
// If the ifi is nil, interfaceAddrTable returns addresses for all
// network interfaces. Otherwise it returns addresses for a specific
// interface.
func interfaceAddrTable(ifi *Interface) ([]Addr, error) {
var ifcs []Interface
if ifi == nil {
var err error
ifcs, err = interfaceTable(0)
if err != nil {
return nil, err
}
} else {
ifcs = []Interface{*ifi}
}
var addrs []Addr
for _, ifc := range ifcs {
status := ifc.Name + "/status"
statusf, err := open(status)
if err != nil {
return nil, err
}
defer statusf.close()
// Read but ignore first line as it only contains the table header.
// See https://9p.io/magic/man2html/3/ip
if _, ok := statusf.readLine(); !ok {
return nil, errors.New("cannot read header line for interface: " + status)
}
for line, ok := statusf.readLine(); ok; line, ok = statusf.readLine() {
fields := getFields(line)
if len(fields) < 1 {
return nil, errors.New("cannot parse IP address for interface: " + status)
}
addr := fields[0]
ip := ParseIP(addr)
if ip == nil {
return nil, errors.New("cannot parse IP address for interface: " + status)
}
// The mask is represented as CIDR relative to the IPv6 address.
// Plan 9 internal representation is always IPv6.
maskfld := fields[1]
maskfld = maskfld[1:]
pfxlen, _, ok := dtoi(maskfld)
if !ok {
return nil, errors.New("cannot parse network mask for interface: " + status)
}
var mask IPMask
if ip.To4() != nil { // IPv4 or IPv6 IPv4-mapped address
mask = CIDRMask(pfxlen-8*len(v4InV6Prefix), 8*IPv4len)
}
if ip.To16() != nil && ip.To4() == nil { // IPv6 address
mask = CIDRMask(pfxlen, 8*IPv6len)
}
addrs = append(addrs, &IPNet{IP: ip, Mask: mask})
}
}
return addrs, nil
}
// interfaceMulticastAddrTable returns addresses for a specific
// interface.
func interfaceMulticastAddrTable(ifi *Interface) ([]Addr, error) {
return nil, nil
}