追蹤platform_driver, 以serial8250 為例

參考網站:
http://www.dzsc.com/data/html/2009-12-31/81251.html

2.6 起引入了一套新的驅動管理和註冊機制:platform_device和platform_driver。
Linux中大部分的設備驅動,都可以使用這套機制,設備用platform_device表示,驅動用platform_driver進行註冊。
Linux platform. driver機制和傳統的device driver 機制(通過driver_register函數進行註冊)相比,
一個十分明顯的優勢在於platform機制將設備本身的資源註冊進內核,由內核統一管理,
在驅動程序中使用這些資源時通過platform. device提供的標準接口進行申請並使用。
這樣提高了驅動和資源管理的獨立性,並且擁有較好的可移植性和安全性(這些標準接口是安全的)。
platform機制的本身使用並不復雜,由兩部分組成:platform_device和platfrom_driver。
通過platform機制開發底層設備驅動的大致流程如圖所示。

定義platform_device => 註冊platform_device => 定義platform_driver => 註冊platform_driver

– platform_device結構體用來描述設備的名稱、資源信息等。該結構被定義在include/linux/platform_device.h中

struct platform_device {
const char * name;
int id;
struct device dev;
u32 num_resources;
struct resource * resource;

const struct platform_device_id *id_entry;

/* arch specific additions */
struct pdev_archdata archdata;
};

通過調用函數platform_add_devices()向系統中添加該設備了,該函數內部調用platform_device_register( )進行設備註冊。
要注意的是,這裡的platform_device設備的註冊過程必須在相應設備驅動加載之前被調用,即執行platform_driver_register()之前,
原因是驅動註冊時需要匹配內核中所有已註冊的設備名。

—————————————————————————–

驅動程式 檔案8250.c
在初始化時:
serial8250_isa_devs = platform_device_alloc(“serial8250”,
PLAT8250_DEV_LEGACY);

—————————————————————————–
接下來來看platform_driver結構體的原型定義,在include/linux/platform_device.h中,代碼如下:

struct platform_driver {
int (*probe)(struct platform_device *);
int (*remove)(struct platform_device *);
void (*shutdown)(struct platform_device *);
int (*suspend)(struct platform_device *, pm_message_t state);
int (*resume)(struct platform_device *);
struct device_driver driver;
const struct platform_device_id *id_table;
};

內核提供的platform_driver結構體的註冊函數為platform_driver_register(),其原型定義在driver/base/platform.c文件中
具體實現代碼如下:

int platform_driver_register(struct platform_driver *drv)
{
drv->driver.bus = &platform_bus_type;
if (drv->probe)
drv->driver.probe = platform_drv_probe;
if (drv->remove)
drv->driver.remove = platform_drv_remove;
if (drv->shutdown)
drv->driver.shutdown = platform_drv_shutdown;

return driver_register(&drv->driver);
}

總結,通常情況下只要和內核本身運行依賴性不大的外圍設備,相對獨立的,擁有各自獨自的資源
(地址總線(BUS)是將信息以一個或多個源部件傳送到一個或多個目的部件的一組傳輸線。
通俗的說,就是多個部件間的公共連線,用於在各個部件之間傳輸信息。
人們常常以MHz表示的速度來描述總線(BUS)頻率和IRQs),
都可以用platform_driver實現。如:LCD,網卡、USB、UART等,都可以用platfrom_driver寫,
而timer,irq等小系統之內的設備則最好不用platfrom_driver機制。

分類: 未分類

發表迴響