RM新时代网站-首页

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

Linux驅(qū)動開發(fā)-編寫按鍵驅(qū)動

DS小龍哥-嵌入式技術(shù) ? 2022-09-17 15:08 ? 次閱讀

【摘要】 這篇文章介紹,如何使用雜項(xiàng)設(shè)備框架編寫一個簡單的按鍵驅(qū)動,完成編寫、編譯、安裝、測試等流程,了解一個雜項(xiàng)字符設(shè)備驅(qū)動的開發(fā)流程。

1. 雜項(xiàng)設(shè)備注冊函數(shù)

這篇文章介紹,如何使用雜項(xiàng)設(shè)備框架編寫一個簡單的按鍵驅(qū)動,完成編寫、編譯、安裝、測試等流程,了解一個雜項(xiàng)字符設(shè)備驅(qū)動的開發(fā)流程。

下面是雜項(xiàng)字符設(shè)備的接口:

struct miscdevice  {
	int minor;   /*次設(shè)備號 10 20 */
	const char *name; /*設(shè)備節(jié)點(diǎn)的名稱*/
	const struct file_operations *fops; /*文件操作集合*/
	struct list_head list; //鏈表
	struct device *parent; 
	struct device *this_device;
	const char *nodename;
	umode_t mode;
};

//注冊雜項(xiàng)字符設(shè)備
extern int misc_register(struct miscdevice * misc);
//注銷雜項(xiàng)字符設(shè)備
extern int misc_deregister(struct miscdevice *misc);

按鍵需要將值傳遞給應(yīng)用層,需要使用到copy_to_user函數(shù),這個函數(shù)還有一個配對的copy_from_user,下面介紹這兩個函數(shù)的詳細(xì)功能和參數(shù):

#include 

unsigned long copy_to_user(void __user *to, const void *from, unsigned long n)
函數(shù)功能: 將驅(qū)動層數(shù)據(jù)拷貝到應(yīng)用層。
函數(shù)參數(shù):
void __user *to  用戶空間的地址--到哪里去
const void *from  驅(qū)動層的地址--從哪里來
unsigned long n  拷貝的大小
返回值:  0表示成功。 >0表示未拷貝成功的數(shù)量。

unsigned long copy_from_user(void *to, const void __user *from, unsigned long n)
函數(shù)功能: 將應(yīng)用層的數(shù)據(jù)拷貝到驅(qū)動層。
函數(shù)參數(shù):
void  *to  驅(qū)動空間的地址--拷貝到哪里去
const void __user *from  用戶空間的地址--從哪里來
unsigned long n  拷貝的大小
返回值:  0表示成功。 >0表示未拷貝成功的數(shù)量。

2. 編寫按鍵驅(qū)動

使用雜項(xiàng)設(shè)備注冊按鍵驅(qū)動,應(yīng)用層使用read接口讀取按鍵值。

編寫驅(qū)動之前需要先找到按鍵的原理圖,找到按鍵接到CPU那個IO上的。

image-20211226144955867image-20211226145036869

然后再查閱數(shù)據(jù)手冊,找到這個GPIO口的寄存器地址,寄存器的配置頁面,方便初始化配置GPIO口為輸入模式。

image-20211226145124296

2.1 按鍵驅(qū)動源代碼

#include 
#include 
#include 
#include 
#include 
#include 

/* 按鍵的寄存器*/
static unsigned int *GPX3CON;
static unsigned int *GPX3DAT;

static int tiny4412_open(struct inode *inode, struct file *file)
{
	printk("tiny4412_open-->ok\n");
	return 0;
}

/*應(yīng)用層的函數(shù): 
int key_val;
read(fd,&key_val,4)
ssize_t read(int fd, void *buf, size_t count);
*/
static ssize_t tiny4412_read(struct file *file, char __user *buf, size_t size, loff_t *seek)
{
	int key_val=0;
	if(!(*GPX3DAT&1<<2)) //判斷按鍵是否按下
	{
		key_val=0x1;
	}
	else if(!(*GPX3DAT&1<<3)) //判斷按鍵是否按下
	{
		key_val=0x2;
	}
	else if(!(*GPX3DAT&1<<4)) //判斷按鍵是否按下
	{
		key_val=0x3;
	}
	else if(!(*GPX3DAT&1<<5)) //判斷按鍵是否按下
	{
		key_val=0x4;
	}

	/*數(shù)據(jù)拷貝函數(shù): 給應(yīng)用層空間賦值--將驅(qū)動層的數(shù)據(jù)拷貝給應(yīng)用層*/
	/*copy_to_user(void __user *to, const void *from, unsigned long n)*/
    int error;
	error=copy_to_user(buf,&key_val,4);
	if(error>0)
	{
		printk("數(shù)據(jù)拷貝失敗.\n");
	}
	return 0;
}
static ssize_t tiny4412_write(struct file *file, const char __user *buf, size_t size, loff_t *seek)
{
	
	return 0;
}
static int tiny4412_release(struct inode *inode, struct file *file)
{
	printk("tiny4412_release-->ok\n");
	return 0;
}

static struct file_operations fops=
{
	.open=tiny4412_open,
	.read=tiny4412_read,
	.write=tiny4412_write,
	.release=tiny4412_release
};

/*
Linux內(nèi)核管理驅(qū)動---設(shè)備號
設(shè)備號是一個unsigned int 的變量--32位。
設(shè)備號=主設(shè)備號+次設(shè)備號
*/
static struct miscdevice misc=
{
	.minor = MISC_DYNAMIC_MINOR,  /*次設(shè)備號填255表示自動分配     主設(shè)備號固定為10*/
	.name = "tiny4412_key",  /*/dev目錄下文件名稱*/
	.fops = &fops, /*文件操作接口*/
};

static int __init tiny4412_key_init(void)
{
    /*轉(zhuǎn)換物理地址*/
	GPX3CON=ioremap(0x11000C60,4);
	GPX3DAT=ioremap(0x11000C64,4);

	/*配置GPIO口模式--配置按鍵*/	
	*GPX3CON&=0xFF0000FF;	
	
	/*1. 雜項(xiàng)設(shè)備的注冊函數(shù)*/
	misc_register(&misc);
    printk("按鍵: 驅(qū)動安裝成功\n");
    return 0;
}

static void __exit tiny4412_key_exit(void)
{
	/*2. 雜項(xiàng)設(shè)備的注銷函數(shù)*/
	misc_deregister(&misc);

	/*取消轉(zhuǎn)換*/
	iounmap(GPX3CON);
	iounmap(GPX3DAT);
    printk("按鍵: 驅(qū)動卸載成功\n");
}

module_init(tiny4412_key_init); /*驅(qū)動入口--安裝驅(qū)動的時候執(zhí)行*/
module_exit(tiny4412_key_exit); /*驅(qū)動出口--卸載驅(qū)動的時候執(zhí)行*/

MODULE_LICENSE("GPL");  /*設(shè)置模塊的許可證--GPL*/

2.2 makefile文件

編譯驅(qū)動的makefile代碼。

KER_DRI=/home/wbyq/work/linux-3.5/linux-3.5
all:
	make -C $(KER_DRI) M=`pwd` modules
	cp *.ko /home/wbyq/work/rootfs/code -f
	make -C $(KER_DRI) M=`pwd` modules clean
	arm-linux-gcc app.c -o app
	cp app /home/wbyq/work/rootfs/code -f
	rm app -f
obj-m += miscdev_key_drv.o

2.3 應(yīng)用層驅(qū)動測試代碼

編譯完運(yùn)行時,傳入按鍵的設(shè)備節(jié)點(diǎn)文件.

#include 
#include 
#include 
#include 

int main(int argc,char **argv)
{
    if(argc!=2)
    {
        printf("./app <設(shè)備節(jié)點(diǎn)文件>\n");
        return 0;
    }
    /*1. 打開設(shè)備文件*/
    int fd=open(argv[1],O_RDWR);
    if(fd<0)
    {
        printf("%s 設(shè)備驅(qū)動打開失敗.\n",argv[1]);
        return 0;
    }
    /*2.讀寫數(shù)據(jù)*/
    int key_val;
    while(1)
    {
        read(fd,&key_val,4);//讀取按鍵值
		if(key_val)
		{
			printf("%#x\n",key_val);
		}
    }
    /*3. 關(guān)閉文件*/
    close(fd);
    return 0;
}

2.4 驅(qū)動安裝流程

[root@wbyq code]# ls
tiny4412_key_drv.ko
[root@wbyq code]# 
[root@wbyq code]# insmod tiny4412_key_drv.ko 
[  173.340000] 驅(qū)動測試: 驅(qū)動安裝成功
[root@wbyq code]# lsmod
hello_drv 616 0 - Live 0xbf000000 (O)
[root@wbyq code]# modinfo tiny4412_key_drv.ko 
filename:       tiny4412_key_drv.ko
license:        GPL
depends:        
vermagic:       3.5.0-FriendlyARM SMP preempt mod_unload ARMv7 p2v8 
[root@wbyq code]# rmmod tiny4412_key_drv.ko 
[  391.075000] 驅(qū)動測試: 驅(qū)動卸載成功
[root@wbyq code]#
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 驅(qū)動
    +關(guān)注

    關(guān)注

    12

    文章

    1838

    瀏覽量

    85262
  • 函數(shù)
    +關(guān)注

    關(guān)注

    3

    文章

    4327

    瀏覽量

    62569
  • 按鍵
    +關(guān)注

    關(guān)注

    4

    文章

    223

    瀏覽量

    57595
收藏 人收藏

    評論

    相關(guān)推薦

    linux內(nèi)核中通用HID觸摸驅(qū)動

    linux內(nèi)核中,為HID觸摸面板實(shí)現(xiàn)了一個通用的驅(qū)動程序,位于/drivers/hid/hid-multitouch.c文件中。hid觸摸驅(qū)動是以struct hid_driver實(shí)現(xiàn),首先定義一個描述hid觸摸
    的頭像 發(fā)表于 10-29 10:55 ?541次閱讀
    <b class='flag-5'>linux</b>內(nèi)核中通用HID觸摸<b class='flag-5'>驅(qū)動</b>

    LSP 2.10 DaVinci Linux驅(qū)動程序

    電子發(fā)燒友網(wǎng)站提供《LSP 2.10 DaVinci Linux驅(qū)動程序.pdf》資料免費(fèi)下載
    發(fā)表于 10-09 09:30 ?0次下載
    LSP 2.10 DaVinci <b class='flag-5'>Linux</b><b class='flag-5'>驅(qū)動</b>程序

    北京迅為RK3568開發(fā)板嵌入式學(xué)習(xí)之Linux驅(qū)動全新更新-CAN+

    北京迅為RK3568開發(fā)板嵌入式學(xué)習(xí)之Linux驅(qū)動全新更新-CAN+
    的頭像 發(fā)表于 09-04 15:29 ?494次閱讀
    北京迅為RK3568<b class='flag-5'>開發(fā)</b>板嵌入式學(xué)習(xí)之<b class='flag-5'>Linux</b><b class='flag-5'>驅(qū)動</b>全新更新-CAN+

    linux系統(tǒng)的設(shè)備驅(qū)動一般分幾類

    Linux系統(tǒng)的設(shè)備驅(qū)動是操作系統(tǒng)與硬件設(shè)備之間的橋梁,負(fù)責(zé)實(shí)現(xiàn)操作系統(tǒng)與硬件設(shè)備之間的通信和控制。Linux系統(tǒng)的設(shè)備驅(qū)動可以分為以下幾類: 字符設(shè)備
    的頭像 發(fā)表于 08-30 15:13 ?409次閱讀

    Linux設(shè)備驅(qū)動程序分類有哪些

    Linux設(shè)備驅(qū)動程序是操作系統(tǒng)與硬件設(shè)備之間的橋梁,負(fù)責(zé)實(shí)現(xiàn)硬件設(shè)備與操作系統(tǒng)之間的通信和控制。Linux設(shè)備驅(qū)動程序的分類繁多,可以根據(jù)不同的標(biāo)準(zhǔn)進(jìn)行分類。 按硬件類型分類
    的頭像 發(fā)表于 08-30 15:11 ?530次閱讀

    linux驅(qū)動程序如何加載進(jìn)內(nèi)核

    Linux系統(tǒng)中,驅(qū)動程序是內(nèi)核與硬件設(shè)備之間的橋梁。它們允許內(nèi)核與硬件設(shè)備進(jìn)行通信,從而實(shí)現(xiàn)對硬件設(shè)備的控制和管理。 驅(qū)動程序的編寫 驅(qū)動
    的頭像 發(fā)表于 08-30 15:02 ?440次閱讀

    linux驅(qū)動程序主要有哪些功能

    Linux驅(qū)動程序是操作系統(tǒng)與硬件設(shè)備之間進(jìn)行通信的橋梁,負(fù)責(zé)實(shí)現(xiàn)硬件設(shè)備與操作系統(tǒng)之間的數(shù)據(jù)交換和控制。Linux驅(qū)動程序的主要功能包括以下幾個方面: 設(shè)備識別與初始化
    的頭像 發(fā)表于 08-30 14:47 ?347次閱讀

    linux驅(qū)動程序的編譯方法是什么

    Linux驅(qū)動程序的編譯方法主要包括兩種: 與內(nèi)核一起編譯 和 編譯成獨(dú)立的內(nèi)核模塊 。以下是對這兩種方法的介紹: 一、與內(nèi)核一起編譯 與內(nèi)核一起編譯意味著將驅(qū)動程序的源代碼直接集成到Linu
    的頭像 發(fā)表于 08-30 14:46 ?536次閱讀

    linux驅(qū)動程序的編譯方法有哪兩種

    Collection)或其他C/C++編譯器來編譯源代碼文件。這種方法較為原始,需要開發(fā)者手動指定編譯器選項(xiàng)、包含路徑、庫文件等。然而,在Linux驅(qū)動開發(fā)中,由于
    的頭像 發(fā)表于 08-30 14:39 ?640次閱讀

    linux驅(qū)動程序運(yùn)行在什么空間

    Linux 驅(qū)動程序是操作系統(tǒng)的一部分,負(fù)責(zé)管理硬件設(shè)備與操作系統(tǒng)之間的交互。驅(qū)動程序運(yùn)行在內(nèi)核空間(Kernel Space),這是操作系統(tǒng)的核心部分,與用戶空間(User Space)相對。內(nèi)核
    的頭像 發(fā)表于 08-30 14:37 ?392次閱讀

    Linux 驅(qū)動開發(fā)與應(yīng)用開發(fā),你知道多少?

    一、Linux驅(qū)動開發(fā)與應(yīng)用開發(fā)的區(qū)別開發(fā)層次不同:Linux
    的頭像 發(fā)表于 08-30 12:16 ?747次閱讀
    <b class='flag-5'>Linux</b> <b class='flag-5'>驅(qū)動</b><b class='flag-5'>開發(fā)</b>與應(yīng)用<b class='flag-5'>開發(fā)</b>,你知道多少?

    虹科技術(shù) Linux環(huán)境再升級:PLIN驅(qū)動程序正式發(fā)布

    Linux驅(qū)動程序領(lǐng)域再添新成員,PLIN驅(qū)動程序現(xiàn)已正式發(fā)布。
    的頭像 發(fā)表于 06-28 13:34 ?357次閱讀
    虹科技術(shù) <b class='flag-5'>Linux</b>環(huán)境再升級:PLIN<b class='flag-5'>驅(qū)動</b>程序正式發(fā)布

    AOSP源碼定制-內(nèi)核驅(qū)動編寫

    有時候?yàn)榱朔治鲆恍さ臋z測,需要在內(nèi)核層面對讀寫相關(guān)的操作進(jìn)行監(jiān)控,每次去修改對應(yīng)的內(nèi)核源碼編譯重刷過于耗時耗力,這里就來嘗試編寫一個內(nèi)核驅(qū)動,載入后監(jiān)控讀寫。
    的頭像 發(fā)表于 04-23 11:15 ?1199次閱讀
    AOSP源碼定制-內(nèi)核<b class='flag-5'>驅(qū)動</b><b class='flag-5'>編寫</b>

    ArmSoM系列板卡 嵌入式Linux驅(qū)動開發(fā)實(shí)戰(zhàn)指南 之 字符設(shè)備驅(qū)動

    字符設(shè)備驅(qū)動 本章,我們將學(xué)習(xí)字符設(shè)備使用、字符設(shè)備驅(qū)動相關(guān)的概念,理解字符設(shè)備驅(qū)動程序的基本框架,并從源碼上分析字符設(shè)備驅(qū)動實(shí)現(xiàn)和管理等。 主要分為下面五部分:
    的頭像 發(fā)表于 04-10 09:53 ?1049次閱讀
    ArmSoM系列板卡 嵌入式<b class='flag-5'>Linux</b><b class='flag-5'>驅(qū)動</b><b class='flag-5'>開發(fā)</b>實(shí)戰(zhàn)指南 之 字符設(shè)備<b class='flag-5'>驅(qū)動</b>

    嵌入式Linux開發(fā)的三種方式

    嵌入式Linux開發(fā)主要有三種方式:裸機(jī)開發(fā)、SDK開發(fā)驅(qū)動開發(fā)
    的頭像 發(fā)表于 01-22 14:22 ?961次閱讀
    RM新时代网站-首页