如何在内核和用户空间之间创建“netlink”?

时间:2009-04-30 06:40:55

标签: c linux linux-kernel device-driver netlink

我想使用netlink在应用程序和内核空间之间进行通信。我的Linux内核版本是2.6.28,以下是我的错误代码:

nf_sock=netlink_kernel_create(NL_PROTO,0,nl_user_skb,THIS_MODULE);

缩写错误消息为:

error: too few arguments to function 'netlink_kernel_create'

在文件<linux/netlink.h>中,函数netlink_kernel_create()定义为

extern struct sock *netlink_kernel_create(struct net *net,int unit,unsigned int groups,void (*input)(struct sk_buff *skb),struct mutex *cb_mutex,struct module *module)

我不明白第一个参数net的用途。有人可以解释一下我应该在这里使用什么吗?

4 个答案:

答案 0 :(得分:4)

struct net包含有关网络命名空间的信息,网络命名空间是一组可用于进程的网络资源。请注意,可能有多个网络命名空间(即网络堆栈的多个实例),但大多数驱动程序使用init_net命名空间。

您的电话应该类似于以下内容

nf_sock = netlink_kernel_create(&init_net,
                                NETLINK_USERSOCK,
                                0,
                                nl_rcv_func,
                                NULL,
                                THIS_MODULE);

其中nl_rcv_func是一个以struct sk_buff *skb为唯一参数并处理收到的netlink消息的函数。

答案 1 :(得分:2)

你似乎一直在关注像this one这样的指南,这些指南(来自2005年)可能已经被内核的开发所超越。似乎从内核端创建netlink的内部API已经发生了变化。

检查本地内核树中的Documentation /文件夹以获取一些(希望更新鲜的)文档,或者阅读代码本身。您还可以trawl Linux内核邮件列表存档,以便提及似乎已发生的更改。

Here是2.6.29的实际实现,如果您更倾向于向后解决(当然还没有在您自己的树中检查过这个)。

答案 2 :(得分:1)

是的,struct net确实用于net命名空间,但是总是使用init_net是不合适的,你应该注册你自己的pernet_operations,如下所示:

static struct pernet_operations fib_net_ops = {
        .init = fib_net_init,
        .exit = fib_net_exit,
};

static int __net_init fib_net_init(struct net *net)
{
        int error;

#ifdef CONFIG_IP_ROUTE_CLASSID
        net->ipv4.fib_num_tclassid_users = 0;
#endif
        error = ip_fib_net_init(net);
        if (error < 0)
                goto out;
        error = nl_fib_lookup_init(net);
        if (error < 0)
                goto out_nlfl;
        error = fib_proc_init(net);
        if (error < 0)
                goto out_proc;
out:
        return error;

out_proc:
        nl_fib_lookup_exit(net);
out_nlfl:
        ip_fib_net_exit(net);
        goto out;
}

static int __net_init nl_fib_lookup_init(struct net *net)
{
        struct sock *sk;
        struct netlink_kernel_cfg cfg = {
                .input  = nl_fib_input,
        };

        sk = netlink_kernel_create(net, NETLINK_FIB_LOOKUP, &cfg);
        if (sk == NULL)
                return -EAFNOSUPPORT;
        net->ipv4.fibnl = sk;
        return 0;
}

最后:

register_pernet_subsys(&fib_net_ops);

答案 3 :(得分:-3)

我建议ioctl进行内核/用户通信。 ioctl接口是标准的,并且内核之间更新的可能性很小。