net: add bare Interface implementation
Этот коммит содержится в:
		
							родитель
							
								
									a360c82b40
								
							
						
					
					
						коммит
						d87ff838eb
					
				
					 2 изменённых файлов: 305 добавлений и 0 удалений
				
			
		
							
								
								
									
										252
									
								
								src/net/interface.go
									
										
									
									
									
										Обычный файл
									
								
							
							
						
						
									
										252
									
								
								src/net/interface.go
									
										
									
									
									
										Обычный файл
									
								
							| 
						 | 
				
			
			@ -0,0 +1,252 @@
 | 
			
		|||
// The following is copied from Go 1.16 official implementation.
 | 
			
		||||
 | 
			
		||||
// 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 (
 | 
			
		||||
	"errors"
 | 
			
		||||
	"sync"
 | 
			
		||||
	"time"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
var (
 | 
			
		||||
	errInvalidInterface         = errors.New("invalid network interface")
 | 
			
		||||
	errInvalidInterfaceIndex    = errors.New("invalid network interface index")
 | 
			
		||||
	errInvalidInterfaceName     = errors.New("invalid network interface name")
 | 
			
		||||
	errNoSuchInterface          = errors.New("no such network interface")
 | 
			
		||||
	errNoSuchMulticastInterface = errors.New("no such multicast network interface")
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
// Interface represents a mapping between network interface name
 | 
			
		||||
// and index. It also represents network interface facility
 | 
			
		||||
// information.
 | 
			
		||||
type Interface struct {
 | 
			
		||||
	Index        int          // positive integer that starts at one, zero is never used
 | 
			
		||||
	MTU          int          // maximum transmission unit
 | 
			
		||||
	Name         string       // e.g., "en0", "lo0", "eth0.100"
 | 
			
		||||
	HardwareAddr HardwareAddr // IEEE MAC-48, EUI-48 and EUI-64 form
 | 
			
		||||
	Flags        Flags        // e.g., FlagUp, FlagLoopback, FlagMulticast
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type Flags uint
 | 
			
		||||
 | 
			
		||||
const (
 | 
			
		||||
	FlagUp           Flags = 1 << iota // interface is up
 | 
			
		||||
	FlagBroadcast                      // interface supports broadcast access capability
 | 
			
		||||
	FlagLoopback                       // interface is a loopback interface
 | 
			
		||||
	FlagPointToPoint                   // interface belongs to a point-to-point link
 | 
			
		||||
	FlagMulticast                      // interface supports multicast access capability
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
var flagNames = []string{
 | 
			
		||||
	"up",
 | 
			
		||||
	"broadcast",
 | 
			
		||||
	"loopback",
 | 
			
		||||
	"pointtopoint",
 | 
			
		||||
	"multicast",
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (f Flags) String() string {
 | 
			
		||||
	s := ""
 | 
			
		||||
	for i, name := range flagNames {
 | 
			
		||||
		if f&(1<<uint(i)) != 0 {
 | 
			
		||||
			if s != "" {
 | 
			
		||||
				s += "|"
 | 
			
		||||
			}
 | 
			
		||||
			s += name
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	if s == "" {
 | 
			
		||||
		s = "0"
 | 
			
		||||
	}
 | 
			
		||||
	return s
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Addrs returns a list of unicast interface addresses for a specific
 | 
			
		||||
// interface.
 | 
			
		||||
func (ifi *Interface) Addrs() ([]Addr, error) {
 | 
			
		||||
	if ifi == nil {
 | 
			
		||||
		return nil, &OpError{Op: "route", Net: "ip+net", Source: nil, Addr: nil, Err: errInvalidInterface}
 | 
			
		||||
	}
 | 
			
		||||
	ifat, err := interfaceAddrTable(ifi)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		err = &OpError{Op: "route", Net: "ip+net", Source: nil, Addr: nil, Err: err}
 | 
			
		||||
	}
 | 
			
		||||
	return ifat, err
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// MulticastAddrs returns a list of multicast, joined group addresses
 | 
			
		||||
// for a specific interface.
 | 
			
		||||
func (ifi *Interface) MulticastAddrs() ([]Addr, error) {
 | 
			
		||||
	if ifi == nil {
 | 
			
		||||
		return nil, &OpError{Op: "route", Net: "ip+net", Source: nil, Addr: nil, Err: errInvalidInterface}
 | 
			
		||||
	}
 | 
			
		||||
	ifat, err := interfaceMulticastAddrTable(ifi)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		err = &OpError{Op: "route", Net: "ip+net", Source: nil, Addr: nil, Err: err}
 | 
			
		||||
	}
 | 
			
		||||
	return ifat, err
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Interfaces returns a list of the system's network interfaces.
 | 
			
		||||
func Interfaces() ([]Interface, error) {
 | 
			
		||||
	ift, err := interfaceTable(0)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, &OpError{Op: "route", Net: "ip+net", Source: nil, Addr: nil, Err: err}
 | 
			
		||||
	}
 | 
			
		||||
	if len(ift) != 0 {
 | 
			
		||||
		zoneCache.update(ift, false)
 | 
			
		||||
	}
 | 
			
		||||
	return ift, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// InterfaceAddrs returns a list of the system's unicast interface
 | 
			
		||||
// addresses.
 | 
			
		||||
//
 | 
			
		||||
// The returned list does not identify the associated interface; use
 | 
			
		||||
// Interfaces and Interface.Addrs for more detail.
 | 
			
		||||
func InterfaceAddrs() ([]Addr, error) {
 | 
			
		||||
	ifat, err := interfaceAddrTable(nil)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		err = &OpError{Op: "route", Net: "ip+net", Source: nil, Addr: nil, Err: err}
 | 
			
		||||
	}
 | 
			
		||||
	return ifat, err
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// InterfaceByIndex returns the interface specified by index.
 | 
			
		||||
//
 | 
			
		||||
// On Solaris, it returns one of the logical network interfaces
 | 
			
		||||
// sharing the logical data link; for more precision use
 | 
			
		||||
// InterfaceByName.
 | 
			
		||||
func InterfaceByIndex(index int) (*Interface, error) {
 | 
			
		||||
	if index <= 0 {
 | 
			
		||||
		return nil, &OpError{Op: "route", Net: "ip+net", Source: nil, Addr: nil, Err: errInvalidInterfaceIndex}
 | 
			
		||||
	}
 | 
			
		||||
	ift, err := interfaceTable(index)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, &OpError{Op: "route", Net: "ip+net", Source: nil, Addr: nil, Err: err}
 | 
			
		||||
	}
 | 
			
		||||
	ifi, err := interfaceByIndex(ift, index)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		err = &OpError{Op: "route", Net: "ip+net", Source: nil, Addr: nil, Err: err}
 | 
			
		||||
	}
 | 
			
		||||
	return ifi, err
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func interfaceByIndex(ift []Interface, index int) (*Interface, error) {
 | 
			
		||||
	for _, ifi := range ift {
 | 
			
		||||
		if index == ifi.Index {
 | 
			
		||||
			return &ifi, nil
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	return nil, errNoSuchInterface
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// InterfaceByName returns the interface specified by name.
 | 
			
		||||
func InterfaceByName(name string) (*Interface, error) {
 | 
			
		||||
	if name == "" {
 | 
			
		||||
		return nil, &OpError{Op: "route", Net: "ip+net", Source: nil, Addr: nil, Err: errInvalidInterfaceName}
 | 
			
		||||
	}
 | 
			
		||||
	ift, err := interfaceTable(0)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, &OpError{Op: "route", Net: "ip+net", Source: nil, Addr: nil, Err: err}
 | 
			
		||||
	}
 | 
			
		||||
	if len(ift) != 0 {
 | 
			
		||||
		zoneCache.update(ift, false)
 | 
			
		||||
	}
 | 
			
		||||
	for _, ifi := range ift {
 | 
			
		||||
		if name == ifi.Name {
 | 
			
		||||
			return &ifi, nil
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	return nil, &OpError{Op: "route", Net: "ip+net", Source: nil, Addr: nil, Err: errNoSuchInterface}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// An ipv6ZoneCache represents a cache holding partial network
 | 
			
		||||
// interface information. It is used for reducing the cost of IPv6
 | 
			
		||||
// addressing scope zone resolution.
 | 
			
		||||
//
 | 
			
		||||
// Multiple names sharing the index are managed by first-come
 | 
			
		||||
// first-served basis for consistency.
 | 
			
		||||
type ipv6ZoneCache struct {
 | 
			
		||||
	sync.RWMutex                // guard the following
 | 
			
		||||
	lastFetched  time.Time      // last time routing information was fetched
 | 
			
		||||
	toIndex      map[string]int // interface name to its index
 | 
			
		||||
	toName       map[int]string // interface index to its name
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
var zoneCache = ipv6ZoneCache{
 | 
			
		||||
	toIndex: make(map[string]int),
 | 
			
		||||
	toName:  make(map[int]string),
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// update refreshes the network interface information if the cache was last
 | 
			
		||||
// updated more than 1 minute ago, or if force is set. It reports whether the
 | 
			
		||||
// cache was updated.
 | 
			
		||||
func (zc *ipv6ZoneCache) update(ift []Interface, force bool) (updated bool) {
 | 
			
		||||
	zc.Lock()
 | 
			
		||||
	defer zc.Unlock()
 | 
			
		||||
	now := time.Now()
 | 
			
		||||
	if !force && zc.lastFetched.After(now.Add(-60*time.Second)) {
 | 
			
		||||
		return false
 | 
			
		||||
	}
 | 
			
		||||
	zc.lastFetched = now
 | 
			
		||||
	if len(ift) == 0 {
 | 
			
		||||
		var err error
 | 
			
		||||
		if ift, err = interfaceTable(0); err != nil {
 | 
			
		||||
			return false
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	zc.toIndex = make(map[string]int, len(ift))
 | 
			
		||||
	zc.toName = make(map[int]string, len(ift))
 | 
			
		||||
	for _, ifi := range ift {
 | 
			
		||||
		zc.toIndex[ifi.Name] = ifi.Index
 | 
			
		||||
		if _, ok := zc.toName[ifi.Index]; !ok {
 | 
			
		||||
			zc.toName[ifi.Index] = ifi.Name
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	return true
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (zc *ipv6ZoneCache) name(index int) string {
 | 
			
		||||
	if index == 0 {
 | 
			
		||||
		return ""
 | 
			
		||||
	}
 | 
			
		||||
	updated := zoneCache.update(nil, false)
 | 
			
		||||
	zoneCache.RLock()
 | 
			
		||||
	name, ok := zoneCache.toName[index]
 | 
			
		||||
	zoneCache.RUnlock()
 | 
			
		||||
	if !ok && !updated {
 | 
			
		||||
		zoneCache.update(nil, true)
 | 
			
		||||
		zoneCache.RLock()
 | 
			
		||||
		name, ok = zoneCache.toName[index]
 | 
			
		||||
		zoneCache.RUnlock()
 | 
			
		||||
	}
 | 
			
		||||
	if !ok { // last resort
 | 
			
		||||
		name = uitoa(uint(index))
 | 
			
		||||
	}
 | 
			
		||||
	return name
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (zc *ipv6ZoneCache) index(name string) int {
 | 
			
		||||
	if name == "" {
 | 
			
		||||
		return 0
 | 
			
		||||
	}
 | 
			
		||||
	updated := zoneCache.update(nil, false)
 | 
			
		||||
	zoneCache.RLock()
 | 
			
		||||
	index, ok := zoneCache.toIndex[name]
 | 
			
		||||
	zoneCache.RUnlock()
 | 
			
		||||
	if !ok && !updated {
 | 
			
		||||
		zoneCache.update(nil, true)
 | 
			
		||||
		zoneCache.RLock()
 | 
			
		||||
		index, ok = zoneCache.toIndex[name]
 | 
			
		||||
		zoneCache.RUnlock()
 | 
			
		||||
	}
 | 
			
		||||
	if !ok { // last resort
 | 
			
		||||
		index, _, _ = dtoi(name)
 | 
			
		||||
	}
 | 
			
		||||
	return index
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										53
									
								
								src/net/interface_tinygo.go
									
										
									
									
									
										Обычный файл
									
								
							
							
						
						
									
										53
									
								
								src/net/interface_tinygo.go
									
										
									
									
									
										Обычный файл
									
								
							| 
						 | 
				
			
			@ -0,0 +1,53 @@
 | 
			
		|||
// +build tinygo
 | 
			
		||||
 | 
			
		||||
package net
 | 
			
		||||
 | 
			
		||||
const (
 | 
			
		||||
	tinyGoInterfaceName = "tinygo-undefined"
 | 
			
		||||
	maxTransmissionUnit = 1500 // Ethernet?
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
// DE:AD:BE:EF:FE:FF
 | 
			
		||||
var defaultMAC = HardwareAddr{0xDE, 0xAD, 0xBE, 0xEF, 0xFE, 0xFF}
 | 
			
		||||
 | 
			
		||||
// 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) {
 | 
			
		||||
	i, err := readInterface(0)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, err
 | 
			
		||||
	}
 | 
			
		||||
	return []Interface{*i}, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func readInterface(i int) (*Interface, error) {
 | 
			
		||||
	if i != 0 {
 | 
			
		||||
		return nil, errInvalidInterfaceIndex
 | 
			
		||||
	}
 | 
			
		||||
	ifc := &Interface{
 | 
			
		||||
		Index:        i + 1, // Offset the index by one to suit the contract
 | 
			
		||||
		Name:         tinyGoInterfaceName,
 | 
			
		||||
		MTU:          maxTransmissionUnit,
 | 
			
		||||
		HardwareAddr: defaultMAC,
 | 
			
		||||
		Flags:        0, // No flags since interface is not implemented.
 | 
			
		||||
	}
 | 
			
		||||
	return ifc, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func interfaceCount() (int, error) {
 | 
			
		||||
	return 1, 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) {
 | 
			
		||||
	return nil, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// interfaceMulticastAddrTable returns addresses for a specific
 | 
			
		||||
// interface.
 | 
			
		||||
func interfaceMulticastAddrTable(ifi *Interface) ([]Addr, error) {
 | 
			
		||||
	return nil, nil
 | 
			
		||||
}
 | 
			
		||||
		Загрузка…
	
	Создание таблицы
		
		Сослаться в новой задаче