Chinaunix

标题: 驱动模块中如何才能调用socket族函数sys_sendto? [打印本页]

作者: shaohui973    时间: 2013-07-01 18:34
标题: 驱动模块中如何才能调用socket族函数sys_sendto?
写了一个程序,模块中使用内核定时器,时间到了就把用户注册上来的数据往指定的socket发送。
在加载时,提示:unkown symbol sys_sendto

早听说,内核不导出所有的符号给模块使用了;今天遇到了...  

这个要咋整呢?
作者: 瀚海书香    时间: 2013-07-01 19:54
回复 1# shaohui973
写了一个程序,模块中使用内核定时器,时间到了就把用户注册上来的数据往指定的socket发送。
在加载时,提示:unkown symbol sys_sendto

早听说,内核不导出所有的符号给模块使用了;今天遇到了...  


系统调用是给应用层用的,内核发送数据包使用ip_send_skb等类似的函数,或者通过NF_HOOK将数据包包放到NF_INET_LOCAL_OUT链上。
作者: 瀚海书香    时间: 2013-07-01 20:27
回复 1# shaohui973
刚才看了下之前的代码,里面用的是kernel_sendmsg函数

   
作者: shaohui973    时间: 2013-07-01 22:33
我的是raw socket,最后调用的是raw_sendmsg,但是不管怎么样,里面的函数的符号都不是到处给内核模块用的,我insmod的话还是会失败的呀。况且,有些函数根本就没有头文件去申明。

如果直接写成编译连接进内核的模块,然后直接调用系统调用sys_sendto会不会就可以了呢?
作者: 帅绝人寰    时间: 2013-07-02 09:45
连试都不试试?


我刚写过一个tcp server的demo code, 一点问题都没有
作者: shaohui973    时间: 2013-07-02 09:57
回复 5# 帅绝人寰


    你怎么操作的呢?
我正在弄将模块编译进内核...
作者: 帅绝人寰    时间: 2013-07-02 11:01
告诉你一绝招: 看kenrel code本身。

例如, ocfs2是一个集群文件系统, 需要用socket连接、发包, 它又可以build为lkm
作者: 瀚海书香    时间: 2013-07-02 12:55
回复 4# shaohui973
看一下这个链接http://blog.chinaunix.net/uid-20662820-id-3784251.html 在内核开启一个线程定时发送数据包。。。

   
作者: shaohui973    时间: 2013-07-02 15:43
回复 8# 瀚海书香
这种情况不适合我,我的socket是用户空间的程序创建的,socketfd也是由用户空间传递过来的。这样的话,内核模块根据fd来查找sock,sk之类的函数引用不了吧。

   
作者: shaohui973    时间: 2013-07-02 15:45
回复 7# 帅绝人寰


    这个需要时间消化这一块代码。我还没看到这里,之前看的是从上电到中断初始化完成(X86).


作者: shaohui973    时间: 2013-07-02 15:46
我把驱动当成内核的一部份编进去了,起来后/dev/下的也创建了,这个应该可行吧
作者: 瀚海书香    时间: 2013-07-02 16:28
回复 9# shaohui973
这种情况不适合我,我的socket是用户空间的程序创建的,socketfd也是由用户空间传递过来的。这样的话,内核模块根据fd来查找sock,sk之类的函数引用不了吧。


应该可行。
方案:
   1. 用户态传入内核态进程的PID和文件fd.
   2. 内核态模块根据PID获取到对应的task_struct
   3. 根据task_struct和fd,获取到socket
               下面是一个demo代码
               struct socket *sockfd_lookup_task(int fd, struct task_struct *task, int *err)
               {
                        struct file *file;
                        struct socket *sock;
                        struct files_struct *files = task->files;
                        rcu_read_lock();
                        file = fcheck_file(files, fd);
                        if (file) {
                                if (file->f_mode & FMODE_PATH || !atomic_long_inc_not_zero(&file->f_count))
                                          file = NULL;
                        }
                        rcu_read_unlock();
                        if (!file)
                                return -1;
                        sock = sock_from_file(file, err);
                        if (!sock)
                               fput(file);
                        return sock;
               }
   4. 调用kernel_sendmsg向socket发送数据包
   
作者: shaohui973    时间: 2013-07-03 15:30
回复 12# 瀚海书香


    谢谢版主,我试下。
作者: 瀚海书香    时间: 2013-07-03 17:28
回复 13# shaohui973
今天抽空写了个代码验证了一下,是可行的。只要应用层创建socket,bind,connect。然后将pid和fd传入内核模块就可以了。

   
作者: 瀚海书香    时间: 2013-07-04 11:26
回复 1# shaohui973

看我的博文http://blog.chinaunix.net/uid-20662820-id-3787086.html
   




欢迎光临 Chinaunix (http://bbs.chinaunix.net/) Powered by Discuz! X3.2