网桥创建

在网桥创建时(brctl addbr br_test),内核为其生成一个随机的mac地址,并保证此地址的多播位为零(首字节第一个bit位:addr[0] &= 0xfe),并且设置上首字节的第二个bit位,表明为本地生成的mac地址(addr[0] |= 0x02)。


static inline void eth_hw_addr_random(struct net_device *dev)
{
    dev->addr_assign_type = NET_ADDR_RANDOM;
    eth_random_addr(dev->dev_addr);
}


手动设置网桥MAC地址

用户通过IOCTL修改网桥MAC地址(ifconfig br_test hw eth 10:20:30:40:50:60),此地址不能为全零或全F地址,也不能为多播地址,否则修改不成功。最终通过函数br_stp_change_bridge_id修改网桥MAC地址。修改之后,会将变量addr_assign_type赋值为NET_ADDR_SET,表明此时的MAC地址为用户设置。


int dev_set_mac_address(struct net_device *dev, struct sockaddr *sa)
{
    const struct net_device_ops *ops = dev->netdev_ops;

	//调用网桥注册函数br_set_mac_address修改MAC地址。
    err = ops->ndo_set_mac_address(dev, sa);
    dev->addr_assign_type = NET_ADDR_SET;
}
static int br_set_mac_address(struct net_device *dev, void *p)
{
    if (!ether_addr_equal(dev->dev_addr, addr->sa_data)) {
	/* Mac address will be changed in br_stp_change_bridge_id(). */
        br_stp_change_bridge_id(br, addr->sa_data);
     }
}


增减网桥子接口

在增加/减少桥接口(brctl addif br_test eth0)时,调用br_stp_recalculate_bridge_id函数,判断是否需要更新网桥mac地址。前提条件是用户未手动设置过网桥MAC地址(addr_assign_type判断)。接着遍历网桥的子接口列表,找到子接口中最小的MAC地址,网桥使用此地址。


int br_add_if(struct net_bridge *br, struct net_device *dev)
{
    changed_addr = br_stp_recalculate_bridge_id(br);
}

bool br_stp_recalculate_bridge_id(struct net_bridge *br)
{
    /* user has chosen a value so keep it */
    if (br->dev->addr_assign_type == NET_ADDR_SET)
        return false;

    list_for_each_entry(p, &br->port_list, list) {
        if (addr == br_mac_zero ||
            memcmp(p->dev->dev_addr, addr, ETH_ALEN) < 0)
            addr = p->dev->dev_addr;
    }
    br_stp_change_bridge_id(br, addr);
}


内核版本

linux-4.14.4



GitHub 加速计划 / li / linux-dash
10.39 K
1.2 K
下载
A beautiful web dashboard for Linux
最近提交(Master分支:2 个月前 )
186a802e added ecosystem file for PM2 4 年前
5def40a3 Add host customization support for the NodeJS version 4 年前
Logo

旨在为数千万中国开发者提供一个无缝且高效的云端环境,以支持学习、使用和贡献开源项目。

更多推荐