我正在尝试根据此示例http://www.electronicsfaq.com/2014/02/generic-netlink-sockets-example-code将结构从 LKM 发送/接收到用户区(反之亦然) 。 .html _
我向内核发送一个结构。它接收到结构体,处理信息,并将结构体发送回用户区。但是,如果结构有 2 个字段,我可以发送和接收,但是当我再添加一个字段时,我会收到一个空结构。自 2 天前以来,我已经探索了一些变体,但没有成功。我很困惑,我不确定问题出在哪里?这是代码:
nl_user.c:
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <unistd.h>
#include <poll.h>
#include <string.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <signal.h>
#include "global.h"
#include <linux/genetlink.h>
#define GENLMSG_DATA(glh) ((void *)(NLMSG_DATA(glh) + GENL_HDRLEN))
#define GENLMSG_PAYLOAD(glh) (NLMSG_PAYLOAD(glh, 0) - GENL_HDRLEN)
#define NLA_DATA(na) ((void *)((void*)(na) + NLA_HDRLEN))
//Variables used for netlink
int nl_fd; //netlink socket's file descriptor
struct sockaddr_nl nl_address; //netlink socket address
int nl_family_id;
int nl_rxtx_length; //Number of bytes sent or received via send() or recv()
struct nlattr *nl_na; //pointer to netlink attributes structure within the payload
Response p;
struct {
struct nlmsghdr n;
struct genlmsghdr g;
char buf[256];
} nl_request_msg, nl_response_msg;
int main(void) {
struct {
struct nlmsghdr n;
struct genlmsghdr g;
Response buf;
} my_nl_request_msg, my_nl_response_msg;
//Step 1: Open the socket. Note that protocol = NETLINK_GENERIC
nl_fd = socket(AF_NETLINK, SOCK_RAW, NETLINK_GENERIC);
if (nl_fd < 0) {
perror("socket()");
return -1;
}
//Step 2: Bind the socket.
memset(&nl_address, 0, sizeof(nl_address));
nl_address.nl_family = AF_NETLINK;
nl_address.nl_groups = 0;
if (bind(nl_fd, (struct sockaddr *) &nl_address, sizeof(nl_address)) < 0) {
perror("bind()");
close(nl_fd);
return -1;
}
//Step 3. Resolve the family ID corresponding to the string "CONTROL_EXMPL"
//Populate the netlink header
nl_request_msg.n.nlmsg_type = GENL_ID_CTRL;
nl_request_msg.n.nlmsg_flags = NLM_F_REQUEST;
nl_request_msg.n.nlmsg_seq = 0;
nl_request_msg.n.nlmsg_pid = getpid();
nl_request_msg.n.nlmsg_len = NLMSG_LENGTH(GENL_HDRLEN);
//Populate the payload's "family header" : which in our case is genlmsghdr
nl_request_msg.g.cmd = CTRL_CMD_GETFAMILY;
nl_request_msg.g.version = 0x1;
//Populate the payload's "netlink attributes"
nl_na = (struct nlattr *) GENLMSG_DATA(&nl_request_msg);
nl_na->nla_type = CTRL_ATTR_FAMILY_NAME;
nl_na->nla_len = strlen("CONTROL_EXMPL") + 1 + NLA_HDRLEN;
strcpy(NLA_DATA(nl_na), "CONTROL_EXMPL");
nl_request_msg.n.nlmsg_len += NLMSG_ALIGN(nl_na->nla_len);
memset(&nl_address, 0, sizeof(nl_address));
nl_address.nl_family = AF_NETLINK;
//Send the family ID request message to the netlink controller
nl_rxtx_length = sendto(nl_fd, (char *) &nl_request_msg, nl_request_msg.n.nlmsg_len,
0, (struct sockaddr *) &nl_address, sizeof(nl_address));
if (nl_rxtx_length != nl_request_msg.n.nlmsg_len) {
perror("sendto()");
close(nl_fd);
return -1;
}
//Wait for the response message
nl_rxtx_length = recv(nl_fd, &nl_response_msg, sizeof(nl_response_msg), 0);
if (nl_rxtx_length < 0) {
perror("recv()");
return -1;
}
//Validate response message
if (!NLMSG_OK((&nl_response_msg.n), nl_rxtx_length)) {
fprintf(stderr, "family ID request : invalid message\n");
return -1;
}
if (nl_response_msg.n.nlmsg_type == NLMSG_ERROR) { //error
fprintf(stderr, "family ID request : receive error\n");
return -1;
}
//Extract family ID
nl_na = (struct nlattr *) GENLMSG_DATA(&nl_response_msg);
nl_na = (struct nlattr *) ((char *) nl_na + NLA_ALIGN(nl_na->nla_len));
if (nl_na->nla_type == CTRL_ATTR_FAMILY_ID) {
nl_family_id = *(__u16 *) NLA_DATA(nl_na);
}
//Step 4. Send own custom message
memset(&my_nl_request_msg, 0, sizeof(my_nl_request_msg));
memset(&my_nl_response_msg, 0, sizeof(my_nl_response_msg));
my_nl_request_msg.n.nlmsg_len = NLMSG_LENGTH(GENL_HDRLEN);
my_nl_request_msg.n.nlmsg_type = nl_family_id;
my_nl_request_msg.n.nlmsg_flags = NLM_F_REQUEST;
my_nl_request_msg.n.nlmsg_seq = 60;
my_nl_request_msg.n.nlmsg_pid = getpid();
my_nl_request_msg.g.cmd = 1; //corresponds to DOC_EXMPL_C_ECHO;
nl_na = (struct nlattr *) GENLMSG_DATA(&my_nl_request_msg);
nl_na->nla_type = 2; // corresponds to DOC_EXMPL_A_MSG
//nl_na->nla_len = sizeof(MESSAGE_TO_KERNEL)+NLA_HDRLEN; //Message length
//memcpy(NLA_DATA(nl_na), MESSAGE_TO_KERNEL, sizeof(MESSAGE_TO_KERNEL));
p.dataSize = 32;
nl_na->nla_len = sizeof(p)+NLA_HDRLEN; //Message length
memcpy(NLA_DATA(nl_na), &p, sizeof(p));
my_nl_request_msg.n.nlmsg_len += NLMSG_ALIGN(nl_na->nla_len);
memset(&nl_address, 0, sizeof(nl_address));
nl_address.nl_family = AF_NETLINK;
//Send the custom message
nl_rxtx_length = sendto(nl_fd, &my_nl_request_msg,my_nl_request_msg.n.nlmsg_len, 0, (struct sockaddr *) &nl_address, sizeof(nl_address));
if (nl_rxtx_length != my_nl_request_msg.n.nlmsg_len) {
perror("sendto()");
close(nl_fd);
return -1;
}
printf("Sent to kernel: %d\n",p.dataSize);
//Receive reply from kernel
nl_rxtx_length = recv(nl_fd, &my_nl_response_msg, sizeof(my_nl_response_msg), 0);
if (nl_rxtx_length < 0) {
perror("recv()");
return -1;
}
//Validate response message
if (my_nl_response_msg.n.nlmsg_type == NLMSG_ERROR) { //Error
printf("Error while receiving reply from kernel: NACK Received\n");
close(nl_fd);
return -1;
}
if (nl_rxtx_length < 0) {
printf("Error while receiving reply from kernel\n");
close(nl_fd);
return -1;
}
if (!NLMSG_OK((&my_nl_response_msg.n), nl_rxtx_length)) {
printf("Error while receiving reply from kernel: Invalid Message\n");
close(nl_fd);
return -1;
}
//Parse the reply message
nl_rxtx_length = GENLMSG_PAYLOAD(&my_nl_response_msg.n);
nl_na = (struct nlattr *) GENLMSG_DATA(&my_nl_response_msg);
Response *r = (Response *)NLA_DATA(nl_na);
printf("Kernel replied: %d\n",r->dataSize);
//Step 5. Close the socket and quit
close(nl_fd);
return 0;
}
nl_kernel.c:
#include "global.h"
#include <net/genetlink.h>
#include <linux/module.h>
#include <linux/kernel.h>
/* attributes (variables):
* the index in this enum is used as a reference for the type,
* userspace application has to indicate the corresponding type
* the policy is used for security considerations
*/
enum {
DOC_EXMPL_A_UNSPEC,
DOC_EXMPL_A_MSG,
DOC_EXMPL_A_MSG2,
__DOC_EXMPL_A_MAX,
};
#define DOC_EXMPL_A_MAX (__DOC_EXMPL_A_MAX - 1)
static struct nla_policy doc_exmpl_genl_policy[DOC_EXMPL_A_MAX + 1] = {
[DOC_EXMPL_A_MSG] = { .type = NLA_NUL_STRING },
[DOC_EXMPL_A_MSG2] = { .type = NLA_UNSPEC, .len = sizeof(Response) },
};
#define VERSION_NR 1
//family definition
static struct genl_family doc_exmpl_gnl_family = {
.id = GENL_ID_GENERATE, //Genetlink should generate an id
.hdrsize = 0,
.name = "CONTROL_EXMPL",
.version = VERSION_NR, //Version number
.maxattr = DOC_EXMPL_A_MAX,
};
/* commands: enumeration of all commands (functions),
* used by userspace application to identify command to be executed
*/
enum {
DOC_EXMPL_C_UNSPEC,
DOC_EXMPL_C_ECHO,
__DOC_EXMPL_C_MAX,
};
#define DOC_EXMPL_C_MAX (__DOC_EXMPL_C_MAX - 1)
//An echo command, receives a message, prints it and sends another message back
int doc_exmpl_echo(struct sk_buff *skb_2, struct genl_info *info) {
struct nlattr *na;
struct sk_buff *skb;
int rc;
void *msg_head;
char * mydata;
Response *req;
if (info == NULL) {
goto out;
}
if (info->attrs[DOC_EXMPL_A_MSG]) {
na = info->attrs[DOC_EXMPL_A_MSG];
mydata = (char *)nla_data(na);
//req = (Response *)nla_data(na);
if (req == NULL) {
printk("error while receiving data\n");
} else {
//printk("received: %d\n", req->dataSize);
printk("received: %s\n", mydata);
}
}
else if(info->attrs[DOC_EXMPL_A_MSG2]){
na = info->attrs[DOC_EXMPL_A_MSG2];
req = (Response *)nla_data(na);
if (req == NULL) {
printk("error while receiving data\n");
} else {
printk("received2: %d\n", req->dataSize);
}
}else {
printk("no info->attrs %i - %i \n", DOC_EXMPL_A_MSG, DOC_EXMPL_A_MSG2);
}
//Send a message back
//Allocate some memory, since the size is not yet known use NLMSG_GOODSIZE
skb = genlmsg_new(NLMSG_GOODSIZE, GFP_KERNEL);
if (skb == NULL) {
goto out;
}
//Create the message headers
/* arguments of genlmsg_put:
struct sk_buff *,
int (sending) pid,
int sequence number,
struct genl_family *,
int flags,
u8 command index (why do we need this?)
*/
msg_head = genlmsg_put(skb, 0, info->snd_seq+1, &doc_exmpl_gnl_family, 0, DOC_EXMPL_C_ECHO);
if (msg_head == NULL) {
rc = -ENOMEM;
goto out;
}
req->dataSize *= 10;
printk("req %d\n", req->dataSize);
//Add a DOC_EXMPL_A_MSG attribute (actual value to be sent)
rc = nla_put(skb, DOC_EXMPL_A_MSG2, sizeof(req),req);
if (rc != 0) {
goto out;
}
//Finalize the message
genlmsg_end(skb, msg_head);
//Send the message back
rc = genlmsg_unicast(genl_info_net(info), skb,info->snd_pid );
if (rc != 0) {
goto out;
}
return 0;
out:
printk("An error occured in doc_exmpl_echo:\n");
return 0;
}
//Commands: mapping between the command enumeration and the actual function
struct genl_ops doc_exmpl_gnl_ops_echo = {
.cmd = DOC_EXMPL_C_ECHO,
.flags = 0,
.policy = doc_exmpl_genl_policy,
.doit = doc_exmpl_echo,
.dumpit = NULL,
};
static int __init gnKernel_init(void) {
int rc;
printk("Generic Netlink Example Module inserted.\n");
//Register the new family
rc = genl_register_family(&doc_exmpl_gnl_family);
if (rc != 0) {
goto failure;
}
//Register functions (commands) of the new family
rc = genl_register_ops(&doc_exmpl_gnl_family, &doc_exmpl_gnl_ops_echo);
if (rc != 0) {
printk("Register ops: %i\n",rc);
genl_unregister_family(&doc_exmpl_gnl_family);
goto failure;
}
return 0;
failure:
printk("An error occured while inserting the generic netlink example module\n");
return -1;
}
static void __exit gnKernel_exit(void) {
int ret;
printk("Generic Netlink Example Module unloaded.\n");
//Unregister the functions
ret = genl_unregister_ops(&doc_exmpl_gnl_family, &doc_exmpl_gnl_ops_echo);
if(ret != 0) {
printk("Unregister ops: %i\n",ret);
return;
}
//Unregister the family
ret = genl_unregister_family(&doc_exmpl_gnl_family);
if(ret !=0) {
printk("Unregister family %i\n",ret);
}
}
module_init(gnKernel_init);
module_exit(gnKernel_exit);
MODULE_LICENSE("GPL");
全局.h
#ifndef __GLOBAL_H
#define __GLOBAL_H
typedef struct _Response Response;
struct _Response
{
//int index;
int dataSize; /* ammount of data in bytes of the response */
char data[4096];
};
#endif
注意:我在 Oracle VB 中使用的是 Debian 2.6.32-5(这是必要的)。