高效学习Linux内核——从源码中的宏下手

Posted 小螃蟹ya

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了高效学习Linux内核——从源码中的宏下手相关的知识,希望对你有一定的参考价值。

在阅读Linux内核代码时,在代码的开头总是有各种各样的宏定义,了解熟悉这些宏定义,对于代码的理解和快速修改代码提供了便利。有些宏相对简单,一眼便可以了解作用,有些就稍微复杂一些,需要一些逻辑,今天就给大家总结了一些常见的宏定义,帮助大家更快,更好的理解内核代码。

一、什么是宏

在C语言中,可以采用命令#define来定义宏。该命令允许把一个名称指定成任何所需的文本,例如一个常量值或者一条语句。在定义了宏之后,无论宏名称出现在源代码的何处,预处理器都会把它用定义时指定的文本替换掉。


 替换列表和标识符列表都是将字符串token化以后的列表。区别在于标识符列表使用,作为不同参数之间的分割符。每一个参数都是一个token化的列表。在宏中空白符只起到分割token的作用,空白符的多少对于预处理器是没有意义的。

二、内核中常见的宏

__CONCAT宏

"##"用于粘贴两个参数,"#"用于替换参数:


 BUILD_BUG_ON宏


 !!(e)对e的结果进行两次求非。如果e为0,则结果为0;如果e不为0,则结果为1。所以上述表达式的结果有两种:

1,condition为真时,sizeof(char[-1]),产生错误,编译不通过

2,condition为假时,sizeof(char[1]),编译通过

例如,在飞凌6Q的3.0.35的内核中,/driver/usb/storage/uas.c/usb接口u盘的驱动中就有使用该宏


 BUILD_BUG_ON_ZERO(e)宏


 检查表达式e是否为0,为0编译通过且返回0;如果不为0,则编译不通过


 如果e为0,则该结构体拥有一个int型的数据域,并且规定它所占的位的个数为0。


如果e非0,结构体的int型数据域的位域将变为一个负数,产生语法的错误

typecheck宏

该宏在/include/linux/typecheck.h中定义


 用于检查x是否为type类型,如果不是会抛出(warning: comparison of distinct pointer types lacks a cast)例如,在内核中/include/linux/rwlock.h中便使用到了该宏,用于读写锁的定义中。


 __is_constexpr宏


 判断x是否为整数常量表达式:如果x是常量表达式,则(void )((long)(x) 0l)是一个空指针常量,就会使用第三个操作数即((int *)8)的类型。如果不是常量表达式,则会使用第二个操作数void类型。

roundup宏

该宏在/include/linux/kernel.h中被定义。


 返回一个能够整除y并且大于x,最接近x的值,向上取整,可用于地址的内存对齐。

clamp宏


判断val是否在lo和hi的范围内,如果小于lo,返回lo,如果大于hi则返回hi,如果在lo和hi之间就返回val。

swap宏


利用typeof获取要交换变量的类型

container_of宏


根据一个结构体变量中的成员变量来获取整个结构体变量的指针

ALIGN对齐宏


 对齐是采用上对齐的方式,例如0x123以16对齐,结果是0x130,因为对齐常在分配内存时使用,所以分配的要比需要的大。

mdelay宏


忙等待函数,在延迟过程中无法运行其他任务,会占用CPU时间,延迟时间是准确的

三、总结

像开头说的,内核源码中有各种各样的宏,在这里不能一一罗列。宏的使用使得代码开发更加标准化,了解更多的宏,无疑可以更好的更深入的理解代码含义,如对此有兴趣,可在网上搜索更多的内核宏,了解更多关于Linux内核的知识


联系我们: 

服    务   热   线 :400-699-6866

飞凌嵌入式官网:https://www.forlinx.com/

官    方   商   城:https://forlinx.taobao.com/   作者:飞凌嵌入式 https://www.bilibili.com/read/cv12897286 出处:bilibili

以上是关于高效学习Linux内核——从源码中的宏下手的主要内容,如果未能解决你的问题,请参考以下文章

10+ 张图来探究 Linux 内核,以及如何高效学习

Linux内核(17) - 高效学习Linux驱动开发

学习 C 中的宏的建议

Git学习之路-Git简介

Git的学习

如何高效的学习 TensorFlow 代码