github.com/Prakhar-Agarwal-byte/moby@v0.0.0-20231027092010-a14e3e8ab87e/libnetwork/drivers/bridge/setup_device_linux.go (about)

     1  package bridge
     2  
     3  import (
     4  	"context"
     5  	"fmt"
     6  	"os"
     7  	"path/filepath"
     8  
     9  	"github.com/Prakhar-Agarwal-byte/moby/libnetwork/netutils"
    10  	"github.com/containerd/log"
    11  	"github.com/vishvananda/netlink"
    12  )
    13  
    14  // SetupDevice create a new bridge interface/
    15  func setupDevice(config *networkConfiguration, i *bridgeInterface) error {
    16  	// We only attempt to create the bridge when the requested device name is
    17  	// the default one. The default bridge name can be overridden with the
    18  	// DOCKER_TEST_CREATE_DEFAULT_BRIDGE env var. It should be used only for
    19  	// test purpose.
    20  	var defaultBridgeName string
    21  	if defaultBridgeName = os.Getenv("DOCKER_TEST_CREATE_DEFAULT_BRIDGE"); defaultBridgeName == "" {
    22  		defaultBridgeName = DefaultBridgeName
    23  	}
    24  	if config.BridgeName != defaultBridgeName && config.DefaultBridge {
    25  		return NonDefaultBridgeExistError(config.BridgeName)
    26  	}
    27  
    28  	// Set the bridgeInterface netlink.Bridge.
    29  	i.Link = &netlink.Bridge{
    30  		LinkAttrs: netlink.LinkAttrs{
    31  			Name: config.BridgeName,
    32  		},
    33  	}
    34  
    35  	// Set the bridge's MAC address. Requires kernel version 3.3 or up.
    36  	hwAddr := netutils.GenerateRandomMAC()
    37  	i.Link.Attrs().HardwareAddr = hwAddr
    38  	log.G(context.TODO()).Debugf("Setting bridge mac address to %s", hwAddr)
    39  
    40  	if err := i.nlh.LinkAdd(i.Link); err != nil {
    41  		log.G(context.TODO()).WithError(err).Errorf("Failed to create bridge %s via netlink", config.BridgeName)
    42  		return err
    43  	}
    44  
    45  	return nil
    46  }
    47  
    48  func setupDefaultSysctl(config *networkConfiguration, i *bridgeInterface) error {
    49  	// Disable IPv6 router advertisements originating on the bridge
    50  	sysPath := filepath.Join("/proc/sys/net/ipv6/conf/", config.BridgeName, "accept_ra")
    51  	if _, err := os.Stat(sysPath); err != nil {
    52  		log.G(context.TODO()).
    53  			WithField("bridge", config.BridgeName).
    54  			WithField("syspath", sysPath).
    55  			Info("failed to read ipv6 net.ipv6.conf.<bridge>.accept_ra")
    56  		return nil
    57  	}
    58  	if err := os.WriteFile(sysPath, []byte{'0', '\n'}, 0o644); err != nil {
    59  		log.G(context.TODO()).WithError(err).Warn("unable to disable IPv6 router advertisement")
    60  	}
    61  	return nil
    62  }
    63  
    64  // SetupDeviceUp ups the given bridge interface.
    65  func setupDeviceUp(config *networkConfiguration, i *bridgeInterface) error {
    66  	err := i.nlh.LinkSetUp(i.Link)
    67  	if err != nil {
    68  		return fmt.Errorf("Failed to set link up for %s: %v", config.BridgeName, err)
    69  	}
    70  
    71  	// Attempt to update the bridge interface to refresh the flags status,
    72  	// ignoring any failure to do so.
    73  	if lnk, err := i.nlh.LinkByName(config.BridgeName); err == nil {
    74  		i.Link = lnk
    75  	} else {
    76  		log.G(context.TODO()).Warnf("Failed to retrieve link for interface (%s): %v", config.BridgeName, err)
    77  	}
    78  	return nil
    79  }