绑定驱动程序如何从从属接口获取RX数据包

问题描述 投票:5回答:2

我有一个有关如何绑定驱动程序从从属接口获取RX数据包的问题。我发现绑定使用dev_add_pack()来设置LACPDU和ARP数据包的处理程序,但是我没有找到其他处理程序(对于其他数据包类型)。

您能帮我解决这个问题吗?

c linux-kernel network-programming linux-device-driver
2个回答
2
投票

绑定驱动程序注册自己的Rx处理程序,当从属接口被奴役到绑定主机时,在bond_enslave()中,您可以看到:

res = netdev_rx_handler_register(slave_dev, bond_handle_frame,
                                 new_slave);

因此,在bond_handle_frame()中,它将劫持从属接口接收到的数据包,以便绑定主设备将改为接收这些数据包:

static rx_handler_result_t bond_handle_frame(struct sk_buff **pskb)
{
        struct sk_buff *skb = *pskb;
        struct slave *slave;
        struct bonding *bond;
        int (*recv_probe)(const struct sk_buff *, struct bonding *,
                          struct slave *);
        int ret = RX_HANDLER_ANOTHER;

        skb = skb_share_check(skb, GFP_ATOMIC);
        if (unlikely(!skb))
                return RX_HANDLER_CONSUMED;

        *pskb = skb;

        slave = bond_slave_get_rcu(skb->dev);
        bond = slave->bond;

        if (bond->params.arp_interval)
                slave->dev->last_rx = jiffies;

        recv_probe = ACCESS_ONCE(bond->recv_probe);
        if (recv_probe) {
                ret = recv_probe(skb, bond, slave);
                if (ret == RX_HANDLER_CONSUMED) {
                        consume_skb(skb);
                        return ret;
                }
        }

        if (bond_should_deliver_exact_match(skb, slave, bond)) {
                return RX_HANDLER_EXACT;
        }

        skb->dev = bond->dev;

        if (bond->params.mode == BOND_MODE_ALB &&
            bond->dev->priv_flags & IFF_BRIDGE_PORT &&
            skb->pkt_type == PACKET_HOST) {

                if (unlikely(skb_cow_head(skb,
                                          skb->data - skb_mac_header(skb)))) {
                        kfree_skb(skb);
                        return RX_HANDLER_CONSUMED;
                }
                memcpy(eth_hdr(skb)->h_dest, bond->dev->dev_addr, ETH_ALEN);
        }

        return ret;
}

1
投票

我检查了绑定代码,发现当驱动程序在这些模式下工作时,如果没有某些类型(LACPDU,ARP),驱动程序不会检查传入的RX数据包。驱动程序使用dev_add_pack()函数为此数据包设置处理程序。

实际上,您可以使用nf_register_hook()来设置全局挂钩,该接口为设置自己的用于拦截数据包的网络过滤程序提供接口。似乎nf_register_hook()比dev_add_pack()更强大,但是我认为在使用nf_register_hook()时需要格外小心,因为在挂接条件错误的情况下,它可能会丢弃很多数据包。

© www.soinside.com 2019 - 2024. All rights reserved.