如何在用户空间中使用NETLINK_KOBJECT_UEVENT协议?

时间:2017-06-20 20:42:56

标签: linux-kernel netlink

让我们考虑一下这个示例代码:

#include <linux/netlink.h>
#include <sys/socket.h>
#include <string.h>
#include <unistd.h>
#include <stdio.h>

#define BUF_SIZE 4096

int main() {
    int fd, res;
    unsigned int i, len;
    char buf[BUF_SIZE];
    struct sockaddr_nl nls;

    fd = socket(PF_NETLINK, SOCK_RAW, NETLINK_KOBJECT_UEVENT);
    if (fd == -1) {
        return 1;
    }

    memset(&nls, 0, sizeof(nls));
    nls.nl_family = AF_NETLINK;
    nls.nl_pid = getpid();
    nls.nl_groups = 1;

    res = bind(fd, (struct sockaddr *)&nls, sizeof(nls));
    if (res == -1) {
        return 2;
    }

    while (1) {
        len = recv(fd, buf, sizeof(buf), 0);

        printf("============== Received %d bytes\n", len);
        for (i = 0; i < len; ++i) {
            if (buf[i] == 0) {
                printf("[0x00]\n");
            } else if (buf[i] < 33 || buf[i] > 126) {
                printf("[0x%02hhx]", buf[i]);
            } else {
                printf("%c", buf[i]);
            }
        }
        printf("<END>\n");
    }

    close(fd);
    return 0;
}

它在netlink套接字上侦听与hotplug相关的事件。基本上,它的工作原理。然而,即使在谷歌上搜索整个晚上,阅读不同的文档和手册以及通过示例工作之后,我也不清楚某些部分。

基本上,我有两个问题。

  1. sockaddr_nl.nl_groups的不同值是什么意思?至少对于NETLINK_KOBJECT_UEVENT协议。
  2. 如果为邮件分配的缓冲区太小,则会简单地截断该邮件(您可以使用BUF_SIZE大小来查看该邮件)。这个缓冲区大小应该是什么,不丢失任何数据?是否有可能知道传入消息的用户空间长度以分配足够的空间?
  3. 我希望直接答案作为对内核代码的引用。

1 个答案:

答案 0 :(得分:1)

  1. 这些值表示不同的多播组。一个netlink套接字可以具有31个不同的多播组(0表示单播),可以将多播消息发送到该组。对于NETLINK_KOBJECT_UEVENT,它似乎固定为1 see f.ex. here

  2. 您应该能够使用level设置为SOL_SOCKEToptname设置为SO_RCVBUF的{​​{3}}。