?? file_dev.c
字號:
/** linux/fs/file_dev.c** (C) 1991 Linus Torvalds*/#include <errno.h> // 錯誤號頭文件。包含系統中各種出錯號。(Linus 從minix 中引進的)#include <fcntl.h> // 文件控制頭文件。用于文件及其描述符的操作控制常數符號的定義。#include <linux/sched.h> // 調度程序頭文件,定義了任務結構task_struct、初始任務0 的數據,// 還有一些有關描述符參數設置和獲取的嵌入式匯編函數宏語句。#include <linux/kernel.h> // 內核頭文件。含有一些內核常用函數的原形定義。#include <asm/segment.h> // 段操作頭文件。定義了有關段寄存器操作的嵌入式匯編函數。#define MIN(a,b) (((a)<(b))?(a):(b)) // 取a,b 中的最小值。#define MAX(a,b) (((a)>(b))?(a):(b)) // 取a,b 中的最大值。//// 文件讀函數 - 根據i 節點和文件結構,讀設備數據。// 由i 節點可以知道設備號,由filp 結構可以知道文件中當前讀寫指針位置。buf 指定用戶態中// 緩沖區的位置,count 為需要讀取的字節數。返回值是實際讀取的字節數,或出錯號(小于0)。intfile_read (struct m_inode *inode, struct file *filp, char *buf, int count){ int left, chars, nr; struct buffer_head *bh;// 若需要讀取的字節計數值小于等于零,則返回。 if ((left = count) <= 0) return 0;// 若還需要讀取的字節數不等于0,就循環執行以下操作,直到全部讀出。 while (left) {// 根據i 節點和文件表結構信息,取數據塊文件當前讀寫位置在設備上對應的邏輯塊號nr。若nr 不// 為0,則從i 節點指定的設備上讀取該邏輯塊,如果讀操作失敗則退出循環。若nr 為0,表示指定// 的數據塊不存在,置緩沖塊指針為NULL。 if (nr = bmap (inode, (filp->f_pos) / BLOCK_SIZE)) { if (!(bh = bread (inode->i_dev, nr))) break; } else bh = NULL;// 計算文件讀寫指針在數據塊中的偏移值nr,則該塊中可讀字節數為(BLOCK_SIZE-nr),然后與還需// 讀取的字節數left 作比較,其中小值即為本次需讀的字節數chars。若(BLOCK_SIZE-nr)大則說明// 該塊是需要讀取的最后一塊數據,反之則還需要讀取一塊數據。 nr = filp->f_pos % BLOCK_SIZE; chars = MIN (BLOCK_SIZE - nr, left);// 調整讀寫文件指針。指針前移此次將讀取的字節數chars。剩余字節計數相應減去chars。 filp->f_pos += chars; left -= chars;// 若從設備上讀到了數據,則將p 指向讀出數據塊緩沖區中開始讀取的位置,并且復制chars 字節// 到用戶緩沖區buf 中。否則往用戶緩沖區中填入chars 個0 值字節。 if (bh) { char *p = nr + bh->b_data; while (chars-- > 0) put_fs_byte (*(p++), buf++); brelse (bh); } else { while (chars-- > 0) put_fs_byte (0, buf++); } }// 修改該i 節點的訪問時間為當前時間。返回讀取的字節數,若讀取字節數為0,則返回出錯號。 inode->i_atime = CURRENT_TIME; return (count - left) ? (count - left) : -ERROR;}//// 文件寫函數 - 根據i 節點和文件結構信息,將用戶數據寫入指定設備。// 由i 節點可以知道設備號,由filp 結構可以知道文件中當前讀寫指針位置。buf 指定用戶態中// 緩沖區的位置,count 為需要寫入的字節數。返回值是實際寫入的字節數,或出錯號(小于0)。intfile_write (struct m_inode *inode, struct file *filp, char *buf, int count){ off_t pos; int block, c; struct buffer_head *bh; char *p; int i = 0;/** ok, append may not work when many processes are writing at the same time* but so what. That way leads to madness anyway.*//** ok,當許多進程同時寫時,append 操作可能不行,但那又怎樣。不管怎樣那樣做會* 導致混亂一團。*/// 如果是要向文件后添加數據,則將文件讀寫指針移到文件尾部。否則就將在文件讀寫指針處寫入。 if (filp->f_flags & O_APPEND) pos = inode->i_size; else pos = filp->f_pos;// 若已寫入字節數i 小于需要寫入的字節數count,則循環執行以下操作。 while (i < count) {// 創建數據塊號(pos/BLOCK_SIZE)在設備上對應的邏輯塊,并返回在設備上的邏輯塊號。如果邏輯// 塊號=0,則表示創建失敗,退出循環。 if (!(block = create_block (inode, pos / BLOCK_SIZE))) break;// 根據該邏輯塊號讀取設備上的相應數據塊,若出錯則退出循環。 if (!(bh = bread (inode->i_dev, block))) break;// 求出文件讀寫指針在數據塊中的偏移值c,將p 指向讀出數據塊緩沖區中開始讀取的位置。置該// 緩沖區已修改標志。 c = pos % BLOCK_SIZE; p = c + bh->b_data; bh->b_dirt = 1;// 從開始讀寫位置到塊末共可寫入c=(BLOCK_SIZE-c)個字節。若c 大于剩余還需寫入的字節數// (count-i),則此次只需再寫入c=(count-i)即可。 c = BLOCK_SIZE - c; if (c > count - i) c = count - i;// 文件讀寫指針前移此次需寫入的字節數。如果當前文件讀寫指針位置值超過了文件的大小,則// 修改i 節點中文件大小字段,并置i 節點已修改標志。 pos += c; if (pos > inode->i_size) { inode->i_size = pos; inode->i_dirt = 1; }// 已寫入字節計數累加此次寫入的字節數c。從用戶緩沖區buf 中復制c 個字節到高速緩沖區中p// 指向開始的位置處。然后釋放該緩沖區。 i += c; while (c-- > 0) *(p++) = get_fs_byte (buf++); brelse (bh); }// 更改文件修改時間為當前時間。 inode->i_mtime = CURRENT_TIME;// 如果此次操作不是在文件尾添加數據,則把文件讀寫指針調整到當前讀寫位置,并更改i 節點修改// 時間為當前時間。 if (!(filp->f_flags & O_APPEND)) { filp->f_pos = pos; inode->i_ctime = CURRENT_TIME; }// 返回寫入的字節數,若寫入字節數為0,則返回出錯號-1。 return (i ? i : -1);}
?? 快捷鍵說明
復制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -