尝试在C中编写netlink套接字程序时出错

时间:2013-11-04 07:47:25

标签: c sockets netlink

首先,您可以从my previous post

找到我的情况详情

我在过去一天左右的时间里一直在尝试netlink socket程序。

我并没有试图让它听起来像我没有从我在互联网上找到的其他人的示例代码中获取所有这些代码,因为我们都知道我做到了这一点。

但我一直在尝试对代码进行反向工程(参见结束),经过数小时的调试后,我终于让它编译没有错误。但在尝试运行它之后,我收到消息“Segmentation Fault(Core Dumped)”。

现在我知道这个错误与尝试从无效的内存位置读取有关但我老实说不明白我拼接的一半代码所以如果有人能指出我哪里出错了我会很感激它

如果你能解释我在尝试通过netlink套接字发送消息时的正确/错误,我也非常感谢你。比如说,我会把信息放在哪里?因为我真的不知道我现在用我的代码发送什么消息。任何帮助表示赞赏!

#include <sys/socket.h>       /*  socket definitions        */
#include <sys/types.h>        /*  socket types              */
#include <linux/netlink.h>  /* netlink functions */
#include <sys/uio.h>        /* scatter-gather definition for iovec */
#include <stdlib.h>
#include <stdio.h>
#include <string.h>


/*  Global constants  */

#define ECHO_PORT          (2002)
#define MAX_LINE           (1000)

int socketHolder; // declare variable to hold Socket

// Set up configuration for destination which holds the socket to be used to communicate with the Kernel
struct sockaddr_nl dest; // declare variable to hold Socket Address
// Set up configuration for the netlink message header
struct nlmsghdr *netlinkHeader; // declare variable to hold message to be sent
// Set up configuration for the scatter-gather function for expediting writing to buffer
struct iovec iov;
// Set up configuration for the message header for simplifying message parameters to be sent
struct msghdr message;

int main()
{

memset(&dest, 0, sizeof(dest)); // set socket Address to all 0
dest.nl_family = AF_NETLINK; // set socket family to AF_NETLINK
dest.nl_pad = 0; // defaults to 0
dest.nl_pid = 0; // set address of netlink socket to 0 because we're going to the kernel
dest.nl_groups = 0; // set to 0 because we're doing unicast

netlinkHeader->nlmsg_pid = 0; // set sender port number to 0 because we're in kernel

iov.iov_base = (caddr_t)netlinkHeader; // set address of buffer
iov.iov_len = netlinkHeader->nlmsg_len; // set length of buffer

message.msg_name = (caddr_t) &dest; // set socket name
message.msg_namelen = sizeof(dest); // set length of name
message.msg_iov = &iov; // I have no idea what this is
message.msg_iovlen = 1; // number of iov blocks
message.msg_control = NULL;  // ignore
message.msg_controllen = 0;  // ignore
message.msg_flags = 0;  //ignore

socketHolder = socket(AF_NETLINK, SOCK_RAW, NETLINK_GENERIC); // declare Socket and assign to holder variable

bind(socketHolder, (struct sockaddr *) &dest, sizeof(dest)); // bind the destination settings to socketHolder

int sender = sendmsg(socketHolder, &message, 0); // send the message

}

2 个答案:

答案 0 :(得分:1)

不幸的是,我对netlink不太熟悉。但是你的错误很可能会在这里:

netlinkHeader->nlmsg_pid = 0; // set sender port number to 0 because we're in kernel

因为你从未初始化netlinkHeader,即确保指针指向允许读写的内存,这是未定义的行为。实际上,由于您正在访问某个随机地址,并且您的操作系统会阻止您的进程破坏事物,因此通常会导致分段错误。

不知道为什么这是一个指针,你应该尝试将它作为struct nlmsghdr类型的实际实例,因为它听起来好像你想操纵地址。

此外,注释暗示代码来自内核代码,而您的代码不是内核代码,这可能意味着还存在其他问题。

答案 1 :(得分:1)

如果您确定导致分段错误的行会有所帮助,但是从您显示的代码中看起来似乎是下面的行。

netlinkHeader->nlmsg_pid = 0;

您尚未初始化netlinkHeader指针指向某个有效的内存位置,因此当您尝试将值0写入属性netlinkHeader->nlmsg_pid时,它会导致您看到的错误