- 論壇徽章:
- 0
|
平時(shí)網(wǎng)絡(luò)部分的東西碰的多些,這塊一開始還真不知道怎么寫,因?yàn)榭隙ê驮谟脩艨臻g下是不同的。google過(guò)后,得到以下答案。一般可以用兩種方法:第一種是用系統(tǒng)調(diào)用。第二種方法是filp->open()等函數(shù)。下面分別來(lái)說(shuō)下這兩種方法。
1 利用系統(tǒng)調(diào)用:
sys_open,sys_write,sys_read等。
其實(shí)分析過(guò)sys_open可以知道,最后調(diào)用的也是filp->open。
sys_open ==> do_sys_open ==> filp->open
在linuxsir上的一個(gè)帖子,上面一個(gè)版主說(shuō):sys_open和進(jìn)程緊密相關(guān),往往不在內(nèi)核中使用。
而其實(shí)sys_open最后也是調(diào)用了filp->open。
其實(shí)好像Linux2.6.20后面就不推薦使用sys_open,那我們這里就就后者進(jìn)行詳細(xì)的介紹
2 filp->open等函數(shù)。
在模塊中,用戶空間的open,read,write,llseek等函數(shù)都是不可以使用的。應(yīng)該使用其在內(nèi)核中對(duì)應(yīng)的函數(shù)?梢允褂胒ilp->open配合struct file里的read/write來(lái)進(jìn)行對(duì)文件的讀寫操作。
例子1:
- #include <linux/kernel.h>
- #include <linux/module.h>
- #include <linux/fs.h>
- #include <asm/uaccess.h>
- #include <linux/mm.h>
- MODULE_AUTHOR("Kenthy@163.com.");
- MODULE_DESCRIPTION("Kernel study and test.");
- void fileread(const char * filename)
- {
- struct file *filp;
- struct inode *inode;
- mm_segment_t fs;
- off_t fsize;
- char *buf;
- unsigned long magic;
- printk("<1>start....\n");
- filp=filp_open(filename,O_RDONLY,0);
- inode=filp->f_dentry->d_inode;
-
- magic=inode->i_sb->s_magic;
- printk("<1>file system magic:%li \n",magic);
- printk("<1>super blocksize:%li \n",inode->i_sb->s_blocksize);
- printk("<1>inode %li \n",inode->i_ino);
- fsize=inode->i_size;
- printk("<1>file size:%i \n",(int)fsize);
- buf=(char *) kmalloc(fsize+1,GFP_ATOMIC);
- fs=get_fs();
- set_fs(KERNEL_DS);
- filp->f_op->read(filp,buf,fsize,&(filp->f_pos));
- set_fs(fs);
- buf[fsize]='\0';
- printk("<1>The File Content is:\n");
- printk("<1>%s",buf);
- filp_close(filp,NULL);
- }
- void filewrite(char* filename, char* data)
- {
- struct file *filp;
- mm_segment_t fs;
- filp = filp_open(filename, O_RDWR|O_APPEND, 0644);
- if(IS_ERR(filp))
- {
- printk("open error...\n");
- return;
- }
- fs=get_fs();
- set_fs(KERNEL_DS);
- filp->f_op->write(filp, data, strlen(data),&filp->f_pos);
- set_fs(fs);
- filp_close(filp,NULL);
- }
- int init_module()
- {
- char *filename="/root/test1.c";
- printk("<1>Read File from Kernel.\n");
- fileread(filename);
- filewrite(filename, "kernel write test\n");
- return 0;
- }
- void cleanup_module()
- {
- printk("<1>Good,Bye!\n");
- }
復(fù)制代碼
eg2:
- #include<linux/module.h>
- #include<linux/kernel.h>
- #include<linux/init.h>
- #include<linux/types.h>
- #include<linux/fs.h>
- #include<linux/string.h>
- #include<asm/uaccess.h> /* get_fs(),set_fs(),get_ds() */
- #define FILE_DIR "/root/test.txt"
- MODULE_LICENSE("GPL");
- MODULE_AUTHOR("kenthy@163.com");
- char *buff = "module read/write test";
- char tmp[100];
- static struct file *filp = NULL;
- static int __init wr_test_init(void)
- {
- mm_segment_t old_fs;
- ssize_t ret;
-
- filp = filp_open(FILE_DIR, O_RDWR | O_CREAT, 0644);
-
- // if(!filp)
- if(IS_ERR(filp))
- printk("open error...\n");
-
- old_fs = get_fs();
- set_fs(get_ds());
-
- filp->f_op->write(filp, buff, strlen(buff), &filp->f_pos);
-
- filp->f_op->llseek(filp,0,0);
- ret = filp->f_op->read(filp, tmp, strlen(buff), &filp->f_pos);
-
- set_fs(old_fs);
-
- if(ret > 0)
- printk("%s\n",tmp);
- else if(ret == 0)
- printk("read nothing.............\n");
- else
- {
- printk("read error\n");
- return -1;
- }
- return 0;
- }
- static void __exit wr_test_exit(void)
- {
- if(filp)
- filp_close(filp,NULL);
- }
- module_init(wr_test_init);
- module_exit(wr_test_exit);
復(fù)制代碼
3.Makefile
- obj-m := os_attack.o
- KDIR := /lib/modules/$(uname -r)/build/
- PWD := $(shell pwd)
- all:module
- module:
- $(MAKE) -C $(KDIR) M=$(PWD) modules
- clean:
- rm -rf *.ko *.mod.c *.o Module.* modules.* .*.cmd .tmp_versions
復(fù)制代碼
注意:
在調(diào)用filp->f_op->read和filp->f_op->write等對(duì)文件的操作之前,應(yīng)該先設(shè)置FS。
默認(rèn)情況下,filp->f_op->read或者filp->f_op->write會(huì)對(duì)傳進(jìn)來(lái)的參數(shù)buff進(jìn)行指針檢查。如果不是在用戶空間會(huì)拒絕訪問(wèn)。因?yàn)槭窃趦?nèi)核模塊中,所以buff肯定不在用戶空間,所以要增大其尋址范圍。
拿filp->f_op->write為例來(lái)說(shuō)明:
filp->f_op->write最終會(huì)調(diào)用access_ok ==> range_ok.
而range_ok會(huì)判斷訪問(wèn)的地址是否在0 ~ addr_limit之間。如果在,則ok,繼續(xù)。如果不在,則禁止訪問(wèn)。而內(nèi)核空間傳過(guò)來(lái)的buff肯定大于addr_limit。所以要set_fs(get_ds())。
這些函數(shù)在asm/uaccess.h中定義。以下是這個(gè)頭文件中的部分內(nèi)容:
#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的值。這樣就使得從模塊中傳遞進(jìn)去的參數(shù)也可以正常使用了。
在寫測(cè)試模塊的時(shí)候,要實(shí)現(xiàn)的功能是寫進(jìn)去什么,然后讀出來(lái)放在tmp數(shù)組中。但寫完了以后filp->f_ops已經(jīng)在末尾了,這個(gè)時(shí)候讀是什么也讀不到的,如果想要讀到數(shù)據(jù),則應(yīng)該改變filp->f-ops的值,這就要用到filp->f_op->llseek函數(shù)了。上網(wǎng)查了下,其中的參數(shù)需要記下筆記:
系統(tǒng)調(diào)用:
off_t sys_lseek(unsigned int fd, off_t offset, unsigned int origin)
offset是偏移量。
若origin是SEEK_SET(0),則將該文件的位移量設(shè)置為距文件開始處offset 個(gè)字節(jié)。
若origin是SEEK_CUR(1),則將該文件的位移量設(shè)置為其當(dāng)前值加offset, offset可為正或負(fù)。
若origin是SEEK_END(2),則將該文件的位移量設(shè)置為文件長(zhǎng)度加offset, offset可為正或負(fù)。
ok,that's all. |
評(píng)分
-
查看全部評(píng)分
|