即使類似的話題已經存在,我注意到,它的歷史可以追溯到兩年了,因此,我想這是比較合適的,打開一個新的一個...從Linux內核發送UDP數據包
我想弄清楚如何從Linux Kernel(3.3.4)發送UDP數據包,以便監視隨機數生成器(/drivers/char/random.c)的行爲。到目前爲止,由於sock_create和sock_sendmsg函數的原因,我設法監視了一些東西。你可以在這封郵件的結尾找到我使用的典型代碼。 (您可能還想下載完整的修改過的隨機文件here。)
通過在適當的random.c函數中插入此代碼,我可以爲每個訪問發送一個UDP數據包到/ dev/random和/ dev/urandom,以及隨機數生成器用於收集熵的每個鍵盤/鼠標事件。但是,當我嘗試監視磁盤事件時,它根本不起作用:它在引導期間會產生內核恐慌。
因此,這裏是我的主要問題:你知道爲什麼我的代碼插入磁盤事件函數時會導致如此多的麻煩嗎?(add_disk_randomness)
另外,我已經讀過netpoll API,它應該可以處理這種UDP-in-kernel問題。不幸的是,除了2005年相當有趣但過時的紅帽演示外,我還沒有找到任何相關文檔。您認爲我應該使用此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);
一般而言,最好不要在內核中執行任何可以在用戶空間中執行的操作 - 將信息通過日誌記錄機制或sysfs公開給用戶空間可能會更好,然後讓守護程序將其發送到遠程系統。 –
當一個相似的話題已經存在時,鏈接到它。你做了一個合理的工作,解釋了爲什麼你認爲現有的問題不夠好(我可能會說一些關於新內核版本的內容等等)。但如果現有問題容易獲得,那麼答案就有可能集中於自那時起發生的變化。 –
@BenVoigt Thx爲您的建議。這是[上一個主題](http://stackoverflow.com/questions/1814485/sending-udp-packet-in-linux-kernel)。 – tvuillemin