2009-07-26 234 views
73

我知道关于为什么不应该从内核读取/写入文件的所有讨论,而是如何使用/ proc或netlink来做到这一点。无论如何,我想读/写。我也读过 Driving Me Nuts - Things You Never Should Do in the Kernel如何在Linux内核模块中读取/写入文件?

但是,问题是2.6.30不导出sys_read()。而是它包裹在SYSCALL_DEFINE3。所以,如果我在我的模块中使用,我得到以下警告:

WARNING: "sys_read" [xxx.ko] undefined! 
WARNING: "sys_open" [xxx.ko] undefined! 

显然insmod无法加载模块,因为连接不正确地发生。

问题:

  • 如何阅读2.6.22(其中不出口sys_read()/sys_open())后的内核中/写?
  • 一般来说,如何在内核中使用包含在宏SYSCALL_DEFINEn()中的系统调用?

回答

94

你应该知道,你应该尽可能地避免文件I/O。其主要思路是走“更深层次”,并呼吁VFS level functions,而不是直接在系统调用处理程序:

包括:

#include <linux/fs.h> 
#include <asm/segment.h> 
#include <asm/uaccess.h> 
#include <linux/buffer_head.h> 

开幕(类似打开文件)的文件:

struct file *file_open(const char *path, int flags, int rights) 
{ 
    struct file *filp = NULL; 
    mm_segment_t oldfs; 
    int err = 0; 

    oldfs = get_fs(); 
    set_fs(get_ds()); 
    filp = filp_open(path, flags, rights); 
    set_fs(oldfs); 
    if (IS_ERR(filp)) { 
     err = PTR_ERR(filp); 
     return NULL; 
    } 
    return filp; 
} 

关闭文件(类似于关闭):

void file_close(struct file *file) 
{ 
    filp_close(file, NULL); 
} 

从文件中读取数据(与prea类似d):

int file_read(struct file *file, unsigned long long offset, unsigned char *data, unsigned int size) 
{ 
    mm_segment_t oldfs; 
    int ret; 

    oldfs = get_fs(); 
    set_fs(get_ds()); 

    ret = vfs_read(file, data, size, &offset); 

    set_fs(oldfs); 
    return ret; 
} 

将数据写入到文件中(类似于PWRITE):

int file_write(struct file *file, unsigned long long offset, unsigned char *data, unsigned int size) 
{ 
    mm_segment_t oldfs; 
    int ret; 

    oldfs = get_fs(); 
    set_fs(get_ds()); 

    ret = vfs_write(file, data, size, &offset); 

    set_fs(oldfs); 
    return ret; 
} 

同步功能变化(类似于FSYNC)的文件:

int file_sync(struct file *file) 
{ 
    vfs_fsync(file, 0); 
    return 0; 
} 

[编辑]最初,我建议使用file_fsync,它在更新的内核版本中消失了。感谢这位可怜的人提出改变,但他们的改变被拒绝了。编辑在我审查之前被拒绝。

+2

谢谢。我正在考虑通过复制sys_read/sys_open功能来做类似的事情。但这是很大的帮助。好奇心,有什么方法可以使用SYSCALL_DEFINE声明的系统调用吗? – Methos 2009-07-26 12:48:32