文章目录

  • 全系列传送门
  • 引言
  • 什么是unlocked_ioctl接口?
  • unlocked_ioctl和read/write函数有什么相同和不同
  • unlocked_ioctl接口命令规则
  • 命令的合成宏与分解宏
    • 合成宏
    • 分解宏
    • 测试程序
  • 使用_IO
    • module_leds.c
    • Makefile
    • app.c
    • 结果
  • 使用_IOW
  • 使用_IOR
    • 实验结果
  • 完整代码
    • module_leds.c
    • app.c

全系列传送门

Linux嵌入式驱动开发01——第一个驱动Hello World(附源码)

Linux嵌入式驱动开发02——驱动编译到内核

Linux嵌入式驱动开发03——杂项设备驱动(附源码)

Linux嵌入式驱动开发04——应用层和内核层数据传输

Linux嵌入式驱动开发05——物理地址到虚拟地址映射

Linux嵌入式驱动开发06——第一个相对完整的驱动实践编写

Linux嵌入式驱动开发07——GPIO驱动过程记录(飞凌开发板)

Linux嵌入式驱动开发08——字符设备(步步为营)

Linux嵌入式驱动开发09——平台总线详解及实战

Linux嵌入式驱动开发10——设备树开发详解

Linux嵌入式驱动开发11——平台总线模型修改为设备树实例

Linux嵌入式驱动开发12——pinctl和gpio子系统实践操作

Linux嵌入式驱动开发13——ioctl接口(gpio控制使用)

Linux嵌入式驱动开发14——中断的原理以及按键中断的实现(tasklet中断下文)

Linux嵌入式驱动开发15——等待队列和工作队列

Linux嵌入式驱动开发16——按键消抖实验(内核定时器)

Linux嵌入式驱动开发17——输入子系统

Linux嵌入式驱动开发18——I2C通信

引言

我们从平台总线模型,然后到pinctrl和gpio子系统,会发现步骤逐渐的规范,代码也逐渐的简单,也越来越能体会到linux屏蔽底层硬件的优势。

在之前的代码中,在write函数中往内核中写入数据。

并且在应用层传入内核层的过程,需要执行copy函数,对于我们这里只需要传入0或者1的少量数据来说,有些繁琐。

什么是unlocked_ioctl接口?

unlocked_ioctl就是ioctl接口,但是功能和对应的系统调用均没有发生变化

unlocked_ioctl和read/write函数有什么相同和不同

之前我们使用read/write函数完成写数据或者读数据的操作,ioctl函数也可以往内核中写入命令。

不同点是read/write函数是两个单独的函数,完成单一的读或者写功能,我们的ioctl函数,既可以读,也可以写。

不过read/write函数在读写大数据时候效率比较高。

所以,对于gpio控制led灯或者蜂鸣器等操作时,我们不需要大量的数据读写,可以使用ioctl函数,来简化,而read/write函数专职于大量数据的传输。

unlocked_ioctl接口命令规则

unlocked_ioctl总共32位

第一个分区 0-7,命令的编号,范围是0-255

第二个分区 8-15 命令的幻数。

(第一个分区和第二个分区主要作用就是用来区分命令的。)

第三个分区 16-29 表示传递的数据大小

第四个分区 30-31 代表读写的方向
00:表示用户程序和驱动数据没有数据传递
10:表示用户程序从驱动里面读取数据
01:表示用户程序向驱动里面写入数据

11:先写数据到驱动,然后再从驱动把数据读出来(不常用)

命令的合成宏与分解宏

合成宏

分解宏

测试程序

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>#include <sys/ioctl.h>#define CMD_TEST0 _IO('L', 0)                       // 'L'是幻数 0是编号
#define CMD_TEST1 _IO('L', 1)
#define CMD_TEST2 _IOW('L', 2, int)                 // 第三个参数size,不是数据大小,而是数据类型,这里是四个字节的话,就是int
#define CMD_TEST3 _IOR('L', 3, int)                 // 这四个命令幻数虽然一样,但是编号不同,所以不同。int main(int argc, char *argv[])
{printf("30-31 is %d\n", _IOC_DIR(CMD_TEST0));printf("30-31 is %d\n", _IOC_DIR(CMD_TEST3));return 0;
}

我们先来测试一下代码的功能,这里使用_IOC_DIR分解命令的方向,CMD_TEST0是定义命令编号,没有数据操作,所以_IOC_DIR分解后30-31的数据应该是00

CMD_TEST3是_IOR(‘L’, 3, int),是读取,所以分解后应该读到是读对应的10,对应十进制也就是2

接下来继续测试_IOC_TYPE函数

    printf("8-15 is %c\n", _IOC_TYPE(CMD_TEST0));printf("8-15 is %c\n", _IOC_TYPE(CMD_TEST1));

执行后可以看到返回了8-15位的幻数

printf("编号 0-7 is %d\n", _IOC_NR(CMD_TEST2));

使用_IO

模拟一个闪灯的操作

module_leds.c

#include <linux/init.h>
#include <linux/module.h>
#include <linux/miscdevice.h>
#include <linux/fs.h>
#include <linux/uaccess.h>
#include <linux/io.h>#define CMD_TEST0 _IO('L', 0)                       // 'L'是幻数 0是编号
#define CMD_TEST1 _IO('L', 1) int misc_open (struct inode *inode, struct file *file){printk("hello misc_open!!!\n");return 0;
}int misc_release(struct inode *inode, struct file *file){printk("bye bye misc_release!!!\n");return 0;}ssize_t misc_read(struct file *file, const char __user *ubuf, size_t size, loff_t *loff_t){char kbuf[64] = "copy to user!!!\n";if( copy_to_user(ubuf, kbuf, size) != 0 ){printk("copy_to_user error!!!\n");return -1;}printk("hello misc_read!!!\n");return 0;
}ssize_t misc_write(struct file *file, const char __user *ubuf, size_t size, loff_t *loff_t){char kbuf[64] = "copy from user!!!\n";printk("hello misc_write!!!\n");if( copy_from_user(kbuf, ubuf, size) != 0 ){printk("copy_from_user error!!!\n");return -1;}printk("buf is:%s\n", kbuf);return 0;
}long misc_ioctl(struct file *file, unsigned int cmd, unsigned long value)
{switch (cmd){case CMD_TEST0:printk("LED_ON!!!\n");case CMD_TEST1:printk("LED_OFF!!!\n");break;default:break;}return 0;
}struct file_operations misc_fops = {.owner = THIS_MODULE,.open = misc_open,.release = misc_release,.read = misc_read,.write = misc_write,.unlocked_ioctl = misc_ioctl,
};struct miscdevice misc_dev = {.minor = MISC_DYNAMIC_MINOR,.name = "hello_misc",.fops = &misc_fops
};static int misc_init(void)
{int ret;ret = misc_register(&misc_dev);if(ret < 0){printk("misc_register failed!!!\n");return -1;}printk("misc_register succeed!!!\n");           // 在内核中无法使用c语言库,所以不用printfreturn 0;
}static void misc_exit(void)
{misc_deregister(&misc_dev);printk("misc exit!!!\n");
}module_init(misc_init);
module_exit(misc_exit);MODULE_LICENSE("GPL");              //声明模块拥有开源许可

Makefile

# 开发板Linux内核的实际路径
# KDIR变量
KDIR:=/work/linux-4.1.15#  获取当前目录
PWD:=$(shell pwd)# obj-m表示将 chrdevbase.c这个文件 编译为 chrdevbase.ko模块。
obj-m += module_leds.o# 编译成模块
all:make -C $(KDIR) M=$(PWD) modulesclean:make -C $(KDIR) M=$(PWD) clean

app.c

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>#include <sys/ioctl.h>#define CMD_TEST0 _IO('L', 0)                       // 'L'是幻数 0是编号
#define CMD_TEST1 _IO('L', 1) int main(int argc, char *argv[])
{int fd;char buff[64] = {0};fd = open("/dev/hello_misc", O_RDWR);       // 打开节点时候触发open函数if(fd < 0){perror("open error\n");                 // perror在应用中打印return fd;}while(1){ioctl(fd, CMD_TEST0);                   // 触发驱动中的.unlocked_ioctlsleep(2);ioctl(fd, CMD_TEST1);sleep(2);}return 0;
}

结果

在测试结果中可以看到每次ioctl都会进入到驱动中的misc_ioctl中进行判断,然后打印出来预期的结果

使用_IOW

这里只修改了两个地方的代码
module_leds.c

long misc_ioctl(struct file *file, unsigned int cmd, unsigned long value)
{switch (cmd){case CMD_TEST2:printk("LED_ON!!!\n");printk("value is %d\n", value);break;case CMD_TEST3:printk("LED_OFF!!!\n");printk("value is %d\n", value);break;default:break;}return 0;
}

app.c

    while(1){ioctl(fd, CMD_TEST2, 0);                   // 触发驱动中的.unlocked_ioctlsleep(2);                               // 每隔两秒钟ioctl(fd, CMD_TEST3, 1);sleep(2);}

这里的一一对应的关系如图所示

这样,我们的实验结果就是,可以正常的写入数据

使用_IOR

有些时候我们需要读取按键的值,或者引脚的状态,使用_IOR就会很方便

在应用函数里,设置如下,传入value的地址到ioctl函数

    while(1){ioctl(fd, CMD_TEST4, &value);                   // 触发驱动中的.unlocked_ioctlprintf("app value is %d\n", value);sleep(2);                               // 每隔两秒钟}

在驱动函数中,依然要使用copy_to_user函数,复制数据从底层到应用层中,所以,在这里第一个参数就是应用层的value的地址,因为是int型指针,所以要强制转换一下,然后第二个参数就是底层的val值,第三个就是底层val的长度大小

    case CMD_TEST4:val = 12;if( copy_to_user((int *)value, &val, sizeof(val)) != 0 ){printk("copy_to_user error!!!\n");return -1;}break;

实验结果


我们程序中的val可以替换为底层io口的状态。

完整代码

module_leds.c

#include <linux/init.h>
#include <linux/module.h>
#include <linux/miscdevice.h>
#include <linux/fs.h>
#include <linux/uaccess.h>
#include <linux/io.h>#define CMD_TEST0 _IO('L', 0)                       // 'L'是幻数 0是编号
#define CMD_TEST1 _IO('L', 1) #define CMD_TEST2 _IOW('L', 2, int)
#define CMD_TEST3 _IOW('L', 3, int) #define CMD_TEST4 _IOR('L', 4, int) int misc_open (struct inode *inode, struct file *file){printk("hello misc_open!!!\n");return 0;
}int misc_release(struct inode *inode, struct file *file){printk("bye bye misc_release!!!\n");return 0;}ssize_t misc_read(struct file *file, const char __user *ubuf, size_t size, loff_t *loff_t){char kbuf[64] = "copy to user!!!\n";if( copy_to_user(ubuf, kbuf, size) != 0 ){printk("copy_to_user error!!!\n");return -1;}printk("hello misc_read!!!\n");return 0;
}ssize_t misc_write(struct file *file, const char __user *ubuf, size_t size, loff_t *loff_t){char kbuf[64] = "copy from user!!!\n";printk("hello misc_write!!!\n");if( copy_from_user(kbuf, ubuf, size) != 0 ){printk("copy_from_user error!!!\n");return -1;}printk("buf is:%s\n", kbuf);return 0;
}long misc_ioctl(struct file *file, unsigned int cmd, unsigned long value)
{int val;switch (cmd){case CMD_TEST2:printk("LED_ON!!!\n");printk("value is %d\n", value);break;case CMD_TEST3:printk("LED_OFF!!!\n");printk("value is %d\n", value);break;case CMD_TEST4:val = 12;if( copy_to_user((int *)value, &val, sizeof(val)) != 0 ){printk("copy_to_user error!!!\n");return -1;}break;default:break;}return 0;
}struct file_operations misc_fops = {.owner = THIS_MODULE,.open = misc_open,.release = misc_release,.read = misc_read,.write = misc_write,.unlocked_ioctl = misc_ioctl,
};struct miscdevice misc_dev = {.minor = MISC_DYNAMIC_MINOR,.name = "hello_misc",.fops = &misc_fops
};static int misc_init(void)
{int ret;ret = misc_register(&misc_dev);if(ret < 0){printk("misc_register failed!!!\n");return -1;}printk("misc_register succeed!!!\n");           // 在内核中无法使用c语言库,所以不用printfreturn 0;
}static void misc_exit(void)
{misc_deregister(&misc_dev);printk("misc exit!!!\n");
}module_init(misc_init);
module_exit(misc_exit);MODULE_LICENSE("GPL");              //声明模块拥有开源许可

app.c

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>#include <sys/ioctl.h>#define CMD_TEST0 _IO('L', 0)                       // 'L'是幻数 0是编号
#define CMD_TEST1 _IO('L', 1) #define CMD_TEST2 _IOW('L', 2, int)
#define CMD_TEST3 _IOW('L', 3, int) #define CMD_TEST4 _IOR('L', 4, int) int main(int argc, char *argv[])
{int fd;int value;fd = open("/dev/hello_misc", O_RDWR);       // 打开节点时候触发open函数if(fd < 0){perror("open error\n");                 // perror在应用中打印return fd;}while(1){ioctl(fd, CMD_TEST4, &value);                   // 触发驱动中的.unlocked_ioctlprintf("app value is %d\n", value);sleep(2);                               // 每隔两秒钟}return 0;
}

Linux嵌入式驱动开发13——ioctl接口(gpio控制使用)相关推荐

  1. Linux嵌入式驱动开发07——GPIO驱动过程记录(飞凌开发板)

    文章目录 全系列传送门 1. 在/arch/arm/boot/dts/imx6q-pinfunc.h查找 2. 在设备树配置文件中添加设备节点定义以及其引脚定义 3. 修改设备树文件添加配置 4. d ...

  2. Linux嵌入式驱动开发零基础入门集合(STM32过渡到Linux嵌入式)

    Linux嵌入式驱动开发01--第一个驱动Hello World(附源码) Linux嵌入式驱动开发02--驱动编译到内核 Linux嵌入式驱动开发03--杂项设备驱动(附源码) Linux嵌入式驱动 ...

  3. Linux嵌入式驱动开发02——驱动编译到内核

    文章目录 全系列传送门 make menuconfig图形化配置界面 1. 怎么进入到make menuconfig图形化界面? 2. make menuconfig图形化界面的操作 3. 退出 4. ...

  4. Linux嵌入式驱动开发01——第一个驱动Hello World(附源码)

    文章目录 全系列传送门 引言 驱动介绍 Hello World 1. 包含头文件 2. 驱动模块的入口和出口 3. 声明信息 4. 功能实现 完整代码 编译 第一种方法 第二种方法 编译成模块 第一步 ...

  5. 嵌入式 Linux 内核驱动开发【The first day: 36093万字】

    嵌入式 Linux 内核驱动开发[1] 嵌入式 Linux 内核驱动开发前言 第1章 Linux 内核裁剪和定制 [1]Linux 内核开发简介 [2] Linux 源码阅读工具 [1.2.1]Sou ...

  6. 《Linux设备驱动开发详解 A》一一2.3 接口与总线

    本节书摘来华章计算机出版社<Linux设备驱动开发详解 A>一书中的第2章,第2.3节,作者:宋宝华 更多章节内容可以访问云栖社区"华章计算机"公众号查看.1 2.3 ...

  7. linux cached释放_正点原子Linux第四十一章嵌入式Linux LED驱动开发实验

    1)资料下载:点击资料即可下载 2)对正点原子Linux感兴趣的同学可以加群讨论:935446741 3)关注正点原子公众号,获取最新资料更新 第四十一章嵌入式Linux LED驱动开发实验 上一章我 ...

  8. 【正点原子Linux连载】第四十一章 嵌入式Linux LED驱动开发实验 -摘自【正点原子】I.MX6U嵌入式Linux驱动开发指南V1.0

    1)实验平台:正点原子阿尔法Linux开发板 2)平台购买地址:https://item.taobao.com/item.htm?id=603672744434 2)全套实验源码+手册+视频下载地址: ...

  9. 【正点原子MP157连载】第二十一章 嵌入式Linux LED驱动开发实验-摘自【正点原子】STM32MP1嵌入式Linux驱动开发指南V1.7

    1)实验平台:正点原子STM32MP157开发板 2)购买链接:https://item.taobao.com/item.htm?&id=629270721801 3)全套实验源码+手册+视频 ...

最新文章

  1. java 投票ip限制_java 限定网站在指定IP段访问
  2. android 学习笔记之图形算法
  3. Kali Linux打开多个终端窗口
  4. HDFS--Hadoop分布式文件系统
  5. freebsd 运维人员
  6. Unet项目解析(3): run_testing.py
  7. iOS四种多线程(swift和oc)
  8. 复制选中的listbox内容
  9. perclos嘴巴_一种基于视频分析的疲劳状态检测方法及装置与流程
  10. 走不远的共享滑板车!
  11. 程序员如何快速迁移 10 亿级数据?
  12. 单节2A锂电池充电芯片方案,PD和QC快充充电器5-12V输入
  13. doc 问卷调查模板表_调查问卷模板.doc
  14. html ul动态添加li,javaScript动态添加Li元素
  15. 整理苹果官网上iOS的各种辅助功能
  16. 在图3-30 中,某学院的以太网交换机有三个接口分别和学院三个系的以太网相连,另外三个接口分别和电子邮件服务器、万维网服务器以及一个连接互联网的路由器相连。图中的A,B和C都是100Mbit/s以太网
  17. 联想小新笔记本电脑显示很暗, 教你如何将将其亮度调到最高
  18. android获取热点主机ip和连接热点手机ip
  19. IPv6下的DAD检测
  20. 用友企业空间 - http://upesn.com

热门文章

  1. 项目管理之工作分解结构(WBS)
  2. 线序检测视觉算设计过程及效果展示
  3. IE高级技巧不完全手册
  4. Docker系列 安装个人RSS服务TTRSS 手机完美适配
  5. Java Web实现用户登录功能
  6. 手机录屏录音不用愁,这些方法无需ROOT就能内录声音
  7. 如何将图片批量重命名001开始?
  8. SpringBoot对接微信小程序支付功能开发(二,支付回调功能)
  9. FusionCharts flash透明度设置
  10. GIS基础简介:基本概念、互联网坐标系、WebGIS实操