14.5. 類

2018-02-24 15:50 更新

14.5.?類

我們在本章中要考察最后的設(shè)備模型概念是類.一個類是一個設(shè)備的高級視圖, 它抽象出低級的實(shí)現(xiàn)細(xì)節(jié). 驅(qū)動可以見到一個SCSI 磁盤或者一個 ATA 磁盤, 在類的級別, 它們都是磁盤. 類允許用戶空間基于它們做什么來使用設(shè)備, 而不是它們?nèi)绾伪贿B接或者它們?nèi)绾喂ぷ?

幾乎所有的類都在 sysfs 中在 /sys/class 下出現(xiàn). 因此, 例如, 所有的網(wǎng)絡(luò)接口可在 /sys/class/net 下發(fā)現(xiàn), 不管接口類型. 輸入設(shè)備可在 /sys/class/input 下, 以及串行設(shè)備在 /sys/class/tty. 一個例外是塊設(shè)備, 由于歷史的原因在 /sys/block.

類成員關(guān)系常常由高級的代碼處理, 不必要驅(qū)動的明確的支持. 當(dāng) sbull 驅(qū)動( 見 16 章) 創(chuàng)建一個虛擬磁盤設(shè)備, 它自動出現(xiàn)在 /sys/block. snull 網(wǎng)絡(luò)驅(qū)動(見 17 章)沒有做任何特殊事情給它的接口在 /sys/class/net 中出現(xiàn). 將有多次, 但是, 當(dāng)驅(qū)動結(jié)束直接處理類.

在許多情況, 類子系統(tǒng)是最好的輸出信息到用戶空間的方法. 當(dāng)一個子系統(tǒng)創(chuàng)建一個類, 它完全擁有這個類, 因此沒有必要擔(dān)心哪個模塊擁有那里發(fā)現(xiàn)的屬性. 它也用極少的時間徘徊于更加面向硬件的 sysfs 部分來了解, 它不是一個直接瀏覽的好地方. 用戶會更加高興地在 /sys/class/some-widget 中發(fā)現(xiàn)信息, 而不是, /sys/device/pci0000:00/0000:00:10.0/usb2/2-0:1.0.

驅(qū)動核心輸出 2 個清晰的接口來管理類. class_simple 函數(shù)設(shè)計來盡可能容易地添加新類到系統(tǒng). 它們的主要目的, 常常, 是暴露包含設(shè)備號的屬性來使能設(shè)備節(jié)點(diǎn)的自動創(chuàng)建. 常用的類接口更加復(fù)雜但是同時提供更多特性. 我們從簡單版本開始.

14.5.1.?class_simple 接口

class_simple 接口意圖是易于使用, 以至于沒人會抱怨沒有暴露至少一個包含設(shè)備的被分配的號的屬性. 使用這個接口只不過是一對函數(shù)調(diào)用, 沒有通常的和 Linux 設(shè)備模型關(guān)聯(lián)的樣板.

第一步是創(chuàng)建類自身. 使用一個對 class_simple_create 的調(diào)用來完成:


struct class_simple *class_simple_create(struct module *owner, char *name);

這個函數(shù)使用給定的名子創(chuàng)建一個類. 這個操作可能失敗, 當(dāng)然, 因此在繼續(xù)之前返回值應(yīng)當(dāng)一直被檢查( 使用 IS_ERR, 在第 1 章的"指針和錯誤值"一節(jié)中描述過).

一個簡單的類可被銷毀, 使用:


void class_simple_destroy(struct class_simple *cs); 

創(chuàng)建一個簡單類的真實(shí)目的是添加設(shè)備給它; 這個任務(wù)使用:


struct class_device *class_simple_device_add(struct class_simple *cs, dev_t devnum, struct device *device, const char *fmt, ...); 

這里, cs 是之前創(chuàng)建的簡單類, devnum 是分配的設(shè)備號, device 是代表這個設(shè)備的 struct device, 其他的參數(shù)是一個 printk-風(fēng)格 的格式串和參數(shù)來創(chuàng)建設(shè)備名子. 這個調(diào)用添加一項(xiàng)到類, 包含一個屬性, dev, 含有設(shè)備號. 如果設(shè)備參數(shù)是非 NULL, 一個符號連接( 稱為 device )指向在 /sys/devices 下的設(shè)備的入口.

可能添加其他的屬性到設(shè)備入口. 它只是使用 class_device_create_file, 我們在下一節(jié)和完整類子系統(tǒng)所剩下的內(nèi)容討論.

當(dāng)設(shè)備進(jìn)出時類產(chǎn)生熱插拔事件. 如果你的驅(qū)動需要添加變量到環(huán)境中給用戶空間事件處理者, 可以建立一個熱插拔回調(diào), 使用:


int class_simple_set_hotplug(struct class_simple *cs,
 int (*hotplug)(struct class_device *dev,
 char **envp, int num_envp,
 char *buffer, int buffer_size)); 

當(dāng)你的設(shè)備離開時, 類入口應(yīng)當(dāng)被去除, 使用:


void class_simple_device_remove(dev_t dev); 

注意, 由 class_simple_device_add 返回的 class_device 結(jié)構(gòu)這里不需要; 設(shè)備號(它當(dāng)然應(yīng)當(dāng)是唯一的)足夠了.

14.5.2.?完整的類接口

class_simple 接口滿足許多需要, 但是有時需要更多靈活性. 下面的討論描述如何使用完整的類機(jī)制, class_simple 正是基于此. 它是簡短的: 類函數(shù)和結(jié)構(gòu)遵循設(shè)備模型其他部分相同的模式, 因此這里沒有什么真正是新的.

14.5.2.1.?管理類

一個類由一個 struct class 的實(shí)例來定義:


struct class {
 char *name;
 struct class_attribute *class_attrs;
 struct class_device_attribute *class_dev_attrs;
 int (*hotplug)(struct class_device *dev, char **envp,
 int num_envp, char *buffer, int buffer_size);
 void (*release)(struct class_device *dev);
 void (*class_release)(struct class *class);
 /* Some fields omitted */
};

每個類需要一個唯一的名子, 它是這個類如何在 /sys/class 中出現(xiàn). 當(dāng)這個類被注冊, 由 class_attrs 所指向的數(shù)組中列出的所有屬性被創(chuàng)建. 還有一套缺省屬性給每個添加到類中的設(shè)備; class_dev_attrs 指向它們. 有通常的熱插拔函數(shù)來添加變量到環(huán)境中, 當(dāng)事件產(chǎn)生時. 還有 2 個釋放方法: release 在無論何時從類中去除一個設(shè)備時被調(diào)用, 而 class_release 在類自己被釋放時調(diào)用.

注冊函數(shù)是:


int class_register(struct class *cls);
void class_unregister(struct class *cls);

使用屬性的接口不應(yīng)當(dāng)在這點(diǎn)嚇人:


struct class_attribute {
 struct attribute attr;
 ssize_t (*show)(struct class *cls, char *buf);
 ssize_t (*store)(struct class *cls, const char *buf, size_t count); 
}; 
CLASS_ATTR(name, mode, show, store); 
int class_create_file(struct class *cls, const struct class_attribute *attr);
void class_remove_file(struct class *cls, const struct class_attribute *attr);

14.5.2.2.?類設(shè)備

一個類的真正目的是作為一個是該類成員的設(shè)備的容器. 一個成員由 struct class_device 來表示:


struct class_device {
struct kobject kobj;
struct class *class;
struct device *dev;
void *class_data;
char class_id[BUS_ID_SIZE];

 };

class_id 成員持有設(shè)備名子, 如同它在 sysfs 中的一樣. class 指針應(yīng)當(dāng)指向持有這個設(shè)備的類, 并且 dev 應(yīng)當(dāng)指向關(guān)聯(lián)的設(shè)備結(jié)構(gòu). 設(shè)置 dev 是可選的; 如果它是非 NULL, 它用來創(chuàng)建一個符號連接從類入口到對應(yīng)的在 /sys/devices 下的入口, 使得易于在用戶空間找到設(shè)備入口. 類可以使用 class_data 來持有一個私有指針.

通常的注冊函數(shù)已經(jīng)被提供:


int class_device_register(struct class_device *cd);
void class_device_unregister(struct class_device *cd);

類設(shè)備接口也允許重命名一個已經(jīng)注冊的入口:


int class_device_rename(struct class_device *cd, char *new_name); 

類設(shè)備入口有屬性:


struct class_device_attribute {
 struct attribute attr;
 ssize_t (*show)(struct class_device *cls, char *buf);
 ssize_t (*store)(struct class_device *cls, const char *buf,
 size_t count);
};

CLASS_DEVICE_ATTR(name, mode, show, store); 
int class_device_create_file(struct class_device *cls, const struct class_device_attribute *attr);
void class_device_remove_file(struct class_device *cls, const struct class_device_attribute *attr);

一個缺省的屬性集合, 在類的 class_dev_attrs 成員, 被創(chuàng)建當(dāng)類設(shè)備被注冊時; class_device_create_file 可用來創(chuàng)建額外的屬性. 屬性還可以被加入到由 class_simple 接口創(chuàng)建的類設(shè)備.

14.5.2.3.?類接口

類子系統(tǒng)有一個額外的在 Linux 設(shè)備模型其他部分找不到的概念. 這個機(jī)制稱為一個接口, 但是它是, 也許, 最好作為一種觸發(fā)機(jī)制可用來在設(shè)備進(jìn)入或離開類時得到通知.

一個接口被表示, 使用:


struct class_interface {
 struct class *class;
 int (*add) (struct class_device *cd);
 void (*remove) (struct class_device *cd); 
}; 

接口可被注冊或注銷, 使用:


int class_interface_register(struct class_interface *intf);
void class_interface_unregister(struct class_interface *intf);

一個接口的功能是簡單明了的. 無論何時一個類設(shè)備被加入到在 class_interface 結(jié)構(gòu)中指定的類時, 接口的 add 函數(shù)被調(diào)用. 這個函數(shù)可進(jìn)行任何額外的這個設(shè)備需要的設(shè)置; 這個設(shè)置常常采取增加更多屬性的形式, 但是其他的應(yīng)用都可能. 當(dāng)設(shè)備被從類中去除, remove 方法被調(diào)用來進(jìn)行任何需要的清理.

可注冊多個接口給一個類.

以上內(nèi)容是否對您有幫助:
在線筆記
App下載
App下載

掃描二維碼

下載編程獅App

公眾號
微信公眾號

編程獅公眾號