當(dāng)前位置:首頁(yè) > 單片機(jī) > 單片機(jī)
[導(dǎo)讀]設(shè)備驅(qū)動(dòng)程序是操作系統(tǒng)內(nèi)核和機(jī)器硬件之間的接口,它為應(yīng)用程序屏蔽硬件的細(xì)節(jié),一般來(lái)說(shuō),Linux的設(shè)備驅(qū)動(dòng)程序需要完成如下功能:  ·設(shè)備初始化、釋放;  ·提供各類(lèi)設(shè)備服務(wù);  ·負(fù)責(zé)內(nèi)核和設(shè)備之間的數(shù)據(jù)

設(shè)備驅(qū)動(dòng)程序是操作系統(tǒng)內(nèi)核和機(jī)器硬件之間的接口,它為應(yīng)用程序屏蔽硬件的細(xì)節(jié),一般來(lái)說(shuō),Linux的設(shè)備驅(qū)動(dòng)程序需要完成如下功能:

  ·設(shè)備初始化、釋放;

  ·提供各類(lèi)設(shè)備服務(wù);

  ·負(fù)責(zé)內(nèi)核和設(shè)備之間的數(shù)據(jù)交換;

  ·檢測(cè)和處理設(shè)備工作過(guò)程中出現(xiàn)的錯(cuò)誤。

  Linux下的設(shè)備驅(qū)動(dòng)程序被組織為一組完成不同任務(wù)的函數(shù)的集合,通過(guò)這些函數(shù)使得Windows的設(shè)備操作猶如文件一般。在應(yīng)用程序看來(lái),硬件設(shè)備只是一個(gè)設(shè)備文件,應(yīng)用程序可以象操作普通文件一樣對(duì)硬件設(shè)備進(jìn)行操作,如open ()、close ()、read ()、write () 等。

  Linux主要將設(shè)備分為二類(lèi):字符設(shè)備和塊設(shè)備。字符設(shè)備是指設(shè)備發(fā)送和接收數(shù)據(jù)以字符的形式進(jìn)行;而塊設(shè)備則以整個(gè)數(shù)據(jù)緩沖區(qū)的形式進(jìn)行。在對(duì)字符設(shè)備發(fā)出讀/寫(xiě)請(qǐng)求時(shí),實(shí)際的硬件I/O一般就緊接著發(fā)生了;而塊設(shè)備則不然,它利用一塊系統(tǒng)內(nèi)存作緩沖區(qū),當(dāng)用戶進(jìn)程對(duì)設(shè)備請(qǐng)求能滿足用戶的要求,就返回請(qǐng)求的數(shù)據(jù),如果不能,就調(diào)用請(qǐng)求函數(shù)來(lái)進(jìn)行實(shí)際的I/O操作。塊設(shè)備主要針對(duì)磁盤(pán)等慢速設(shè)備。

  1.內(nèi)存分配

  由于Linux驅(qū)動(dòng)程序在內(nèi)核中運(yùn)行,因此在設(shè)備驅(qū)動(dòng)程序需要申請(qǐng)/釋放內(nèi)存時(shí),不能使用用戶級(jí)的malloc/free函數(shù),而需由內(nèi)核級(jí)的函數(shù)kmalloc/kfree () 來(lái)實(shí)現(xiàn),kmalloc()函數(shù)的原型為:

void kmalloc (size_t size ,int priority);

  參數(shù)size為申請(qǐng)分配內(nèi)存的字節(jié)數(shù),kmalloc最多只能開(kāi)辟128k的內(nèi)存;參數(shù)priority說(shuō)明若kmalloc()不能馬上分配內(nèi)存時(shí)用戶進(jìn)程要采用的動(dòng)作:GFP_KERNEL 表示等待,即等kmalloc()函數(shù)將一些內(nèi)存安排到交換區(qū)來(lái)滿足你的內(nèi)存需要,GFP_ATOMIC 表示不等待,如不能立即分配到內(nèi)存則返回0 值;函數(shù)的返回值指向已分配內(nèi)存的起始地址,出錯(cuò)時(shí),返回0。

  kmalloc ()分配的內(nèi)存需用kfree()函數(shù)來(lái)釋放,kfree ()被定義為:

# define kfree (n) kfree_s( (n) ,0)

  其中kfree_s () 函數(shù)原型為:

void kfree_s (void * ptr ,int size);

  參數(shù)ptr為kmalloc()返回的已分配內(nèi)存的指針,size是要釋放內(nèi)存的字節(jié)數(shù),若為0 時(shí),由內(nèi)核自動(dòng)確定內(nèi)存的大小。

  2.中斷

  許多設(shè)備涉及到中斷操作,因此,在這樣的設(shè)備的驅(qū)動(dòng)程序中需要對(duì)硬件產(chǎn)生的中斷請(qǐng)求提供中斷服務(wù)程序。與注冊(cè)基本入口點(diǎn)一樣,驅(qū)動(dòng)程序也要請(qǐng)求內(nèi)核將特定的中斷請(qǐng)求和中斷服務(wù)程序聯(lián)系在一起。在Linux中,用request_irq()函數(shù)來(lái)實(shí)現(xiàn)請(qǐng)求:

int request_irq (unsigned int irq ,void( * handler) int ,unsigned long type ,char * name);

   參數(shù)irq為要中斷請(qǐng)求號(hào),參數(shù)handler為指向中斷服務(wù)程序的指針,參數(shù)type 用來(lái)確定是正常中斷還是快速中斷(正常中斷指中斷服務(wù)子程序返回后,內(nèi)核可以執(zhí)行調(diào)度程序來(lái)確定將運(yùn)行哪一個(gè)進(jìn)程;而快速中斷是指中斷服務(wù)子程序返回后,立即執(zhí)行被中斷程序,正常中斷type 取值為0 ,快速中斷type 取值為SA_INTERRUPT),參數(shù)name是設(shè)備驅(qū)動(dòng)程序的名稱。

3.字符設(shè)備驅(qū)動(dòng)

  我們必須為字符設(shè)備提供一個(gè)初始化函數(shù),該函數(shù)用來(lái)完成對(duì)所控設(shè)備的初始化工作,并調(diào)用register_chrdev() 函數(shù)注冊(cè)字符設(shè)備。假設(shè)有一字符設(shè)備"exampledev",則其init 函數(shù)為:

void exampledev_init(void)
{
if (register_chrdev(MAJOR_NUM, " exampledev ", &exampledev_fops))
TRACE_TXT("Device exampledev driver registered error");
else
TRACE_TXT("Device exampledev driver registered successfully");
…//設(shè)備初始化
}

  其中,register_chrdev函數(shù)中的參數(shù)MAJOR_NUM為主設(shè)備號(hào),"exampledev"為設(shè)備名,exampledev_fops 為包含基本函數(shù)入口點(diǎn)的結(jié)構(gòu)體,類(lèi)型為file_operations。當(dāng)執(zhí)行exampledev_init時(shí),它將調(diào)用內(nèi)核函數(shù) register_chrdev,把驅(qū)動(dòng)程序的基本入口點(diǎn)指針存放在內(nèi)核的字符設(shè)備地址表中,在用戶進(jìn)程對(duì)該設(shè)備執(zhí)行系統(tǒng)調(diào)用時(shí)提供入口地址。

  較早版本內(nèi)核的file_operations結(jié)構(gòu)體定義為(代碼及圖示):

struct file_operations
{
 int (*lseek)();
 int (*read)();
 int (*write)();
 int (*readdir)();
 int (*select)();
 int (*ioctl)();
 int (*mmap)();
 int (*open)();
 void(*release)();
 int (*fsync)();
 int (*fasync)();
 int (*check_media_change)();
 void(*revalidate)();
};

  隨著內(nèi)核功能的加強(qiáng),file_operations結(jié)構(gòu)體也變得更加龐大。但是大多數(shù)的驅(qū)動(dòng)程序只是利用了其中的一部分,對(duì)于驅(qū)動(dòng)程序中無(wú)需提供的功能,只需要把相應(yīng)位置的值設(shè)為NULL。對(duì)于字符設(shè)備來(lái)說(shuō),要提供的主要入口有:open ()、release ()、read ()、write ()、ioctl ()等。

  open()函數(shù) 對(duì)設(shè)備特殊文件進(jìn)行open()系統(tǒng)調(diào)用時(shí),將調(diào)用驅(qū)動(dòng)程序的open () 函數(shù):

int (*open)(struct inode * inode,struct file *filp);

   其中參數(shù)inode為設(shè)備特殊文件的inode (索引結(jié)點(diǎn)) 結(jié)構(gòu)的指針,參數(shù)filp是指向這一設(shè)備的文件結(jié)構(gòu)的指針。open()的主要任務(wù)是確定硬件處在就緒狀態(tài)、驗(yàn)證次設(shè)備號(hào)的合法性(次設(shè)備號(hào)可以用 MINOR(inode-> i_rdev) 取得)、控制使用設(shè)備的進(jìn)程數(shù)、根據(jù)執(zhí)行情況返回狀態(tài)碼(0表示成功,負(fù)數(shù)表示存在錯(cuò)誤) 等;

  release()函數(shù) 當(dāng)最后一個(gè)打開(kāi)設(shè)備的用戶進(jìn)程執(zhí)行close ()系統(tǒng)調(diào)用時(shí),內(nèi)核將調(diào)用驅(qū)動(dòng)程序的release () 函數(shù):

void (*release) (struct inode * inode,struct file *filp) ;

  release 函數(shù)的主要任務(wù)是清理未結(jié)束的輸入/輸出操作、釋放資源、用戶自定義排他標(biāo)志的復(fù)位等。

  read()函數(shù) 當(dāng)對(duì)設(shè)備特殊文件進(jìn)行read() 系統(tǒng)調(diào)用時(shí),將調(diào)用驅(qū)動(dòng)程序read() 函數(shù):

ssize_t (*read) (struct file * filp, char * buf, size_t count, loff_t * offp);

  參數(shù)buf是指向用戶空間緩沖區(qū)的指針,由用戶進(jìn)程給出,count 為用戶進(jìn)程要求讀取的字節(jié)數(shù),也由用戶給出。

  read() 函數(shù)的功能就是從硬設(shè)備或內(nèi)核內(nèi)存中讀取或復(fù)制count個(gè)字節(jié)到buf 指定的緩沖區(qū)中。在復(fù)制數(shù)據(jù)時(shí)要注意,驅(qū)動(dòng)程序運(yùn)行在內(nèi)核中,而buf指定的緩沖區(qū)在用戶內(nèi)存區(qū)中,是不能直接在內(nèi)核中訪問(wèn)使用的,因此,必須使用特殊的復(fù)制函數(shù)來(lái)完成復(fù)制工作,這些函數(shù)在include/asm/uaccess.h中被聲明:

unsigned long copy_to_user (void * to, void * from, unsigned long len);

  此外,put_user()函數(shù)用于內(nèi)核空間和用戶空間的單值交互(如char、int、long)。

  write( ) 函數(shù) 當(dāng)設(shè)備特殊文件進(jìn)行write () 系統(tǒng)調(diào)用時(shí),將調(diào)用驅(qū)動(dòng)程序的write () 函數(shù):

ssize_t (*write) (struct file *, const char *, size_t, loff_t *);

  write ()的功能是將參數(shù)buf 指定的緩沖區(qū)中的count 個(gè)字節(jié)內(nèi)容復(fù)制到硬件或內(nèi)核內(nèi)存中,和read() 一樣,復(fù)制工作也需要由特殊函數(shù)來(lái)完成:

unsigned long copy_from_user(void *to, const void *from, unsigned long n);

  此外,get_user()函數(shù)用于內(nèi)核空間和用戶空間的單值交互(如char、int、long)。

  ioctl() 函數(shù) 該函數(shù)是特殊的控制函數(shù),可以通過(guò)它向設(shè)備傳遞控制信息或從設(shè)備取得狀態(tài)信息,函數(shù)原型為:

int (*ioctl) (struct inode * inode,struct file * filp,unsigned int cmd,unsigned long arg);

  參數(shù)cmd為設(shè)備驅(qū)動(dòng)程序要執(zhí)行的命令的代碼,由用戶自定義,參數(shù)arg 為相應(yīng)的命令提供參數(shù),類(lèi)型可以是整型、指針等。

  同樣,在驅(qū)動(dòng)程序中,這些函數(shù)的定義也必須符合命名規(guī)則,按照本文約定,設(shè)備"exampledev"的驅(qū)動(dòng)程序的這些函數(shù)應(yīng)分別命名為 exampledev_open、exampledev_ release、exampledev_read、exampledev_write、exampledev_ioctl,因此設(shè)備 "exampledev"的基本入口點(diǎn)結(jié)構(gòu)變量exampledev_fops 賦值如下(對(duì)較早版本的內(nèi)核):

struct file_operations exampledev_fops {
 NULL ,
 exampledev_read ,
 exampledev_write ,
 NULL ,
 NULL ,
 exampledev_ioctl ,
 NULL ,
 exampledev_open ,
 exampledev_release ,
 NULL ,
 NULL ,
 NULL ,
 NULL
} ;

  就目前而言,由于file_operations結(jié)構(gòu)體已經(jīng)很龐大,我們更適合用GNU擴(kuò)展的C語(yǔ)法來(lái)初始化exampledev_fops:

struct file_operations exampledev_fops = {
 read: exampledev _read,
 write: exampledev _write,
 ioctl: exampledev_ioctl ,
 open: exampledev_open ,
 release : exampledev_release ,
};

  看看第一章電路板硬件原理圖,板上包含四個(gè)用戶可編程的發(fā)光二極管(LED),這些LED連接在ARM處理器的可編程I/O口(GPIO)上,現(xiàn)在來(lái)編寫(xiě)這些LED的驅(qū)動(dòng):

#include
#include
#include
#include
#include
#include
#include
#define DEVICE_NAME "leds" /*定義led 設(shè)備的名字*/
#define LED_MAJOR 231 /*定義led 設(shè)備的主設(shè)備號(hào)*/
static unsigned long led_table[] =
{
 /*I/O 方式led 設(shè)備對(duì)應(yīng)的硬件資源*/
 GPIO_B10, GPIO_B8, GPIO_B5, GPIO_B6,
};
/*使用ioctl 控制led*/
static int leds_ioctl(struct inode *inode, struct file *file, unsigned int cmd,
unsigned long arg)
{
 switch (cmd)
 {
  case 0:
  case 1:
   if (arg > 4)
   {
    return -EINVAL;
   }
   write_gpio_bit(led_table[arg], !cmd);
  default:
   return -EINVAL;
 }
}
static struct file_operations leds_fops =
{
 owner: THIS_MODULE, ioctl: leds_ioctl,
};
static devfs_handle_t devfs_handle;
static int __init leds_init(void)
{
 int ret;
 int i;
 /*在內(nèi)核中注冊(cè)設(shè)備*/
 ret = register_chrdev(LED_MAJOR, DEVICE_NAME, &leds_fops);
 if (ret < 0)
 {
  printk(DEVICE_NAME " can't register major numbern");
  return ret;
 }
 devfs_handle = devfs_register(NULL, DEVICE_NAME, DEVFS_FL_DEFAULT, LED_MAJOR,
0, S_IFCHR | S_IRUSR | S_IWUSR, &leds_fops, NULL);
 /*使用宏進(jìn)行端口初始化,set_gpio_ctrl 和write_gpio_bit 均為宏定義*/
 for (i = 0; i < 8; i++)
 {
  set_gpio_ctrl(led_table[i] | GPIO_PULLUP_EN | GPIO_MODE_OUT);
  write_gpio_bit(led_table[i], 1);
 }
 printk(DEVICE_NAME " initializedn");
 return 0;
}

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請(qǐng)聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請(qǐng)及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車(chē)的華為或?qū)⒋呱龈蟮莫?dú)角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

倫敦2024年8月29日 /美通社/ -- 英國(guó)汽車(chē)技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車(chē)工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車(chē)。 SODA V工具的開(kāi)發(fā)耗時(shí)1.5...

關(guān)鍵字: 汽車(chē) 人工智能 智能驅(qū)動(dòng) BSP

北京2024年8月28日 /美通社/ -- 越來(lái)越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時(shí)企業(yè)卻面臨越來(lái)越多業(yè)務(wù)中斷的風(fēng)險(xiǎn),如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報(bào)道,騰訊和網(wǎng)易近期正在縮減他們對(duì)日本游戲市場(chǎng)的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)開(kāi)幕式在貴陽(yáng)舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語(yǔ)權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點(diǎn): 有效應(yīng)對(duì)環(huán)境變化,經(jīng)營(yíng)業(yè)績(jī)穩(wěn)中有升 落實(shí)提質(zhì)增效舉措,毛利潤(rùn)率延續(xù)升勢(shì) 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長(zhǎng) 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競(jìng)爭(zhēng)力 堅(jiān)持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競(jìng)爭(zhēng)優(yōu)勢(shì)...

關(guān)鍵字: 通信 BSP 電信運(yùn)營(yíng)商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺(tái)與中國(guó)電影電視技術(shù)學(xué)會(huì)聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會(huì)上宣布正式成立。 活動(dòng)現(xiàn)場(chǎng) NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長(zhǎng)三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會(huì)上,軟通動(dòng)力信息技術(shù)(集團(tuán))股份有限公司(以下簡(jiǎn)稱"軟通動(dòng)力")與長(zhǎng)三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉