日本黄色一级经典视频|伊人久久精品视频|亚洲黄色色周成人视频九九九|av免费网址黄色小短片|黄色Av无码亚洲成年人|亚洲1区2区3区无码|真人黄片免费观看|无码一级小说欧美日免费三级|日韩中文字幕91在线看|精品久久久无码中文字幕边打电话

當(dāng)前位置:首頁 > 嵌入式 > 嵌入式軟件

作者:劉洪濤,華清遠見嵌入式培訓(xùn)中心高級講師,ARM公司授權(quán)ATC講師。

在上篇《利用udev、sys動態(tài)創(chuàng)建設(shè)備結(jié)點》的記錄中,設(shè)備驅(qū)動中主要依靠下面兩個功能完成的:

1、在/sys/class下創(chuàng)建farsight_class類

my_class =class_create(THIS_module, "farsight_class");

2、在farsight_class中創(chuàng)建新的class設(shè)備

class_device_create(my_class,NULL, devno, NULL,"farsight_dev");

然后會在/sys中出現(xiàn)如圖的文件結(jié)構(gòu):


其中”dev”和uevent都是“屬性”,可以讀取dev獲取設(shè)備的主次設(shè)備號;也可以對uevent操作;讓內(nèi)核發(fā)出“add”事件用于熱插拔。如:




注:這里寫入任何值都會導(dǎo)致“add”事件的產(chǎn)生,udevmonitor檢測時現(xiàn)象如下:

UEVENT[1220019773.507374] add????? /class/farsight_class/farsight_dev (farsight_class)

那么上述功能實現(xiàn)的原理是什么呢?現(xiàn)在就要過度到本文的主題ktype的使用了。先認識下這個結(jié)構(gòu)

kype的結(jié)構(gòu)定義為:

STruct kobj_type {

void (*release)(struct kobject *);

struct sysfs_ops *sysfs_ops;/*提供實現(xiàn)以下屬性的方法*/

struct attribute **default_attrs; /*用于保存類型屬性列表(指針的指針) */

};

其中 attribute定義為:

struct attribute {

char *name;/*屬性的名字(在kobject的sysfs 目錄中顯示,如上文的dev、uvent)*/

struct module *owner;/*指向模塊的指針(如果有), 此模塊負責(zé)實現(xiàn)這個屬性*/

mode_t mode; /*屬性的保護位,modes 的宏定義在 <linux/stat.h>:例如S_IRUGO 為只讀屬性等等*/

}; /*default_attrs 列表中的最后一個元素必須用 0 填充*/

sysfs 系統(tǒng)中的屬性讀寫是由 kobj_type->sysfs_ops 成員中的函數(shù)完成的:

struct sysfs_ops {

ssize_t (*show)(struct kobject *kobj, struct attribute *attr, char *buffer);

ssize_t (*store)(struct kobject *kobj, struct attribute *attr, const char *buffer, size_t size);

};

當(dāng)用戶空間讀取一個屬性時(如:#cat dev),內(nèi)核會使用指向 kobject 的指針(kobj)和正確的屬性結(jié)構(gòu)(*attr)來調(diào)用show 方法,該方法將給定屬性值編碼進緩沖(buffer)(注意不要越界( PAGE_SIZE 字節(jié))), 并返回實際數(shù)據(jù)長度。

也可對所有 kobject (通常指在一個kset關(guān)聯(lián)的范圍內(nèi))關(guān)聯(lián)的屬性使用同一個 show 方法,用傳遞到函數(shù)的 attr 指針來判斷所請求的屬性。有的 show 方法包含對屬性名字的檢查。有的show 方法會將屬性結(jié)構(gòu)嵌入另一個結(jié)構(gòu)(本文舉的例子就是用的這種方法), 這個結(jié)構(gòu)包含需要返回屬性值的信息,這時可用container_of 獲得上層結(jié)構(gòu)的指針以返回屬性值的信息。

當(dāng)用戶空間寫入一個屬性時(如echo “hello” > event)內(nèi)核會使用指向 kobject 的指針(kobj)和正確的屬性結(jié)構(gòu)(*attr)來調(diào)用store 方法。

store 方法將存在緩沖(buffer)的數(shù)據(jù)( size為數(shù)據(jù)的長度,不能超過 PAGE_SIZE )解碼并保存新值到屬性(*attr), 返回實際解碼的字節(jié)數(shù)。store 方法只在擁有屬性的寫權(quán)限時才能被調(diào)用。此時注意:接收來自用戶空間的數(shù)據(jù)一定要驗證其合法性。如果到數(shù)據(jù)不匹配, 返回一個負的錯誤值。

每一個 kobject 需要有一個關(guān)聯(lián)的 kobj_type 結(jié)構(gòu),指向這個結(jié)構(gòu)的指針能在 2 個不同的地方找到:

(1)kobject 結(jié)構(gòu)自身包含一個成員(ktype)指向kobj_type ;

struct kobject {

……

struct kobj_type * ktype;/*負責(zé)對該kobject類型進行跟蹤的struct kobj_type的指針*/

……

}

(2)如果這個 kobject 是一個 kset 的成員, kset 會提供kobj_type 指針。

struct kset {

struct kobj_type * ktype; /*指向該kset對象類型的指針*/

……

}

訪問屬性的時候到底是用的哪個kobj_type呢?

下面這個函數(shù)用以查找指定kobject的kobj_type 指針:

static inline struct kobj_type * get_ktype(struct kobject * k)

{

if (k->kset && k->kset->ktype)

return k->kset->ktype;

else

return k->ktype;

}

上面可以看出,kset中的ktype這個類型優(yōu)先于 kobject 自身中的 ktype 。因此在典型的應(yīng)用中, 在 struct kobject 中的 ktype 成員被設(shè)為 NULL, 而 kset 中的ktype是實際被使用的。

下面通過跟蹤class_device_create(my_class,NULL, devno, NULL,"farsight_dev");來確定ktype的使用。

1、

struct class_device *class_device_create(……)

{

……

retval = class_device_register(class_dev);

}

2、

int class_device_register(struct class_device *class_dev)

{

class_device_initialize(class_dev);

return class_device_add(class_dev);

}

3、

void class_device_initialize(struct class_device *class_dev)

{

kobj_set_kset_s(class_dev, class_obj_subsys);

kobject_init(&class_dev->kobj);

INIT_LIST_HEAD(&class_dev->node);

}

4、

#define kobj_set_kset_s(obj,subsys)

(obj)->kobj.kset = &(subsys).kset

從中可以看出名為“farsight_dev”的kobject對應(yīng)的kset是class_obj_subsys.kset

5、

看看class_obj_subsys的定義

static decl_subsys(class_obj, &ktype_class_device, &class_uevent_ops);

#define decl_subsys(_name,_type,_uevent_ops)

struct subsystem _name##_subsys = {

.kset = {

.kobj = { .name = __stringify(_name) },

.ktype = _type,

.uevent_ops =_uevent_ops,

}

}

所以kset對應(yīng)的ktype為ktype_class_device

6、

static struct kobj_type ktype_class_device = {

.sysfs_ops????? = &class_dev_sysfs_ops,

.release??? = class_dev_release,

};

7、

static struct sysfs_ops class_dev_sysfs_ops = {

.show???? = class_device_attr_show,

.store????? = class_device_attr_store,

};

在操作上文中的“dev”或“uevent”時都是操作這個class_dev_sysfs_ops

8、

class_device_attr_show(struct kobject * kobj, struct attribute * attr,

char * buf)

{

struct class_device_attribute * class_dev_attr = to_class_dev_attr(attr);

struct class_device * cd = to_class_dev(kobj);

ssize_t ret = 0;

if (class_dev_attr->show)

ret = class_dev_attr->show(cd, buf);

return ret;

}

class_device_attr_store(struct kobject * kobj, struct attribute * attr,

const char * buf, size_t count)

{

struct class_device_attribute * class_dev_attr = to_class_dev_attr(attr);

struct class_device * cd = to_class_dev(kobj);

ssize_t ret = 0;

if (class_dev_attr->store)

ret = class_dev_attr->store(cd, buf, count);

return ret;

}

可以看出操作函數(shù)會根據(jù)to_class_dev_attr(attr);找出對應(yīng)attr的class_dev_attr并調(diào)用其對應(yīng)的show或store

9、

再跟蹤一下上文中的dev及uvent屬性的定義及其對應(yīng)的操作函數(shù)

attr->attr.name = "dev";

attr->attr.mode = S_IRUGO;

attr->attr.owner = parent_class->owner;

attr->show = show_dev;

error = class_device_create_file(class_dev, attr);

static ssize_t show_dev(struct class_device *class_dev, char *buf)

{

return print_dev_t(buf, class_dev->devt);//上文中將打印出“252:0”

}

class_dev->uevent_attr.attr.name = "uevent";

class_dev->uevent_attr.attr.mode = S_IWUSR;

class_dev->uevent_attr.attr.owner = parent_class->owner;

class_dev->uevent_attr.store = store_uevent;

error = class_device_create_file(class_dev, &class_dev->uevent_attr);

static ssize_t store_uevent(struct class_device *class_dev,

const char *buf, size_t count)

{

kobject_uevent(&class_dev->kobj, KOBJ_ADD);

return count;

}

可以看出無論寫入什么值都會觸發(fā)KOBJ_ADD事件,內(nèi)核調(diào)用kobject_uevent函數(shù)發(fā)送netlink message給用戶空間用戶層,用戶空間可以用udev通過取到此事件,從而處理熱插拔事件。

“本文由華清遠見http://www.embedu.org/index.htm提供”



華清遠見

本站聲明: 本文章由作者或相關(guān)機構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

LED驅(qū)動電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: 驅(qū)動電源

在工業(yè)自動化蓬勃發(fā)展的當(dāng)下,工業(yè)電機作為核心動力設(shè)備,其驅(qū)動電源的性能直接關(guān)系到整個系統(tǒng)的穩(wěn)定性和可靠性。其中,反電動勢抑制與過流保護是驅(qū)動電源設(shè)計中至關(guān)重要的兩個環(huán)節(jié),集成化方案的設(shè)計成為提升電機驅(qū)動性能的關(guān)鍵。

關(guān)鍵字: 工業(yè)電機 驅(qū)動電源

LED 驅(qū)動電源作為 LED 照明系統(tǒng)的 “心臟”,其穩(wěn)定性直接決定了整個照明設(shè)備的使用壽命。然而,在實際應(yīng)用中,LED 驅(qū)動電源易損壞的問題卻十分常見,不僅增加了維護成本,還影響了用戶體驗。要解決這一問題,需從設(shè)計、生...

關(guān)鍵字: 驅(qū)動電源 照明系統(tǒng) 散熱

根據(jù)LED驅(qū)動電源的公式,電感內(nèi)電流波動大小和電感值成反比,輸出紋波和輸出電容值成反比。所以加大電感值和輸出電容值可以減小紋波。

關(guān)鍵字: LED 設(shè)計 驅(qū)動電源

電動汽車(EV)作為新能源汽車的重要代表,正逐漸成為全球汽車產(chǎn)業(yè)的重要發(fā)展方向。電動汽車的核心技術(shù)之一是電機驅(qū)動控制系統(tǒng),而絕緣柵雙極型晶體管(IGBT)作為電機驅(qū)動系統(tǒng)中的關(guān)鍵元件,其性能直接影響到電動汽車的動力性能和...

關(guān)鍵字: 電動汽車 新能源 驅(qū)動電源

在現(xiàn)代城市建設(shè)中,街道及停車場照明作為基礎(chǔ)設(shè)施的重要組成部分,其質(zhì)量和效率直接關(guān)系到城市的公共安全、居民生活質(zhì)量和能源利用效率。隨著科技的進步,高亮度白光發(fā)光二極管(LED)因其獨特的優(yōu)勢逐漸取代傳統(tǒng)光源,成為大功率區(qū)域...

關(guān)鍵字: 發(fā)光二極管 驅(qū)動電源 LED

LED通用照明設(shè)計工程師會遇到許多挑戰(zhàn),如功率密度、功率因數(shù)校正(PFC)、空間受限和可靠性等。

關(guān)鍵字: LED 驅(qū)動電源 功率因數(shù)校正

在LED照明技術(shù)日益普及的今天,LED驅(qū)動電源的電磁干擾(EMI)問題成為了一個不可忽視的挑戰(zhàn)。電磁干擾不僅會影響LED燈具的正常工作,還可能對周圍電子設(shè)備造成不利影響,甚至引發(fā)系統(tǒng)故障。因此,采取有效的硬件措施來解決L...

關(guān)鍵字: LED照明技術(shù) 電磁干擾 驅(qū)動電源

開關(guān)電源具有效率高的特性,而且開關(guān)電源的變壓器體積比串聯(lián)穩(wěn)壓型電源的要小得多,電源電路比較整潔,整機重量也有所下降,所以,現(xiàn)在的LED驅(qū)動電源

關(guān)鍵字: LED 驅(qū)動電源 開關(guān)電源

LED驅(qū)動電源是把電源供應(yīng)轉(zhuǎn)換為特定的電壓電流以驅(qū)動LED發(fā)光的電壓轉(zhuǎn)換器,通常情況下:LED驅(qū)動電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: LED 隧道燈 驅(qū)動電源
關(guān)閉