0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

Linux驱动开发-编写按键驱动

DS小龙哥-嵌入式技术 2022-09-17 15:08 次阅读

【摘要】 这篇文章介绍,如何使用杂项设备框架编写一个简单的按键驱动,完成编写、编译、安装、测试等流程,了解一个杂项字符设备驱动的开发流程。

1. 杂项设备注册函数

这篇文章介绍,如何使用杂项设备框架编写一个简单的按键驱动,完成编写、编译、安装、测试等流程,了解一个杂项字符设备驱动的开发流程。

下面是杂项字符设备的接口:

struct miscdevice  {
	int minor;   /*次设备号 10 20 */
	const char *name; /*设备节点的名称*/
	const struct file_operations *fops; /*文件操作集合*/
	struct list_head list; //链表
	struct device *parent; 
	struct device *this_device;
	const char *nodename;
	umode_t mode;
};

//注册杂项字符设备
extern int misc_register(struct miscdevice * misc);
//注销杂项字符设备
extern int misc_deregister(struct miscdevice *misc);

按键需要将值传递给应用层,需要使用到copy_to_user函数,这个函数还有一个配对的copy_from_user,下面介绍这两个函数的详细功能和参数:

#include 

unsigned long copy_to_user(void __user *to, const void *from, unsigned long n)
函数功能: 将驱动层数据拷贝到应用层。
函数参数:
void __user *to  用户空间的地址--到哪里去
const void *from  驱动层的地址--从哪里来
unsigned long n  拷贝的大小
返回值:  0表示成功。 >0表示未拷贝成功的数量。

unsigned long copy_from_user(void *to, const void __user *from, unsigned long n)
函数功能: 将应用层的数据拷贝到驱动层。
函数参数:
void  *to  驱动空间的地址--拷贝到哪里去
const void __user *from  用户空间的地址--从哪里来
unsigned long n  拷贝的大小
返回值:  0表示成功。 >0表示未拷贝成功的数量。

2. 编写按键驱动

使用杂项设备注册按键驱动,应用层使用read接口读取按键值。

编写驱动之前需要先找到按键的原理图,找到按键接到CPU那个IO上的。

image-20211226144955867image-20211226145036869

然后再查阅数据手册,找到这个GPIO口的寄存器地址,寄存器的配置页面,方便初始化配置GPIO口为输入模式。

image-20211226145124296

2.1 按键驱动源代码

#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;
}

/*应用层的函数: 
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;
	}

	/*数据拷贝函数: 给应用层空间赋值--将驱动层的数据拷贝给应用层*/
	/*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("数据拷贝失败.\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内核管理驱动---设备号
设备号是一个unsigned int 的变量--32位。
设备号=主设备号+次设备号
*/
static struct miscdevice misc=
{
	.minor = MISC_DYNAMIC_MINOR,  /*次设备号填255表示自动分配     主设备号固定为10*/
	.name = "tiny4412_key",  /*/dev目录下文件名称*/
	.fops = &fops, /*文件操作接口*/
};

static int __init tiny4412_key_init(void)
{
    /*转换物理地址*/
	GPX3CON=ioremap(0x11000C60,4);
	GPX3DAT=ioremap(0x11000C64,4);

	/*配置GPIO口模式--配置按键*/	
	*GPX3CON&=0xFF0000FF;	
	
	/*1. 杂项设备的注册函数*/
	misc_register(&misc);
    printk("按键: 驱动安装成功\n");
    return 0;
}

static void __exit tiny4412_key_exit(void)
{
	/*2. 杂项设备的注销函数*/
	misc_deregister(&misc);

	/*取消转换*/
	iounmap(GPX3CON);
	iounmap(GPX3DAT);
    printk("按键: 驱动卸载成功\n");
}

module_init(tiny4412_key_init); /*驱动入口--安装驱动的时候执行*/
module_exit(tiny4412_key_exit); /*驱动出口--卸载驱动的时候执行*/

MODULE_LICENSE("GPL");  /*设置模块的许可证--GPL*/

2.2 makefile文件

编译驱动的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 应用层驱动测试代码

编译完运行时,传入按键的设备节点文件.

#include 
#include 
#include 
#include 

int main(int argc,char **argv)
{
    if(argc!=2)
    {
        printf("./app <设备节点文件>\n");
        return 0;
    }
    /*1. 打开设备文件*/
    int fd=open(argv[1],O_RDWR);
    if(fd<0)
    {
        printf("%s 设备驱动打开失败.\n",argv[1]);
        return 0;
    }
    /*2.读写数据*/
    int key_val;
    while(1)
    {
        read(fd,&key_val,4);//读取按键值
		if(key_val)
		{
			printf("%#x\n",key_val);
		}
    }
    /*3. 关闭文件*/
    close(fd);
    return 0;
}

2.4 驱动安装流程

[root@wbyq code]# ls
tiny4412_key_drv.ko
[root@wbyq code]# 
[root@wbyq code]# insmod tiny4412_key_drv.ko 
[  173.340000] 驱动测试: 驱动安装成功
[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] 驱动测试: 驱动卸载成功
[root@wbyq code]#
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 驱动
    +关注

    关注

    11

    文章

    1717

    浏览量

    84350
  • 函数
    +关注

    关注

    3

    文章

    3881

    浏览量

    61310
  • 按键
    +关注

    关注

    4

    文章

    220

    浏览量

    57338
收藏 人收藏

    评论

    相关推荐

    linux安装网卡驱动教程

    Linux系统中安装网卡驱动是一个比较基础的操作,下面我将为你详细讲解如何安装网卡驱动。 第一步,检查网卡型号和驱动支持情况:首先,你需要确定你的网卡型号,并查看该网卡型号在
    的头像 发表于 11-17 11:11 1639次阅读

    一文总结linux的platform驱动

    linux设备驱动中,有许多没有特定总线的外设驱动,在实际开发中,又需要使用到总线、驱动和设备模型这三个概念,故而
    的头像 发表于 10-16 16:45 412次阅读
    一文总结<b class='flag-5'>linux</b>的platform<b class='flag-5'>驱动</b>

    S3C2440开发板上实现按键点亮LED驱动开发的详细过程

    电子发烧友网站提供《S3C2440开发板上实现按键点亮LED驱动开发的详细过程.pdf》资料免费下载
    发表于 10-11 09:36 0次下载
    S3C2440<b class='flag-5'>开发</b>板上实现<b class='flag-5'>按键</b>点亮LED<b class='flag-5'>驱动</b><b class='flag-5'>开发</b>的详细过程

    Linux模块相关命令 Linux驱动模块的编写与挂载

    Linux模块相关命令 Linux驱动模块的编写与挂载
    发表于 10-01 12:20 198次阅读
    <b class='flag-5'>Linux</b>模块相关命令 <b class='flag-5'>Linux</b><b class='flag-5'>驱动</b>模块的<b class='flag-5'>编写</b>与挂载

    OpenHarmony:全流程讲解如何编写RTC平台驱动以及应用程序

    1、程序介绍本程序是基于OpenHarmony标准系统编写的平台驱动案例:RTC目前已在凌蒙派-RK3568开发板跑通。
    的头像 发表于 09-19 10:14 511次阅读
    OpenHarmony:全流程讲解如何<b class='flag-5'>编写</b>RTC平台<b class='flag-5'>驱动</b>以及应用程序

    基于OpenHarmony编写GPIO平台驱动和应用程序

    该程序是基于OpenHarmony标准系统编写的基础外设类:GPIO驱动
    的头像 发表于 09-12 10:04 472次阅读
    基于OpenHarmony<b class='flag-5'>编写</b>GPIO平台<b class='flag-5'>驱动</b>和应用程序

    基于DWC_ether_qos的以太网驱动开发-收发驱动编写与调试

    本文转自公众号,欢迎关注 基于DWC_ether_qos的以太网驱动开发-收发驱动编写与调试 (qq.com) https://mp.weixin.qq.com
    的头像 发表于 09-05 08:47 1368次阅读
    基于DWC_ether_qos的以太网<b class='flag-5'>驱动</b><b class='flag-5'>开发</b>-收发<b class='flag-5'>驱动</b><b class='flag-5'>编写</b>与调试

    Linux系统驱动开发之字符设备虚拟设备实验

    本系列图文教程均以全志H3开发板为实验板设计,字符设备驱动开发是最基础的驱动开发。其本质是按字节进行读写操作,读写数据是分先后顺序的。LED
    发表于 08-07 16:16 332次阅读
    <b class='flag-5'>Linux</b>系统<b class='flag-5'>驱动</b><b class='flag-5'>开发</b>之字符设备虚拟设备实验

    Linux Regmap 驱动框架

    层,regmap 向驱动编写人员提供的 API 接口,驱动编写人员使用这些API 接口来操作具体的芯片设备,也是驱动
    的头像 发表于 07-06 17:29 741次阅读
    <b class='flag-5'>Linux</b> Regmap <b class='flag-5'>驱动</b>框架

    基于状态机的按键驱动设计

    按键作为单片机的输入设备,可以向单片机输入数据、传输命令等,是设置参数和控制设备的常用接口。所以,学会按键驱动也是初学者必不可少的能力。说到按键驱动
    的头像 发表于 07-04 11:43 684次阅读
    基于状态机的<b class='flag-5'>按键</b><b class='flag-5'>驱动</b>设计

    基于Linux使用spidev驱动OLED

    如果不想编写spi设备驱动,那么linux内核提供了一个通用的spidev设备驱动,提供统一的字符设备操作,那么只需要在应用层读写和控制即可。以SPI OLED为例子,使用spidev
    发表于 06-16 10:36 2723次阅读
    基于<b class='flag-5'>Linux</b>使用spidev<b class='flag-5'>驱动</b>OLED

    Linux reset子系统及驱动实例

    上篇讲了Linux clock驱动,今天说说Linux的reset驱动
    发表于 05-31 16:16 610次阅读
    <b class='flag-5'>Linux</b> reset子系统及<b class='flag-5'>驱动</b>实例

    Linux clock子系统及驱动实例

    Linux驱动中,操作时钟只需要简单调用内核提供的通用接口即可,clock驱动通常是由芯片厂商开发的,在Linux启动时clock
    发表于 05-31 16:10 462次阅读
    <b class='flag-5'>Linux</b> clock子系统及<b class='flag-5'>驱动</b>实例

    Linux之PWM驱动

    本文主要讲述了Linux的PWM驱动框架、实现方法、驱动添加方法和调试方法。
    发表于 05-25 09:19 408次阅读
    <b class='flag-5'>Linux</b>之PWM<b class='flag-5'>驱动</b>

    汇编驱动IMX6ULL LED灯

    用汇编编写正点原子Linux开发板的Led驱动
    的头像 发表于 05-01 09:19 25.8w次阅读
    汇编<b class='flag-5'>驱动</b>IMX6ULL LED灯