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

news/2025/1/15 18:59:44/

                              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 的首页,持续学…