file operations中各項解析

2021-06-29 15:19:07 字數 3867 閱讀 3090

linux裝置驅動中file_operations結構體分析

struct module *owner

第乙個 file_operations 成員根本不是乙個操作; 它是乙個指向擁有這個結構的模組的指標. 這個成員用來在它的操作還在被使用時阻止模組被解除安裝. 幾乎所有時間中, 它被簡單初始化為 this_module, 乙個在 中定義的巨集.

loff_t (*llseek) (struct file *, loff_t, int);

llseek 方法用作改變檔案中的當前讀/寫位置, 並且新位置作為(正的)返回值. loff_t 引數是乙個"long offset", 並且就算在 32位平台上也至少 64 位寬. 錯誤由乙個負返回值指示. 如果這個函式指標是 null, seek 呼叫會以潛在地無法預知的方式修改 file 結構中的位置計數器( 在"file 結構" 一節中描述).

ssize_t (*read) (struct file *, char __user *, size_t, loff_t *);

用來從裝置中獲取資料. 在這個位置的乙個空指標導致 read 系統呼叫以 -einval("invalid argument") 失敗. 乙個非負返回值代表了成功讀取的位元組數( 返回值是乙個 "signed size" 型別, 常常是目標平台本地的整數型別).

ssize_t (*aio_read)(struct kiocb *, char __user *, size_t, loff_t);

初始化乙個非同步讀 -- 可能在函式返回前不結束的讀操作. 如果這個方法是 null, 所有的操作會由 read 代替進行(同步地).

ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *);

傳送資料給裝置. 如果 null, -einval 返回給呼叫 write 系統呼叫的程式. 如果非負, 返回值代表成功寫的位元組數.

ssize_t (*aio_write)(struct kiocb *, const char __user *, size_t, loff_t *);

初始化裝置上的乙個非同步寫.

int (*readdir) (struct file *, void *, filldir_t);

對於裝置檔案這個成員應當為 null; 它用來讀取目錄, 並且僅對檔案系統有用.

unsigned int (*poll) (struct file *, struct poll_table_struct *);

poll 方法是 3 個系統呼叫的後端: poll, epoll, 和 select, 都用作查詢對乙個或多個檔案描述符的讀或寫是否會阻塞. poll 方法應當返回乙個位掩碼指示是否非阻塞的讀或寫是可能的, 並且, 可能地, 提供給核心資訊用來使呼叫程序睡眠直到 i/o 變為可能. 如果乙個驅動的 poll 方法為 null, 裝置假定為不阻塞地可讀可寫.

int (*ioctl) (struct inode *, struct file *, unsigned int, unsigned long);

ioctl 系統呼叫提供了發出裝置特定命令的方法(例如格式化軟盤的乙個磁軌, 這不是讀也不是寫). 另外, 幾個 ioctl 命令被核心識別而不必引用 fops 表. 如果裝置不提供 ioctl 方法, 對於任何未事先定義的請求(-enotty, "裝置無這樣的 ioctl"), 系統呼叫返回乙個錯誤.

int (*mmap) (struct file *, struct vm_area_struct *);

mmap 用來請求將裝置記憶體對映到程序的位址空間. 如果這個方法是 null, mmap 系統呼叫返回 -enodev.

int (*open) (struct inode *, struct file *);

儘管這常常是對裝置檔案進行的第乙個操作, 不要求驅動宣告乙個對應的方法. 如果這個項是 null, 裝置開啟一直成功, 但是你的驅動不會得到通知.

int (*flush) (struct file *);

flush 操作在程序關閉它的裝置檔案描述符的拷貝時呼叫; 它應當執行(並且等待)裝置的任何未完成的操作. 這個必須不要和使用者查詢請求的 fsync 操作混淆了. 當前, flush 在很少驅動中使用; scsi 磁帶驅動使用它, 例如, 為確保所有寫的資料在裝置關閉前寫到磁帶上. 如果 flush 為 null, 核心簡單地忽略使用者應用程式的請求.

int (*release) (struct inode *, struct file *);

在檔案結構被釋放時引用這個操作. 如同 open, release 可以為 null.

int (*fsync) (struct file *, struct dentry *, int);

這個方法是 fsync 系統呼叫的後端, 使用者呼叫來重新整理任何掛著的資料. 如果這個指標是 null, 系統呼叫返回 -einval.

int (*aio_fsync)(struct kiocb *, int);

這是 fsync 方法的非同步版本.

int (*fasync) (int, struct file *, int);

這個操作用來通知裝置它的 fasync 標誌的改變. 非同步通知是乙個高階的主題, 在第 6 章中描述. 這個成員可以是null 如果驅動不支援非同步通知.

int (*lock) (struct file *, int, struct file_lock *);

lock 方法用來實現檔案加鎖; 加鎖對常規檔案是必不可少的特性, 但是裝置驅動幾乎從不實現它.

ssize_t (*readv) (struct file *, const struct iovec *, unsigned long, loff_t *);

ssize_t (*writev) (struct file *, const struct iovec *, unsigned long, loff_t *);

這些方法實現發散/匯聚讀和寫操作. 應用程式偶爾需要做乙個包含多個記憶體區的單個讀或寫操作; 這些系統呼叫允許它們這樣做而不必對資料進行額外拷貝. 如果這些函式指標為 null, read 和 write 方法被呼叫( 可能多於一次 ).

ssize_t (*sendfile)(struct file *, loff_t *, size_t, read_actor_t, void *);

這個方法實現 sendfile 系統呼叫的讀, 使用最少的拷貝從乙個檔案描述符搬移資料到另乙個. 例如, 它被乙個需要傳送檔案內容到乙個網路連線的 web 伺服器使用. 裝置驅動常常使 sendfile 為 null.

ssize_t (*sendpage) (struct file *, struct page *, int, size_t, loff_t *, int);

sendpage 是 sendfile 的另一半; 它由核心呼叫來傳送資料, 一次一頁, 到對應的檔案. 裝置驅動實際上不實現 sendpage.

這個方法的目的是在程序的位址空間找乙個合適的位置來對映在底層裝置上的記憶體段中. 這個任務通常由記憶體管理**進行; 這個方法存在為了使驅動能強制特殊裝置可能有的任何的對齊請求. 大部分驅動可以置這個方法為 null.[10]

int (*check_flags)(int)

這個方法允許模組檢查傳遞給 fnctl(f_setfl...) 呼叫的標誌.

int (*dir_notify)(struct file *, unsigned long);

這個方法在應用程式使用 fcntl 來請求目錄改變通知時呼叫. 只對檔案系統有用; 驅動不需要實現 dir_notify.

scull 裝置驅動只實現最重要的裝置方法. 它的 file_operations 結構是如下初始化的:

struct file_operations scull_fops = ;
sunsea1026

另外一篇講字元裝置驅動幾個資料結構的:

file operations中各項解析

linux裝置驅動中file operations結構體分析 struct module owner 第乙個 file operations 成員根本不是乙個操作 它是乙個指向擁有這個結構的模組的指標.這個成員用來在它的操作還在被使用時阻止模組被解除安裝.幾乎所有時間中,它被簡單初始化為 this ...

關於file operations結構體

結構體file operations 在標頭檔案linux fs.h 中定義,用來儲存驅動核心模組提供的對裝置進行各種操作的函式的指標。該結構體的每個域都對應著驅動核心模組用來處理某個被請求的事務的函式的位址。舉個例子,每個字元裝置需要定義乙個用來讀取裝置資料的函式。結構體file operatio...

關於file operations結構體

結構體 file operations 在標頭檔案 linux fs.h 中定義,用來儲存驅動核心模組提供的對裝置進行各種操作的函式的指標。該結構體的每個域都對應著驅動核心模組用來處理某個被請求的 事務的函式的位址。舉個例子,每個字元裝置需要定義乙個用來讀取裝置資料的函式。結構體 file oper...