内核模块中对文件的读写

平时网络部分的东西碰的多些,这块一开始还真不知道怎么写,因为肯定和在用户空间下是不同的。google过后,得到以下答案。一般可以用两种方法:第一种是用系统调用。第二种方法是filp->open()等函数。下面分别来说下这两种方法。
http://blog.csdn.net/stephen_yin/article/details/6574353

1 利用系统调用:
sys_open,sys_write,sys_read等。
其实分析过sys_open可以知道,最后调用的也是filp->open。
sys_open ==> do_sys_open ==> filp->open
Linuxsir上的一个帖子,上面一个版主说:sys_open和进程紧密相关,往往不在内核中使用。
而其实sys_open最后也是调用了filp->open。
其实好像Linux2.6.20后面就不推荐使用sys_open,那我们这里就就后者进行详细的介绍

2 filp->open等函数。
在模块中,用户空间的open,read,write,llseek等函数都是不可以使用的。应该使用其在内核中对应的函数。可以使用filp->open配合struct file里的read/write来进行对文件的读写操作。

 

例子1:

  1. #include <linux/kernel.h>   
  2. #include <linux/module.h>   
  3. #include <linux/fs.h>   
  4. #include <asm/uaccess.h>   
  5. #include <linux/mm.h>   
  6. MODULE_AUTHOR("Kenthy@163.com.");   
  7. MODULE_DESCRIPTION("Kernel study and test.");   
  8.   
  9. void fileread(const char * filename)   
  10. {   
  11.   struct file *filp;   
  12.   struct inode *inode;   
  13.   mm_segment_t fs;   
  14.   off_t fsize;   
  15.   char *buf;   
  16.   unsigned long magic;   
  17.   printk("<1>start..../n");   
  18.   filp=filp_open(filename,O_RDONLY,0);   
  19.   inode=filp->f_dentry->d_inode;    
  20.     
  21.   magic=inode->i_sb->s_magic;   
  22.   printk("<1>file system magic:%li /n",magic);   
  23.   printk("<1>super blocksize:%li /n",inode->i_sb->s_blocksize);   
  24.   printk("<1>inode %li /n",inode->i_ino);   
  25.   fsize=inode->i_size;   
  26.   printk("<1>file size:%i /n",(int)fsize);   
  27.   buf=(char *) kmalloc(fsize+1,GFP_ATOMIC);   
  28.   fs=get_fs();   
  29.   set_fs(KERNEL_DS);   
  30.   filp->f_op->read(filp,buf,fsize,&(filp->f_pos));   
  31.   set_fs(fs);   
  32.   buf[fsize]='/0';   
  33.   printk("<1>The File Content is:/n");   
  34.   printk("<1>%s",buf);   
  35.   
  36.   filp_close(filp,NULL);   
  37. }   
  38. void filewrite(char* filename, char* data)  
  39. {  
  40.   struct file *filp;   
  41.  mm_segment_t fs;  
  42.  filp = filp_open(filename, O_RDWR|O_APPEND, 0644);  
  43.  if(IS_ERR(filp))  
  44.     {  
  45.       printk("open error.../n");   
  46.       return;  
  47.         }     
  48.   fs=get_fs();  
  49.   set_fs(KERNEL_DS);  
  50.   filp->f_op->write(filp, data, strlen(data),&filp->f_pos);  
  51.   set_fs(fs);  
  52.   filp_close(filp,NULL);  
  53. }  
  54. int init_module()   
  55. {   
  56.   char *filename="/root/test1.c";   
  57.   printk("<1>Read File from Kernel./n");   
  58.   fileread(filename);   
  59.   filewrite(filename, "kernel write test/n");  
  60.   return 0;   
  61. }   
  62. void cleanup_module()   
  63. {   
  64.   printk("<1>Good,Bye!/n");   
  65. }   
 

例子2

  1. #include<linux/module.h>  
  2. #include<linux/kernel.h>  
  3. #include<linux/init.h>  
  4. #include<linux/types.h>  
  5. #include<linux/fs.h>  
  6. #include<linux/string.h>  
  7. #include<asm/uaccess.h> /* get_fs(),set_fs(),get_ds() */  
  8. #define FILE_DIR "/root/test.txt"  
  9. MODULE_LICENSE("GPL");  
  10. MODULE_AUTHOR("kenthy@163.com");  
  11. char *buff = "module read/write test";  
  12. char tmp[100];  
  13. static struct file *filp = NULL;  
  14. static int __init wr_test_init(void)  
  15. {  
  16.     mm_segment_t old_fs;  
  17.     ssize_t ret;  
  18.       
  19.     filp = filp_open(FILE_DIR, O_RDWR | O_CREAT, 0644);  
  20.       
  21.     //    if(!filp)  
  22.     if(IS_ERR(filp))  
  23.         printk("open error.../n");  
  24.       
  25.     old_fs = get_fs();  
  26.     set_fs(get_ds());  
  27.       
  28.     filp->f_op->write(filp, buff, strlen(buff), &filp->f_pos);  
  29.       
  30.     filp->f_op->llseek(filp,0,0);  
  31.     ret = filp->f_op->read(filp, tmp, strlen(buff), &filp->f_pos);  
  32.       
  33.     set_fs(old_fs);  
  34.       
  35.     if(ret > 0)  
  36.         printk("%s/n",tmp);  
  37.     else if(ret == 0)  
  38.         printk("read nothing............./n");  
  39.     else   
  40.         {  
  41.             printk("read error/n");  
  42.             return -1;  
  43.         }  
  44.     return 0;  
  45. }  
  46. static void __exit wr_test_exit(void)  
  47. {  
  48.     if(filp)  
  49.         filp_close(filp,NULL);  
  50. }  
  51. module_init(wr_test_init);  
  52. module_exit(wr_test_exit);  
 

3.Makefile

  1. obj-m := os_attack.o  
  2. KDIR := /lib/modules/$(uname -r)/build/  
  3. PWD := $(shell pwd)  
  4. all:module  
  5. module:  
  6.         $(MAKE) -C $(KDIR) M=$(PWD) modules  
  7.   
  8. clean:  
  9.         rm -rf *.ko *.mod.c *.o Module.* modules.* .*.cmd .tmp_versions  
 

 

注意:
在调用filp->f_op->read和filp->f_op->write等对文件的操作之前,应该先设置FS。
默认情况下,filp->f_op->read或者filp->f_op->write会对传进来的参数buff进行指针检查。如果不是在用户空间会拒绝访问。因为是在内核模块中,所以buff肯定不在用户空间,所以要增大其寻址范围。

拿filp->f_op->write为例来说明:
filp->f_op->write最终会调用access_ok ==> range_ok.
而range_ok会判断访问的地址是否在0 ~ addr_limit之间。如果在,则ok,继续。如果不在,则禁止访问。而内核空间传过来的buff肯定大于addr_limit。所以要set_fs(get_ds())。
这些函数在asm/uaccess.h中定义。以下是这个头文件中的部分内容:

#define MAKE_MM_SEG(s)    ((mm_segment_t) { (s) })

#define KERNEL_DS    MAKE_MM_SEG(-1UL)
#define USER_DS        MAKE_MM_SEG(PAGE_OFFSET)

#define get_ds()    (KERNEL_DS)
#define get_fs()    (current_thread_info()->addr_limit)
#define set_fs(x)    (current_thread_info()->addr_limit = (x))

#define segment_eq(a, b)    ((a).seg == (b).seg)


可以看到set_fs(get_ds())改变了addr_limit的值。这样就使得从模块中传递进去的参数也可以正常使用了。

在写测试模块的时候,要实现的功能是写进去什么,然后读出来放在tmp数组中。但写完了以后filp->f_ops已经在末尾了,这个时候读是什么也读不到的,如果想要读到数据,则应该改变filp->f-ops的值,这就要用到filp->f_op->llseek函数了。上网查了下,其中的参数需要记下笔记:


系统调用:
off_t sys_lseek(unsigned int fd, off_t offset, unsigned int origin)
offset是偏移量。
若origin是SEEK_SET(0),则将该文件的位移量设置为距文件开始处offset 个字节。
若origin是SEEK_CUR(1),则将该文件的位移量设置为其当前值加offset, offset可为正或负。
若origin是SEEK_END(2),则将该文件的位移量设置为文件长度加offset, offset可为正或负。

ok,that's all.

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值