3. 字符設(shè)備驅(qū)動(dòng)-總線設(shè)備驅(qū)動(dòng)模型寫法

一讹挎、引言:

?? ? ? ?在 字符設(shè)備驅(qū)動(dòng)的傳統(tǒng)寫法 中,介紹了字符設(shè)備的傳統(tǒng)寫法躁锡。從代碼中我們可以看到午绳,使用的引腳,寫死在代碼中了映之,如果更改硬件資源拦焚,比如將GPIO3_4改成GPIO3_5,那就需要重新編譯這個(gè)驅(qū)動(dòng)程序杠输,如果該驅(qū)動(dòng)程序是放在內(nèi)核里的赎败,那么就需要重新編譯內(nèi)核。并且蠢甲,在需要更改硬件資源的時(shí)候都需要去閱讀驅(qū)動(dòng)源碼僵刮,對(duì)于沒有寫驅(qū)動(dòng)能力的人來說,這也是挺痛苦的鹦牛。在Linux內(nèi)核持續(xù)發(fā)展中搞糕,改進(jìn)了寫驅(qū)動(dòng)程序的方法,使用總線設(shè)備驅(qū)動(dòng)模型能岩。

?? ? ? ?為了方便對(duì)比學(xué)習(xí)寞宫,我們?cè)谏瞎?jié)傳統(tǒng)字符設(shè)備寫法基礎(chǔ)上進(jìn)行修改萧福±椋總線設(shè)備驅(qū)動(dòng)模型將驅(qū)動(dòng)程序分成了兩部分led_devled_drv ;dev部分指定硬件資源,drv分配設(shè)置fileoperations結(jié)構(gòu)體然后根據(jù)硬件資源來操作硬件 膏燕。

二钥屈、BUS - dev

?? ? ? ?現(xiàn)在我們來看下,內(nèi)核中又是如何指定硬件資源的坝辫?以面向?qū)ο蟮乃枷氲姆椒ㄅ窬停趦?nèi)核里它定義一個(gè)dev的時(shí)候,也是去分配設(shè)置某個(gè)結(jié)構(gòu)體近忙,這個(gè)結(jié)構(gòu)體就是平臺(tái)設(shè)備竭业。
平臺(tái)設(shè)備使用struct platform_device來描述:

 struct platform_device {
  const char       * name;           //設(shè)備名稱,要與platform_driver的name一樣,
                                        //這樣總線才能匹配成功
  u32          id;                   //id號(hào),插入總線下相同name的設(shè)備編號(hào)(一個(gè)驅(qū)動(dòng)可以有多個(gè)設(shè)備),
                                        //如果只有一個(gè)設(shè)備填-1
  struct  device  dev;               //內(nèi)嵌的具體的device結(jié)構(gòu)體,其中成員platform_data,是個(gè)void *類型,
                                        //可以給平臺(tái)driver提供各種數(shù)據(jù)(比如:GPIO引腳等等)
  u32 num_resources;                 //資源數(shù)量及舍,
  struct resource         * resource;    //資源結(jié)構(gòu)體,保存設(shè)備的信息
};

resource資源未辆,就是用來記錄地址,地址等資源锯玛,供drv使用咐柜。
其中resource資源結(jié)構(gòu)體,如下:

struct resource {
         resource_size_t start;                    //起始資源,如果是地址的話,必須是物理地址
         resource_size_t end;                      //結(jié)束資源,如果是地址的話,必須是物理地址
         const char *name;                         //資源名
         unsigned long flags;                      //資源的標(biāo)志
         //比如IORESOURCE_MEM,表示地址資源, IORESOURCE_IRQ表示中斷引腳... ...

         struct resource *parent, *sibling, *child;   //資源拓?fù)渲羔樃浮⑿秩敛小⒆?可以構(gòu)成鏈表
};

分析到這里拙友,可以看出,對(duì)于 bus-dev 這邊歼郭,就是去定義一個(gè)一個(gè)的platform_device,然后去注冊(cè)到bus總線上遗契。對(duì)于 bus-drv 那邊,也類似病曾,定義了一個(gè)一個(gè)的platform_driver然后注冊(cè)到bus總線上姊途。
????? 內(nèi)核里有那么多的platform_device,上百個(gè)都有可能,同時(shí)也存在那么多個(gè)platform_driver知态;問題來了捷兰,platform_driver該從哪個(gè)platform_device里獲得指定的硬件資源呢?或者說负敏,對(duì)于指定的platform_device又是給內(nèi)核中那么多的platform_driver中的哪個(gè)提供硬件資源的描述呢贡茅?他們之間需要有個(gè)匹配,在BUS(我們平時(shí)使用的一般為platform_bus_type)里有個(gè)match函數(shù) 其做,就是用來匹配drv和dev顶考。如果匹配,則調(diào)用drv->probe函數(shù)妖泄;至于probe函數(shù)里做什么驹沿,由驅(qū)動(dòng)開發(fā)者決定;總線設(shè)備驅(qū)動(dòng)模型不過提供了這樣一種機(jī)制蹈胡。它并不是驅(qū)動(dòng)程序的核心渊季,核心仍然是drv里的分配朋蔫、設(shè)置、注冊(cè)file_operations結(jié)構(gòu)體却汉。
????? 接下來看總線里的match函數(shù)是如何確定dev和drv是否匹配的驯妄。總線下面掛載著一系列的dev和一系列的drv合砂,設(shè)備青扔、驅(qū)動(dòng)匹配時(shí)就是通過match函數(shù)來兩兩比較的,一旦match成功翩伪,則調(diào)用drv里的probe函數(shù)微猖。

機(jī)制講完,開始寫代碼缘屹。

2.1 編寫 led_dev

2.1.1 首先分配設(shè)置一個(gè)平臺(tái) dev

static struct platform_device led_dev = {
    .name         = "myled",                           //對(duì)應(yīng)的platform_driver驅(qū)動(dòng)的名字
    .id       = -1,                                    //表示只有一個(gè)設(shè)備
    .num_resources    = ARRAY_SIZE(led_resource), //資源數(shù)量,ARRAY_SIZE()函數(shù):獲取數(shù)量
    .resource     = led_resource,  //資源數(shù)組led_resource
    .dev = { 
        .release = led_release,   //釋放函數(shù)励两,必須向內(nèi)核提供一個(gè)release函數(shù), 、
                                //否則卸載時(shí),內(nèi)核找不到該函數(shù)會(huì)報(bào)錯(cuò)
    },
};

.name 設(shè)置平臺(tái)設(shè)備中的名字囊颅,用于與平臺(tái)驅(qū)動(dòng)匹配当悔;
.resource 資源,用于描述設(shè)備信息踢代,具體類型通過resource中的flags標(biāo)識(shí)盲憎;資源類型主要有:

好像都沒有指明引腳的資源,那怎么辦胳挎?反正這個(gè)平臺(tái)資源是自己使用的饼疙,我們先假設(shè)其為MEM資源,在驅(qū)動(dòng)解析使用時(shí)慕爬,并不把他當(dāng)成是MEM窑眯,內(nèi)存資源,直接當(dāng)成一個(gè)引腳

static struct resource led_resource[] = {
    [0] = {
        .start = S3C2440_GPF(5),
        .end   = S3C2440_GPF(5),
        .flags = IORESOURCE_MEM,
    },
};

2.1.2 在入口時(shí)注冊(cè)

platform_device_register(&led_dev);

2.1.3 出口時(shí)卸載

platform_device_unregister(&led_dev);

完整的led_dev.c如下:

#include <linux/module.h>
#include <linux/version.h>
#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/types.h>
#include <linux/interrupt.h>
#include <linux/list.h>
#include <linux/timer.h>
#include <linux/init.h>
#include <linux/serial_core.h>
#include <linux/platform_device.h>

#define S3C2440_GPA(n)  (0<<16 | n)
#define S3C2440_GPB(n)  (1<<16 | n)
#define S3C2440_GPC(n)  (2<<16 | n)
#define S3C2440_GPD(n)  (3<<16 | n)
#define S3C2440_GPE(n)  (4<<16 | n)
#define S3C2440_GPF(n)  (5<<16 | n)
#define S3C2440_GPG(n)  (6<<16 | n)
#define S3C2440_GPH(n)  (7<<16 | n)
#define S3C2440_GPI(n)  (8<<16 | n)
#define S3C2440_GPJ(n)  (9<<16 | n)

/* 分配/設(shè)置/注冊(cè)一個(gè)platform_device */

static struct resource led_resource[] = {
    [0] = {
        .start = S3C2440_GPF(5),
        .end   = S3C2440_GPF(5),
        .flags = IORESOURCE_MEM,
    },
};

static void led_release(struct device * dev)
{
}


static struct platform_device led_dev = {
    .name         = "myled",
    .id       = -1,
    .num_resources    = ARRAY_SIZE(led_resource),
    .resource     = led_resource,
    .dev = { 
        .release = led_release, 
    },
};

static int led_dev_init(void)
{
    platform_device_register(&led_dev);
    return 0;
}

static void led_dev_exit(void)
{
    platform_device_unregister(&led_dev);
}

module_init(led_dev_init);
module_exit(led_dev_exit);

MODULE_LICENSE("GPL");

2.2 編寫 led_drv

2.2.1 首先還是先分配設(shè)置一個(gè)平臺(tái) drv

struct platform_driver led_drv = {
    .probe      = led_probe,
    .remove     = led_remove,
    .driver     = {
        .name   = "myled",
    }
};

?????? 經(jīng)過對(duì)平臺(tái)總線機(jī)制的分析医窿,我們知道在平臺(tái)drv中也有個(gè)名字磅甩,就是用這個(gè)名字("myled")來和平臺(tái)dev中的名字做匹配的,一旦匹配姥卢,則調(diào)用平臺(tái)drv中的probe函數(shù)卷要。接下來編寫probe函數(shù),其參數(shù)中有個(gè) platform_device 平臺(tái) dev 独榴,在傳統(tǒng)的字符設(shè)備寫法中僧叉,在入口函數(shù)中直接注冊(cè)了字符設(shè)備;file_operations中的open棺榔、write直接使用了led_pin瓶堕,led_pin是直接寫死在驅(qū)動(dòng)代碼中的;現(xiàn)在症歇,我們要使用平臺(tái)總線的方法來寫郎笆,這個(gè)引腳資源需要從對(duì)應(yīng)的平臺(tái) dev 里來獲得這個(gè)資源谭梗,確定這個(gè)引腳。

static struct file_operations myled_oprs = {
    .owner = THIS_MODULE,
    .open  = led_open,
    .write = led_write,
    .release = led_release,
};

static int led_probe(struct platform_device *pdev)
{
    struct resource     *res;

    /* 根據(jù)platform_device的資源進(jìn)行ioremap 
        參數(shù) 0代表IORESOURCE_MEM這類資源中的第0個(gè)题画,
        把他取出來后res->start,代表的就是引腳了*/
    res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
    led_pin = res->start;

    major = register_chrdev(0, "myled", &myled_oprs);

    led_class = class_create(THIS_MODULE, "myled");
    device_create(led_class, NULL, MKDEV(major, 0), NULL, "led"); /* /dev/led */
    
    return 0;
}

在remove里面來unregister:

static int led_remove(struct platform_device *pdev)
{
    unregister_chrdev(major, "myled");
    device_destroy(led_class,  MKDEV(major, 0));
    class_destroy(led_class);
    
    return 0;
}

接下來的就還是原本的那套德频,完整代碼如下:

#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <linux/init.h>
#include <linux/delay.h>
#include <linux/uaccess.h>
#include <asm/irq.h>
#include <asm/io.h>
#include <linux/of.h>
#include <linux/of_device.h>
#include <linux/of_platform.h>
#include <linux/platform_device.h>

#define S3C2440_GPA(n)  (0<<16 | n)
#define S3C2440_GPB(n)  (1<<16 | n)
#define S3C2440_GPC(n)  (2<<16 | n)
#define S3C2440_GPD(n)  (3<<16 | n)
#define S3C2440_GPE(n)  (4<<16 | n)
#define S3C2440_GPF(n)  (5<<16 | n)
#define S3C2440_GPG(n)  (6<<16 | n)
#define S3C2440_GPH(n)  (7<<16 | n)
#define S3C2440_GPI(n)  (8<<16 | n)
#define S3C2440_GPJ(n)  (9<<16 | n)

static int led_pin;
static volatile unsigned int *gpio_con;
static volatile unsigned int *gpio_dat;

/* 123. 分配/設(shè)置/注冊(cè)file_operations 
 * 4. 入口
 * 5. 出口
 */

static int major;
static struct class *led_class;

static unsigned int gpio_base[] = {
    0x56000000, /* GPACON */
    0x56000010, /* GPBCON */
    0x56000020, /* GPCCON */
    0x56000030, /* GPDCON */
    0x56000040, /* GPECON */
    0x56000050, /* GPFCON */
    0x56000060, /* GPGCON */
    0x56000070, /* GPHCON */
    0,          /* GPICON */
    0x560000D0, /* GPJCON */
};

static int led_open (struct inode *node, struct file *filp)
{
    /* 把LED引腳配置為輸出引腳 */
    /* GPF5 - 0x56000050 */
    int bank = led_pin >> 16;
    int base = gpio_base[bank];

    int pin = led_pin & 0xffff;
    gpio_con = ioremap(base, 8);
    if (gpio_con) {
        printk("ioremap(0x%x) = 0x%x\n", base, gpio_con);
    }
    else {
        return -EINVAL;
    }
    
    gpio_dat = gpio_con + 1;

    *gpio_con &= ~(3<<(pin * 2));
    *gpio_con |= (1<<(pin * 2));  

    return 0;
}

static ssize_t led_write (struct file *filp, const char __user *buf, size_t size, loff_t *off)
{
    /* 根據(jù)APP傳入的值來設(shè)置LED引腳 */
    unsigned char val;
    int pin = led_pin & 0xffff;
    
    copy_from_user(&val, buf, 1);

    if (val)
    {
        /* 點(diǎn)燈 */
        *gpio_dat &= ~(1<<pin);
    }
    else
    {
        /* 滅燈 */
        *gpio_dat |= (1<<pin);
    }

    return 1; /* 已寫入1個(gè)數(shù)據(jù) */
}

static int led_release (struct inode *node, struct file *filp)
{
    printk("iounmap(0x%x)\n", gpio_con);
    iounmap(gpio_con);
    return 0;
}


static struct file_operations myled_oprs = {
    .owner = THIS_MODULE,
    .open  = led_open,
    .write = led_write,
    .release = led_release,
};

static int led_probe(struct platform_device *pdev)
{
    struct resource     *res;

    /* 根據(jù)platform_device的資源進(jìn)行ioremap 
        參數(shù) 0代表IORESOURCE_MEM這類資源中的第0個(gè)苍息,
        把他取出來后res->start,代表的就是引腳了*/
    res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
    led_pin = res->start;

    major = register_chrdev(0, "myled", &myled_oprs);

    led_class = class_create(THIS_MODULE, "myled");
    device_create(led_class, NULL, MKDEV(major, 0), NULL, "led"); /* /dev/led */
    
    return 0;
}

static int led_remove(struct platform_device *pdev)
{
    unregister_chrdev(major, "myled");
    device_destroy(led_class,  MKDEV(major, 0));
    class_destroy(led_class);
    
    return 0;
}


struct platform_driver led_drv = {
    .probe      = led_probe,
    .remove     = led_remove,
    .driver     = {
        .name   = "myled",
    }
};


static int myled_init(void)
{
    platform_driver_register(&led_drv);
    return 0;
}

static void myled_exit(void)
{
    platform_driver_unregister(&led_drv);
}

module_init(myled_init);
module_exit(myled_exit);

MODULE_LICENSE("GPL");

2.3 makefile

KERN_DIR = /work/system/linux-4.19-rc3

all:
    make -C $(KERN_DIR) M=`pwd` modules 

clean:
    make -C $(KERN_DIR) M=`pwd` modules clean
    rm -rf modules.order

obj-m   += led_drv.o
obj-m   += led_dev.o

2.4 test.c

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>

/* ledtest on
  * ledtest off
  */
int main(int argc, char **argv)
{
    int fd;
    unsigned char val = 1;
    fd = open("/dev/led", O_RDWR);
    if (fd < 0)
    {
        printf("can't open!\n");
    }
    if (argc != 2)
    {
        printf("Usage :\n");
        printf("%s <on|off>\n", argv[0]);
        return 0;
    }

    if (strcmp(argv[1], "on") == 0)
    {
        val  = 1;
    }
    else
    {
        val = 0;
    }
    
    write(fd, &val, 1);
    return 0;
}

2.5 測試

2.5.1 加載led_drv

2.5.2 加載led_dev

2.5.3 運(yùn)行測試代碼

接下來壹置,如果想更變LED引腳竞思,就不需要再修改led_drv.c了,直接在led_dev.c里更改相應(yīng)引腳就OK了钞护。

三盖喷、寫在最后

賦個(gè)圖介紹下,注冊(cè)平臺(tái)dev难咕、注冊(cè)平臺(tái)drv所觸發(fā)的match课梳、probe過程是怎樣進(jìn)行的:

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市余佃,隨后出現(xiàn)的幾起案子暮刃,更是在濱河造成了極大的恐慌,老刑警劉巖爆土,帶你破解...
    沈念sama閱讀 211,123評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件椭懊,死亡現(xiàn)場離奇詭異,居然都是意外死亡步势,警方通過查閱死者的電腦和手機(jī)氧猬,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,031評(píng)論 2 384
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來坏瘩,“玉大人盅抚,你說我怎么就攤上這事【蠓” “怎么了泉哈?”我有些...
    開封第一講書人閱讀 156,723評(píng)論 0 345
  • 文/不壞的土叔 我叫張陵,是天一觀的道長破讨。 經(jīng)常有香客問我丛晦,道長,這世上最難降的妖魔是什么提陶? 我笑而不...
    開封第一講書人閱讀 56,357評(píng)論 1 283
  • 正文 為了忘掉前任烫沙,我火速辦了婚禮,結(jié)果婚禮上隙笆,老公的妹妹穿的比我還像新娘锌蓄。我一直安慰自己升筏,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,412評(píng)論 5 384
  • 文/花漫 我一把揭開白布瘸爽。 她就那樣靜靜地躺著您访,像睡著了一般。 火紅的嫁衣襯著肌膚如雪剪决。 梳的紋絲不亂的頭發(fā)上灵汪,一...
    開封第一講書人閱讀 49,760評(píng)論 1 289
  • 那天,我揣著相機(jī)與錄音柑潦,去河邊找鬼享言。 笑死,一個(gè)胖子當(dāng)著我的面吹牛渗鬼,可吹牛的內(nèi)容都是我干的览露。 我是一名探鬼主播,決...
    沈念sama閱讀 38,904評(píng)論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼譬胎,長吁一口氣:“原來是場噩夢啊……” “哼差牛!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起堰乔,我...
    開封第一講書人閱讀 37,672評(píng)論 0 266
  • 序言:老撾萬榮一對(duì)情侶失蹤多糠,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后浩考,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體夹孔,經(jīng)...
    沈念sama閱讀 44,118評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,456評(píng)論 2 325
  • 正文 我和宋清朗相戀三年析孽,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了搭伤。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,599評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡袜瞬,死狀恐怖怜俐,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情邓尤,我是刑警寧澤拍鲤,帶...
    沈念sama閱讀 34,264評(píng)論 4 328
  • 正文 年R本政府宣布,位于F島的核電站汞扎,受9級(jí)特大地震影響季稳,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜澈魄,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,857評(píng)論 3 312
  • 文/蒙蒙 一景鼠、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧痹扇,春花似錦铛漓、人聲如沸溯香。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,731評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽玫坛。三九已至,卻和暖如春包晰,著一層夾襖步出監(jiān)牢的瞬間湿镀,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,956評(píng)論 1 264
  • 我被黑心中介騙來泰國打工杜窄, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留肠骆,地道東北人算途。 一個(gè)月前我還...
    沈念sama閱讀 46,286評(píng)論 2 360
  • 正文 我出身青樓塞耕,卻偏偏與公主長得像,于是被迫代替她去往敵國和親嘴瓤。 傳聞我的和親對(duì)象是個(gè)殘疾皇子扫外,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,465評(píng)論 2 348