LinuxKernel(一)
首先,回顾一下基础的宏操作:
C语言宏
#与##
#的作用是字符串化:在一个宏中的参数前面使用一个#,预处理器会把这个参数转换为一个字符数组#define ERROR_LOG(info) fprintf(stderr,"error:"#info"\n");
则有:
ERROR_LOG("add"); ---> fprintf(stderr,"error: "add"\n");
ERROR_LOG(devied =0); ---> fprintf(stderr,"error: devied=0\n");
#是一种分隔连接方式,它的作用是先分隔,然后进行强制连接。例如:
#define XNAME(n) x##n
那么
XNAME(4)就会展开为x4.
do{/*codes*/}while(0)
采用这种方式是为了防范在使用宏过程中出现错误,主要有如下几点:
(1)空的宏定义避免warning:
#define foo() do{}while(0)
(2)存在一个独立的block,可以用来进行变量定义,进行比较复杂的实现。
(3)如果出现在判断语句过后的宏,这样可以保证作为一个整体来是实现:
#define foo() \
action1(); \
action2();
在遇到分支语句时:
if(NULL == pPtr)
foo();
foo()中的两个语句就不会都被执行。
(4)为何不用单独的{}
#define switch(x,y) {int tmp; tmp=x;x=y;y=tmp;}
if(x>y)
switch(x,y);
else
op();
在把宏引入代码中,会多出一个分号,从而会报错。
变参宏: ··· 与__VA_ARGS__
某些函数接受可变的参数例如printf(),在头文件stdvar.h中有工具可以自定义变参宏。
把宏参数列表中最后的参数用···省略,而__VA_ARGS__可用在替换部分,表面省略号代表的东西。
#define PR(···) printf(__VA_ARGS__)
例如:
PR("THIS IS __VA_ARGS__");
会被展开为:
printf("THIS IS __VA_ARGS__");
预定义符号
| 符号 | 样例值 | 含义 |
|---|---|---|
__FILE__ |
"test.c" |
进行编译的文件名 |
__LINE__ |
25 |
当前行的行号 |
__DATE__ |
"Jan 31 2001" |
被编译的日期 |
__TIME__ |
"23:17:24" |
被编译的时间 |
__STDC__ |
1 |
是否遵循ANSI C |
__FUNCTION__ |
main |
所在函数名称 |
这些宏与编译器有关,有些支持有些不支持.
如下程序:

运行结果为:

注意到: 如果用函数或内联函数,每次的行号便都会相同。
下面是内核中,常见的两个宏:
Linux常用的两个宏
offsetof
该宏的定义如下:
#define __offsetof__(type, member) ( ( size_t ) & ( ( type * ) 0 )->member )
作用是获取结构体某成员变量的偏移量。
分析如下:
(type *)0将0转化为该类型的指针,即地址为0x00000000((type *)0)->member访问成员member&(((type *)0)->member)获取该成员地址(也就是其偏移量)(size_t)&(((type *)0)->member)将地址转化为size_t类型 即偏移量
这里访问0指针为何不会报错,取决于gcc对于该过程的优化,不会直接访问空间而是直接获得地址.
container_of
该宏的定义如下:
#define __container_of__(ptr, type, member) ({\
const typeof ( ( ( type * ) 0 ) -> member ) *__mptr=(ptr);\
( type * )( ( char * )__mptr - __offsetof__( type, member ) );\
})
要理解这段宏,需要知道几个GCC C EXTENSIONS,查阅GCC MANUAL:
1.Statements and Declarations in Expressions


2.Referring to a Type with typeof

手册中也给出了一个典型的用法示例:

这段宏的分析如下:
typeof()为GNU C,获得变量类型typeof (((type *)0 )->member)起始地址为0再获取member最后返回member类型const typeof (((type *)0 )->member) * __mptr=(ptr)定义__mptr 指针,指向ptr指向的地址,并成为常量指针(char *)__mptr__mptr转化为字符型指针(运算以1个字节为单位)- __offsetof__(type,member))减去该成员的偏移量(type*)( ( char * )__mptr - __offsetof__(type,member))最后转化为指向该类型的指针(指向该类型的首地址)
关于上述两个宏的一段程序如下:
#include <stdio.h>
#include <string.h>
/**
* 获取结构体变量成员的偏移量
* @param type 类型(struct)
* @param member 成员
*/
#define __offsetof__(type, member) ( ( size_t ) & ( ( type * ) 0 )->member )
/**
* 获取指向整个结构体的指针
* @param ptr 指向成员(member)变量的指针
* @param type 类型(struct)
* @param member 成员变量
*/
#define __container_of__(ptr, type, member) ({\
const typeof ( ( ( type * ) 0 ) -> member ) *__mptr=(ptr);\
( type * )( ( char * )__mptr - __offsetof__( type, member ) );\
})
typedef struct Student {
char gender;
int id;
int age;
char name[20];
double score;
} Stu;
int main() {
int gender_offset,id_offset,age_offset,name_offset,score_offset;
gender_offset = __offsetof__(struct Student, gender);
id_offset = __offsetof__(struct Student, id);
age_offset = __offsetof__(struct Student, age);
name_offset = __offsetof__(struct Student, name);
score_offset = __offsetof__(struct Student, score);
printf("%d\t%d\t%d\t%d\t%d\n", gender_offset, id_offset, age_offset, name_offset, score_offset);
Stu stu;
Stu *pstu;
stu.gender = '1';
stu.id = 9527;
stu.age = 18;
stu.score = 98.2;
strcpy(stu.name, "elioyang");
pstu = __container_of__(&stu.id, Stu, id);
printf("gender=%c\n", pstu->gender);
printf("age=%d\n", pstu->age);
printf("name=%s\n", pstu->name);
printf("score=%lf", pstu->score);
return 0;
}
运行结果如下:
0 4 8 12 32
gender=1
age=18
name=elioyang
score=98.200000
LinuxKernel(一)的更多相关文章
- visual studio 阅读 linux-kernel
@2018-12-13 [小记] 使用 visual studio 阅读 linux-kernel 方法 a. 文件 ---> 新建 --->从现有代码创建项目 b. 指定项目存储位置,命 ...
- LinuxKernel优秀博客
1.vanbreaker的专栏 2.LinuxKernel Exploration 3.DroidPhone的专栏 4.Linux内核研究以及学习文档和ARM学习以及研究的开放文档 [力荐] 5. ...
- linux-kernel 学习计划
[资料] http://www.ibm.com/developerworks/cn/views/linux/libraryview.jsp http://www.kerneltravel.net/ [ ...
- linux-kernel/CodingStyle
https://www.kernel.org/doc/Documentation/zh_CN/CodingStyle Chinese translated version of Documentati ...
- 如何在LinuxKernel中操作file(set_fs與get_fs)
在Kernel 中,照理說能存取至 0 ~ 4GB.但是實作層面卻是只能讓我們使用到3GB ~ 4GB 這會導致我們無法使用open(),write()這些在user space下的function. ...
- 在阿里云中编译Linux4.5.0内核 - Ubuntu内核编译教程
实验环境:Ubnuntu 64位(推荐使用14.04)+Xshell 阿里云现在提供的云服务器很好用的,用来编译内核性能也不错.本文介绍最基本的内核编译方法,为了方便,所有操作均在root用户下进行. ...
- C语言位域
转载自 http://tonybai.com/2013/05/21/talk-about-bitfield-in-c-again/ 再谈C语言位域 五 21 bigwhite技术志 bitfield, ...
- linux I/O复用
转载自:哈维.dpkirin url:http://blog.csdn.NET/zhang_shuai_2011/article/details/7675797 http://blog.csdn.Ne ...
- Linux内核分析作业7:Linux内核如何装载和启动一个可执行程序
1.可执行文件的格式 在 Linux 平台下主要有以下三种可执行文件格式: 1.a.out(assembler and link editor output 汇编器和链接编辑器的输出) ...
随机推荐
- sqlsugar入门(2)-C#方法与sugar自定义函数的区别
1.使用tostring获取当天数据 var list = ssc.Queryable<Student>().Where(o => o.CreateTime.Value.ToStri ...
- CF1271E Common Number
数学+二分 连续打了3场$codeforces$,深深的被各种模拟贪心分类讨论的$C$,$D$题给恶心到了 还有永远看到题一脸懵的$B$题 首先考虑画出不同函数值迭代转移的关系,要注意考虑连边是否能成 ...
- 遗传算法(Genetic Algorithm)——基于Java实现
一.遗传算法原理介绍 遗传算法(Genetic Algorithm)是模拟达尔文生物进化论的自然选择和遗传学机理的生物进化过程的计算模型,是一种通过模拟自然进化过程搜索最优解的方法.遗传算法是从代表问 ...
- 20 HTTP1.0和HTTP1.1
20 HTTP1.0和HTTP1.1 推荐: http://blog.csdn.net/elifefly/article/details/3964766 请求头Host字段,一个服务器多个网站 长链接 ...
- excel导出csv包括逗号等的处理
/** * @Title: trimRubbishChar * @Description: 导出的时候需要对一格的内容进行检查,看是否有非法字符,以免串行 * @Since: 2016年8月2日 下午 ...
- Netlink 内核实现分析 3
Netlink IPC 数据结构 #define NETLINK_ROUTE 0 /* Routing/device hook */ #define NETLINK_UNUSED 1 /* Unuse ...
- python中 try、except、finally 的执行顺序(转)
def test1(): try: print('to do stuff') raise Exception('hehe') print('to return in try') return 'try ...
- python之路《七》文件的处理
其实我觉得python的方便之处就是简便特别是对数据的处理上 当我们要处理一些数据的时候免不了的就是要处理许多的文件那么下面就让我们来看一下文件的处理 1.打开文件 python的文件打开方式较为简单 ...
- cephfs根据存储池显示df容量
前言 如果用cephfs比较多,应该都知道,在cephfs的客户端进行mount以后,看到的容量显示的是集群的总的容量,也就是你的总的磁盘空间是多少这个地方显示的就是多少 这个一直都是这样显示的,我们 ...
- echarts折线图,数据切换时(最近七天)绘图不合理现象
echarts折线图,当进行数据切换时存在绘制不合理的问题,数据没错,但绘制不对. 两个0之间的连线应该是平滑直线,如图: 正确的显示: 解决: 在myCharts.setOption(option) ...