成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

學習深入Linux Sysfs編程與源碼分析

運維 系統運維
從源代碼中理解 Linux Sysfs 屬性的用途更多的 Linux Sysfs 屬性的功能只能靠閱讀源代碼來理解。還是以上文提到的 scsi_host 的 scan 屬性來理解,這個功能沒有任何文檔上有描述,因此只能去讀源代碼。

Linux經過長時間的發展,很多用戶都很了解Linux Sysfs了,本人很喜歡Linux Sysfs,下面就這個問題來詳細說說吧。Linux Sysfs 源碼分析和編程實踐。Linux的發行遵守GNU(GNU is Not UNIX)的通用公共許可證,遵循公共版權許可證(GPL,General Public License),秉承“自由的思想,開放的源碼”原則。

從源代碼中理解 Linux Sysfs 屬性的用途更多的 Linux Sysfs 屬性的功能只能靠閱讀源代碼來理解。還是以上文提到的 scsi_host 的 scan 屬性來理解,這個功能沒有任何文檔上有描述,因此只能去讀源代碼。

在內核中, Linux Sysfs屬性一般是由 __ATTR 系列的宏來聲明的,如對設備的使用 DEVICE_ATTR ,對總線使用 BUS_ATTR ,對驅動使用 DRIVER_ATTR ,對類別(class)使用 CLASS_ATTR, 這四個高級的宏來自于 , 都是以更低層的來自 中的 __ATTR/__ATRR_RO 宏實現; 因此我們在內核源碼樹中相應位置 drivers/scsi/ 找到這幾個宏的使用情況,可以得到在 drivers/scsi/scsi_sysfs.c 中:
static ssize_t
store_scan(struct device *dev, struct device_attribute *attr,
const char *buf, size_t count)
struct Scsi_Host *shost = class_to_shost(dev);
int res;
res = scsi_scan(shost, buf);
if (res == 0)
res = count;
return res;
static DEVICE_ATTR(scan, S_IWUSR, NULL, store_scan);

DEVICE_ATTR 宏聲明有四個參數,分別是名稱、權限位、讀函數、寫函數。這里對應的,名稱是 scan, 權限是只有屬主可寫(S_IWUSR)、沒有讀函數、只有寫函數。因此讀寫功能與權限位是對應的,因為 DEVICE_ATTR 把權限位聲明與真正的讀寫是否實現放在了一起,減少了出現不一致的可能。(上文提到 /proc/scsi/scsi 接口的權限位聲明與其功能不對應,這與注冊 proc 接口的函數設計中的不一致是有關系的,權限位聲明與功能實現不在代碼中同一個位置,因此易出錯。雖然修復 /proc/scsi/scsi 的權限位錯誤很容易,但內核團隊中多年來一直沒有人發現或未有人去修正這個 BUG,應該是與 /proc/scsi/ 接口的過時有關,過時的功能會在未來某個內核版本中去除。)

上面的 scan 屬性寫入功能是在 store_scan 函數中實現的,這個接口的四個參數中, buf/count 代表用戶寫入過來的字符串,它把 buf 進一步傳給了 scsi_scan 函數;如果進一步分析 scsi_scan 函數實現可以知道,它期望從 buf 中接受三個或四個整型值(也接受"-"作為通配符),分別代表 host, channel, id 三個值,(第四個整數在早期內核中曾代表 lun 號碼,但在較新內核中第四個數字被忽略,僅作為向后兼容保留接受四個整數),然后對具體的 (host, channel, id) 進行重新掃描以發現這個 SCSI 控制器上的設備變動。

添加 Linux Sysfs 支持

如果你正在開發的設備驅動程序中需要與用戶層的接口,一般可選的方法有:
注冊虛擬的字符設備文件,以這個虛擬設備上的 read/write/ioctl 等接口與用戶交互;但 read/write 一般只能做一件事情, ioctl 可以根據 cmd 參數做多個功能,但其缺點是很明顯的: ioctl 接口無法直接在 Shell 腳本中使用,為了使用 ioctl 的功能,還必須編寫配套的 C語言的虛擬設備操作程序, ioctl 的二進制數據接口也是造成大小端問題 (big endian與little endian)、32位/64位不可移植問題的根源;
注冊 proc 接口,接受用戶的 read/write/ioctl 操作;同樣的,一個 proc 項通常使用其 read/write/ioctl 接口,它所存在的問題與上面的虛擬字符設備的的問題相似;

注冊 Linux Sysfs 屬性;

最重要的是,添加虛擬字符設備支持和注冊 proc 接口支持這兩者所需要增加的代碼量都并不少,***的方法還是使用 Linux Sysfs屬性支持,一切在用戶層是可見的透明,且增加的代碼量是最少的,可維護性也***;方法就是使用 頭文件提供的這四個宏,分別應用于總線/類別/驅動/設備四種內核數據結構對象上:
#define BUS_ATTR(_name, _mode, _show, _store)   \
struct bus_attribute bus_attr_##_name = __ATTR(_name, _mode, _show, _store)
#define CLASS_ATTR(_name, _mode, _show, _store) \
struct class_attribute class_attr_##_name = __ATTR(_name, _mode, _show, _store)
#define DRIVER_ATTR(_name, _mode, _show, _store)\
struct driver_attribute driver_attr_##_name =\
__ATTR(_name, _mode, _show, _store)
#define DEVICE_ATTR(_name, _mode, _show, _store) \
struct device_attribute dev_attr_##_name = __ATTR(_name, _mode, _show, _store)
 
總線(BUS)和類別(CLASS)屬性一般用于新設計的總線和新設計的類別,這兩者一般是不用的;因為你的設備一般是以PCI等成熟的常規方式連接到主機,而不會去新發明一種類型;使用驅動屬性和設備屬性的區別就在于:看你的 Linux Sysfs屬性設計是針對整個驅動有效的還是針對這份驅動所可能支持的每個設備分別有效。

從頭文件中還可以找到 show/store 函數的原型,注意到它和虛擬字符設備或 proc 項的 read/write 的作用很類似,但有一點不同是 show/store 函數上的 buf/count 參數是在Linux Sysfs 層已作了用戶區/內核區的內存復制,虛擬字符設備上常見的 __user 屬性在這里并不需要,因而也不需要多一次 copy_from_user/copy_to_user, 在 show/store 函數參數上的 buf/count 參數已經是內核區的地址,可以直接操作。
上面四種都是 Linux Sysfs 統一設備模型所添加的高級接口,如果使用 Linux Sysfs所提供的底層接口的話,則還有下面兩個,定義來自 :(上面的總線/類別/驅動/設備四個接口都是以這里的__ATTR實現的)
#define __ATTR(_name,_mode,_show,_store) { \
.attr = {.name = __stringify(_name), .mode = _mode },   \
.show   = _show,  \
.store  = _store, \
#define __ATTR_RO(_name) { \
.attr   = { .name = __stringify(_name), .mode = 0444 }, \
.show   = _name##_show,\

上面這些宏都是在注冊總線/類別/驅動/設備時作為缺省屬性而使用的,在實際應用中還有一種情況是根據條件動態添加屬性,如 PCI 設備上的 resource{0,1,2,...} 屬性文件,因為一個 PCI 設備上的可映射資源究竟有多少無法預知,也只能以條件判斷的方式動態添加上。
int __must_check sysfs_create_file(struct kobject *kobj,
const struct attribute *attr);
int __must_check sysfs_create_bin_file(struct kobject *kobj,struct bin_attribute *attr); 
這兩個函數可以對一個 kobject 動態添加上文本屬性或二進制屬性,這也是唯一可以添加二進制屬性的方法。

二進制屬性與普通文本屬性的區別在于二進制屬性 struct bin_attribute 中內嵌一個 struct attribute 結構體對象,因此具有普通屬性的所有功能特征;二進制屬性上多一個 size 用來描述此二進制文件的大小,而普通屬性文件的大小總是 4096, 準確地說,應該是一個內存頁的大小,因為從當前 Linux Sysfs 內核實現來說,它分配一個內存頁面來作為 (buf/count) 的緩沖區;二進制屬性比普通屬性多內存映射(mmap)接口的支持;

編程示例,對 LDD3 一書中的 lddbus 驅動程序的 Linux Sysfs 改進

首先,這個程序本身是針對當時作者寫書的年代的內核(2.6.11)而編寫的,在當前的 Fedora10 系統 (2.6.27.5-117.fc10.i686) 上甚至無法編譯編譯通過;因此首先需要將它移植過來至少達到可運行狀態;附件的壓縮包中含有修改過的 lddbus, sculld 的源代碼和修改過程的四個patch:

***個 0001-ldd3-examples-build-on-fedora-10-2.6.27.5-117.fc10.i.patch 是將 lddbus 和 sculld 移植到 Fedora10 內核上可運行,這其中主要是一此內核 API 的變化;

第二個 0002-port-dmem-proc-entry-to-use-sysfs-entry.patch 演示了怎樣將原有的 proc 接口改進成為 Linux Sysfs 屬性接口的,從這個 patch 中可以看到刪除的代碼多而新增加的代碼少,這說明對于相同的功能,使用 Linux Sysfs 編程接口的代碼量更少,而且 sysfs 代碼看起來也比 proc 更為整潔:打印每個設備的調試信息可以做成每個設備上分別有自己的接口,而不是統一的一個 proc 接口;設備屬性文件最終出現的位置如 "/sys/devices/ldd0/sculld0/dmem"; static ssize_t sculld_show_dmem(struct device *ddev,
struct device_attribute *attr, char *buf)
/* 其中打印每個設備調試信息的代碼復制自原proc接口 */
static DEVICE_ATTR(dmem, S_IRUGO, sculld_show_dmem, NULL);
static int __init sculld_register_dev(struct sculld_dev *dev, int index)
/* 創建此device屬性文件 */
ret |= device_create_file(&dev->ldev.dev, &dev_attr_dmem);

第三個 0003-add-.gitignore.patch 是增加了 .gitignore 文件,屏蔽一些編譯生成的臨時文件;

第四個 0004-port-qset-get-set-ioctl-to-use-sysfs-entry.patch 演示了怎樣把基于 ioctl 的操作接口改進成為基于 Linux Sysfs 接口,由于原來的 ioctl 接口設置和獲取 qset 信息是表示整個驅動模塊級的變量,它用來控制整個驅動程序而非驅動所支持的單個的設備,因此這個 qset 屬性使用 DRIVER_ATTR 來添加更為合適; ssize_t sculld_show_qset(struct device_driver *driver, char *buf)
return snprintf(buf, PAGE_SIZE, "%d\n", sculld_qset);
ssize_t sculld_store_qset(struct device_driver *driver, const char *buf,
size_t count)sculld_qset = simple_strtol(buf, NULL, 0);return count;
/* 聲明一個權限為0644的可同時讀寫的driver屬性 */
static DRIVER_ATTR(qset, S_IRUGO | S_IWUSR, sculld_show_qset, sculld_store_qset);
/* 創建此driver屬性文件 */
result = driver_create_file(&sculld_driver.driver, &driver_attr_qset);
驅動屬性最終出現如 "/sys/bus/ldd/drivers/sculld/qset" ,這里聲明的是同時可讀寫的,權限位 0644 與其保持一致。 6446 0 -rw-r--r-- 1 root root 4096 12月 14 07:44 /sys/bus/ldd/drivers/sculld/qset

【編輯推薦】

  1. 為你分析Linux操作系統和Windows系統
  2. 闡述Linux操作系統安裝與認識支援的硬體
  3. 詳解Linux操作系統中使用Windows分區
  4. Linux技術受風河與NEC垂青
  5. 多樣性的Linux桌面
責任編輯:佚名 來源: CSDN
相關推薦

2009-12-22 13:26:03

Linux sysfs

2009-12-11 09:42:54

Linux內核源碼進程調度

2009-12-11 09:47:23

Linux內核源碼進程調度

2016-10-26 20:49:24

ReactJavascript前端

2024-04-29 09:06:46

線程初始化源碼

2010-01-12 17:55:03

C++程序

2011-06-28 14:11:33

JavaScript

2017-02-28 18:26:09

Linuxinput子系統編程

2009-12-22 13:05:15

Linux sysfs

2023-12-13 10:01:15

數據結構c++編程

2010-03-08 14:53:48

Linux分區

2009-06-16 10:51:14

Java源碼

2024-10-14 13:30:20

2023-03-15 21:46:17

中間件Java網絡編程

2018-10-31 15:54:47

Java線程池源碼

2009-07-03 11:14:57

2009-11-30 16:46:29

學習Linux

2019-07-08 20:00:35

Linux內核模塊

2009-12-22 12:14:11

Linux內核

2016-10-21 13:03:18

androidhandlerlooper
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产午夜视频 | 激情一区二区三区 | 亚洲三级av| 国产色| 日本不卡视频 | 亚洲成人99 | 2019精品手机国产品在线 | 亚洲少妇综合网 | 国产成人综合在线 | 免费在线黄| 在线一区二区国产 | 久在线视频播放免费视频 | 国产成在线观看免费视频 | 在线色网站 | 色伊人网 | 91福利在线观看 | 久久久999成人 | 中文字幕国产 | 日韩精品一区二区三区免费观看 | 亚洲男女视频在线观看 | 99精品在线观看 | 97国产精品视频人人做人人爱 | 国产精品福利视频 | 一区精品国产欧美在线 | 久久99久久久久 | 成人免费在线视频 | 日韩在线欧美 | 欧美激情综合五月色丁香小说 | 亚州春色| 久久久www成人免费无遮挡大片 | 成人中文网| 国产乱码久久久 | 日韩av一区二区在线观看 | 在线观看成年人视频 | 91亚洲国产成人久久精品网站 | 日韩欧美一区二区三区 | 欧美xxxx色视频在线观看免费 | 人人干人人看 | 久久久久九九九女人毛片 | www.一级片 | 成人国产精品 |