ARM驱动学习之21_字符驱动

news/2024/9/19 4:55:16/ 标签: arm开发, 学习

                              ARM驱动学习之21_字符驱动

操作步骤:

file_operations中的函数比较多,
选取用的比较多的函数简单介绍,
后
面的驱动教程中调用了对应的函数• int (*open) (struct inode *, struct file *)
– 打开函数
• int (*release) (struct inode *, struct file *)
– 释放close函数
• long (*unlocked_ioctl) (struct file *, unsigned int, unsigned long)
– io控制函数
• ssize_t (*read) (struct file *, char __user *, size_t, loff_t *)
– 读函数
• ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *)
– 写函数
• loff_t (*llseek) (struct file *, loff_t, int)
– 定位函数1.修改文件名为char_driver并修改Makefile里面的文件。定义函数如下:
/*打开操作*/
static int chardevnode_open(struct inode * inode,struct file * file){printk(KERN_EMERG "chardevnode_open is opened! \n");return 0;
}/*关闭操作*/
static int chardevnode_release(struct inode *inode,struct file *file){printk(KERN_EMERG "chardevnode_release is release! \n");return 0;     
}/*IO操作*/
static long chardevnode_ioctl (struct file *file, unsigned int cmd, unsigned long arg){printk(KERN_EMERG "chardevnode_ioctl is success! cmd is %d,arg is %d \n",cmd,arg);return 0;  
}static ssize_t chardevnode_read (struct file *file , char __user *buf, size_t count, loff_t *f_ops){return 0; 
}static ssize_t chardevnode_write (struct file *file , const char __user *buf, size_t count, loff_t *f_ops){return 0; 
}static loff_t chardevnode_llseek (struct file *file , loff_t offset, int ence){return 0; 
}2.在struct file_operations my_fops = {.owner = THIS_MODULE,.open = chardevnode_open,.release = chardevnode_release,.unlocked_ioctl = chardevnode_ioctl,.read = chardevnode_read,.write = chardevnode_write,.llseek = chardevnode_llseek,
};

2.修改invoke_hello.c文件:

1.将“invoke_hello.c”改为“invoke_char_driver.c”
在文件中添加如下内容:
char *hello_node0 = "/dev/chardevnode0";
char *hello_node1 = "/dev/chardevnode1";
2.修改函数名为hello_node0 和 hello_node1;
if((fd = open(hello_node0,O_RDWR|O_NDELAY)) < 0)
{printf();
}close(fd);编译应用命令如下
– arm-none-linux-gnueabi-gcc -o invoke_char_driver invoke_char_driver.c -static

代码,设备节点:

#include <linux/init.h>
/*包含初始化宏定义的头文件,代码中的module_init和module_exit在此文件中*/
#include <linux/module.h>
/*包含初始化加载模块的头文件,代码中的MODULE_LICENSE在此头文件中*/
/*定义module_param module_param_array的头文件*/
#include <linux/moduleparam.h>
/*定义module_param module_param_array中perm的头文件*/
#include <linux/stat.h>
/*字符设备函数*/
#include <linux/fs.h>
/*MDKDEV转换设备号数据类型宏定义*/
#include <linux/kdev_t.h>
/*定义字符设备的结构体*/
#include <linux/cdev.h>
/*分配内存空间函数头文件*/
#include <linux/slab.h>#include <linux/device.h>#define DEVICE_NAME "chardevnode"
#define DEVICE_MINOR_NUM 2
#define DEV_MAJOR 0
#define DEV_MINOR 0#define REGDEV_SIZE 3000MODULE_LICENSE("Dual BSD/GPL");
/*声明是开源的,没有内核版本限制*/
MODULE_AUTHOR("iTOPEET_dz");
/*声明作者*/static int numdev_major = DEV_MAJOR ;//主设备号
static int numdev_minor = DEV_MINOR ;//次设备号module_param(numdev_major,int,S_IRUSR);
module_param(numdev_minor,int,S_IRUSR);struct reg_dev
{char *data;unsigned long size;struct cdev cdev;
};
struct reg_dev *my_devices;
static struct class * myclass;/*打开操作*/
static int chardevnode_open(struct inode * inode,struct file * file){printk(KERN_EMERG "chardevnode_open is success! \n");return 0;
}/*关闭操作*/
static int chardevnode_release(struct inode *inode,struct file *file){printk(KERN_EMERG "chardevnode_release is success! \n");return 0;     
}/*IO操作*/
static long chardevnode_ioctl (struct file *file, unsigned int cmd, unsigned long arg){printk(KERN_EMERG "chardevnode_ioctl is success! cmd is %d,arg is %d \n",cmd,arg);return 0;  
}ssize_t chardevnode_read (struct file *file , char __user *buf, size_t count, loff_t *f_ops){return 0; 
}ssize_t chardevnode_write (struct file *file , const char __user *buf, size_t count, loff_t *f_ops){return 0; 
}loff_t chardevnode_llseek (struct file *file , loff_t offset, int ence){return 0; 
}struct file_operations my_fops = {.owner = THIS_MODULE,.open = chardevnode_open,.release = chardevnode_release,.unlocked_ioctl = chardevnode_ioctl,.read = chardevnode_read,.write = chardevnode_write,.llseek = chardevnode_llseek,
};/*设备注册到系统*/
static void reg_init_cdev(struct reg_dev *dev,int index){int err;int devno = MKDEV(numdev_major,numdev_minor + index);/*数据初始化*/cdev_init(&dev -> cdev,&my_fops);dev -> cdev.owner = THIS_MODULE;dev -> cdev.ops = &my_fops;err = cdev_add(&dev -> cdev,devno,1);if(err){printk(KERN_EMERG "failed,cdev_add is %d ,index is %d\n",err,index);}else{printk(KERN_EMERG "cdev_add %d success\n",index);printk(KERN_EMERG "numdev_minor  is %d",numdev_minor);}}static int Ascdev_init(void)
{int ret = 0;int i = 0;printk(KERN_EMERG "numdev_major is %d!\n",numdev_major);printk(KERN_EMERG "numdev_minor is %d!\n",numdev_minor);dev_t num_dev;if(numdev_major){num_dev = MKDEV(numdev_major,numdev_minor);//宏命令,用于处理各种设备号相关的数据//设备注册ret = register_chrdev_region(num_dev,DEVICE_MINOR_NUM,DEVICE_NAME);if(ret < 0){printk(KERN_EMERG "register_chrdev_region req is %d is failed \n",num_dev );}printk(KERN_EMERG "register_chrdev_region %d is success \n",numdev_major);	}else{ret = alloc_chrdev_region(&num_dev,numdev_minor,DEVICE_MINOR_NUM,DEVICE_NAME);if(ret < 0){printk(KERN_EMERG "alloc_chrdev_region req is %d is failed \n",num_dev );return -1;}numdev_major = MAJOR(num_dev);//huoqu printk(KERN_EMERG "numdev_major is %d \n",numdev_major);}myclass = class_create(THIS_MODULE,DEVICE_NAME);my_devices = kmalloc(DEVICE_MINOR_NUM * sizeof(struct reg_dev),GFP_KERNEL);	if(!my_devices){ret = -ENOMEM;goto fail;}	memset(my_devices,0,DEVICE_MINOR_NUM * sizeof(struct reg_dev));//初始化缓存为0;for(i = 0;i < DEVICE_MINOR_NUM;i ++){my_devices[i].data = kmalloc(REGDEV_SIZE,GFP_KERNEL);memset(my_devices[i].data,0,REGDEV_SIZE);reg_init_cdev(&my_devices[i],i);device_create(myclass,NULL,MKDEV(numdev_major,numdev_minor + i),NULL,DEVICE_NAME"%d",i);//这个 i 可以创建chardevnode0,chardevnode1		}printk(KERN_EMERG "Ascdev enter!\n");/*打印信息,KERN_EMERG表示紧急信息*/return 0;fail:unregister_chrdev_region(MKDEV(numdev_major,numdev_minor),DEVICE_MINOR_NUM);printk(KERN_EMERG "kmalloc is fail\n");return ret;	}static void Ascdev_exit(void)
{	int i;for(i = 0;i < DEVICE_MINOR_NUM;i++){cdev_del(&(my_devices[i].cdev));	device_destroy(myclass,MKDEV(numdev_major,numdev_minor + i));}class_destroy(myclass);
//	kfree(myclass);	unregister_chrdev_region(MKDEV(numdev_major,numdev_minor),DEVICE_MINOR_NUM);printk(KERN_EMERG "Ascdev exit!\n");
}module_init(Ascdev_init);
/*初始化函数*/
module_exit(Ascdev_exit);
/*卸载函数*/

应用:

/*************************************************************************
> File Name: Invoke.c
> Author: 
> Mail: 
> Created Time: Fri 27 Sep 2019 10:51:55 PM CST
************************************************************************/#include<stdio.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include <unistd.h>
#include <sys/ioctl.h>int main(void)
{char * hello_node0 = "/dev/chardevnode0";char * hello_node1 = "/dev/chardevnode1";int fd; if(fd = open(hello_node0,O_RDWR|O_NDELAY) < 0){printf("hello_node0 open %s is failed\r\n",hello_node0);}else{printf("hello_node0 open  %s is success\r\n",hello_node0);}close(fd);if(fd = open(hello_node1,O_RDWR|O_NDELAY) < 0){printf("hello_node1 open %s is failed\r\n",hello_node1);}else{printf("hello_node1 open  %s is success\r\n",hello_node1);}	close(fd);return 0;
}

运行结果:

[root@iTOP-4412]# ls
chardevicenode.ko   invoke_char_driver
[root@iTOP-4412]# insmod chardevicenode.ko                                                                     
[ 1951.543724] numdev_major is 0!
[ 1951.545312] numdev_minor is 0!
[ 1951.548335] numdev_major is 248 
[ 1951.553133] cdev_add 0 success
[ 1951.554733] numdev_minor  is 0
[ 1951.559293] cdev_add 1 success
[ 1951.561050] numdev_minor  is 0
[ 1951.565035] Ascdev enter!
[root@iTOP-4412]# ls /sys/class                                                                                
android_usb   gpsdrv        kovaplus      net           scsi_device   ump
arvo          graphics      lcd           power_supply  scsi_disk     usb_device
backlight     i2c-adapter   lirc          ppp           scsi_generic  video4linux
bdi           i2c-dev       mali          pyra          scsi_host
block         ieee80211     mdio_bus      rc            sound
bluetooth     input         mem           rc522         spi_master
chardevnode   kone          misc          regulator     switch
firmware      koneplus      mmc_host      rtc           tty
[root@iTOP-4412]# ls /dev/chardevnode*                                                                         
/dev/chardevnode0  /dev/chardevnode1
[root@iTOP-4412]# ./invoke_char_driver                                                                         
[ 2065.729172] chardevnode_open is success! 
[ 2065.732435] chardevnode_open is success! 
[ 2065.735821] chardevnode_release is success! 
[ 2065.740939] chardevnode_release is success! 
hello_node0 open  /dev/chardevnode0 is successhello_node1 open  /dev/chardevnode1 is success[root@iTOP-4412]# 
[root@iTOP-4412]# rmmod chardevicenode                                                                         
[ 2072.553457] Ascdev exit!


http://www.ppmy.cn/news/1526262.html

相关文章

【人工智能】AI创业的前沿思考 | 从垂直领域到通用智能模型AGI的崛起

写在前面&#xff1a; &#x1f31f; 欢迎光临 清流君 的博客小天地&#xff0c;这里是我分享技术与心得的温馨角落。&#x1f4dd; 个人主页&#xff1a;清流君_CSDN博客&#xff0c;期待与您一同探索 移动机器人 领域的无限可能。 &#x1f50d; 本文系 清流君 原创之作&…

p11 日志,元数据,进程的查看

直接运行docker run -d centos这个时候回启动容器&#xff0c;但是因为容器里面没有前台进程所以这个时候docker会把没用的进程给停止掉&#xff0c;可以看到docker ps命令没有查看到任何的正在运行的容器 但是如果说你使用 -it命令进入到了容器里面&#xff0c;这个他就不会…

??Ansible介绍

文章目录 一、Ansible基本概述1、什么是以及特性1&#xff09;是什么&#xff1a;2&#xff09;功能 2、架构下充当的角色&#xff1a;3、同软件对比Ansible与SaltStack*YAML的基本语法 Ansible与其他同类软件对比 4、Ansible的架构组成5、Ansible的执行流程 二、简单测试Ansib…

IP池对数据爬取工作的帮助

在数据爬取的过程中&#xff0c;IP池&#xff08;也称为代理IP池&#xff09;是一个极为重要的工具&#xff0c;它为数据抓取工作提供了多方面的支持和便利。本文将详细探讨IP池在数据爬取工作中的具体作用&#xff0c;以及它如何帮助提升数据抓取的效率、稳定性和合规性。 一…

[产品管理-21]:NPDP新产品开发 - 19 - 产品设计与开发工具 - 详细设计与规格定义

目录 前言&#xff1a; 一、详细设计与规格定义概述 1、产品详细设计 2、规格定义 3、详细设计与规格定义的关系 4、实际应用中的注意事项 二、详细设计与规格定义主要工具 2.1 质量功能展开QFD - 需求跟踪矩阵 1、QFD的基本原理 2、QFD的实施步骤 3、QFD的优势与应…

3. Python计算水仙花数

Python计算水仙花数 一、什么是水仙花数&#xff1f; 百度答案 二、怎样使用Python计算水仙花数&#xff1f; 这里需要for循环&#xff0c;if判断&#xff0c;需要range()函数&#xff0c;需要知道怎么求个位数&#xff0c;十位数&#xff0c;百位数… 1. For循环 语句结…

网络穿透:TCP 打洞、UDP 打洞与 UPnP

在现代网络中&#xff0c;很多设备都处于 NAT&#xff08;网络地址转换&#xff09;或防火墙后面&#xff0c;这使得直接访问这些设备变得困难。在这种情况下&#xff0c;网络穿透技术就显得非常重要。本文将介绍三种常用的网络穿透技术&#xff1a;TCP 打洞、UDP 打洞和 UPnP。…

【Elasticsearch系列五】Java API

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…

C++系列-函数对象/仿函数

函数对象/仿函数 &#x1f4a2;什么是仿函数&#x1f4a2;仿函数的使用&#x1f4a2;&#x1f4a2;像普通函数一样使用&#x1f4a2;&#x1f4a2;可以有自己的状态&#x1f4a2;&#x1f4a2;可以作为函数的参数&#x1f4a2;&#x1f4a2;可以作为模板参数&#x1f4a2;&…

ElementUI 快速入门:使用 Vue 脚手架搭建项目

文章目录 一 . ElementUI 的基本安装1.1 通过 Vue 脚手架创建项目1.2 在 vue 脚手架中安装 ElementUI1.3 编写页面 ElementUI 是 Vue.js 的强大 UI 框架&#xff0c;让前端界面开发变得简单高效。本教程将带你从安装到实战&#xff0c;快速掌握 ElementUI 的核心技巧。 核心内容…

手机玩机常识____展讯芯片刷机平台ResearchDownload的一些基本常识与问题解决

展讯ResearchDownload工具 展讯芯片的刷机工具--ResearchDownload下载工具"是一款专为用户设计的高效、便捷的下载管理软件&#xff0c;它能够帮助用户快速、稳定地从互联网上获取各种文件。这款工具以其强大的功能和良好的用户体验&#xff0c;在众多展讯芯片下载工具中脱…

【大数据方案】智慧大数据平台总体建设方案书(word原件)

第1章 总体说明 1.1 建设背景 1.2 建设目标 1.3 项目建设主要内容 1.4 设计原则 第2章 对项目的理解 2.1 现状分析 2.2 业务需求分析 2.3 功能需求分析 第3章 大数据平台建设方案 3.1 大数据平台总体设计 3.2 大数据平台功能设计 3.3 平台应用 第4章 政策标准保障体系 4.1 政策…

校园水电费管理|基于java的校园水电费管理小程序系统 (源码+数据库+文档)

校园水电费管理 目录 基于java的校园水电费管理小程序系统 一、前言 二、系统设计 三、系统功能设计 小程序端 后台功能模块 四、数据库设计 五、核心代码 六、论文参考 七、最新计算机毕设选题推荐 八、源码获取&#xff1a; 博主介绍&#xff1a;✌️大厂码农|毕…

Spring Boot-API版本控制问题

在现代软件开发中&#xff0c;API&#xff08;应用程序接口&#xff09;版本控制是一项至关重要的技术。随着应用的不断迭代&#xff0c;API 的改动不可避免&#xff0c;如何在引入新版本的同时保证向后兼容&#xff0c;避免对现有用户的影响&#xff0c;是每个开发者需要考虑的…

掌握MATLAB中的图形用户界面布局管理器

在MATLAB中&#xff0c;图形用户界面&#xff08;GUI&#xff09;的设计对于创建专业且用户友好的应用至关重要。布局管理器在GUI设计中扮演着核心角色&#xff0c;它们负责在窗口中自动管理和调整控件的位置和大小。本文将详细介绍MATLAB中的布局管理器&#xff0c;包括它们的…

[PICO VR眼镜]眼动追踪串流Unity开发与使用方法,眼动追踪打包报错问题解决(Eye Tracking)

前言 最近在做一个工作需要用到PICO4 Enterprise VR头盔里的眼动追踪功能&#xff0c;但是遇到了如下问题&#xff1a; 在Unity里面没法串流调试眼动追踪功能&#xff0c;根本获取不到Device&#xff0c;只能将整个场景build成APK&#xff0c;安装到头盔里&#xff0c;才能在…

Oracle 11gR2打PSU补丁详细教程

1 说明 Oracle的PSU&#xff08;Patch Set Update&#xff09;补丁是Oracle公司为了其数据库产品定期发布的更新包&#xff0c;通常每季度发布一次。PSU包含了该季度内收集的一系列安全更新&#xff08;CPU&#xff1a;Critical Patch Update&#xff09;以及一些重要的错误修…

6芯7芯可旋转电连接器航空插头

概述 可旋转电航空插头是一种能够在旋转或相对运动的部件间稳定传输电气信号或电源的装置&#xff0c;广泛应用于航空航天、自动化设备、医疗设备等多个领域。它的核心在于精密的接触系统&#xff0c;由旋转端和固定端两部分组成&#xff0c;通过金属触点或导电环实现电气连接。…

Unity 粒子系统参数说明

一、Particle System 1. Duration&#xff08;持续时间&#xff09; 粒子系统运行一次所需的时间。它决定粒子系统持续播放的时间长度。 2. Looping&#xff08;循环播放&#xff09; 如果启用&#xff0c;粒子系统将在播放完一次后自动重新开始播放&#xff0c;直到你停止它…

SpringBoot框架Web开发

1. 控制器开发 (1) JSON的支持 Spring Boot默认支持JSON格式的数据交换&#xff0c;主要通过以下方式实现&#xff1a; 依赖&#xff1a;spring-boot-starter-web 依赖包已经包括了 Jackson&#xff0c;用于将Java对象序列化为JSON&#xff0c;以及将JSON反序列化为Java对象…