/* SetupNetwork() */ func SetupNetwork(addr string) error { // Bring up loop back interface. lo, e := netlink.LinkByName("lo") if e != nil { return fmt.Errorf("Failed to find loopback interface: %v", e) } if e := netlink.LinkSetUp(lo); e != nil { return fmt.Errorf("Failed to setup loopback interface: %v", e) } if len(addr) > 0 { veth, e := netlink.LinkByName("veth0") if e != nil { return fmt.Errorf("Failed to find veth interface: %v", e) } addr, e := netlink.ParseAddr(addr) if e != nil { return fmt.Errorf("Failed to parse NetworkAddr: %v", e) } netlink.AddrAdd(veth, addr) if e := netlink.LinkSetUp(veth); e != nil { return fmt.Errorf("Network link failed to come up: %v", e) } } return nil }
// SetupVeth sets up a virtual ethernet link. // Should be in container netns. func SetupVeth(contVethName string, mtu int, hostNS *os.File) (hostVeth, contVeth netlink.Link, err error) { var hostVethName string hostVethName, contVeth, err = makeVeth(contVethName, mtu) if err != nil { return } if err = netlink.LinkSetUp(contVeth); err != nil { err = fmt.Errorf("failed to set %q up: %v", contVethName, err) return } hostVeth, err = netlink.LinkByName(hostVethName) if err != nil { err = fmt.Errorf("failed to lookup %q: %v", hostVethName, err) return } if err = netlink.LinkSetUp(hostVeth); err != nil { err = fmt.Errorf("failed to set %q up: %v", contVethName, err) return } if err = netlink.LinkSetNsFd(hostVeth, int(hostNS.Fd())); err != nil { err = fmt.Errorf("failed to move veth to host netns: %v", err) return } return }
// Wire up a tap interface for communicating with the guest. Returns the name // of the created tap interface. func wireTapInterface(config *netConfig) string { // Drop link on eth0 before configuring anything eth0, err := netlink.LinkByName("eth0") if err != nil { log.Fatalf("LinkByName(eth0): %v", err) } if err := netlink.LinkSetDown(eth0); err != nil { log.Fatalf("LinkSetDown(eth0): %v", err) } // Flush any L3 addresses on eth0 if err := flushAddresses(eth0); err != nil { log.Fatalf("flushAddresses(eth0): %v", err) } // Generate and set random MAC address for eth0 eth0Addr := generateHardwareAddr() if err := netlink.LinkSetHardwareAddr(eth0, eth0Addr); err != nil { log.Fatalf("LinkSetHardwareAddr(eth0): %v", err) } // Create "tap0" (interface to guest) tap0Attrs := netlink.NewLinkAttrs() tap0Attrs.Name = "tap0" tap0 := &netlink.Tuntap{tap0Attrs, netlink.TUNTAP_MODE_TAP} if err := netlink.LinkAdd(tap0); err != nil { log.Fatalf("LinkAdd(tap0): %v", err) } // Create a new bridge, br0 and add eth0 and tap0 to it br0Attrs := netlink.NewLinkAttrs() br0Attrs.Name = "br0" br0 := &netlink.Bridge{br0Attrs} if err := netlink.LinkAdd(br0); err != nil { log.Fatalf("LinkAdd(br0): %v", err) } if err := netlink.LinkSetMaster(eth0, br0); err != nil { log.Fatalf("LinkSetMaster(eth0, br0): %v", err) } if err := netlink.LinkSetMaster(tap0, br0); err != nil { log.Fatalf("LinkSetMaster(tap0, br0): %v", err) } // Set all links up if err := netlink.LinkSetUp(tap0); err != nil { log.Fatalf("LinkSetUp(tap0): %v", err) } if err := netlink.LinkSetUp(eth0); err != nil { log.Fatalf("LinkSetUp(eth0): %v", err) } if err := netlink.LinkSetUp(br0); err != nil { log.Fatalf("LinkSetUp(br0): %v", err) } return tap0Attrs.Name }
func routingUp() { veth, err := netlink.LinkByName("myveth1") if err != nil { panic(err) } err = netlink.LinkSetUp(veth) if err != nil { panic(err) } addr, _ := netlink.ParseAddr("172.19.80.2/24") err = netlink.AddrAdd(veth, addr) if err != nil { panic(err) } routes := createRoutes(veth) for _, route := range routes { fmt.Println("Adding route", route) err := netlink.RouteAdd(route) if err != nil { fmt.Println(err) // panic(err) } } }
func endpointCreate() { hostIfName := "myveth0" containerIfName := "myveth1" veth, err := netlink.LinkByName("myveth0") if err == nil && veth != nil { return } veth = &netlink.Veth{ LinkAttrs: netlink.LinkAttrs{Name: hostIfName, TxQLen: 0}, PeerName: containerIfName} if err := netlink.LinkAdd(veth); err != nil { panic(err) } la := netlink.NewLinkAttrs() la.Name = "mycbridge" mybridge := &netlink.Bridge{la} err = netlink.LinkSetMaster(veth, mybridge) if err != nil { panic(err) } err = netlink.LinkSetUp(veth) if err != nil { panic(err) } }
func configureIface(ifname string, ipn ip.IP4Net, mtu int) error { iface, err := netlink.LinkByName(ifname) if err != nil { return fmt.Errorf("failed to lookup interface %v", ifname) } err = netlink.AddrAdd(iface, &netlink.Addr{IPNet: ipn.ToIPNet(), Label: ""}) if err != nil { return fmt.Errorf("failed to add IP address %v to %v: %v", ipn.String(), ifname, err) } err = netlink.LinkSetMTU(iface, mtu) if err != nil { return fmt.Errorf("failed to set MTU for %v: %v", ifname, err) } err = netlink.LinkSetUp(iface) if err != nil { return fmt.Errorf("failed to set interface %v to UP state: %v", ifname, err) } // explicitly add a route since there might be a route for a subnet already // installed by Docker and then it won't get auto added err = netlink.RouteAdd(&netlink.Route{ LinkIndex: iface.Attrs().Index, Scope: netlink.SCOPE_UNIVERSE, Dst: ipn.Network().ToIPNet(), }) if err != nil && err != syscall.EEXIST { return fmt.Errorf("failed to add route (%v -> %v): %v", ipn.Network().String(), ifname, err) } return nil }
func (d *Driver) Join(r *dknet.JoinRequest) (*dknet.JoinResponse, error) { // create and attach local name to the bridge localVethPair := vethPair(truncateID(r.EndpointID)) if err := netlink.LinkAdd(localVethPair); err != nil { log.Errorf("failed to create the veth pair named: [ %v ] error: [ %s ] ", localVethPair, err) return nil, err } // Bring the veth pair up err := netlink.LinkSetUp(localVethPair) if err != nil { log.Warnf("Error enabling Veth local iface: [ %v ]", localVethPair) return nil, err } bridgeName := d.networks[r.NetworkID].BridgeName err = d.addOvsVethPort(bridgeName, localVethPair.Name, 0) if err != nil { log.Errorf("error attaching veth [ %s ] to bridge [ %s ]", localVethPair.Name, bridgeName) return nil, err } log.Infof("Attached veth [ %s ] to bridge [ %s ]", localVethPair.Name, bridgeName) // SrcName gets renamed to DstPrefix + ID on the container iface res := &dknet.JoinResponse{ InterfaceName: dknet.InterfaceName{ SrcName: localVethPair.PeerName, DstPrefix: containerEthName, }, Gateway: d.networks[r.NetworkID].Gateway, } log.Debugf("Join endpoint %s:%s to %s", r.NetworkID, r.EndpointID, r.SandboxKey) return res, nil }
// addUplink adds a dummy uplink to ofnet agent func addUplink(ofa *OfnetAgent, linkName string, ofpPortNo uint32) (*netlink.Veth, error) { link := &netlink.Veth{ LinkAttrs: netlink.LinkAttrs{ Name: linkName, TxQLen: 100, MTU: 1400, }, PeerName: linkName + "peer", } // delete old link if it exists.. and ignore error netlink.LinkDel(link) time.Sleep(100 * time.Millisecond) if err := netlink.LinkAdd(link); err != nil { return nil, err } // add it to ofnet err := ofa.AddUplink(ofpPortNo, linkName) if err != nil { return nil, err } time.Sleep(time.Second) // mark the link as up if err := netlink.LinkSetUp(link); err != nil { return nil, err } return link, nil }
// ConfigureIface takes the result of IPAM plugin and // applies to the ifName interface func configureIface(conf *NetConf) error { ifName := conf.IfName link, err := netlink.LinkByName(ifName) if err != nil { return fmt.Errorf("failed to lookup %q: %v", ifName, err) } if err := netlink.LinkSetUp(link); err != nil { return fmt.Errorf("failed to set %q UP: %v", ifName, err) } i, n, err := net.ParseCIDR(conf.IPAddr) if err != nil { return fmt.Errorf("failed to parse ip address :%s", conf.IPAddr) } addr := &net.IPNet{IP: i, Mask: n.Mask} nlAddr := &netlink.Addr{IPNet: addr, Label: ""} if err = netlink.AddrAdd(link, nlAddr); err != nil { return fmt.Errorf("failed to add IP addr to %q: %v", ifName, err) } gw := net.ParseIP(conf.Gateway) if gw == nil { return fmt.Errorf("parse gateway: %s return nil", conf.Gateway) } if err = ip.AddDefaultRoute(gw, link); err != nil { // we skip over duplicate routes as we assume the first one wins if !os.IsExist(err) { return fmt.Errorf("failed to add default route via %v dev %v: %v", gw, ifName, err) } } return nil }
func createInterface(t *testing.T, name string, nws ...string) { // Add interface link := &netlink.Bridge{ LinkAttrs: netlink.LinkAttrs{ Name: "test", }, } bips := []*net.IPNet{} for _, nw := range nws { bip, err := types.ParseCIDR(nw) if err != nil { t.Fatal(err) } bips = append(bips, bip) } if err := netlink.LinkAdd(link); err != nil { t.Fatalf("Failed to create interface via netlink: %v", err) } for _, bip := range bips { if err := netlink.AddrAdd(link, &netlink.Addr{IPNet: bip}); err != nil { t.Fatal(err) } } if err := netlink.LinkSetUp(link); err != nil { t.Fatal(err) } }
// attach a container network interface to an external network func (v *veth) attach(n *configs.Network) (err error) { brl, err := netlink.LinkByName(n.Bridge) if err != nil { return err } br, ok := brl.(*netlink.Bridge) if !ok { return fmt.Errorf("Wrong device type %T", brl) } host, err := netlink.LinkByName(n.HostInterfaceName) if err != nil { return err } if err := netlink.LinkSetMaster(host, br); err != nil { return err } if err := netlink.LinkSetMTU(host, n.Mtu); err != nil { return err } if n.HairpinMode { if err := netlink.LinkSetHairpin(host, true); err != nil { return err } } if err := netlink.LinkSetUp(host); err != nil { return err } return nil }
func loopbackUp() error { iface, err := netlink.LinkByName("lo") if err != nil { return err } return netlink.LinkSetUp(iface) }
// setLinkUp sets the link up func setLinkUp(name string) error { iface, err := netlink.LinkByName(name) if err != nil { return err } return netlink.LinkSetUp(iface) }
func (l *DHCPLease) acquire() error { c, err := newDHCPClient(l.link) if err != nil { return err } defer c.Close() if (l.link.Attrs().Flags & net.FlagUp) != net.FlagUp { log.Printf("Link %q down. Attempting to set up", l.link.Attrs().Name) if err = netlink.LinkSetUp(l.link); err != nil { return err } } pkt, err := backoffRetry(func() (*dhcp4.Packet, error) { ok, ack, err := c.Request() switch { case err != nil: return nil, err case !ok: return nil, fmt.Errorf("DHCP server NACK'd own offer") default: return &ack, nil } }) if err != nil { return err } return l.commit(pkt) }
func ensureBridge(brName string, mtu int) (*netlink.Bridge, error) { br := &netlink.Bridge{ LinkAttrs: netlink.LinkAttrs{ Name: brName, MTU: mtu, }, } if err := netlink.LinkAdd(br); err != nil { if err != syscall.EEXIST { return nil, fmt.Errorf("could not add %q: %v", brName, err) } // it's ok if the device already exists as long as config is similar br, err = bridgeByName(brName) if err != nil { return nil, err } } if err := netlink.LinkSetUp(br); err != nil { return nil, err } return br, nil }
// ConfigureIface takes the result of IPAM plugin and // applies to the ifName interface func ConfigureIface(ifName string, res *types.Result) error { link, err := netlink.LinkByName(ifName) if err != nil { return fmt.Errorf("failed to lookup %q: %v", ifName, err) } if err := netlink.LinkSetUp(link); err != nil { return fmt.Errorf("failed to set %q UP: %v", ifName, err) } // TODO(eyakubovich): IPv6 addr := &netlink.Addr{IPNet: &res.IP4.IP, Label: ""} if err = netlink.AddrAdd(link, addr); err != nil { return fmt.Errorf("failed to add IP addr to %q: %v", ifName, err) } for _, r := range res.IP4.Routes { gw := r.GW if gw == nil { gw = res.IP4.Gateway } if err = ip.AddRoute(&r.Dst, gw, link); err != nil { // we skip over duplicate routes as we assume the first one wins if !os.IsExist(err) { return fmt.Errorf("failed to add route '%v via %v dev %v': %v", r.Dst, gw, ifName, err) } } } return nil }
func applyNetConf(link netlink.Link, netConf config.InterfaceConfig) error { if netConf.DHCP { log.Infof("Running DHCP on %s", link.Attrs().Name) cmd := exec.Command("dhcpcd", "-A4", "-e", "force_hostname=true", link.Attrs().Name) cmd.Stdout = os.Stdout cmd.Stderr = os.Stderr if err := cmd.Run(); err != nil { log.Error(err) } } else if netConf.IPV4LL { if err := AssignLinkLocalIP(link); err != nil { log.Error("IPV4LL set failed") return err } } else if netConf.Address == "" { return nil } else { addr, err := netlink.ParseAddr(netConf.Address) if err != nil { return err } if err := netlink.AddrAdd(link, addr); err != nil { log.Error("addr add failed") return err } log.Infof("Set %s on %s", netConf.Address, link.Attrs().Name) } if netConf.MTU > 0 { if err := netlink.LinkSetMTU(link, netConf.MTU); err != nil { log.Error("set MTU Failed") return err } } if err := netlink.LinkSetUp(link); err != nil { log.Error("failed to setup link") return err } if netConf.Gateway != "" { gatewayIp := net.ParseIP(netConf.Gateway) if gatewayIp == nil { return errors.New("Invalid gateway address " + netConf.Gateway) } route := netlink.Route{ Scope: netlink.SCOPE_UNIVERSE, Gw: net.ParseIP(netConf.Gateway), } if err := netlink.RouteAdd(&route); err != nil { log.Error("gateway set failed") return err } log.Infof("Set default gateway %s", netConf.Gateway) } return nil }
func testBridgeSimpleSetup(t *testing.T, nsPrefix string, br *netlink.Bridge) ( *netlink.Bridge, []*netlink.Veth, []netns.NsHandle, func()) { links, nets, cleanup := testNetnsPair(t, nsPrefix) if br == nil { br = &netlink.Bridge{ LinkAttrs: netlink.LinkAttrs{ Name: "br0", }, } if err := netlink.LinkAdd(br); err != nil { cleanup() t.Fatal(err) } } cleanup2 := func() { netlink.LinkDel(br) cleanup() } if err := netlink.LinkSetMaster(links[0], br); err != nil { cleanup2() t.Fatal(err) } if err := netlink.LinkSetMaster(links[1], br); err != nil { cleanup2() t.Fatal(err) } if err := netlink.LinkSetUp(br); err != nil { cleanup2() t.Fatal(err) } return br, links, nets, cleanup2 }
func ensureBridge(brName string, mtu int) (*netlink.Bridge, error) { br := &netlink.Bridge{ LinkAttrs: netlink.LinkAttrs{ Name: brName, MTU: mtu, // Let kernel use default txqueuelen; leaving it unset // means 0, and a zero-length TX queue messes up FIFO // traffic shapers which use TX queue length as the // default packet limit TxQLen: -1, }, } if err := netlink.LinkAdd(br); err != nil { if err != syscall.EEXIST { return nil, fmt.Errorf("could not add %q: %v", brName, err) } // it's ok if the device already exists as long as config is similar br, err = bridgeByName(brName) if err != nil { return nil, err } } if err := netlink.LinkSetUp(br); err != nil { return nil, err } return br, nil }
func (n *networkNamespace) AddInterface(i *Interface) error { n.Lock() i.DstName = fmt.Sprintf("%s%d", i.DstName, n.nextIfIndex) n.nextIfIndex++ n.Unlock() runtime.LockOSThread() defer runtime.UnlockOSThread() origns, err := netns.Get() if err != nil { return err } defer origns.Close() f, err := os.OpenFile(n.path, os.O_RDONLY, 0) if err != nil { return fmt.Errorf("failed get network namespace %q: %v", n.path, err) } defer f.Close() // Find the network interface identified by the SrcName attribute. iface, err := netlink.LinkByName(i.SrcName) if err != nil { return err } // Move the network interface to the destination namespace. nsFD := f.Fd() if err := netlink.LinkSetNsFd(iface, int(nsFD)); err != nil { return err } if err = netns.Set(netns.NsHandle(nsFD)); err != nil { return err } defer netns.Set(origns) // Down the interface before configuring if err := netlink.LinkSetDown(iface); err != nil { return err } // Configure the interface now this is moved in the proper namespace. if err := configureInterface(iface, i); err != nil { return err } // Up the interface. if err := netlink.LinkSetUp(iface); err != nil { return err } n.Lock() n.sinfo.Interfaces = append(n.sinfo.Interfaces, i) n.Unlock() return nil }
// Enable a netlink interface func (driver *driver) interfaceUP(name string) error { iface, err := netlink.LinkByName(name) if err != nil { log.Debugf("Error retrieving a link named [ %s ]", iface.Attrs().Name) return err } return netlink.LinkSetUp(iface) }
func linkUp(link netlink.Link, netConf InterfaceConfig) error { if err := netlink.LinkSetUp(link); err != nil { log.Errorf("failed to setup link: %v", err) return err } return nil }
func SetLinkUp(iface string) error { link, err := netlink.LinkByName(iface) if err != nil { return err } return netlink.LinkSetUp(link) }
// create and attach a veth to the Weave bridge func CreateAndAttachVeth(name, peerName, bridgeName string, mtu int, keepTXOn bool, init func(peer netlink.Link) error) (*netlink.Veth, error) { bridge, err := netlink.LinkByName(bridgeName) if err != nil { return nil, fmt.Errorf(`bridge "%s" not present; did you launch weave?`, bridgeName) } if mtu == 0 { mtu = bridge.Attrs().MTU } veth := &netlink.Veth{ LinkAttrs: netlink.LinkAttrs{ Name: name, MTU: mtu}, PeerName: peerName, } if err := netlink.LinkAdd(veth); err != nil { return nil, fmt.Errorf(`could not create veth pair %s-%s: %s`, name, peerName, err) } cleanup := func(format string, a ...interface{}) (*netlink.Veth, error) { netlink.LinkDel(veth) return nil, fmt.Errorf(format, a...) } switch bridgeType := DetectBridgeType(bridgeName, DatapathName); bridgeType { case Bridge, BridgedFastdp: if err := netlink.LinkSetMasterByIndex(veth, bridge.Attrs().Index); err != nil { return cleanup(`unable to set master of %s: %s`, name, err) } if bridgeType == Bridge && !keepTXOn { if err := EthtoolTXOff(peerName); err != nil { return cleanup(`unable to set tx off on %q: %s`, peerName, err) } } case Fastdp: if err := odp.AddDatapathInterface(bridgeName, name); err != nil { return cleanup(`failed to attach %s to device "%s": %s`, name, bridgeName, err) } default: return cleanup(`invalid bridge configuration`) } if init != nil { peer, err := netlink.LinkByName(peerName) if err != nil { return cleanup("unable to find peer veth %s: %s", peerName, err) } if err := init(peer); err != nil { return cleanup("initializing veth: %s", err) } } if err := netlink.LinkSetUp(veth); err != nil { return cleanup("unable to bring veth up: %s", err) } return veth, nil }
func (driver *driver) joinEndpoint(w http.ResponseWriter, r *http.Request) { var j join if err := json.NewDecoder(r.Body).Decode(&j); err != nil { sendError(w, "Could not decode JSON encode payload", http.StatusBadRequest) return } Log.Debugf("Join request: %+v", &j) endID := j.EndpointID // create and attach local name to the bridge local := vethPair(endID[:5]) if err := netlink.LinkAdd(local); err != nil { Log.Error(err) errorResponsef(w, "could not create veth pair") return } var bridge *netlink.Bridge if maybeBridge, err := netlink.LinkByName(WeaveBridge); err != nil { Log.Error(err) errorResponsef(w, `bridge "%s" not present`, WeaveBridge) return } else { var ok bool if bridge, ok = maybeBridge.(*netlink.Bridge); !ok { Log.Errorf("%s is %+v", WeaveBridge, maybeBridge) errorResponsef(w, `device "%s" not a bridge`, WeaveBridge) return } } if netlink.LinkSetMaster(local, bridge) != nil || netlink.LinkSetUp(local) != nil { errorResponsef(w, `unable to bring veth up`) return } ifname := &iface{ SrcName: local.PeerName, DstPrefix: "ethwe", ID: 0, } res := &joinResponse{ InterfaceNames: []*iface{ifname}, } if driver.nameserver != "" { routeToDNS := &staticRoute{ Destination: driver.nameserver + "/32", RouteType: types.CONNECTED, NextHop: "", InterfaceID: 0, } res.StaticRoutes = []*staticRoute{routeToDNS} } objectResponse(w, res) Log.Infof("Join endpoint %s:%s to %s", j.NetworkID, j.EndpointID, j.SandboxKey) }
func (driver *driver) JoinEndpoint(j *api.JoinRequest) (*api.JoinResponse, error) { endID := j.EndpointID maybeBridge, err := netlink.LinkByName(WeaveBridge) if err != nil { return nil, errorf(`bridge "%s" not present; did you launch weave?`, WeaveBridge) } // create and attach local name to the bridge local := vethPair(endID[:5]) local.Attrs().MTU = maybeBridge.Attrs().MTU if err := netlink.LinkAdd(local); err != nil { return nil, errorf("could not create veth pair: %s", err) } switch maybeBridge.(type) { case *netlink.Bridge: if err := netlink.LinkSetMasterByIndex(local, maybeBridge.Attrs().Index); err != nil { return nil, errorf(`unable to set master: %s`, err) } case *netlink.GenericLink: if maybeBridge.Type() != "openvswitch" { Log.Errorf("device %s is %+v", WeaveBridge, maybeBridge) return nil, errorf(`device "%s" is of type "%s"`, WeaveBridge, maybeBridge.Type()) } odp.AddDatapathInterface(WeaveBridge, local.Name) case *netlink.Device: Log.Warnf("kernel does not report what kind of device %s is, just %+v", WeaveBridge, maybeBridge) // Assume it's our openvswitch device, and the kernel has not been updated to report the kind. odp.AddDatapathInterface(WeaveBridge, local.Name) default: Log.Errorf("device %s is %+v", WeaveBridge, maybeBridge) return nil, errorf(`device "%s" not a bridge`, WeaveBridge) } if err := netlink.LinkSetUp(local); err != nil { return nil, errorf(`unable to bring veth up: %s`, err) } ifname := &api.InterfaceName{ SrcName: local.PeerName, DstPrefix: "ethwe", } response := &api.JoinResponse{ InterfaceName: ifname, } if !driver.noMulticastRoute { multicastRoute := api.StaticRoute{ Destination: "224.0.0.0/4", RouteType: types.CONNECTED, } response.StaticRoutes = append(response.StaticRoutes, multicastRoute) } Log.Infof("Join endpoint %s:%s to %s", j.NetworkID, j.EndpointID, j.SandboxKey) return response, nil }
/* Set interface named @iface up (if not already running) */ func SetUp(iface string) { dev, err := netlink.LinkByName(iface) if err != nil { log.Fatalf("failed to look up interface %s: %s", iface, err) } log.Printf("bringing up interface %s", iface) if err := netlink.LinkSetUp(dev); err != nil { log.Fatalf("failed to set up interface %s: %s", iface, err) } }
// setupTapDevice creates persistent macvtap device // and returns a newly created netlink.Link structure // using part of pod hash and interface number in interface name func setupMacVTapDevice(podID types.UUID, config MacVTapNetConf, interfaceNumber int) (netlink.Link, error) { master, err := netlink.LinkByName(config.Master) if err != nil { return nil, errwrap.Wrap(fmt.Errorf("cannot find master device '%v'", config.Master), err) } var mode netlink.MacvlanMode switch config.Mode { // if not set - defaults to bridge mode as in: // https://github.com/coreos/rkt/blob/master/Documentation/networking.md#macvlan case "", "bridge": mode = netlink.MACVLAN_MODE_BRIDGE case "private": mode = netlink.MACVLAN_MODE_PRIVATE case "vepa": mode = netlink.MACVLAN_MODE_VEPA case "passthru": mode = netlink.MACVLAN_MODE_PASSTHRU default: return nil, fmt.Errorf("unsupported macvtap mode: %v", config.Mode) } mtu := master.Attrs().MTU if config.MTU != 0 { mtu = config.MTU } interfaceName := fmt.Sprintf("rkt-%s-vtap%d", podID.String()[0:4], interfaceNumber) link := &netlink.Macvtap{ Macvlan: netlink.Macvlan{ LinkAttrs: netlink.LinkAttrs{ Name: interfaceName, MTU: mtu, ParentIndex: master.Attrs().Index, }, Mode: mode, }, } if err := netlink.LinkAdd(link); err != nil { return nil, errwrap.Wrap(errors.New("cannot create macvtap interface"), err) } // TODO: duplicate following lines for ipv6 support, when it will be added in other places ipv4SysctlValueName := fmt.Sprintf(IPv4InterfaceArpProxySysctlTemplate, interfaceName) if _, err := cnisysctl.Sysctl(ipv4SysctlValueName, "1"); err != nil { // remove the newly added link and ignore errors, because we already are in a failed state _ = netlink.LinkDel(link) return nil, errwrap.Wrap(fmt.Errorf("failed to set proxy_arp on newly added interface %q", interfaceName), err) } if err := netlink.LinkSetUp(link); err != nil { // remove the newly added link and ignore errors, because we already are in a failed state _ = netlink.LinkDel(link) return nil, errwrap.Wrap(errors.New("cannot set up macvtap interface"), err) } return link, nil }
func (driver *driver) joinEndpoint(w http.ResponseWriter, r *http.Request) { var j join if err := json.NewDecoder(r.Body).Decode(&j); err != nil { sendError(w, "Could not decode JSON encode payload", http.StatusBadRequest) return } log.Debugf("Join request: %+v", &j) endID := j.EndpointID // unique name while still on the common netns preMoveName := endID[:5] mode, err := setVlanMode(macvlanMode) if err != nil { log.Errorf("error parsing vlan mode [ %v ]: %s", mode, err) return } // Get the link for the master index (Example: the docker host eth iface) hostEth, err := netlink.LinkByName(macvlanEthIface) if err != nil { log.Warnf("Error looking up the parent iface [ %s ] mode: [ %s ] error: [ %s ]", macvlanEthIface, mode, err) } macvlan := &netlink.Macvlan{ LinkAttrs: netlink.LinkAttrs{ Name: preMoveName, ParentIndex: hostEth.Attrs().Index, }, Mode: mode, } if err := netlink.LinkAdd(macvlan); err != nil { log.Errorf("failed to create Macvlan: [ %v ] with the error: %s", macvlan, err) log.Error("Ensure there are no existing [ ipvlan ] type links and remove with 'ip link del <link_name>'," + " also check `/var/run/docker/netns/` for orphaned links to unmount and delete, then restart the plugin") return } log.Infof("Created Macvlan port: [ %s ] using the mode: [ %s ]", macvlan.Name, macvlanMode) // Set the netlink iface MTU, default is 1500 if err := netlink.LinkSetMTU(macvlan, defaultMTU); err != nil { log.Errorf("Error setting the MTU [ %d ] for link [ %s ]: %s", defaultMTU, macvlan.Name, err) } // Bring the netlink iface up if err := netlink.LinkSetUp(macvlan); err != nil { log.Warnf("failed to enable the [ macvlan ] netlink link: [ %v ]", macvlan, err) } // SrcName gets renamed to DstPrefix on the container iface ifname := &InterfaceName{ SrcName: macvlan.Name, DstPrefix: containerIfacePrefix, } res := &joinResponse{ InterfaceName: *ifname, Gateway: gatewayIP, } objectResponse(w, res) log.Debugf("Join endpoint %s:%s to %s", j.NetworkID, j.EndpointID, j.SandboxKey) }
func (driver *driver) JoinEndpoint(j *netApi.JoinRequest) (*netApi.JoinResponse, error) { log.Debugf("Join endpoint request: %+v", j) log.Debugf("Joining endpoint %s:%s to %s", j.NetworkID, j.EndpointID, j.SandboxKey) tempName := j.EndpointID[:4] hostName := "vethr" + j.EndpointID[:4] veth := &netlink.Veth{ LinkAttrs: netlink.LinkAttrs{ Name: hostName, TxQLen: 0, }, PeerName: tempName, } log.Debugf("Adding link %+v", veth) if err := netlink.LinkAdd(veth); err != nil { log.Errorf("Unable to add link %+v:%+v", veth, err) return nil, err } if err := netlink.LinkSetMTU(veth, 1500); err != nil { log.Errorf("Error setting the MTU %s", err) } log.Debugf("Bringing link up %+v", veth) if err := netlink.LinkSetUp(veth); err != nil { log.Errorf("Unable to bring up %+v: %+v", veth, err) return nil, err } ep := driver.network.endpoints[j.EndpointID] ep.iface = hostName iface, _ := netlink.LinkByName(hostName) routeAdd(ep.ipv4Address, iface) for _, ipa := range ep.ipAliases { routeAdd(ipa, iface) } respIface := &netApi.InterfaceName{ SrcName: tempName, DstPrefix: "eth", } sandboxRoute := netApi.StaticRoute{ Destination: "0.0.0.0/0", RouteType: 1, // CONNECTED NextHop: "", } resp := &netApi.JoinResponse{ InterfaceName: respIface, DisableGatewayService: true, StaticRoutes: []netApi.StaticRoute{sandboxRoute}, } log.Infof("Join Request Response %+v", resp) return resp, nil }