คุณควรทราบว่าคุณควรหลีกเลี่ยงไฟล์ I / O จากภายในเคอร์เนล Linux เมื่อเป็นไปได้ แนวคิดหลักคือไป "ลึกระดับหนึ่ง" และเรียกใช้ฟังก์ชันระดับ VFSแทนตัวจัดการ syscall โดยตรง:
รวมถึง:
#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);
}
การอ่านข้อมูลจากไฟล์ (คล้ายกับ pread):
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 ซึ่งหายไปในเวอร์ชันเคอร์เนลที่ใหม่กว่า ต้องขอบคุณชายผู้น่าสงสารที่แนะนำการเปลี่ยนแปลง แต่การเปลี่ยนแปลงถูกปฏิเสธ การแก้ไขถูกปฏิเสธก่อนที่ฉันจะตรวจสอบได้