2016-11-24 22 views
2

USBデバイスが(dis)接続しているときに、netlinkソケット経由でカーネルからueventを受信しようとしています。私は、同じ仕事をしてPythonスクリプトを持っており、それは動作しますが、私はCで同じ機能を必要とするこれまでのところ、私はこれを持っている:Netlinkソケットでカーネルのイベントを受け取る方法は?

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

#define NL_MAX_PAYLOAD 8192 

int main() { 
    int nl_socket; 
    struct sockaddr_nl src_addr, dest_addr; 
    struct nlmsghdr* nlh; 
    struct msghdr msg; 
    struct iovec iov; 

    // Prepare source address 
    memset(&src_addr, 0, sizeof(src_addr)); 
    src_addr.nl_family = AF_NETLINK; 
    src_addr.nl_pid = getpid(); 

    // Prepare destination address 
    memset(&dest_addr, 0, sizeof(dest_addr)); 
    dest_addr.nl_family = AF_NETLINK; 
    dest_addr.nl_pid = 0; 
    dest_addr.nl_groups = 0; 

    // Prepare netlink message 
    nlh = (struct nlmsghdr*) malloc(NLMSG_SPACE(NL_MAX_PAYLOAD)); 
    memset(nlh, 0, NLMSG_SPACE(NL_MAX_PAYLOAD)); 
    nlh->nlmsg_len = NLMSG_SPACE(NL_MAX_PAYLOAD); 
    nlh->nlmsg_pid = getpid(); 
    nlh->nlmsg_flags = 0; 
    iov.iov_base = (void *) nlh; 
    iov.iov_len = nlh->nlmsg_len; 
    msg.msg_iov = &iov; 
    msg.msg_iovlen = 1; 

    nl_socket = socket(AF_NETLINK, SOCK_DGRAM,  NETLINK_KOBJECT_UEVENT); 
    if (nl_socket < 0) { 
     printf("Failed to create socket for DeviceFinder"); 
     exit(1); 
    } 

    bind(nl_socket, (struct sockaddr*) &src_addr, sizeof(src_addr)); 

    while (1) { 
     int r = recvmsg(nl_socket, &msg, 0); 
     printf("%i\n", r); 
     if (r < 0) { 
      perror(""); 
     } 
    } 
} 

これは小さな、作業例を作成するには、コンテキストから取り出されます。

問題は、私はただ何も受け取っていないということです。私は何のエラーも出ません。recvmsgの呼び出しは、私がデバイスをプラグインすると決して戻ってこないのです。どうして?代わりに、私はRECVを使用していますrecvmmsgを使用するので、私はプログラムを少し変更した

答えて

1

受光部が誤っていた、次はueventsを出力するコードです:

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

#define NL_MAX_PAYLOAD 8192 

int main() { 
    int nl_socket; 
    struct sockaddr_nl src_addr; 
    char msg[NL_MAX_PAYLOAD]; 
    int ret; 

    // Prepare source address 
    memset(&src_addr, 0, sizeof(src_addr)); 
    src_addr.nl_family = AF_NETLINK; 
    src_addr.nl_pid = getpid(); 
    src_addr.nl_groups = -1; 

    nl_socket = socket(AF_NETLINK, (SOCK_DGRAM | SOCK_CLOEXEC),  NETLINK_KOBJECT_UEVENT); 
    if (nl_socket < 0) { 
     printf("Failed to create socket for DeviceFinder"); 
     exit(1); 
    } 

    ret = bind(nl_socket, (struct sockaddr*) &src_addr, sizeof(src_addr)); 
    if (ret) { 
     printf("Failed to bind netlink socket.."); 
     close(nl_socket); 
     return 1; 
    } 

    printf("Waiting for events now...\n"); 
    while (1) { 
     int r = recv(nl_socket, msg, sizeof(msg), MSG_DONTWAIT); 
     if (r == -1) 
      continue; 
     if (r < 0) { 
      perror(""); 
      continue; 
     } 
     printf("length:%i\n msg:%s", r, msg); 

    } 
} 
+0

ありがとう、何らかの理由で多くのメッセージが空ですが、これは完全に機能します。私が必要なものは大丈夫です。 –

0

のrecvよりも、この優れた使用のrecvmsg(ただ接続リンク用).ref man recvmsg。 と成功に近いプログラム。

src_addr.nl_family = AF_NETLINK; 
src_addr.nl_pid = getpid(); 
src_addr.nl_groups = -1; <--add this so can receive from kernel broadcast 
関連する問題