<dfn id="is4kg"></dfn>
  • <ul id="is4kg"></ul>
  • <abbr id="is4kg"></abbr>
  • <ul id="is4kg"></ul>
    <bdo id="is4kg"></bdo>

    Rss & SiteMap

    曙海教育集團論壇 http://www.bjzhda.cn

    曙海教育集團論壇
    共1 條記錄, 每頁顯示 10 條, 頁簽: [1]
    [瀏覽完整版]

    標題:linux驅動程序-字符設備驅動開發一

    1樓
    wangxinxin 發表于:2010-11-24 10:41:32
    正在研究linux設備驅動程序,現在把平時的學習心得以筆記的形式發到博客上,方便跟同行們交流與討論!因為是初學者,對linux的認識還不夠深入,所以在博文中會有很多錯誤,我乃拋磚引玉,請大俠們指教!!

        說明:博文的內容主要參考好朋友Tekkaman Ninja同學博客http://blog.chinaunix.net/u1/34474/index.html上的文章。

        linux驅動程序學習-字符設備驅動程序(第三章)

    一、主設備號和次設備號

       對字符設備的訪問是通過文件系統內的設備名稱進行的。那些名稱稱為特殊文件、設備文件或者簡單稱之為文件系統樹的節點,它們通常位于/dev目錄下。

               主設備號:標識設備對應的驅動程序                次設備號:標識確定設備文件所指的設備

       同一個主設備號下有不同的從設備號,對應同一類驅動程序下的不同具體設備,如:同屬于字符設備的有控制臺和串口終端等。

       注意理解:主設備號、次設備號、設備文件之間的關系!!

    二、設備編號的內部表達

      內核用dev_t類型(<linux/types.h>)來保存設備編號,dev_t是一個32位的數,12位表示主設備號,20為表示次設備號。在實際使用中,是通過<linux/kdev_t.h>中定義的宏來轉換格式。

      如果:想獲得主設備號或者次設備號,應使用:MAJOR(dev_t dev)--獲得主設備號  MINOR(dev_t dev)--獲得次設備號

      如果:已知了主設備號與次設備號,想把他轉換成dev_t類型,則使用MKDEV(int major,int minor);

    三、分配和釋放設備編號

      在建立一個字符設備之前,驅動程序首先要做的事情是獲得一個或者多個設備編號。

      有2種情況:一種是在已經知道設備編號的情況下,調用函數分配;一種是先前不知道驅動所需的設備編號,調用函數去分配

    第一種情況:調用函數 int register_chrdev_region(dev_t first, unsigned int count,
    char *name);   //指定設備編號

    第二種情況:調用函數  int alloc_chrdev_region(dev_t *dev, unsigned int firstminor,
    unsigned int count, char *name);   //動態生成設備編號

    釋放設備編號:void unregister_chrdev_region(dev_t first, unsigned int count);      //釋放設備編號

    四、一些重要的數據結構:

      設備編號的注冊是驅動程序代碼必須完成的許多工作中第一件事情而已,后面還有很多事情等著我們去做呢!!大部分基本的驅動程序操作涉及到三個重要的內核數據結構,分別是file_operations、file、inode。下面詳細闡述:

    struct file_operations fops 設備驅動程序接口
    struct file_operations {
    struct module *owner;
    loff_t (*llseek) (struct file *, loff_t, int);
    ssize_t (*read) (struct file *, char *, size_t, loff_t *);
    ssize_t (*write) (struct file *, const char *, size_t, loff_t *);
    int (*readdir) (struct file *, void *, filldir_t);
    unsigned int (*poll) (struct file *, struct poll_table_struct *);
    int (*ioctl) (struct inode *, struct file *, unsigned int, unsigned long);
    int (*mmap) (struct file *, struct vm_area_struct *);
    int (*open) (struct inode *, struct file *);
    int (*flush) (struct file *);
    int (*release) (struct inode *, struct file *);
    int (*fsync) (struct file *, struct dentry *, int datasync);
    int (*fasync) (int, struct file *, int);
    int (*lock) (struct file *, int, struct file_lock *);
    ssize_t (*readv) (struct file *, const struct iovec *, unsigned long, loff_t *);
    ssize_t (*writev) (struct file *, const struct iovec *, unsigned long, loff_t *);
    ssize_t (*sendpage) (struct file *, struct page *, int, size_t, loff_t *, int);
    unsigned long (*get_unmapped_area)(struct file *, unsigned long, unsigned long, unsigned long, unsigned long);
    };
    上面結構體內的每個字段大部分是函數指針,這些函數指針指向驅動程序實現具體操作的函數。我們可以看到上面的指針所指向函數的參數里面有一種結構體很常見:struct file 還有struct inode。

    下面來分析struct file:file結構與用戶空間中的FILE沒有任何關聯,struct file是一個內核結構,他不會出現在用戶程序中。file結構代表一個打開的文件,是由內核在open時創建的,并傳遞給在該文件上進行操作的所有函數,直到最后的close函數,在文件的所有實例都被關閉之后,內核會釋放掉這個數據結構。在這個數據結構中有一個重要的字段:struct file_operations *f_op,內核在執行open操作時,對這個指針賦值,以后需要處理這些操作時就讀取這個指針。filp->f_op中的值決不會為了方便引用而保存起來,也就是說,我們可以在任何時候修改文件的關聯操作,在返回給調用者之后,新的操作方法立即生效。例如:對應于主設備號1的open代碼根據要打開的次設備號替換filp->f_op中的操作。注意:也就是說,struct file與struct file_operations這2個結構體是通過這樣的方式進行相關聯的。

    inode結構:內核用inode結構在內部表示文件,因此它和file結構不同,后者表示打開的文件描述符。對于單個文件,可能會有許多個表示打開的文件描述符的file結構。但他們都指向單個inode結構。對于編寫驅動程序,只有2個字段比較常用:dev_t i_rdev; struct cdev *i_cdev;

    struct cdev表示字符設備的內核的內部結構。當inode指向一個字符設備文件時,該字段包含了指向struct cdev結構的指針。

    內核內部使用struct cdev結構來表示字符設備。在內核調用設備的操作之前,必須分配并注冊一個或者多個上述結構。

    注冊一個獨立的cdev設備的基本過程如下:

     

    1、為struct cdev 分配空間(如果已經將struct cdev 嵌入到自己的設備的特定結構體中,并分配了空間,這步略過!)

    struct cdev *my_cdev = cdev_alloc();

    2、初始化struct cdev

    void cdev_init(struct cdev *cdev, const struct file_operations *fops)

    3、初始化cdev.owner

    cdev.owner = THIS_MODULE;

    4、cdev設置完成,通知內核struct cdev的信息(在執行這步之前必須確定你對struct cdev的以上設置已經完成!)

    int cdev_add(struct cdev *p, dev_t dev, unsigned count)

    從系統中移除一個字符設備:void cdev_del(struct cdev *p)

    /*
     * Set up the char_dev structure for this device.
     */

    static void scull_setup_cdev(struct scull_dev *dev, int index)
    {
        int err, devno = MKDEV(scull_major, scull_minor + index);
        
        cdev_init(&dev->cdev, &scull_fops);
        dev->cdev.owner = THIS_MODULE;
        dev->cdev.ops = &scull_fops
     //這句可以省略,在cdev_init中已經做過
        err = cdev_add (&dev->cdev, devno, 1);
        /* Fail gracefully if need be 這步值得注意*/
        if (err)
            printk(KERN_NOTICE "Error %d adding scull%d", err, index);
    }

    scull模型的結構體:

    /*
     * Representation of scull quantum sets.
     */

    struct scull_qset {
        void **data;
        struct scull_qset *next;
    };

    struct scull_dev {
        struct scull_qset *data; /* Pointer to first quantum set */
        int quantum; /* the current quantum size */
        int qset; /* the current array size */
        unsigned long size; /* amount of data stored here */
        unsigned int access_key; /* used by sculluid and scullpriv */
        struct semaphore sem; /* mutual exclusion semaphore */
        struct cdev cdev;     /* Char device structure        */

    共1 條記錄, 每頁顯示 10 條, 頁簽: [1]

    Copyright © 2000 - 2009 曙海教育集團
    Powered By 曙海教育集團 Version 2.2
    Processed in .02930 s, 2 queries.
    主站蜘蛛池模板: 一道本在线播放| 免费看男女做好爽好硬视频| www日本xxx| 日本h无羞动漫在线观看网站| 亚洲欧美精品中文字幕| 精品乱子伦一区二区三区 | 人人妻人人玩人人澡人人爽| 翁与小莹浴室欢爱51章| 国产精品久久久尹人香蕉| 亚洲av无码一区二区三区dv| 欧美色图亚洲自拍| 四虎亚洲国产成人久久精品| 香蕉久久人人爽人人爽人人片av| 大臿蕉香蕉大视频成人| 久久久久国产精品免费免费不卡| 最近免费中文字幕大全免费版视频| 免费**毛片在线搐放正片| 美国式禁忌免费| 国产精品欧美一区二区三区 | 激情内射亚洲一区二区三区爱妻| 国产一级片播放| 香蕉在线精品视频在线观看2| 国产视频一区在线播放| 一本久久综合亚洲鲁鲁五月天| 新婚张燕被两个局长| 亚洲AV无码无在线观看红杏| 欧美成人看片黄a免费看| 免费看国产曰批40分钟| 精品欧美一区二区三区精品久久| 加勒比精品久久一区二区三区| 久久一区二区三区精品| 日韩精品久久无码人妻中文字幕| 亚洲精品美女久久久久99| 男女一边摸一边做刺激的视频| 国产亚洲精品精品精品| 鸭王3完整版免费完整版在线观看| 国产综合在线观看| spoz是什么意思医学| 少妇被又大又粗又爽毛片| 久久久久亚洲av成人无码| 日本强伦姧人妻一区二区|