|  | // 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 | 
|  | } |