即使已经存在类似的主题,但我注意到它可以追溯到两年前,因此我想重新打开一个更合适......
我试图弄清楚如何从 Linux 内核(3.3.4)发送 UDP 数据包,以监控随机数生成器(/drivers/char/random.c)的行为。到目前为止,由于 sock_create 和 sock_sendmsg 函数,我已经设法监控了一些事情。您可以在此消息的末尾找到我使用的典型代码。(您可能还想在此处下载完整修改后的 random.c 文件。)
通过将此代码插入到适当的 random.c 函数中,我可以为每次访问 /dev/random 和 /dev/urandom 以及随机数生成器用来收集熵的每个键盘/鼠标事件发送一个 UDP 数据包. 但是,当我尝试监视磁盘事件时,它根本不起作用:它在引导期间会产生内核恐慌。
因此,这是我的主要问题:您知道为什么我的代码在插入磁盘事件函数时会引起这么多麻烦吗?(add_disk_randomness)
或者,我读过 netpoll API,它应该处理这种内核中的 UDP 问题。不幸的是,除了 2005 年的一个非常有趣但过时的 Red Hat 演示文稿之外,我没有找到任何相关文档。你认为我应该使用这个 API 吗?如果有,你有什么例子吗?
任何帮助,将不胜感激。提前致谢。
PS:这是我在这里的第一个问题,所以如果我做错了什么,请不要犹豫,告诉我,我会记住它以备将来:)
#include <linux/net.h>
#include <linux/in.h>
#include <linux/netpoll.h>
#define MESSAGE_SIZE 1024
#define INADDR_SEND ((unsigned long int)0x0a00020f) //10.0.2.15
static bool sock_init;
static struct socket *sock;
static struct sockaddr_in sin;
static struct msghdr msg;
static struct iovec iov;
[...]
int error, len;
mm_segment_t old_fs;
char message[MESSAGE_SIZE];
if (sock_init == false)
{
/* Creating socket */
error = sock_create(AF_INET, SOCK_DGRAM, IPPROTO_UDP, &sock);
if (error<0)
printk(KERN_DEBUG "Can't create socket. Error %d\n",error);
/* Connecting the socket */
sin.sin_family = AF_INET;
sin.sin_port = htons(1764);
sin.sin_addr.s_addr = htonl(INADDR_SEND);
error = sock->ops->connect(sock, (struct sockaddr *)&sin, sizeof(struct sockaddr), 0);
if (error<0)
printk(KERN_DEBUG "Can't connect socket. Error %d\n",error);
/* Preparing message header */
msg.msg_flags = 0;
msg.msg_name = &sin;
msg.msg_namelen = sizeof(struct sockaddr_in);
msg.msg_control = NULL;
msg.msg_controllen = 0;
msg.msg_iov = &iov;
msg.msg_control = NULL;
sock_init = true;
}
/* Sending a message */
sprintf(message,"EXTRACT / Time: %llu / InputPool: %4d / BlockingPool: %4d / NonblockingPool: %4d / Request: %4d\n",
get_cycles(),
input_pool.entropy_count,
blocking_pool.entropy_count,
nonblocking_pool.entropy_count,
nbytes*8);
iov.iov_base = message;
len = strlen(message);
iov.iov_len = len;
msg.msg_iovlen = len;
old_fs = get_fs();
set_fs(KERNEL_DS);
error = sock_sendmsg(sock,&msg,len);
set_fs(old_fs);