毛片网站在线看_天堂俺去俺来也www久久婷婷_日韩av免费网站_18性欧美xxxⅹ性满足_一区二区三区韩国免费中文网站 _性xx色xx综合久久久xx_999亚洲国产精


曙海教育集團論壇Linux專區Linux驅動開發 → linux驅動程序-字符設備驅動開發一


  共有10357人關注過本帖樹形打印

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

美女呀,離線,留言給我吧!
wangxinxin
  1樓 個性首頁 | 博客 | 信息 | 搜索 | 郵箱 | 主頁 | UC


加好友 發短信
等級:青蜂俠 帖子:1393 積分:14038 威望:0 精華:0 注冊:2010-11-12 11:08:23
linux驅動程序-字符設備驅動開發一  發帖心情 Post By: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        */


支持(0中立(0反對(0單帖管理 | 引用 | 回復 回到頂部

返回版面帖子列表

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








簽名
毛片网站在线看_天堂俺去俺来也www久久婷婷_日韩av免费网站_18性欧美xxxⅹ性满足_一区二区三区韩国免费中文网站 _性xx色xx综合久久久xx_999亚洲国产精
99视频一区二区| 欧美视频在线不卡| 粉嫩13p一区二区三区| 26uuu精品一区二区三区四区在线| 亚洲桃色在线一区| 国产成人啪免费观看软件| 日韩亚洲欧美一区| 久久国产精品一区二区| 日韩欧美国产精品| 捆绑调教一区二区三区| 欧美一级片免费看| 日本大胆欧美人术艺术动态| 91高清视频在线| 亚洲一区二区高清| 欧美另类一区二区三区| 极品少妇xxxx精品少妇| 久久久国产午夜精品| www.亚洲在线| 亚洲国产精品一区二区www| 欧美色网一区二区| 久久99久久99| 久久亚区不卡日本| 91亚洲午夜精品久久久久久| 国产精品全国免费观看高清 | 欧美精品一区二区三区蜜臀| 国产成人综合亚洲网站| 乱一区二区av| 久久亚区不卡日本| 91成人网在线| 久久99精品国产91久久来源| 亚洲欧美日韩小说| 337p粉嫩大胆噜噜噜噜噜91av| 丁香婷婷深情五月亚洲| 免播放器亚洲一区| 综合激情成人伊人| 国产婷婷色一区二区三区四区| 91福利视频网站| 91视频在线观看| 国产精品18久久久久久久网站| 丝袜美腿亚洲综合| 亚洲在线免费播放| 亚洲欧洲日产国码二区| 久久综合成人精品亚洲另类欧美| 911精品国产一区二区在线| 色婷婷综合久久久久中文一区二区| 男人的天堂亚洲一区| 日韩在线卡一卡二| 日本欧美久久久久免费播放网| 一区二区三区在线免费| 亚洲黄色av一区| 一区二区三区成人| 亚洲高清不卡在线| 婷婷一区二区三区| 久久国产三级精品| 国产麻豆欧美日韩一区| 国产一区美女在线| 国产ts人妖一区二区| av在线不卡网| 欧美人伦禁忌dvd放荡欲情| 欧美一区二区福利在线| 久久亚洲精华国产精华液| 国产精品亲子伦对白| 日本不卡视频在线| 久久精品国产久精国产爱| 国产盗摄一区二区| 色88888久久久久久影院按摩 | 91精品国产综合久久香蕉的特点 | 国产98色在线|日韩| 99精品一区二区| 91精品麻豆日日躁夜夜躁| 欧美激情在线免费观看| 亚洲午夜免费福利视频| 成人午夜碰碰视频| 欧美精品丝袜中出| 亚洲欧洲成人精品av97| 日本vs亚洲vs韩国一区三区二区| 成人av午夜电影| 欧美精品一区男女天堂| 香蕉久久夜色精品国产使用方法| 国产中文字幕一区| 欧美精品123区| 亚洲综合色视频| 国产成人精品亚洲午夜麻豆| 欧美性一区二区| 一区二区三区四区av| av网站一区二区三区| 国产日产欧美精品一区二区三区| 日本一区中文字幕| 欧美一级艳片视频免费观看| 偷拍一区二区三区| 欧美日韩色综合| 日本美女一区二区三区| 日韩精品一区二区三区中文精品| 久久激情五月激情| 欧美一区二区大片| 美女网站一区二区| 欧美大片日本大片免费观看| 免费成人av资源网| 久久精品人人做人人综合| 成人激情综合网站| 有坂深雪av一区二区精品| 欧美影院一区二区三区| 日韩av中文在线观看| 久久影院电视剧免费观看| 成人性生交大片免费看视频在线| 亚洲天堂免费看| 欧洲国内综合视频| 日韩精品久久理论片| 久久九九全国免费| 欧美伊人久久久久久久久影院| 日韩精品电影一区亚洲| 久久午夜羞羞影院免费观看| 99精品在线观看视频| 麻豆一区二区在线| √…a在线天堂一区| 日韩一级免费一区| 99热精品一区二区| 久久99国产精品麻豆| 亚洲免费在线观看视频| 久久免费午夜影院| 日韩欧美一区二区在线视频| 成人avav在线| 国产一区二区三区四区五区入口| 亚洲国产精品影院| 亚洲欧洲av在线| 日本一区二区免费在线观看视频 | 国产精品欧美一区喷水| 日韩一区二区三区电影在线观看| 99re这里都是精品| 国产精品69毛片高清亚洲| 久久国内精品视频| 蜜臀精品一区二区三区在线观看 | 亚洲国产视频一区二区| 亚洲天堂a在线| 亚洲天堂2014| 亚洲自拍另类综合| 洋洋成人永久网站入口| 一区二区三区国产精华| 视频在线观看91| 99视频有精品| 精品婷婷伊人一区三区三| 欧美日韩国产综合一区二区 | 日本一区二区三区dvd视频在线| 国产亚洲综合色| 亚洲免费三区一区二区| 亚洲日本在线天堂| 亚洲精品成人悠悠色影视| 亚洲靠逼com| 日韩精品一卡二卡三卡四卡无卡 | 欧美xxxxx裸体时装秀| 久久精品视频免费| 夜夜嗨av一区二区三区网页| 亚洲r级在线视频| 国产在线精品不卡| 91福利在线看| 欧美精品一区二区三区在线播放 | 欧美亚洲综合色| 精品日产卡一卡二卡麻豆| 中文字幕日韩精品一区| 日产国产欧美视频一区精品| 成人免费黄色在线| 日韩精品专区在线影院观看| 亚洲国产精品精华液2区45| 一区二区不卡在线视频 午夜欧美不卡在 | 舔着乳尖日韩一区| 国产成人精品在线看| 4438x亚洲最大成人网| 中文字幕亚洲成人| 国产盗摄精品一区二区三区在线 | 91亚洲精品久久久蜜桃| 日韩一区二区三区视频在线观看| 欧美韩国一区二区| 国产精一区二区三区| 日韩女优毛片在线| 日韩高清不卡一区二区三区| 欧美精品在线观看播放| 国产日韩欧美麻豆| 国内精品伊人久久久久av一坑 | 91精品国产欧美一区二区18| 亚洲免费观看高清| 91蜜桃免费观看视频| 亚洲欧美电影一区二区| www.视频一区| 亚洲美女屁股眼交3| 欧美三级韩国三级日本三斤| 亚洲成年人影院| 6080亚洲精品一区二区| 美女视频黄 久久| 久久久国产一区二区三区四区小说 | 久久先锋影音av鲁色资源网| 一本久道久久综合中文字幕| 夜夜精品浪潮av一区二区三区| 91国产福利在线| 日韩av一区二| 国产亚洲精品bt天堂精选| 成人黄色电影在线| 亚洲成人激情av| 国产日产欧美一区| 欧美日韩卡一卡二| 成人夜色视频网站在线观看|