總線-設備-驅動 又稱爲 設備驅動模型。html
總線(bus):負責管理掛載對應總線的設備以及驅動;
設備(device):掛載在某個總線的物理設備;
驅動(driver):與特定設備相關的軟件,負責初始化該設備以及提供一些操做該設備的操做方式;
類(class):對於具備相同功能的設備,歸結到一種類別,進行分類管理;node
如下只說 總線-設備-驅動 模式下的操做
總線:linux
總線:數據結構
總線驅動:ide
總線結構體:函數
struct bus_type { const char *name; const struct attribute_group **bus_groups; const struct attribute_group **dev_groups; const struct attribute_group **drv_groups; int (*match)(struct device *dev, struct device_driver *drv); int (*uevent)(struct device *dev, struct kobj_uevent_env *env); int (*probe)(struct device *dev); int (*remove)(struct device *dev); int (*suspend)(struct device *dev, pm_message_t state); int (*resume)(struct device *dev); const struct dev_pm_ops *pm; struct subsys_private *p; };
在實際的驅動開發中,Linux 已經爲咱們編寫好了大部分的總線驅動。
可是內核也提供了註冊總線的 API。ui
bus_register():操作系統
int bus_register(struct bus_type *bus);
。
bus_unregister():指針
int bus_unregister(struct bus_type *bus);
。
當咱們成功註冊總線時,會在 /sys/bus/ 目錄下建立一個新目錄,目錄名爲咱們新註冊的總線名。code
在 /sys/devices 目錄記錄了系統中全部的設備。
/sys 下的全部設備文件和 /sys/dev 下的全部設備節點都是連接文件,實際上都指向了對應的設備文件。
device 結構體:
struct device { const char *init_name; struct device *parent; struct bus_type *bus; struct device_driver *driver; void *platform_data; void *driver_data; struct device_node *of_node; dev_t devt; struct class *class; void (*release)(struct device *dev); const struct attribute_group **groups; /* optional groups */ struct device_private *p; };
在前面的字符設備驅動編寫中,咱們使用到了 device_create() 函數和 device_destroy() 函數來建立和刪除設備。
如今介紹向總線註冊和註銷設備。
向總線註冊設備:
int device_register(struct device *dev);
。
向總線註銷設備:
int device_unregister(struct device *dev);
。
driver 結構體:
struct device_driver { const char *name; struct bus_type *bus; struct module *owner; const char *mod_name; /* used for built-in modules */ bool suppress_bind_attrs; /* disables bind/unbind via sysfs */ const struct of_device_id *of_match_table; const struct acpi_device_id *acpi_match_table; int (*probe) (struct device *dev); int (*remove) (struct device *dev); const struct attribute_group **groups; struct driver_private *p; };
向總線註冊驅動:
int driver_register(struct device_driver *drv);
。
向總線註銷驅動:
int driver_unregister(struct device_driver *drv);
。
數據結構該系統:
註冊流程圖:
系統啓動以後會調用buses_init函數建立/sys/bus文件目錄,這部分系統在開機時已經幫咱們準備好了, 接下去就是經過總線註冊函數bus_register進行總線註冊,註冊完總線後在總線的目錄下生成devices文件夾和drivers文件夾, 最後分別經過device_register以及driver_register函數註冊相對應的設備和驅動。
attribute 結構體:
struct attribute { const char *name; umode_t mode; };
bus_type、device、device_driver 結構體中都包含了一種數據類型 struct attribute_group,它是多個 attribute 文件的集合, 利用它進行初始化,能夠避免一個個註冊 attribute。
struct attribute_group 結構體:
struct attribute_group { const char *name; umode_t (*is_visible)(struct kobject *, struct attribute *, int); struct attribute **attrs; struct bin_attribute **bin_attrs; };
Linux 提供註冊和註銷設備屬性文件的 API。咱們能夠經過這些 API 直接在用戶層進行查詢和修改。
struct device_attribute 結構體:
struct device_attribute { struct attribute attr; ssize_t (*show)(struct device *dev, struct device_attribute *attr, char *buf); ssize_t (*store)(struct device *dev, struct device_attribute *attr, const char *buf, size_t count); }; #define DEVICE_ATTR(_name, _mode, _show, _store) \ struct device_attribute dev_attr_##_name = __ATTR(_name, _mode, _show, _store) extern int device_create_file(struct device *device, const struct device_attribute *entry); extern void device_remove_file(struct device *dev, const struct device_attribute *attr);
Linux 提供註冊和註銷驅動屬性文件的 API。咱們能夠經過這些 API 直接在用戶層進行查詢和修改。
struct driver_attribute 結構體:
struct driver_attribute { struct attribute attr; ssize_t (*show)(struct device_driver *driver, char *buf); ssize_t (*store)(struct device_driver *driver, const char *buf, size_t count);}; #define DRIVER_ATTR_RW(_name) \ struct driver_attribute driver_attr_##_name = __ATTR_RW(_name) #define DRIVER_ATTR_RO(_name) \ struct driver_attribute driver_attr_##_name = __ATTR_RO(_name) #define DRIVER_ATTR_WO(_name) \ struct driver_attribute driver_attr_##_name = __ATTR_WO(_name) extern int __must_check driver_create_file(struct device_driver *driver, const struct driver_attribute *attr); extern void driver_remove_file(struct device_driver *driver, const struct driver_attribute *attr);
struct bus_attribute 結構體:
struct bus_attribute { struct attribute attr; ssize_t (*show)(struct bus_type *bus, char *buf); ssize_t (*store)(struct bus_type *bus, const char *buf, size_t count); }; #define BUS_ATTR(_name, _mode, _show, _store) \ struct bus_attribute bus_attr_##_name = __ATTR(_name, _mode, _show, _store) extern int __must_check bus_create_file(struct bus_type *, struct bus_attribute *); extern void bus_remove_file(struct bus_type *, struct bus_attribute *);
下章筆記就是記錄平臺設備。本次匹配規則就參考 平臺設備驅動 源碼。
最早比較 platform_device.driver_override 和 platform_driver.driver.name。
能夠設置 platform_device 的 driver_override,強制選擇某個 platform_driver。
其次比較 platform_device.name 和 platform_driver.id_table[i].name。
platform_driver.id_table 是 platform_device_id 指針,表示該 drv 支持若干個 device,它裏面列出了各個 device 的 {.name, .driver_data},其中的 name 表示該 drv 支持的設備的名字,driver_data是些提供給該 device 的私有數據。
最後比較 platform_device.name 和 platform_driver.driver.name。
因爲 platform_driver.id_table 可能爲空,因此,接下來就可使用 platform_driver.driver.name 來匹配。
platform_device_register platform_device_add device_add bus_add_device // 放入鏈表 bus_probe_device // probe 枚舉設備,即找到匹配的(dev, drv) device_initial_probe __device_attach bus_for_each_drv(...,__device_attach_driver,...) __device_attach_driver driver_match_device(drv, dev) // 是否匹配 driver_probe_device // 調用 drv 的 probe platform_driver_register __platform_driver_register driver_register bus_add_driver // 放入鏈表 driver_attach(drv) bus_for_each_dev(drv->bus, NULL, drv, __driver_attach); __driver_attach driver_match_device(drv, dev) // 是否匹配 driver_probe_device // 調用 drv 的 probe
總線、設備、驅動都基於驅動模型上實現,方便插入。
模塊三步驟: