2014-03-26 37 views
2

我正在尝试编写android的内核模块与用户空间服务进行通信,以便我的应用程序可以显示关于内核的一些信息。到目前为止,我已经尝试了几个我在网上找到的例子,但都没有成功。这个想法是为了让内核发送更新,然后服务才会接收它们。 ioctl不是解决方案,因为通信需要由事件驱动,并从内核启动。如果有人知道如何做到这一点,那么jni和内核模块的C代码将非常棒。如何使用netlink套接字进行内核 - 用户空间通信Android

+0

一个选择是使用netlink套接字。 http://stackoverflow.com/questions/3299386/how-to-use-netlink-socket-to-communicate-with-a-kernel-module –

+0

已经尝试使用非常相同的代码。在Ubuntu上它工作正常,但是当我在Andoid上尝试它时,它什么都不做。对于更糟糕的部分,它设法创建套接字并绑定(我没有收到错误消息),但是我没有收到内核应该在连接时打印并且也没有从netlink读取的消息。程序仅在读取时阻止... – TheMole

回答

0

我已成功使用netlink套接字在Android中的内核和用户空间之间进行通信。要在非阻塞模式下使用套接字,请使用fcntl和select调用。

+0

当我找到时间时,我会再试一次,因为我们已经改变了使用其他方法的方法。如果真的有效,那么我第一次可能没有做对。谢谢! – TheMole

1

这里是我使用的一个代码片段(不包括错误检查返回值的简洁性),其中netlink套接字通信由内核驱动(即内核是发送者,用户程序是接收者)。

内核模块

#include <net/sock.h> 
#include <net/netlink.h> 
#include <linux/skbuff.h> 
#include <string.h> 

#define MY_GROUP 1 

struct sock* socket; 
struct sk_buff* socket_buff; 

static void nl_receive_callback (struct sk_buff *skb)              
{                          
    nlmsg_free(skb);                     
}                          

static void kernel_send_nl_msg(void)                 
{                          
    struct nlmsghdr *nlsk_mh;                   
    char* msg = "hello kernel"; 

    socket = netlink_kernel_create(&init_net, NETLINK_USERSOCK, 1, nl_receive_callback, NULL, THIS_MODULE);                        

    socket_buff = nlmsg_new(256, GFP_KERNEL);                                                             
    nlsk_mh = nlmsg_put(socket_buff, 0, 0, NLMSG_DONE, strlen(msg), 0);      
    NETLINK_CB(socket_buff).pid = 0; // kernel pid             
    NETLINK_CB(socket_buff).dst_group = MY_GROUP;              
    strcpy(nlmsg_data(nlsk_mh), msg);                 

    nlmsg_multicast(socket, socket_buff, 0, MY_GROUP, GFP_KERNEL);      

    return;                       
} 

你会解雇这件事为你的内核事件,例如的结果以响应ioctl或在您的中断处理中。

用户程序

#include <sys/socket.h> 
#include <linux/netlink.h> 

#define MY_GROUP 1 

void user_recieve_nl_msg(void) 
{ 
    int sock_fd; 
    struct sockaddr_nl user_sockaddr; 
    struct nlmsghdr *nl_msghdr; 
    struct msghdr msghdr; 
    struct iovec iov; 

    char* kernel_msg; 

    sock_fd = socket(PF_NETLINK, SOCK_RAW, NETLINK_USERSOCK); 

    memset(&user_sockaddr, 0, sizeof(user_sockaddr)); 
    user_sockaddr.nl_family = AF_NETLINK; 
    user_sockaddr.nl_pid = getpid(); 
    user_sockaddr.nl_groups = MY_GROUP; 

    bind(sock_fd, (struct sockaddr*)&user_sockaddr, sizeof(user_sockaddr)); 
    while (1) { 
     nl_msghdr = (struct nlmsghdr*) malloc(NLMSG_SPACE(256)); 
     memset(nl_msghdr, 0, NLMSG_SPACE(256)); 

     iov.iov_base = (void*) nl_msghdr; 
     iov.iov_len = NLMSG_SPACE(256); 

     msghdr.msg_name = (void*) &user_sockaddr; 
     msghdr.msg_namelen = sizeof(user_sockaddr); 
     msghdr.msg_iov = &iov; 
     msghdr.msg_iovlen = 1; 

     recvmsg(sock_fd, &msghdr, 0); 

     kernel_msg = (char*)NLMSG_DATA(nl_msghdr); 
     print("Kernel message: %s\n", kernel_msg); // print to android logs 
    } 

    close(sock_fd); 
} 

我在内核nlmsg_multicast()的使用,而不是nlmsg_unicast(),因为做一个单播,内核模块需要知道你的用户进程的进程ID(所以你需要在那里添加一些额外的实现)。

希望有帮助!

相关问题