Linux驱动函数解读
一、kmalloc()、kzalloc()和vmalloc()
这三个函数都可以分配连续的虚拟内存
除此之外,这三个函数的区别有:
1. kmalloc()和kzalloc()函数分配的物理内存也是连续的,而vmalloc()分配的物理内存不一定连续
2. kmalloc()和kzalloc()函数分配的大小需要小于128K,而vmalloc()分配的大小没有限制
3. kmalloc()和kzalloc()分配内存的过程可以是原子操作(使用GFP_ATOMIC),而vmalloc()分配内存时则可能产生阻塞。因此vmalloc()不能从中断上下文调用
4. 在分配DMA内存时(使用GFP_DMA)需要保证物理内存连续,需要使用kmalloc()和kzalloc()
5. kmalloc()和kzalloc()分配内存开销小,因此比vmalloc()要快。内核较多使用的也是kmalloc()和kzalloc()
6. kmalloc()和kzalloc()的差别类似于malloc()和calloc(),kmalloc()只分配,不清0;kzalloc()会进行清0
7. kmalloc()和kzalloc()使用kfree()释放,vmalloc()使用vfree()释放
二、device_create()的调用和设备节点的创建过程
device_create()
-> device_create_vargs
-> kzalloc struct device
-> device_register()
-> device_initialize()
-> device_add()
-> kobject_uevent(&dev->kobj, KOBJ_ADD);
device的注册过程,最终只是调用kobject_uevent来发送一个action,对于uevent分析,读者可以参考:二十三、uevnet机制和U盘自动挂载
那么是谁创建/dev下的设备节点呢?
通常是使用应用程序udev来创建,在嵌入式系统中我们使用busybox的mdev
我们可以在命令行中输入mdev命令查看它的描述信息
我们在制作文件系统过程中(TINY4412移植第五节)使用到了mdev -s。它的作用是在引导期间运行扫描/sys下设备的并根据扫描到的设备信息在/dev创建设备节点
三、container_of()解读
在Linux内核编程中,会经常见到一个宏函数container_of(ptr,type,member),函数原型如下:
#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
...
#define container_of(ptr, type, member) ({ \
const typeof(((type *)0)->member) * __mptr = (ptr); \
(type *)((char *)__mptr - offsetof(type, member)); })
其实原理很简单,它的作用就是:已知结构体type的成员member的地址ptr,求解结构体type的起始地址

如上图,type的起始地址 = ptr - size(这里需要都转换为char *,因为它为单位字节)
而图中的size就是由((size_t) &((TYPE *)0)->MEMBER)求出来的
四、readb()和writeb()系列函数
#define readb_relaxed(c) ({ u8 __r = __raw_readb(c); __r; })
#define readw_relaxed(c) ({ u16 __r = le16_to_cpu((__force __le16) \
__raw_readw(c)); __r; })
#define readl_relaxed(c) ({ u32 __r = le32_to_cpu((__force __le32) \
__raw_readl(c)); __r; })
#define writeb_relaxed(v,c) __raw_writeb(v,c)
#define writew_relaxed(v,c) __raw_writew((__force u16) cpu_to_le16(v),c)
#define writel_relaxed(v,c) __raw_writel((__force u32) cpu_to_le32(v),c)
#define readb(c) ({ u8 __v = readb_relaxed(c); __iormb(); __v; })
#define readw(c) ({ u16 __v = readw_relaxed(c); __iormb(); __v; })
#define readl(c) ({ u32 __v = readl_relaxed(c); __iormb(); __v; })
#define writeb(v,c) ({ __iowmb(); writeb_relaxed(v,c); })
#define writew(v,c) ({ __iowmb(); writew_relaxed(v,c); })
#define writel(v,c) ({ __iowmb(); writel_relaxed(v,c); })
此系列函数用于从内存映射的I/O空间读写数据
readb()和readb_relaxed()从I/O空间读取8位数据(1字节)
readw()和readb_relaxew()从I/O空间读取16位数据(2字节)
readl()和readb_relaxel()从I/O空间读取32位数据(4字节)
writeb()和writeb_relaxed()从I/O空间写入8位数据(1字节)
writew()和writew_relaxed()从I/O空间写入16位数据(2字节)
writel()和writel_relaxed()从I/O空间写入32位数据(4字节)
其中没有relaxed()的系列函数都调用了__iowmb()和__iormb()函数,这两个函数是内存屏障指令,防止编译器优化执行过程。
五、ioctl()的幻数
ioctl()函数的幻数定义如下:
#define GLOBAL_MAGIC 'g'
#define CMD_RESET _IO(GLOBAL_MAGIC, 1)
如果正常定义CMD_RESET 0x1可能会导致不同的设备驱动拥有相同的命令号。如果设备1、设备2都支持0x1命令,就会造成命令码的污染。因此,Linux内核推荐采用幻数来生成命令
Linux驱动函数解读的更多相关文章
- linux驱动的入口函数module_init的加载和释放【转】
本文转载自:http://blog.csdn.net/zhandoushi1982/article/details/4927579 就像你写C程序需要包含C库的头文件那样,Linux内核编程也需要包含 ...
- module_init宏解析 linux驱动的入口函数module_init的加载和释放
linux驱动的入口函数module_init的加载和释放 http://blog.csdn.net/zhandoushi1982/article/details/4927579 void free_ ...
- 基于Linux 3.0.8 Samsung FIMC(S5PV210) 的摄像头驱动框架解读(一)
作者:咕唧咕唧liukun321 来自:http://blog.csdn.net/liukun321 FIMC这个名字应该是从S5PC1x0開始出现的.在s5pv210里面的定义是摄像头接口.可是它相 ...
- Linux驱动实践:中断处理函数如何【发送信号】给应用层?
作 者:道哥,10+年嵌入式开发老兵,专注于:C/C++.嵌入式.Linux. 关注下方公众号,回复[书籍],获取 Linux.嵌入式领域经典书籍:回复[PDF],获取所有原创文章( PDF 格式). ...
- linux驱动工程面试必问知识点
linux内核原理面试必问(由易到难) 简单型 1:linux中内核空间及用户空间的区别?用户空间与内核通信方式有哪些? 2:linux中内存划分及如何使用?虚拟地址及物理地址的概念及彼此之间的转化, ...
- Linux代码的重用与强行卸载Linux驱动
(一)Linux代码的重用 重用=静态重用(将要重用的代码放到其他的文件的头文件中声明)+动态重用(使用另外一个Linux驱动中的资源,例如函数.变量.宏等) 1.编译是由多个文件组成的Linux驱动 ...
- Linux 驱动学习笔记05--字符驱动实例,实现一个共享内存设备的驱动
断断续续学驱动,好不容易有空,做了段字符驱动的例子.主要还是跟书上学习在此记录下来,以后说不定能回过头来温故知新. 首先上驱动源码 gmem.c: /************************* ...
- Linux驱动学习之常用的模块操作命令
1.常用的模块操作命令 (1)lsmod(list module,将模块列表显示),功能是打印出当前内核中已经安装的模块列表 (2)insmod(install module,安装模块),功能是向当前 ...
- Linux驱动学习之什么是驱动?
一.什么是驱动? 1: 驱动一词的字面意思 2: 物理上的驱动 3: 硬件中的驱动 4: linux内核驱动.软件层面上的驱动广义上是指:这一段代码操作了硬件去动,所以这一段代码就叫硬件的驱动程序. ...
随机推荐
- [WEB安全]XXE漏洞总结
目录 0x00 XML基础 0x01 XML文档结构 0x02 DTD 0x03 实体 0x04 XXE漏洞 0x05 总结一些payload 0x06 XXE漏洞修复与防御 0x07 参考链接 0x ...
- OpenFOAM Tutorial Standard Solvers【转载】
转载自:http://www.cnblogs.com/fortran/articles/1996927.html boundaryFoam Steady-state solver for 1D tur ...
- TCP/IP 这猝不及防的爱情
前言 前几天看了老刘的一篇文章,TCP/IP 大明邮差.正好最近也在读<计算机自顶向下>一书 心血来潮,想写一个女版的TCP/IP 正文 一天,我正在百花会上赏花,赏着赏着,就出现了一个令 ...
- android studio的安装信息
默认是会下载sdk等文件的 Preparing "Install Android SDK Tools (revision: 26.1.1)". Downloading https: ...
- Kettle中ETL的效率优化
ETL效率优化 开启数据库日志记录及性能监控 如果我们想要优化一个ETL(KTR或者KJB)的性能,我们首先需要知道的就是它的瓶颈在哪里.而这些信息一般只能在ETL运行的步骤度量中看到,并且是不会持久 ...
- 2018-2019-2 《网络对抗技术》Exp9 Web安全基础 20165114
Exp9 Web安全基础 目录 一.实验内容 二.基础问题回答 (1)SQL注入攻击原理,如何防御 (2)XSS攻击的原理,如何防御 (3)CSRF攻击原理,如何防御 三.实践过程记录 3.1 注入缺 ...
- OpenTK学习笔记(2)-工作窗口的三种方法创建方法(winfrom下类的形式创建)
参考资料: https://www.codeproject.com/Articles/1167212/OpenGL-with-OpenTK-in-Csharp-Part-Initialize-the- ...
- Java 中清空map
本文链接:https://blog.csdn.net/TsuiXh/article/details/87879004在开发中在使用Map时,如果需要将Map作为临时的数据存储和处理,可以不用每次都去新 ...
- Linux笔记整理
[随时更新] ps aux | grep mysql 检测MySQL服务是否在运行 Linux查看某个进程的线程:ps -T -p <pid> 列出了由进程号为<pid>的进程 ...
- paint之文字示例
package com.loaderman.customviewdemo; import android.content.Context; import android.graphics.Canvas ...