lv14 ioctl、printk及多个此设备支持 6

1 ioctl操作实现

对相应设备做指定的控制操作(各种属性的设置获取等等)

long xxx_ioctl (struct file *filp, unsigned int cmd, unsigned long arg);
功能:对相应设备做指定的控制操作(各种属性的设置获取等等)
参数:filp:指向open产生的struct file类型的对象,表示本次ioctl对应的那次opencmd:用来表示做的是哪一个操作arg:和cmd配合用的参数
返回值:成功为0,失败-1

cmd组成

  • dir(direction),ioctl 命令访问模式(属性数据传输方向),占据 2 bit,可以为 IOC_NONE、IOC_READ、IOC_WRITE、IOC_READ | _IOC_WRITE,分别指示了四种访问模式:无数据、读数据、写数据、读写数据

  • type(device type),设备类型,占据 8 bit,在一些文献中翻译为 “幻数” 或者 “魔数”,可以为任意 char 型字符,例如 ‘a’、’b’、’c’ 等等,其主要作用是使 ioctl 命令有唯一的设备标识

  • nr(number),命令编号/序数,占据 8 bit,可以为任意 unsigned char 型数据,取值范围 0~255,如果定义了多个 ioctl 命令,通常

    从 0 开始编号递增

  • size,涉及到 ioctl 函数 第三个参数 arg ,占据 13bit 或者 14bit(体系相关,arm 架构一般为 14 位),指定了 arg 的数据类型及长度,如果在驱动的 ioctl 实现中不检查,通常可以忽略该参数;

#define _IOC(dir,type,nr,size) (((dir)<<_IOC_DIRSHIFT)| \((type)<<_IOC_TYPESHIFT)| \((nr)<<_IOC_NRSHIFT)| \((size)<<_IOC_SIZESHIFT))
//用于解码ioctl数字
/* used to create numbers */
​
// 定义不带参数的 ioctl 命令
#define _IO(type,nr)   _IOC(_IOC_NONE,(type),(nr),0)
​
//定义带读参数的ioctl命令(copy_to_user) size为类型名
#define _IOR(type,nr,size)  _IOC(_IOC_READ,(type),(nr),(_IOC_TYPECHECK(size)))
​
//定义带写参数的 ioctl 命令(copy_from_user) size为类型名
#define _IOW(type,nr,size)  _IOC(_IOC_WRITE,(type),(nr),(_IOC_TYPECHECK(size)))
​
//定义带读写参数的 ioctl 命令 size为类型名
#define _IOWR(type,nr,size) _IOC(_IOC_READ|_IOC_WRITE,(type),(nr),(_IOC_TYPECHECK(size)))
​
//用于解码ioctl数字
/* used to decode ioctl numbers */
#define _IOC_DIR(nr)        (((nr) >> _IOC_DIRSHIFT) & _IOC_DIRMASK)
#define _IOC_TYPE(nr)       (((nr) >> _IOC_TYPESHIFT) & _IOC_TYPEMASK)
#define _IOC_NR(nr)     (((nr) >> _IOC_NRSHIFT) & _IOC_NRMASK)
#define _IOC_SIZE(nr)      (((nr) >> _IOC_SIZESHIFT) & _IOC_SIZEMASK)

头文件位置

1.1 示例:

mychar.h

#ifndef MY_CHAR_H
#define MY_CHAR_H#include <asm/ioctl.h>#define MY_CHAR_MAGIC 'k'#define MYCHAR_IOCTL_GET_MAXLEN _IOR(MY_CHAR_MAGIC,1,int*)
#define MYCHAR_IOCTL_GET_CURLEN _IOR(MY_CHAR_MAGIC,2,int*)#endif

mychar.c

#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <linux/cdev.h>
#include <asm/uaccess.h>#include "mychar.h"#define BUF_LEN 100int major = 11;
int minor = 0;
int mychar_num = 1;//新建结构体类型
struct mychar_dev
{struct cdev mydev;char mydef_buf[BUF_LEN];  //相当于结构体的私有变量int curlen;          //相当于结构体的私有变量
};struct mychar_dev gmydev;int mychar_open(struct inode *pnode, struct file *pfile)
{//利用private_data私有变量来指向全局变量结构体地址pfile->private_data = (void*)(container_of(pnode->i_cdev,struct mychar_dev,mydev));printk("mychar_open is called\n");return 0;
}int mychar_close(struct inode *pnode, struct file *pfile)
{printk("mychar_close is called\n");return 0;
}ssize_t mychar_read(struct file *filp, char __user *pbuf, size_t count, loff_t *ppos)
{int ret = 0;int size = 0;//获取全家变量结构体地址struct mychar_dev *pmydev = (struct mychar_dev *)filp->private_data;if(count > pmydev->curlen){size = pmydev->curlen;}else{size = count;}//将内核空间中的数据复制到用户空间ret = copy_to_user(pbuf,pmydev->mydef_buf,size);if(ret){printk("copy_to_user failed\n");return -1;}//读完之后把后面的内容再拷贝过来,同时更新curlenmemcpy(pmydev->mydef_buf,pmydev->mydef_buf+size,pmydev->curlen - size);pmydev->curlen = pmydev->curlen - size;return size;}ssize_t mychar_write (struct file *filp, const char __user *pbuf, size_t count, loff_t *ppos)
{int size = 0;int ret  = 0;//获取全家变量结构体地址struct mychar_dev *pmydev = (struct mychar_dev *)filp->private_data;if(count > BUF_LEN - pmydev->curlen){size = BUF_LEN - pmydev->curlen;}else{size = count;}//将用户空间中的数据复制到内核空间中ret = copy_from_user(pmydev->mydef_buf + pmydev->curlen, pbuf, size);if(ret){printk("copy_from_user failed\n");return -1;}//更新curlenpmydev->curlen = pmydev->curlen + size;return size;
}long mychar_ioctl(struct file *filp, unsigned int cmd,unsigned long arg)
{int __user *pret = (int *)arg;int maxlen = BUF_LEN;int ret = 0;struct mychar_dev *pmydev = (struct mychar_dev *)filp->private_data;switch(cmd){case MYCHAR_IOCTL_GET_MAXLEN:ret = copy_to_user(pret,&maxlen,sizeof(int));if(ret){printk("copy_to_user MAXLEN failed\n");return -1;}break;case MYCHAR_IOCTL_GET_CURLEN:ret = copy_to_user(pret,&pmydev->curlen,sizeof(int));if(ret){printk("copy_to_user CURLEN failed\n");return -1;}break;default:printk("The cmd is unknow\n");return -1;}return 0;
}//结构体初始化:部分变量赋值初始化
struct file_operations myops = {.owner = THIS_MODULE,.open = mychar_open,.release = mychar_close,.read = mychar_read,.write = mychar_write,.unlocked_ioctl = mychar_ioctl
};int mychar_init(void)
{int ret = 0;dev_t devno = MKDEV(major, minor);/* 申请设备号 */ret = register_chrdev_region(devno, mychar_num, "mychar");if (ret) {ret = alloc_chrdev_region(&devno, minor, mychar_num, "mychar");if (ret) {printk("get devno failed\n");return -1;}major = MAJOR(devno); // 容易遗漏,注意}/* 给struct cdev对象指定操作函数集 */cdev_init(&gmydev.mydev, &myops);/* 将 struct cdev对象添加到内核对应的数据结构里 */gmydev.mydev.owner = THIS_MODULE;cdev_add(&gmydev.mydev, devno, mychar_num);return 0;
}void __exit mychar_exit(void)
{dev_t devno = MKDEV(major, minor);cdev_del(&gmydev.mydev);unregister_chrdev_region(devno, mychar_num);
}//表示支持GPL的开源协议
MODULE_LICENSE("GPL");module_init(mychar_init);
module_exit(mychar_exit);

Makefile

ifeq ($(KERNELRELEASE),)ifeq ($(ARCH),arm)
KERNELDIR ?= /home/linux/Linux_4412/kernel/linux-3.14
ROOTFS ?= /opt/4412/rootfs
else
KERNELDIR ?= /lib/modules/$(shell uname -r)/build
endif
PWD := $(shell pwd)modules:$(MAKE) -C $(KERNELDIR) M=$(PWD) modulesmodules_install:$(MAKE) -C $(KERNELDIR) M=$(PWD) modules INSTALL_MOD_PATH=$(ROOTFS) modules_installclean:rm -rf  *.o  *.ko  .*.cmd  *.mod.*  modules.order  Module.symvers   .tmp_versionselse
CONFIG_MODULE_SIG=n
obj-m += mychar.oendif

testmychar_app.c

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <sys/ioctl.h>#include "mychar.h"
#include <stdio.h>int main(int argc,char *argv[])
{int fd = -1;char buf[8] = "";int max = 0;int cur = 0;if(argc < 2){printf("The argument is too few\n");return 1;}fd = open(argv[1],O_RDWR);if(fd < 0){printf("open %s failed\n",argv[1]);return 2;}ioctl(fd,MYCHAR_IOCTL_GET_MAXLEN,&max);printf("max len is %d\n",max);write(fd,"hello",6);ioctl(fd,MYCHAR_IOCTL_GET_CURLEN,&cur);printf("cur len is %d\n",cur);read(fd,buf,8);printf("buf=%s\n",buf);close(fd);fd = -1;return 0;
}

 编译执行,测试获取设备参数

 

2 printk

//日志级别
#define KERN_EMERG  "<0>"   /* system is unusable           */
#define KERN_ALERT  "<1>"   /* action must be taken immediately */
#define KERN_CRIT   "<2>"   /* critical conditions          */
#define KERN_ERR    "<3>"   /* error conditions         */
​
#define KERN_WARNING    "<4>"   /* warning conditions           */
​
#define KERN_NOTICE "<5>"   /* normal but significant condition */
#define KERN_INFO   "<6>"   /* informational            */
#define KERN_DEBUG  "<7>"   /* debug-level messages         */
​
用法:printk(KERN_INFO"....",....)printk(KERN_INFO"Hello World"); =====> printk("<6>""Hello World") ====> printk("<6>Hello World")

dmesg --level=emerg,alert,crit,err,warn,notice,info,debug 

(dmesg中7个级别对应printk中7个级别)

#define HELLO_DEBUG
#undef PDEBUG
#ifdef HELLO_DEBUG
#define PDEBUG(fmt, args...) printk(KERN_DEBUG fmt, ##args)
#else
#define PDEBUG(fmt, args...)
#endif

3 多个次设备的支持

  • linux支持一个具体的设备同时占用1个主设备号多个次设备号的情况(这种情况主要是根据cdev_add中参数来决定的,一般是1,需要多个次设备号改写参数)
  • 另一种情况是针对一份驱动代码对应多个同类次设备(主设备号一样,次设备不一样的设备)

本节讲得是第二个情况,必须有一个struct cdev来代表它

  • cdev_init
  • cdev.owner赋值
  • cdev_add

以上三个操作对每个具体设备都要进行

3.1 示例

multimychar.c

#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <linux/cdev.h>
#include <asm/uaccess.h>#include "mychar.h"#define BUF_LEN 100
#define MYCHAR_DEV_CNT 3int major = 11;
int minor = 0;
int mychar_num = MYCHAR_DEV_CNT; //支持多个设备//新建结构体类型
struct mychar_dev
{struct cdev mydev;char mydef_buf[BUF_LEN];  //相当于结构体的私有变量int curlen;          //相当于结构体的私有变量
};struct mychar_dev gmydev_arr[MYCHAR_DEV_CNT];  //多个设备实现:创建结构体数组int mychar_open(struct inode *pnode, struct file *pfile)
{//利用private_data私有变量来指向全局变量结构体地址pfile->private_data = (void*)(container_of(pnode->i_cdev,struct mychar_dev,mydev));printk("mychar_open is called\n");return 0;
}int mychar_close(struct inode *pnode, struct file *pfile)
{printk("mychar_close is called\n");return 0;
}ssize_t mychar_read(struct file *filp, char __user *pbuf, size_t count, loff_t *ppos)
{int ret = 0;int size = 0;//获取全家变量结构体地址struct mychar_dev *pmydev = (struct mychar_dev *)filp->private_data;if(count > pmydev->curlen){size = pmydev->curlen;}else{size = count;}//将内核空间中的数据复制到用户空间ret = copy_to_user(pbuf,pmydev->mydef_buf,size);if(ret){printk("copy_to_user failed\n");return -1;}//读完之后把后面的内容再拷贝过来,同时更新curlenmemcpy(pmydev->mydef_buf,pmydev->mydef_buf+size,pmydev->curlen - size);pmydev->curlen = pmydev->curlen - size;return size;}ssize_t mychar_write (struct file *filp, const char __user *pbuf, size_t count, loff_t *ppos)
{int size = 0;int ret  = 0;//获取全家变量结构体地址struct mychar_dev *pmydev = (struct mychar_dev *)filp->private_data;if(count > BUF_LEN - pmydev->curlen){size = BUF_LEN - pmydev->curlen;}else{size = count;}//将用户空间中的数据复制到内核空间中ret = copy_from_user(pmydev->mydef_buf + pmydev->curlen, pbuf, size);if(ret){printk("copy_from_user failed\n");return -1;}//更新curlenpmydev->curlen = pmydev->curlen + size;return size;
}long mychar_ioctl(struct file *filp, unsigned int cmd,unsigned long arg)
{int __user *pret = (int *)arg;int maxlen = BUF_LEN;int ret = 0;struct mychar_dev *pmydev = (struct mychar_dev *)filp->private_data;switch(cmd){case MYCHAR_IOCTL_GET_MAXLEN:ret = copy_to_user(pret,&maxlen,sizeof(int));if(ret){printk("copy_to_user MAXLEN failed\n");return -1;}break;case MYCHAR_IOCTL_GET_CURLEN:ret = copy_to_user(pret,&pmydev->curlen,sizeof(int));if(ret){printk("copy_to_user CURLEN failed\n");return -1;}break;default:printk("The cmd is unknow\n");return -1;}return 0;
}//结构体初始化:部分变量赋值初始化
struct file_operations myops = {.owner = THIS_MODULE,.open = mychar_open,.release = mychar_close,.read = mychar_read,.write = mychar_write,.unlocked_ioctl = mychar_ioctl
};int mychar_init(void)
{int ret = 0;int i = 0;dev_t devno = MKDEV(major, minor);/* 申请设备号 */ret = register_chrdev_region(devno, mychar_num, "mychar");if (ret) {ret = alloc_chrdev_region(&devno, minor, mychar_num, "mychar");if (ret) {printk("get devno failed\n");return -1;}major = MAJOR(devno); // 容易遗漏,注意}for(i = 0;i < MYCHAR_DEV_CNT;i++){devno = MKDEV(major,minor+i);  //设备号需要重新组合/* 给struct cdev对象指定操作函数集 */cdev_init(&gmydev_arr[i].mydev, &myops);/* 将 struct cdev对象添加到内核对应的数据结构里 */gmydev_arr[i].mydev.owner = THIS_MODULE;cdev_add(&gmydev_arr[i].mydev, devno, 1);   //这里需要填1}return 0;
}void __exit mychar_exit(void)
{dev_t devno = MKDEV(major, minor);int i = 0;for(i = 0; i< MYCHAR_DEV_CNT; i ++){cdev_del(&gmydev_arr[i].mydev);}unregister_chrdev_region(devno, mychar_num);
}//表示支持GPL的开源协议
MODULE_LICENSE("GPL");module_init(mychar_init);
module_exit(mychar_exit);

mychar.h(未修改)

#ifndef MY_CHAR_H
#define MY_CHAR_H#include <asm/ioctl.h>#define MY_CHAR_MAGIC 'k'#define MYCHAR_IOCTL_GET_MAXLEN _IOR(MY_CHAR_MAGIC,1,int*)
#define MYCHAR_IOCTL_GET_CURLEN _IOR(MY_CHAR_MAGIC,2,int*)#endif

testmychar_app.c(维修工)

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <sys/ioctl.h>#include "mychar.h"
#include <stdio.h>int main(int argc,char *argv[])
{int fd = -1;char buf[8] = "";int max = 0;int cur = 0;if(argc < 2){printf("The argument is too few\n");return 1;}fd = open(argv[1],O_RDWR);if(fd < 0){printf("open %s failed\n",argv[1]);return 2;}ioctl(fd,MYCHAR_IOCTL_GET_MAXLEN,&max);printf("max len is %d\n",max);write(fd,"hello",6);ioctl(fd,MYCHAR_IOCTL_GET_CURLEN,&cur);printf("cur len is %d\n",cur);read(fd,buf,8);printf("buf=%s\n",buf);close(fd);fd = -1;return 0;
}

Makefile

ifeq ($(KERNELRELEASE),)ifeq ($(ARCH),arm)
KERNELDIR ?= /home/linux/Linux_4412/kernel/linux-3.14
ROOTFS ?= /opt/4412/rootfs
else
KERNELDIR ?= /lib/modules/$(shell uname -r)/build
endif
PWD := $(shell pwd)modules:$(MAKE) -C $(KERNELDIR) M=$(PWD) modulesmodules_install:$(MAKE) -C $(KERNELDIR) M=$(PWD) modules INSTALL_MOD_PATH=$(ROOTFS) modules_installclean:rm -rf  *.o  *.ko  .*.cmd  *.mod.*  modules.order  Module.symvers   .tmp_versionselse
CONFIG_MODULE_SIG=n
obj-m += mychar.oendif

编译运行

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/329166.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ssm基于Java Web的怀旧唱片售卖系统论文

摘 要 传统办法管理信息首先需要花费的时间比较多&#xff0c;其次数据出错率比较高&#xff0c;而且对错误的数据进行更改也比较困难&#xff0c;最后&#xff0c;检索数据费事费力。因此&#xff0c;在计算机上安装怀旧唱片售卖系统软件来发挥其高效地信息处理的作用&#x…

【STM32学习】硬件CRC与传统CRC-32计算的不同点

硬件CRC与传统CRC-32计算的不同点 1、stm32的硬件CRC32与传统CRC-32有何不同&#xff1f;2、解决办法 1、stm32的硬件CRC32与传统CRC-32有何不同&#xff1f; ①STM32F103的硬件CRC校验是对整个32位字进行CRC计算&#xff0c;传统的CRC-32是逐字节的计算。 ②STM32的硬件CRC32的…

C#,入门教程(09)——运算符的基础知识

上一篇&#xff1a; C#&#xff0c;入门教程(08)——基本数据类型及使用的基础知识https://blog.csdn.net/beijinghorn/article/details/123906998 一、算术运算符号 算术运算符号包括&#xff1a;四则运算 加 , 减-, 乘*, 除/与取模%。 // 加法&#xff0c;运算 int va 1 …

【JavaSE】Java中的反射动态代理

本篇文章整理的内容来源于: 反射原理 文章目录 一. 动态代理1. 优点2. 动态代理三要素3. 创建代理对象并使用 二. 反射1. 什么是反射2. 获取字节码文件对象的三种方式(1) Class.forName()获取 (源代码阶段)(2) 通过class属性获取(3) 通过对象获取字节码文件对象 3. 获取构造方…

LeetCode刷题11:滑动窗口解决1423.可获得的最大点数

几张卡牌 排成一行&#xff0c;每张卡牌都有一个对应的点数。点数由整数数组 cardPoints 给出。 每次行动&#xff0c;你可以从行的开头或者末尾拿一张卡牌&#xff0c;最终你必须正好拿 k 张卡牌。 你的点数就是你拿到手中的所有卡牌的点数之和。 给你一个整数数组 cardPoi…

网页设计与制作web前端设计html+css+js成品。电脑网站制作代开发。vscodeDrea 【企业公司宣传网站(HTML静态网页项目实战)附源码】

网页设计与制作web前端设计htmlcssjs成品。电脑网站制作代开发。vscodeDrea 【企业公司宣传网站&#xff08;HTML静态网页项目实战&#xff09;附源码】 https://www.bilibili.com/video/BV1Hp4y1o7RY/?share_sourcecopy_web&vd_sourced43766e8ddfffd1f1a1165a3e72d7605

c语言:用指针找出第一个相同的元素|练习题

一、题目 用指针&#xff0c;找出两数组中第一个相同的元素&#xff0c;并输入该元素 如图&#xff1a; 二、代码截图【带注释】 三、源代码【带注释】 #include <stdio.h> void f(); int main() { int a[5] {5,6,7,8,9}; int b[5] {6,4,6,8,3}; int *pa; …

工智能基础知识总结--什么是TextCNN

什么是TextCNN Yoon Kim在论文(2014 EMNLP) Convolutional Neural Networks for Sentence Classification提出TextCNN&#xff0c;该模型将卷积神经网络CNN应用到文本分类任务&#xff0c;是卷积神经网络应用到文本分析的开创性工作之⼀。 TextCNN的结构 TextCNN的结构图如下&…

【详解】静态库和动态库的认识和使用【Linux】

静态库和动态库的认识和使用 静态库和动态库的概述动静态库的实现静态库动态库库文件名称和引入库的名称 静态库和动态库的概述 静态库&#xff08;.a&#xff09;&#xff1a;程序在编译链接的时候把库的代码链接到可执行文件中。程序运行的时候将不再需要静态库 动态库&#…

Spring Boot 基础知识点1 (含面试题1)

Spring Boot 是一款基于 Spring 框架的开源应用程序开发工具&#xff0c;它旨在简化 Spring 应用程序的配置和开发过程。Spring Boot 提供了一种简单的方式来创建可独立运行的、生产级别的应用程序&#xff0c;并在需要时进行部署。Spring Boot 在微服务架构和云计算环境下得到…

报错curl: (6) Could not resolve host: raw.githubusercontent...的解决办法

我起初想要在macOS系统安装pip包&#xff0c;首先在终端安装homebrew&#xff0c;敲了命令&#xff1a;/usr/bin/ruby -e "$(curl -fsSL https://raw.githubusercontent...)" 之后触发的报错&#xff0c;报错内容&#xff1a;curl: (6) Could not resolve host: raw.…

2024.1.7每日一题

LeetCode 383.赎金信 383. 赎金信 - 力扣&#xff08;LeetCode&#xff09; 题目描述 给你两个字符串&#xff1a;ransomNote 和 magazine &#xff0c;判断 ransomNote 能不能由 magazine 里面的字符构成。 如果可以&#xff0c;返回 true &#xff1b;否则返回 false 。…