1. 程式人生 > >linux裝置驅動--字元裝置驅動

linux裝置驅動--字元裝置驅動

一、linux系統將裝置分為3類:字元裝置、塊裝置、網路裝置。使用驅動程式:

1、字元裝置:是指只能一個位元組一個位元組讀寫的裝置,不能隨機讀取裝置記憶體中的某一資料,讀取資料需要按照先後資料。字元裝置是面向流的裝置,常見的字元裝置有滑鼠、鍵盤、串列埠、控制檯和LED裝置等。
2、塊裝置:是指可以從裝置的任意位置讀取一定長度資料的裝置。塊裝置包括硬碟、磁碟、U盤和SD卡等。

  每一個字元裝置或塊裝置都在/dev目錄下對應一個裝置檔案。linux使用者程式通過裝置檔案(或稱裝置節點)來使用驅動程式操作字元裝置和塊裝置。

二、字元裝置驅動程式基礎:
1、主裝置號和次裝置號(二者一起為裝置號):
  一個字元裝置或塊裝置都有一個主裝置號和一個次裝置號。主裝置號用來標識與裝置檔案相連的驅動程式,用來反映裝置型別。次裝置號被驅動程式用來辨別操作的是哪個裝置,用來區分同類型的裝置。
  linux核心中,裝置號用dev_t來描述,2.6.28中定義如下:
  typedef u_long dev_t;
  在32位機中是4個位元組,高12位表示主裝置號,低12位表示次裝置號。

可以使用下列巨集從dev_t中獲得主次裝置號:                   也可以使用下列巨集通過主次裝置號生成dev_t:
MAJOR(dev_t dev);                              MKDEV(int major,int minor);
MINOR(dev_t dev);

複製程式碼
//巨集定義:#define MINORBITS    20
#define MINORMASK    ((1U << MINORBITS) - 1)
#define MAJOR(dev)    ((unsigned int) ((dev) >> MINORBITS))
#define MINOR(dev)    ((unsigned int) ((dev) & MINORMASK))
#define MKDEV(ma,mi)    (((ma) << MINORBITS) | (mi))
複製程式碼

2、分配裝置號(兩種方法):

(1)靜態申請:
int register_chrdev_region(dev_t from, unsigned count, const char *name);

複製程式碼
/**
 * register_chrdev_region() - register a range of device numbers
 * @from: the first in the desired range of device numbers; must include
 *        the major number.
 * @count: the number of consecutive device numbers required
 * @name: the name of the device or driver.
 *
 * Return value is zero on success, a negative error code on failure.
 
*/
複製程式碼

(2)動態分配:

int alloc_chrdev_region(dev_t *dev, unsigned baseminor, unsigned count, const char *name);

複製程式碼
int alloc_chrdev_region(dev_t *dev, unsigned baseminor, unsigned count, const char *name);
     /**
 * alloc_chrdev_region() - register a range of char device numbers
 * @dev: output parameter for first assigned number
 * @baseminor: first of the requested range of minor numbers
 * @count: the number of minor numbers required
 * @name: the name of the associated device or driver
 *
 * Allocates a range of char device numbers.  The major number will be
 * chosen dynamically, and returned (along with the first minor number)
 * in @dev.  Returns zero or a negative error code.
 */
複製程式碼

登出裝置號

void unregister_chrdev_region(dev_t from, unsigned count);


建立裝置檔案
利用cat /proc/devices檢視申請到的裝置名,裝置號。
(1)使用mknod手工建立:mknod filename type major minor
(2)自動建立;

  利用udev(mdev)來實現裝置檔案的自動建立,首先應保證支援udev(mdev),由busybox配置。在驅動初始化程式碼裡呼叫class_create為該裝置建立一個class,再為每個裝置呼叫device_create建立對應的裝置。

3、字元裝置驅動程式重要的資料結構
(1)struct file:代表一個開啟的檔案描述符,系統中每一個開啟的檔案在核心中都有一個關聯的struct file。它由核心在open時建立,並傳遞給在檔案上操作的任何函式,直到最後關閉。當檔案的所有例項都關閉之後,核心釋放這個資料結構。

//重要成員:     const struct file_operations    *f_op;  //該操作是定義檔案關聯的操作的。核心在執行open時對這個指標賦值。 off_t  f_pos;     //該檔案讀寫位置。void            *private_data;//該成員是系統呼叫時儲存狀態資訊非常有用的資源。

(2)struct inode:用來記錄檔案的物理資訊。它和代表開啟的file結構是不同的。一個檔案可以對應多個file結構,但只有一個inode結構。inode一般作為file_operations結構中函式的引數傳遞過來。
  inode譯成中文就是索引節點。每個儲存裝置或儲存裝置的分割槽(儲存裝置是硬碟、軟盤、U盤 ... ... )被格式化為檔案系統後,應該有兩部份,一部份是inode,另一部份是Block,Block是用來儲存資料用的。而inode呢,就是用來儲存這些資料的資訊,這些資訊包括檔案大小、屬主、歸屬的使用者組、讀寫許可權等。inode為每個檔案進行資訊索引,所以就有了inode的數值。作業系統根據指令,能通過inode值最快的找到相對應的檔案。

dev_t i_rdev;    //對錶示裝置檔案的inode結構,該欄位包含了真正的裝置編號。struct cdev *i_cdev;     //是表示字元裝置的核心的內部結構。當inode指向一個字元裝置檔案時,該欄位包含了指向struct cdev結構的指標。
//我們也可以使用下邊兩個巨集從inode中獲得主裝置號和此裝置號:unsigned int iminor(struct inode *inode);
unsigned int imajor(struct inode *inode);

(3)struct file_operations

複製程式碼
struct file_operations ***_ops={
 .owner =  THIS_MODULE,
 .llseek =  ***_llseek,
 .read =  ***_read,
 .write =  ***_write,
 .ioctl =  ***_ioctl,
 .open =  ***_open,
 .release = ***_release, 
 。。。  。。。
};

struct module *owner;
 /*第一個 file_operations 成員根本不是一個操作; 它是一個指向擁有這個結構的模組的指標.
 這個成員用來在它的操作還在被使用時阻止模組被解除安裝. 幾乎所有時間中, 它被簡單初始化為 
THIS_MODULE, 一個在 <linux/module.h> 中定義的巨集.這個巨集比較複雜,在進行簡單學習操作的時候,一般初始化為THIS_MODULE。*/


loff_t (*llseek) (struct file * filp , loff_t  p,  int  orig);
/*(指標引數filp為進行讀取資訊的目標檔案結構體指標;引數 p 為檔案定位的目標偏移量;引數orig為對檔案定位
的起始地址,這個值可以為檔案開頭(SEEK_SET,0,當前位置(SEEK_CUR,1),檔案末尾(SEEK_END,2))
llseek 方法用作改變檔案中的當前讀/寫位置, 並且新位置作為(正的)返回值.
loff_t 引數是一個"long offset", 並且就算在 32位平臺上也至少 64 位寬. 錯誤由一個負返回值指示.
如果這個函式指標是 NULL, seek 呼叫會以潛在地無法預知的方式修改 file 結構中的位置計數器( 在"file 結構" 一節中描述).*/

ssize_t (*read) (struct file * filp, char __user * buffer, size_t    size , loff_t *  p);
/*(指標引數 filp 為進行讀取資訊的目標檔案,指標引數buffer 為對應放置資訊的緩衝區(即使用者空間記憶體地址),
引數size為要讀取的資訊長度,引數 p 為讀的位置相對於檔案開頭的偏移,在讀取資訊後,這個指標一般都會移動,移動的值為要讀取資訊的長度值)
這個函式用來從裝置中獲取資料. 在這個位置的一個空指標導致 read 系統呼叫以 -EINVAL("Invalid argument") 失敗.
 一個非負返回值代表了成功讀取的位元組數( 返回值是一個 "signed size" 型別, 常常是目標平臺本地的整數型別).*/

ssize_t (*aio_read)(struct kiocb *  , char __user *  buffer, size_t  size ,  loff_t   p);
/*可以看出,這個函式的第一、三個引數和本結構體中的read()函式的第一、三個引數是不同 的,
非同步讀寫的第三個引數直接傳遞值,而同步讀寫的第三個引數傳遞的是指標,因為AIO從來不需要改變檔案的位置。
非同步讀寫的第一個引數為指向kiocb結構體的指標,而同步讀寫的第一引數為指向file結構體的指標,每一個I/O請求都對應一個kiocb結構體);
初始化一個非同步讀 -- 可能在函式返回前不結束的讀操作.如果這個方法是 NULL, 所有的操作會由 read 代替進行(同步地).
(有關linux非同步I/O,可以參考有關的資料,《linux裝置驅動開發詳解》中給出了詳細的解答)*/

ssize_t (*write) (struct file *  filp, const char __user *   buffer, size_t  count, loff_t * ppos);
/*(引數filp為目標檔案結構體指標,buffer為要寫入檔案的資訊緩衝區,count為要寫入資訊的長度,
ppos為當前的偏移位置,這個值通常是用來判斷寫檔案是否越界)
傳送資料給裝置. 如果 NULL, -EINVAL 返回給呼叫 write 系統呼叫的程式. 如果非負, 返回值代表成功寫的位元組數.
(注:這個操作和上面的對檔案進行讀的操作均為阻塞操作)*/

ssize_t (*aio_write)(struct kiocb *, const char __user *  buffer, size_t  count, loff_t * ppos);
/*初始化裝置上的一個非同步寫.引數型別同aio_read()函式;*/

int (*readdir) (struct file *  filp, void *, filldir_t);
/*對於裝置檔案這個成員應當為 NULL; 它用來讀取目錄, 並且僅對檔案系統有用.*/

unsigned int (*poll) (struct file *, struct poll_table_struct *);
/*(這是一個裝置驅動中的輪詢函式,第一個引數為file結構指標,第二個為輪詢表指標)
這個函式返回裝置資源的可獲取狀態,即POLLIN,POLLOUT,POLLPRI,POLLERR,POLLNVAL等巨集的位“或”結果。
每個巨集都表明裝置的一種狀態,如:POLLIN(定義為0x0001)意味著裝置可以無阻塞的讀,POLLOUT(定義為0x0004)意味著裝置可以無阻塞的寫。
(poll 方法是 3 個系統呼叫的後端: poll, epoll, 和 select, 都用作查詢對一個或多個檔案描述符的讀或寫是否會阻塞.
 poll 方法應當返回一個位掩碼指示是否非阻塞的讀或寫是可能的, 並且, 可能地, 提供給核心資訊用來使呼叫程序睡眠直到 I/O 變為可能. 
如果一個驅動的 poll 方法為 NULL, 裝置假定為不阻塞地可讀可寫.
(這裡通常將裝置看作一個檔案進行相關的操作,而輪詢操作的取值直接關係到裝置的響應情況,可以是阻塞操作結果,同時也可以是非阻塞操作結果)*/

int (*ioctl) (struct inode *inode, struct file *filp, unsigned int cmd, unsigned long arg);
/*(inode 和 filp 指標是對應應用程式傳遞的檔案描述符 fd 的值, 和傳遞給 open 方法的相同引數.
cmd 引數從使用者那裡不改變地傳下來, 並且可選的引數 arg 引數以一個 unsigned long 的形式傳遞, 不管它是否由使用者給定為一個整數或一個指標.
如果呼叫程式不傳遞第 3 個引數, 被驅動操作收到的 arg 值是無定義的.
因為型別檢查在這個額外引數上被關閉, 編譯器不能警告你如果一個無效的引數被傳遞給 ioctl, 並且任何關聯的錯誤將難以查詢.)
ioctl 系統呼叫提供了發出裝置特定命令的方法(例如格式化軟盤的一個磁軌, 這不是讀也不是寫). 另外, 幾個 ioctl 命令被核心識別而不必引用 fops 表.
 如果裝置不提供 ioctl 方法, 對於任何未事先定義的請求(-ENOTTY, "裝置無這樣的 ioctl"), 系統呼叫返回一個錯誤.*/

int (*mmap) (struct file *, struct vm_area_struct *);
/*mmap 用來請求將裝置記憶體對映到程序的地址空間. 如果這個方法是 NULL, mmap 系統呼叫返回 -ENODEV.
(如果想對這個函式有個徹底的瞭解,那麼請看有關“程序地址空間”介紹的書籍)*/

int (*open) (struct inode * inode , struct file *  filp ) ;
/*(inode 為檔案節點,這個節點只有一個,無論使用者開啟多少個檔案,都只是對應著一個inode結構;
但是filp就不同,只要開啟一個檔案,就對應著一個file結構體,file結構體通常用來追蹤檔案在執行時的狀態資訊)
 儘管這常常是對裝置檔案進行的第一個操作, 不要求驅動宣告一個對應的方法. 如果這個項是 NULL, 裝置開啟一直成功, 但是你的驅動不會得到通知.
與open()函式對應的是release()函式。*/

int (*flush) (struct file *);
/*flush 操作在程序關閉它的裝置檔案描述符的拷貝時呼叫; 它應當執行(並且等待)裝置的任何未完成的操作.
這個必須不要和使用者查詢請求的 fsync 操作混淆了. 當前, flush 在很少驅動中使用;
 SCSI 磁帶驅動使用它, 例如, 為確保所有寫的資料在裝置關閉前寫到磁帶上. 如果 flush 為 NULL, 核心簡單地忽略使用者應用程式的請求.*/

int (*release) (struct inode *, struct file *);
/*release ()函式當最後一個開啟裝置的使用者程序執行close()系統呼叫的時候,核心將呼叫驅動程式release()函式:
void release(struct inode inode,struct file *file),release函式的主要任務是清理未結束的輸入輸出操作,釋放資源,使用者自定義排他標誌的復位等。
    在檔案結構被釋放時引用這個操作. 如同 open, release 可以為 NULL.*/

int(*synch)(struct file *,struct dentry *,int datasync);
//重新整理待處理的資料,允許程序把所有的髒緩衝區重新整理到磁碟。

int (*aio_fsync)(struct kiocb *, int);
 /*這是 fsync 方法的非同步版本.所謂的fsync方法是一個系統呼叫函式。系統呼叫fsync
把檔案所指定的檔案的所有髒緩衝區寫到磁碟中(如果需要,還包括存有索引節點的緩衝區)。
相應的服務例程獲得檔案物件的地址,並隨後呼叫fsync方法。通常這個方法以呼叫函式__writeback_single_inode()結束,
這個函式把與被選中的索引節點相關的髒頁和索引節點本身都寫回磁碟。*/

int (*fasync) (int, struct file *, int);
//這個函式是系統支援非同步通知的裝置驅動,下面是這個函式的模板:
static int ***_fasync(int fd,struct file *filp,int mode)
{
    struct ***_dev * dev=filp->private_data;
    return fasync_helper(fd,filp,mode,&dev->async_queue);//第四個引數為 fasync_struct結構體指標的指標。
//這個函式是用來處理FASYNC標誌的函式。(FASYNC:表示相容BSD的fcntl同步操作)當這個標誌改變時,驅動程式中的fasync()函式將得到執行。}
/*此操作用來通知裝置它的 FASYNC 標誌的改變. 非同步通知是一個高階的主題, 在第 6 章中描述.
這個成員可以是NULL 如果驅動不支援非同步通知.*/

int (*lock) (struct file *, int, struct file_lock *);
//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 *);
/*這些方法實現發散/匯聚讀和寫操作. 應用程式偶爾需要做一個包含多個記憶體區的單個讀或寫操作;
 這些系統呼叫允許它們這樣做而不必對資料進行額外拷貝. 如果這些函式指標為 NULL, read 和 write 方法被呼叫( 可能多於一次 ).*/

ssize_t (*sendfile)(struct file *, loff_t *, size_t, read_actor_t, void *);
/*這個方法實現 sendfile 系統呼叫的讀, 使用最少的拷貝從一個檔案描述符搬移資料到另一個.
例如, 它被一個需要傳送檔案內容到一個網路連線的 web 伺服器使用. 裝置驅動常常使 sendfile 為 NULL.*/

ssize_t (*sendpage) (struct file *, struct page *, int, size_t, loff_t *, int);
/*sendpage 是 sendfile 的另一半; 它由核心呼叫來發送資料, 一次一頁, 到對應的檔案. 裝置驅動實際上不實現 sendpage.*/

unsigned long (*get_unmapped_area)(struct file *, unsigned long, unsigned long, unsigned long, unsigned long);
/*這個方法的目的是在程序的地址空間找一個合適的位置來對映在底層裝置上的記憶體段中.
這個任務通常由記憶體管理程式碼進行; 這個方法存在為了使驅動能強制特殊裝置可能有的任何的對齊請求. 大部分驅動可以置這個方法為 NULL.[10]*/

int (*check_flags)(int)
//這個方法允許模組檢查傳遞給 fnctl(F_SETFL...) 呼叫的標誌.
int (*dir_notify)(struct file *, unsigned long);
//這個方法在應用程式使用 fcntl 來請求目錄改變通知時呼叫. 只對檔案系統有用; 驅動不需要實現 dir_notify.
複製程式碼

三、字元裝置驅動程式設計

1.設備註冊
在linux2.6核心中,字元裝置使用struct cdev來描述;

複製程式碼
struct cdev
{
  struct kobject kobj;//內嵌的kobject物件  struct module *owner;//所屬模組  struct file_operations *ops;//檔案操作結構體  struct list_head list;
  dev_t dev;//裝置號,長度為32位,其中高12為主裝置號,低20位為此裝置號  unsigned int count;
};
複製程式碼

字元裝置的註冊分為三個步驟:

(1)分配cdev: struct cdev *cdev_alloc(void);
(2)初始化cdev: void cdev_init(struct cdev *cdev, const struct file_operations *fops);
(3)新增cdev: int cdev_add(struct cdev *p, dev_t dev, unsigned count)

View Code

2.裝置操作的實現:file_operations函式集的實現(要明確某個函式什麼時候被呼叫?呼叫來做什麼操作?)
特別注意:驅動程式應用程式的資料交換:
  驅動程式和應用程式的資料交換是非常重要的。file_operations中的read()和write()函式,就是用來在驅動程式和應用程式間交換資料的。通過資料交換,驅動程式和應用程式可以彼此瞭解對方的情況。但是驅動程式和應用程式屬於不同的地址空間。驅動程式不能直接訪問應用程式的地址空間;同樣應用程式也不能直接訪問驅動程式的地址空間,否則會破壞彼此空間中的資料,從而造成系統崩潰,或者資料損壞。安全的方法是使用核心提供的專用函式,完成資料在應用程式空間和驅動程式空間的交換。這些函式對使用者程式傳過來的指標進行了嚴格的檢查和必要的轉換,從而保證使用者程式與驅動程式交換資料的安全性。這些函式有:

unsigned long copy_to_user(void __user *to, const void *from, unsigned long n); 
unsigned long copy_from_user(void *to, const void __user *from, unsigned long n); 
put_user(local,user); 
get_user(local,user);

3.設備註銷:void cdev_del(struct cdev *p);

四、字元裝置驅動小結:

  字元裝置是3大類裝置(字元裝置、塊裝置、網路裝置)中較簡單的一類裝置,其驅動程式中完成的主要工作是初始化、新增和刪除cdev結構體,申請和釋放裝置號,以及填充file_operation結構體中操作函式,並實現file_operations結構體中的read()、write()、ioctl()等重要函式。如圖所示為cdev結構體、file_operations和使用者空間呼叫驅動的關係。

相關推薦

linux驅動---字元裝置的註冊register_chrdev說

首先我們在註冊函式裡面呼叫了register_chrdev(MEM_MAJOR,"mem",&memory_fops),向核心註冊了一個字元裝置。 第一個引數是主裝置號,0代表動態分配,這裡的MEM_MAJOR是1。第二個引數是裝置的名字,第三個引數是檔案操作指標。 每個裝置檔案對應

linux裝置驅動--字元裝置驅動

一、linux系統將裝置分為3類:字元裝置、塊裝置、網路裝置。使用驅動程式: 1、字元裝置:是指只能一個位元組一個位元組讀寫的裝置,不能隨機讀取裝置記憶體中的某一資料,讀取資料需要按照先後資料。字元裝置是面向流的裝置,常見的字元裝置有滑鼠、鍵盤、串列埠、控制檯和LE

linux驅動---字元裝置的註冊register_chrdev說起

首先我們在註冊函式裡面呼叫了register_chrdev(MEM_MAJOR,"mem",&memory_fops),向核心註冊了一個字元裝置。第一個引數是主裝置號,0代表動態分配,這裡的MEM_MAJOR是1。第二個引數是裝置的名字,第三個引數是檔案操作指標。每個

裝置字元裝置、塊裝置不全面比較

A.為什麼Oracle要使用裸裝置呢裸裝置,也叫裸分割槽(原始分割槽),是一種沒有經過格式化,不被Unix通過檔案系統來讀取的特殊字元裝置。本文收集裸裝置和Oracle問答20例。   1.什麼叫做裸裝置?  裸設備,也叫裸分割槽(原始分割槽),是一種沒有經過格式化,不被U

裝置字元裝置的區別

在LINUX裡面,裝置型別分為:字元裝置、塊裝置以及網路裝置, PCI是一種和ISA為一類的匯流排結構,歸屬於網路驅動裝置~~~ 字元裝置、塊裝置主要區別是:在對字元裝置發出讀/寫請求時,實際的硬體I/O一般就緊接著發生了,而塊裝置則不然,它利用一塊系統記憶體作為緩衝區,當

Linux驅動開發(9)——註冊字元裝置

static int scdev_init(void) { int ret = 0,i; dev_t num_dev; printk(KERN_EMERG "numdev_major is %d!\n",numdev_major); printk(KERN_EMERG "

Linux驅動開發(8)——靜態和動態申請字元裝置

先貼一段demo #include <linux/init.h> /*包含初始化巨集定義的標頭檔案,程式碼中的module_init和module_exit在此檔案中*/ #include <linux/module.h> /*包含初始化載入模組的標頭檔案,程

《5.linux驅動開發-第3部分-5.3.字元裝置驅動高階》

《5.linux驅動開發-第3部分-5.3.字元裝置驅動高階》 第一部分、章節目錄 5.3.1.註冊字元裝置驅動新介面1 5.3.2.註冊字元裝置驅動新介面2 5.3.3.註冊字元裝置驅動新介面3 5.3.4.註冊字元裝置驅動新介面4 5.3.5.字元裝置驅動註冊程式碼分析1 5.3.6

《5.linux驅動開發-第2部分-5.2.字元裝置驅動基礎》

《5.linux驅動開發-第2部分-5.2.字元裝置驅動基礎》 第一部分、章節目錄 5.2.1.開啟驅動開發之路 5.2.2.最簡單的模組原始碼分析1 5.2.3.最簡單的模組原始碼分析2 5.2.4.最簡單的模組原始碼分析3 5.2.5.用開發板來除錯模組 5.2.6.字元裝置驅動工作

Linux字元裝置驅動模型--字元裝置的註冊

當我們編寫字元裝置驅動程式的時候,在進行字元裝置物件cdev的分配、初始化,裝置號的註冊這些初始化階段之後,就可以將它加入到系統中,這樣才能使用這個字元裝置。將一個字元裝置加入到系統中呼叫的函式時cdev_add,核心原始碼如下: int cdev_add(struct cdev *

Linux 字元裝置驅動結構(二)—— 自動建立裝置節點

      上一篇我們介紹到建立裝置檔案的方法,利用cat /proc/devices檢視申請到的裝置名,裝置號。 第一種是使用mknod手工建立:mknod filename type major minor 第二種是自動建立裝置節點:利用u

Linux 字元裝置驅動結構(一)—— cdev 結構體、裝置號相關知識解析

一、字元裝置基礎知識 1、裝置驅動分類       linux系統將裝置分為3類:字元裝置、塊裝置、網路裝置。使用驅動程式: 字元裝置:是指只能一個位元組一個位元組讀寫的裝置,不能隨機讀取裝置記憶體中的某一資料,讀取資料需要按照先後資料。

Linux驅動字元裝置工作原理(未完)

字元裝置驅動工作原理 系統整體工作原理 應用層->API->裝置驅動->硬體? API:open、read、write、close等? 驅動原始碼中提供真正的open、read、write、close等函式實體? file_

字元裝置驅動-------Linux異常處理體系結構

  裸機中斷流程 外部觸發 CPU 發生中斷, 強制的跳到異常向量處 跳轉到具體函式 儲存被中斷處的現場(各種暫存器的值) 執行中斷處理函式,處理具體任務 恢復被中斷的現場 Linux處理異常流程   異常發生時,會去異常向量表找到入口

Linux 裝置驅動--- 阻塞型字元裝置驅動 --- O_NONBLOCK --- 非阻塞標誌

阻塞:           在設計簡單字元驅動程式時,要注意一個重要問題.           當一個裝置無法立刻滿足使用者的讀寫請求時應當如何處理?        

Linux字元裝置驅動註冊三種方法以及核心分析

       Linux驅動是使用者訪問底層硬體的橋樑,驅動有可以簡單分成三類:字元裝置、塊裝置、網路裝置。其中最多的是字元裝置,其中字元裝置的註冊方法主要有三種:雜項設備註冊、早期字元設備註冊、標準字元設備註冊。以及詳細介紹各類方法註冊。 開發環境: PC:WMwork

linux misc device字元雜項裝置驅動程式

雜項裝置也是在嵌入式系統中用得比較多的一種裝置驅動。miscdevice共享一個主裝置號MISC_MAJOR(即10),但次裝置號不同。misc裝置其實就是特殊的字元裝置,主裝置編號採用10,並且可自動生成裝置節點。 雜項裝置作為字元裝置的封裝,為字元裝置提供的簡單的程

linux驅動字元裝置

第一部分:字元裝置工作過程1、系統呼叫和驅動程式的關聯關鍵結構體:struct file_operation;file_operation結構體的每一個成員的名字都對應著一個系統呼叫。使用者程序利用系統呼叫在對裝置檔案進行諸如read/write操作時,系統呼叫通過裝置檔案的主裝置號找到相應的裝置驅動程式,然

Linux裝置驅動字元裝置驅動---轉

一、linux系統將裝置分為3類:字元裝置、塊裝置、網路裝置。 應用程式呼叫的流程框圖: 三種裝置的定義分別如下, 字元裝置:只能一個位元組一個位元組的讀寫的裝置,不能隨機讀取裝置記憶體中的某一資料,讀取資料需要按照先後順序進行。字元裝置是面向流的裝置,常見的字

linux字元裝置驅動模型

一.雜項裝置驅動模型 雜項的主裝置號固定為10,只有255個次裝置號。可以直接生成驅動核心。 1.需要確定每個模型都會用到的檔案操作方法集合指標 2.確定核心的結構體 static struct miscdevice abc 確定三個引數,第一個為次裝置號,第二個是次裝