首先,回顾一下基础的宏操作:

C语言宏

###

  1. #的作用是字符串化:在一个宏中的参数前面使用一个#,预处理器会把这个参数转换为一个字符数组

    #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");
  2. #是一种分隔连接方式,它的作用是先分隔,然后进行强制连接。

    例如:

    #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 )

作用是获取结构体某成员变量的偏移量。

分析如下:

  1. (type *)0 将0转化为该类型的指针,即地址为0x00000000
  2. ((type *)0)->member 访问成员member
  3. &(((type *)0)->member) 获取该成员地址(也就是其偏移量)
  4. (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



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

这段宏的分析如下:

  1. typeof()GNU C ,获得变量类型
  2. typeof (((type *)0 )->member) 起始地址为0 再获取member 最后返回member类型
  3. const typeof (((type *)0 )->member) * __mptr=(ptr) 定义 __mptr 指针,指向ptr指向的地址,并成为常量指针
  4. (char *)__mptr __mptr转化为字符型指针(运算以1个字节为单位)
  5. - __offsetof__(type,member)) 减去该成员的偏移量
  6. (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(一)的更多相关文章

  1. visual studio 阅读 linux-kernel

    @2018-12-13 [小记] 使用 visual studio 阅读 linux-kernel 方法 a. 文件 ---> 新建 --->从现有代码创建项目 b. 指定项目存储位置,命 ...

  2. LinuxKernel优秀博客

    1.vanbreaker的专栏 2.LinuxKernel Exploration 3.DroidPhone的专栏 4.Linux内核研究以及学习文档和ARM学习以及研究的开放文档   [力荐] 5. ...

  3. linux-kernel 学习计划

    [资料] http://www.ibm.com/developerworks/cn/views/linux/libraryview.jsp http://www.kerneltravel.net/ [ ...

  4. linux-kernel/CodingStyle

    https://www.kernel.org/doc/Documentation/zh_CN/CodingStyle Chinese translated version of Documentati ...

  5. 如何在LinuxKernel中操作file(set_fs與get_fs)

    在Kernel 中,照理說能存取至 0 ~ 4GB.但是實作層面卻是只能讓我們使用到3GB ~ 4GB 這會導致我們無法使用open(),write()這些在user space下的function. ...

  6. 在阿里云中编译Linux4.5.0内核 - Ubuntu内核编译教程

    实验环境:Ubnuntu 64位(推荐使用14.04)+Xshell 阿里云现在提供的云服务器很好用的,用来编译内核性能也不错.本文介绍最基本的内核编译方法,为了方便,所有操作均在root用户下进行. ...

  7. C语言位域

    转载自 http://tonybai.com/2013/05/21/talk-about-bitfield-in-c-again/ 再谈C语言位域 五 21 bigwhite技术志 bitfield, ...

  8. linux I/O复用

    转载自:哈维.dpkirin url:http://blog.csdn.NET/zhang_shuai_2011/article/details/7675797 http://blog.csdn.Ne ...

  9. Linux内核分析作业7:Linux内核如何装载和启动一个可执行程序

            1.可执行文件的格式 在 Linux 平台下主要有以下三种可执行文件格式: 1.a.out(assembler and link editor output 汇编器和链接编辑器的输出) ...

随机推荐

  1. 使用RS485串口服务器的方法

    1.为什么设备使用RS-485串口通信? RS-485设备可以连接到计算机,并在网络样式配置中的多个位置进行多次丢弃.在需要中继器之前,设备可以距离最远4000英尺(1220米),最多可以连接32个节 ...

  2. Vue项目入门实例

    前言 本文记录Vue2.x + Element-UI + TypeScript语法入门实例 为什么要用TypeScript? 1.TypeScript是JavaScript的超集,利用es6语法,实现 ...

  3. 计算机二级考试:Java

    目录 第 1 章 Java 语言概论 第 2 章 基本数据类型 2.1 概述 2.1.1 标识符 2.1.2 关键字 2.1.3 常量 2.2 基本数据类型 第 3 章 运算符和表达式 3.2 算术运 ...

  4. SPI的学习和ESP8266的SPI通讯测试

    SPI简介: SPI是串行外设接口(Serial Peripheral Interface)的缩写.SPI,是一种高速的,全双工,同步的通信总线,并且在芯片的管脚上只占用四根线,节约了芯片的管脚,同时 ...

  5. php判断手机浏览器和pc浏览器

    <?php public function is_mobile(){ // returns true if one of the specified mobile browsers is det ...

  6. Tab + Swipe+ RecyclerView + Collapsed

    随着Android的不断更新,老旧的布局页面已经过时,这就使得复杂的布局实现起来有些难度,在此记录一下手机中最常见的复杂界面实现方法. 最终效果 本文主要通过分析最新版AS下new project的S ...

  7. Zabbix实现电话告警通知的配置方法分享

    如果要讨论下当下热门的监控系统,我想zabbix应该能够占有自己的一席之地,拥有不小的话语权吧.然而身为一名苦逼的运维,为了不错过重大的告警信息,就需要配置个[电话告警]来进行最快速的通知. zabb ...

  8. TypeError: react__WEBPACK_IMPORTED_MODULE_2___default.a.createClass is not a function

    在看阮一峰的react入门的时候,写到一段代码,但是写完就报错了,经过多方查找,终于解决掉了 错误描述: 解决方法: 将React.createClass换成React.Component, 但是不知 ...

  9. linux的别名(alias/unalias)

    linux中有别名时先找的别名后找命令文件 临时创建是直接用alias. [root@localhost ~]# alias ls=pwd [root@localhost ~]# ls /root 其 ...

  10. 【译】Arc 在 Rust 中是如何工作的

    原文标题:How Arc works in Rust 原文链接:https://medium.com/@DylanKerler1/how-arc-works-in-rust-b06192acd0a6 ...