我正在编写一个使用 netlink 在用户空间和内核空间之间进行通信的 Linux 驱动程序。但是我找不到一些有用的资料,因为netlink已经从Linux kernel >=2.6.24改变了。谁能给我一些关于如何创建netlink套接字的建议。提前致谢!
问问题
12750 次
2 回答
7
下面的代码演示了使用 netlink 从用户空间应用程序向内核模块发送数据的基础知识。此代码适用于具有libnl的 git 版本 (ef8ba32) 的 Linux 2.6.28.9 。有关更多详细信息,请查看 libnl文档和广泛使用 netlink的iw代码。
核心
#include <linux/kernel.h>
#include <linux/module.h>
#include <net/sock.h>
#include <net/netlink.h>
#define MY_MSG_TYPE (0x10 + 2) // + 2 is arbitrary. same value for kern/usr
static struct sock *my_nl_sock;
DEFINE_MUTEX(my_mutex);
static int
my_rcv_msg(struct sk_buff *skb, struct nlmsghdr *nlh)
{
int type;
char *data;
type = nlh->nlmsg_type;
if (type != MY_MSG_TYPE) {
printk("%s: expect %#x got %#x\n", __func__, MY_MSG_TYPE, type);
return -EINVAL;
}
data = NLMSG_DATA(nlh);
printk("%s: %02x %02x %02x %02x %02x %02x %02x %02x\n", __func__,
data[0], data[1], data[2], data[3],
data[4], data[5], data[6], data[7]);
return 0;
}
static void
my_nl_rcv_msg(struct sk_buff *skb)
{
mutex_lock(&my_mutex);
netlink_rcv_skb(skb, &my_rcv_msg);
mutex_unlock(&my_mutex);
}
static int
my_init(void)
{
my_nl_sock = netlink_kernel_create(&init_net, NETLINK_USERSOCK, 0,
my_nl_rcv_msg, NULL, THIS_MODULE);
if (!my_nl_sock) {
printk(KERN_ERR "%s: receive handler registration failed\n", __func__);
return -ENOMEM;
}
return 0;
}
static void
my_exit(void)
{
if (my_nl_sock) {
netlink_kernel_release(my_nl_sock);
}
}
module_init(my_init);
module_exit(my_exit);
用户空间
#include <stdio.h>
#include <stdlib.h>
#include <netlink/netlink.h>
#define MY_MSG_TYPE (0x10 + 2) // + 2 is arbitrary but is the same for kern/usr
int
main(int argc, char *argv[])
{
struct nl_sock *nls;
char msg[] = { 0xde, 0xad, 0xbe, 0xef, 0x90, 0x0d, 0xbe, 0xef };
int ret;
nls = nl_socket_alloc();
if (!nls) {
printf("bad nl_socket_alloc\n");
return EXIT_FAILURE;
}
ret = nl_connect(nls, NETLINK_USERSOCK);
if (ret < 0) {
nl_perror(ret, "nl_connect");
nl_socket_free(nls);
return EXIT_FAILURE;
}
ret = nl_send_simple(nls, MY_MSG_TYPE, 0, msg, sizeof(msg));
if (ret < 0) {
nl_perror(ret, "nl_send_simple");
nl_close(nls);
nl_socket_free(nls);
return EXIT_FAILURE;
} else {
printf("sent %d bytes\n", ret);
}
nl_close(nls);
nl_socket_free(nls);
return EXIT_SUCCESS;
}
于 2009-06-22T17:42:38.963 回答
0
我已经有一段时间没有进行内核编程了。因此,我无法直接举例说明引入了哪些更改。我可以告诉你我如何理解 netlink 函数以及如何使用它们,阅读代码。特别是来自 NetworkManager 或无线扩展 (iwlib) 的代码,因为这是我 2 年前关注的领域。那些 OSS 项目总是在内核的变化上处于领先地位,而且它们的代码也不难理解。
于 2009-05-19T21:34:55.717 回答