- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试运行以下代码,该代码是从 here 复制的。我做了一些更改以使用较旧的内核版本运行它。
当我插入内核模块时,nlmsg_multicast()
失败并在 /var/log/messages
中记录为 nlmsg_multicast() 错误:-3
>。运行用户空间程序时,socket()
失败。
我到底想做的是,
因为,可能会发生用户空间中没有进程可用于回复事件的情况,即使在这种情况下,模块也必须发送事件并等待一段时间才能响应。
是否可以将第一条消息从内核模块发送到用户空间中的进程?我怎样才能做到这一点?
内核模块代码:
Makefile
obj-m := foo.o
KDIR := /lib/modules/$(shell uname -r)/build
PWD := $(shell pwd)
default:
$(MAKE) -C $(KDIR) SUBDIRS=$(PWD) modules
clean:
$(MAKE) -C $(KDIR) SUBDIRS=$(PWD) clean
foo.c
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/netlink.h>
#include <net/netlink.h>
#include <net/net_namespace.h>
/* Protocol family, consistent in both kernel prog and user prog. */
#define MYPROTO NETLINK_USERSOCK
/* Multicast group, consistent in both kernel prog and user prog. */
#define MYGRP 21
static struct sock *nl_sk = NULL;
static void send_to_user(void)
{
struct sk_buff *skb;
struct nlmsghdr *nlh;
char *msg = "Hello from kernel";
int msg_size = strlen(msg) + 1;
int res;
pr_info("Creating skb.\n");
skb = nlmsg_new(NLMSG_ALIGN(msg_size + 1), GFP_KERNEL);
if (!skb) {
pr_err("Allocation failure.\n");
return;
}
nlh = nlmsg_put(skb, 0, 1, NLMSG_DONE, msg_size + 1, 0);
strcpy(nlmsg_data(nlh), msg);
pr_info("Sending skb.\n");
res = nlmsg_multicast(nl_sk, skb, 0, MYGRP, GFP_KERNEL);
if (res < 0)
pr_info("nlmsg_multicast() error: %d\n", res);
else
pr_info("Success.\n");
}
static int __init hello_init(void)
{
pr_info("Inserting hello module.\n");
//nl_sk = netlink_kernel_create(&init_net, MYPROTO, NULL);
nl_sk = netlink_kernel_create(&init_net, MYPROTO, 0, NULL, NULL, THIS_MODULE);
if (!nl_sk) {
pr_err("Error creating socket.\n");
return -10;
}
send_to_user();
netlink_kernel_release(nl_sk);
return 0;
}
static void __exit hello_exit(void)
{
pr_info("Exiting hello module.\n");
}
module_init(hello_init);
module_exit(hello_exit);
MODULE_LICENSE("GPL");
用户空间程序:(使用gcc somename.c
编译)
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <sys/socket.h>
#include <linux/netlink.h>
#include <unistd.h>
/* Protocol family, consistent in both kernel prog and user prog. */
#define MYPROTO NETLINK_USERSOCK
/* Multicast group, consistent in both kernel prog and user prog. */
#define MYMGRP 21
int open_netlink(void)
{
int sock;
struct sockaddr_nl addr;
int group = MYMGRP;
sock = socket(AF_NETLINK, SOCK_RAW, MYPROTO);
if (sock < 0) {
printf("sock < 0.\n");
return sock;
}
memset((void *) &addr, 0, sizeof(addr));
addr.nl_family = AF_NETLINK;
addr.nl_pid = getpid();
/* This doesn't work for some reason. See the setsockopt() below. */
addr.nl_groups = MYMGRP;
if (bind(sock, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
printf("bind < 0.\n");
return -1;
}
/*
* 270 is SOL_NETLINK. See
* http://lxr.free-electrons.com/source/include/linux/socket.h?v=4.1#L314
* and
* https://stackoverflow.com/questions/17732044/
*/
/*if (setsockopt(sock, 270, NETLINK_ADD_MEMBERSHIP, &group, sizeof(group)) < 0) {
printf("setsockopt < 0\n");
return -1;
}*/
return sock;
}
void read_event(int sock)
{
struct sockaddr_nl nladdr;
struct msghdr msg;
struct iovec iov;
char buffer[65536];
int ret;
iov.iov_base = (void *) buffer;
iov.iov_len = sizeof(buffer);
msg.msg_name = (void *) &(nladdr);
msg.msg_namelen = sizeof(nladdr);
msg.msg_iov = &iov;
msg.msg_iovlen = 1;
printf("Ok, listening.\n");
ret = recvmsg(sock, &msg, 0);
if (ret < 0)
printf("ret < 0.\n");
else
printf("Received message payload: %s\n", NLMSG_DATA((struct nlmsghdr *) &buffer));
}
int main(int argc, char *argv[])
{
int nls;
nls = open_netlink();
if (nls < 0)
return nls;
while (1)
read_event(nls);
return 0;
}
感谢您的宝贵时间!
最佳答案
这看起来像是糟糕的设计(因为上层应该依赖于下层,而不是相反)。但是,如果您确信内核无法闲置或使用默认配置运行,直到用户空间可以获取信息,那么首先安装 this tool (可能也想阅读 core guide ),然后执行如下操作:
内核:
#include <linux/module.h>
#include <linux/kernel.h>
#include <net/netlink.h>
#include <net/net_namespace.h>
#define MYPROTO NETLINK_USERSOCK
#define MYGRP 22
static struct sock *nl_sk;
static struct timer_list timer;
void try_send(unsigned long data)
{
struct sk_buff *skb;
struct nlmsghdr *nlh;
char *msg = "Hello from kernel";
int msg_size = strlen(msg) + 1;
int res;
skb = nlmsg_new(NLMSG_ALIGN(msg_size + 1), GFP_ATOMIC);
if (!skb) {
pr_err("Allocation failure.\n");
return;
}
nlh = nlmsg_put(skb, 0, 1, NLMSG_DONE, msg_size + 1, 0);
strcpy(nlmsg_data(nlh), msg);
pr_info("Sending multicast.\n");
res = nlmsg_multicast(nl_sk, skb, 0, MYGRP, GFP_ATOMIC);
if (res < 0) {
pr_info("nlmsg_multicast() error: %d. Will try again later.\n", res);
/* Wait 1 second. */
mod_timer(&timer, jiffies + msecs_to_jiffies(1000));
} else {
pr_info("Success.\n");
}
}
static int handle_netlink_message(struct sk_buff *skb_in, struct nlmsghdr *nl_hdr)
{
char *hello;
hello = NLMSG_DATA(nl_hdr);
pr_info("Userspace says '%s.'\n", hello);
return 0;
}
static void receive_answer(struct sk_buff *skb)
{
netlink_rcv_skb(skb, &handle_netlink_message);
}
static int __init hello_init(void)
{
pr_info("Inserting module.\n");
nl_sk = netlink_kernel_create(&init_net, MYPROTO, 0, receive_answer, NULL, THIS_MODULE);
if (!nl_sk) {
pr_err("Error creating socket.\n");
return -10;
}
init_timer(&timer);
timer.function = try_send;
timer.expires = jiffies + 1000;
timer.data = 0;
add_timer(&timer);
return 0;
}
static void __exit hello_exit(void)
{
del_timer_sync(&timer);
netlink_kernel_release(nl_sk);
pr_info("Exiting module.\n");
}
module_init(hello_init);
module_exit(hello_exit);
MODULE_LICENSE("GPL");
用户(我使用gcc usr.c -I/usr/include/libnl3 -lnl-3 -Wall
进行编译,您的情况可能会有所不同):
#include <netlink/netlink.h>
#include <netlink/msg.h>
#define MYPROTO NETLINK_USERSOCK
#define MYMGRP 22
struct nl_sock *sk;
void respond_to_kernel(void)
{
char *response = "foo bar";
int error;
error = nl_send_simple(sk, 12345, NLMSG_DONE, response, strlen(response) + 1);
if (error < 0) {
printf("nl_send_simple() threw errcode %d.\n", error);
printf("libnl's message: %s", nl_geterror(error));
} else {
printf("Responded %d bytes.\n", error);
}
}
int receive_kernel_request(struct nl_msg *msg, void *arg)
{
char *hello;
hello = nlmsg_data(nlmsg_hdr(msg));
printf("Kernel says '%s'.\n", hello);
respond_to_kernel();
return 0;
}
int prepare_socket(void)
{
int error;
sk = nl_socket_alloc();
if (!sk) {
printf("nl_socket_alloc() returned NULL.\n");
return -1;
}
nl_socket_disable_seq_check(sk);
error = nl_socket_modify_cb(sk, NL_CB_FINISH, NL_CB_CUSTOM, receive_kernel_request, NULL);
if (error < 0) {
printf("Could not register callback function. Errcode: %d\n", error);
goto fail;
}
error = nl_connect(sk, MYPROTO);
if (error < 0) {
printf("Connection failed: %d\n", error);
goto fail;
}
error = nl_socket_add_memberships(sk, MYMGRP, 0);
if (error) {
printf("Could not register to the multicast group. %d\n", error);
goto fail;
}
return 0;
fail:
printf("libnl's message: %s\n", nl_geterror(error));
nl_socket_free(sk);
return error;
}
int wait_for_kernel_message(void)
{
int error;
printf("Waiting for kernel request...\n");
error = nl_recvmsgs_default(sk);
if (error < 0) {
printf("nl_send_simple() threw errcode %d.\n", error);
printf("libnl's message: %s\n", nl_geterror(error));
return error;
}
return 0;
}
void destroy_socket(void)
{
nl_socket_free(sk);
}
int main(int argc, char *argv[])
{
int error;
error = prepare_socket();
if (error)
return error;
error = wait_for_kernel_message();
destroy_socket();
return error;
}
在内核 3.2 上测试。 (抱歉;这是我现在的最低水平。)
关于sockets - 内核模块可以主动通过netlink向用户空间发送消息吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35098943/
我在将 netlink 套接字与新的 netlink 系列一起使用时遇到了一些问题……尤其是对于组。我在内核空间中使用 netlink_broadcast 或 nlmsg_multicast 方法向用
我试图实现的任务实际上非常简单(将字符串“TEST”多播到用户级守护程序),但内核模块无法编译。它因错误而停止: passing argument 4 of ‘genlmsg_multicast_al
我的应用程序应该从内核获取 netlink 多播路由更新。我做了一些研究,发现多播使用不同的系列:RTNL_FAMILY_IPMR组是 RTMGRP_IPV4_MROUTE。 但是如果我使用:
我没有在内核中找到任何文档/说明,要求以网络字节顺序将 netlink 消息中的 16/32-bit 值传递给内核。所以我的问题是在填充网络链接消息时是否必须使用 htonl/htons 函数。有这样
我正在尝试将消息从用户空间发送到内核。我能够通过 netlink 套接字成功发送它。但我想知道我是否应该释放我编写的内核函数中的消息,或者 netlink 套接字 API 正在释放内存? 同样在/pr
我最近又开始开发 C++ 应用程序。我正在使用 NetLink for Sockets 和 Visual Studio 进行开发。 我编写了一个小代码,但出现了一些错误。我已经包含了来自 NetLin
我编写了一个内核模块和用户空间程序,内核模块发送 netlink 多播消息,用户空间程序读取这些消息并将它们打印出来。内核模块和用户空间程序可在此处 (https://github.com/aksha
静态分析器 PVS-Studio 在 nh = (struct nlmsghdr *) buf 中报告, The pointer 'buf' is cast to a more strictly al
我正在使用 netlink和 genetlink Go packages 编写一个实用程序来与 MAC80211_HWSIM netlink 系列功能进行交互。它运作良好……几乎。 我发现,如果我执行
我想使用 netlink 在应用程序和内核空间之间进行通信。我的Linux内核版本是2.6.28,下面是我的错误代码: nf_sock=netlink_kernel_create(NL_PROTO,0
我了解 Netlink 是在 Linux 中通信内核和用户空间的现代且正确的方式。 我有一个需要配置的内核模块,所以我使用 Netlink 让它与用户空间应用程序对话。 一切都很顺利,但在我看来任何用
我正在使用 Linux 作为主要平台学习嵌入式系统编程。我想创建一个设备事件管理服务。该服务是一个用户空间应用程序/守护进程,它将检测连接的硬件模块是否触发了事件。但我的问题是我不知道应该从哪里开始。
我正在开发一个用于嵌入式设备的 C++ 应用程序,该应用程序通过 netlink 套接字监听 USB 热插拔事件。检测事件完美无缺,但另外我想在程序开始时查询已经连接的设备。我能够为网络接口(inte
我编写了一个内核模块,它接受数据(结构形式)并将一些数据(相同结构格式)传回用户空间。我可以成功接收来自用户的消息,但在尝试取消引用从内核接收的数据结构的某些成员时出现段错误。我正在使用 netlin
我试过this program 。该程序向内核发送“hello”,内核再次向用户空间回复“hello”一次。 我的要求:用户必须发送“hello”并且必须每2秒接收一次消息。 我已经尝试过了。 用户应
我一直在尝试在内核模块和进程之间实现非常简单的网络链接通信。但我似乎一直坚持在内核模块上创建 netlink 套接字。正如我从内核 3.6 及更高版本中了解到的,netlink_kernel_crea
我是内核开发新手,在通过 netlink 套接字将数据从内核空间统一转换到用户空间时遇到问题。 send_nat()函数将从内核模块调用,以将用户定义的struct nat_mntr写入netlink
我想以内核作为客户端在内核模块和用户应用程序之间建立连接。换句话说,内核会向用户应用程序发送消息,等待回复,收到回复,然后继续执行。 例如,在内核中我会发送消息然后等待回复。 // inside ke
我正在尝试获取通过 NetLink(Linux 内核)发送的数据 block 的大小。我已尝试使用 size = nlh->nlmsg_len - NLMSG_HDRLEN,但未返回正确的大小。获取消
我的目标是始终在内核内存中拥有一个数组或列表,以便始终可以从内核空间访问它。为此,我使用此处推荐的 netlink 套接字。我关注this example ,它显示了如何发送字符串。我不确定如何在套接
我是一名优秀的程序员,十分优秀!