偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

Linux內(nèi)核里的DebugFS

系統(tǒng) Linux
DebugFS,顧名思義,是一種用于內(nèi)核調(diào)試的虛擬文件系統(tǒng),內(nèi)核開發(fā)者通過debugfs和用戶空間交換數(shù)據(jù)。類似的虛擬文件系統(tǒng)還有procfs和sysfs等,這幾種虛擬文件系統(tǒng)都并不實際存儲在硬盤上,而是Linux內(nèi)核運行起來后才建立起來。

DebugFS,顧名思義,是一種用于內(nèi)核調(diào)試的虛擬文件系統(tǒng),內(nèi)核開發(fā)者通過debugfs和用戶空間交換數(shù)據(jù)。類似的虛擬文件系統(tǒng)還有procfs和sysfs等,這幾種虛擬文件系統(tǒng)都并不實際存儲在硬盤上,而是Linux內(nèi)核運行起來后才建立起來。

通常情況下,最常用的內(nèi)核調(diào)試手段是printk。但printk并不是所有情況都好用,比如打印的數(shù)據(jù)可能過多,我們真正關心的數(shù)據(jù)在大量的輸出里不是那么一目了然;或者我們在調(diào)試時可能需要修改某些內(nèi)核變量,這種情況下printk就無能為力,而如果為了修改某個值重新編譯內(nèi)核或者驅(qū)動又過于低效,此時就需要一個臨時的文件系統(tǒng)可以把我們需要關心的數(shù)據(jù)映射到用戶空間。在過去,procfs可以實現(xiàn)這個目的,到了2.6時代,新引入的sysfs也同樣可以實現(xiàn),但不論是procfs或是sysfs,用它們來實現(xiàn)某些debug的需求,似乎偏離了它們創(chuàng)建的本意。比如procfs,其目的是反映進程的狀態(tài)信息;而sysfs主要用于Linux設備模型。不論是procfs或是sysfs的接口應該保持相對穩(wěn)定,因為用戶態(tài)程序很可能會依賴它們。當然,如果我們只是臨時借用procfs或者sysfs來作debug之用,在代碼發(fā)布之前將相關調(diào)試代碼刪除也無不可。但如果相關的調(diào)試借口要在相當長的一段時間內(nèi)存在于內(nèi)核之中,就不太適合放在procfs和sysfs里了。故此,debugfs應運而生。

默認情況下,debugfs會被掛載在目錄/sys/kernel/debug之下,如果您的發(fā)行版里沒有自動掛載,可以用如下命令手動完成:

  1. # mount -t debugfs none /your/debugfs/dir 

Linux內(nèi)核為debugfs提供了非常簡潔的API,本文接下來將以一個實作為例來介紹,sample code可以從這里下載。

這個實作會在debugfs中建立如下的目錄結構: 

 

 

在debugfs中建立的目錄結構 

其中,a對應模塊中的一個u8類型的變量,b和subdir下面的c都是對應模塊里的一個字符數(shù)組,只是它們的實現(xiàn)方式不同。

在module_init里,我們首先要建立根目錄mydebug:

  1. my_debugfs_root = debugfs_create_dir("mydebug"NULL); 

***個參數(shù)是目錄的名稱,第二個參數(shù)用來指定這個目錄的上級目錄,如果是NULL,則表示是放在debugfs的根目錄里。

子目錄也是用debugfs_create_dir來實現(xiàn):

  1. sub_dir = debugfs_create_dir("subdir", my_debugfs_root); 

建立文件a的代碼非常簡單:

  1. debugfs_create_u8("a", 0644, my_debugfs_root, &a); 

這表示文件名為“a”,文件屬性是0644,父目錄是上面建立的“mydebug”,對應的變量是模塊中的a。

Linux內(nèi)核還提供了其他一些創(chuàng)建debugfs文件的API,請參考本文的附錄。

b是一個32-bytes的字符數(shù)組,在debugfs里,數(shù)組可以用blob wrapper來實現(xiàn)。

  1. char hello[32] = "Hello world!\n"
  2. struct debugfs_blob_wrapper b; 
  3.   
  4. b.data = (void *)hello; 
  5. b.size = strlen(hello) + 1; 
  6. debugfs_create_blob("b", 0644, my_debugfs_root, &b);  

這里需要注意的是,blob wrapper定義的數(shù)據(jù)只能是只讀的。在本例中,雖然我們把文件b的權限設定為0644,但實際這個文件還是只讀的,如果試圖改寫這個文件,系統(tǒng)將提示出錯。

如果需要對內(nèi)核數(shù)組進行寫的動作,blob wrapper就無法滿足要求,我們只能通過自己定義文件操作來實現(xiàn)。在這個實作里,可以參考文件c的實現(xiàn)。c和b在模塊里對應著同一塊字符數(shù)組,不同的是,b是只讀的,而c通過自定義的文件操作同時實現(xiàn)了讀和寫。

  1. static int c_open(struct inode *inode, struct file *filp) 
  2.     filp->private_data = inode->i_private; 
  3.     return 0; 
  4.   
  5. static ssize_t c_read(struct file *filp, char __user *buffer, 
  6.         size_t count, loff_t *ppos) 
  7.     if (*ppos >= 32) 
  8.         return 0; 
  9.     if (*ppos + count > 32) 
  10.         count = 32 - *ppos; 
  11.   
  12.     if (copy_to_user(buffer, hello + *ppos, count)) 
  13.         return -EFAULT; 
  14.   
  15.     *ppos += count
  16.   
  17.     return count
  18.   
  19. static ssize_t c_write(struct file *filp, const char __user *buffer, 
  20.         size_t count, loff_t *ppos) 
  21.     if (*ppos >= 32) 
  22.         return 0; 
  23.     if (*ppos + count > 32) 
  24.         count = 32 - *ppos; 
  25.   
  26.     if (copy_from_user(hello + *ppos, buffer, count)) 
  27.         return -EFAULT; 
  28.   
  29.     *ppos += count
  30.   
  31.     return count
  32.   
  33. struct file_operations c_fops = { 
  34.     .owner = THIS_MODULE, 
  35.     .open = c_open, 
  36.     .read = c_read, 
  37.     .write = c_write, 
  38. }; 
  39.   
  40.   
  41. debugfs_create_file("c", 0644, sub_dir, NULL, &c_fops);  

注:代碼里,c_open其實并沒有任何用處,因為c_read和c_write直接引用了全局變量hello。這里,我們也可以換一種寫法,在read/write函數(shù)里用filp->private_data來引用字符數(shù)組hello。

到這里,三個文件和子目錄已經(jīng)創(chuàng)建完畢。在module_exit中,我們要記得釋放創(chuàng)建的數(shù)據(jù)。

  1. debugfs_remove_recursive(my_debugfs_root); 

debugfs_remove_recursive可以幫我們逐步移除每個分配  

責任編輯:龐桂玉 來源: 嵌入式Linux中文站
相關推薦

2015-07-27 11:34:03

Linux內(nèi)核指針

2016-08-24 20:09:27

Linux數(shù)據(jù)結構位數(shù)組

2020-11-11 14:48:41

Linux內(nèi)核代碼

2013-11-07 13:59:56

Linux內(nèi)核

2013-11-25 14:07:11

Linux內(nèi)核內(nèi)核特性

2013-11-06 13:03:10

Linux內(nèi)核

2013-11-12 11:01:46

Linux內(nèi)核

2013-11-05 09:58:39

Linux內(nèi)核

2021-02-20 06:08:07

LinuxWindows內(nèi)核

2011-01-11 13:45:06

2013-05-13 09:52:52

Windows內(nèi)核Linux內(nèi)核

2010-03-02 09:17:32

Linux local

2013-10-31 14:09:14

Linux內(nèi)核

2011-02-28 10:31:02

Linux 2.6內(nèi)核

2009-10-16 09:45:41

Linux內(nèi)核操作系統(tǒng)

2009-10-29 09:41:01

Linux內(nèi)核DeviceMappe

2022-02-15 14:53:38

Linux內(nèi)核API

2013-11-05 09:55:37

2009-12-11 15:10:22

2011-02-17 10:14:44

LinuxCPU負載
點贊
收藏

51CTO技術棧公眾號