盘点Linux内核源码中使用宏定义的若干技巧(2)

简介: 5. typeof和0指针这个在大名鼎鼎的container_of就有出现,事实上一些面试题有时候也喜欢跟这个沾点边。 点击(此处)折叠或打开#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER) 点击(...
5. typeof和0指针
这个在大名鼎鼎的container_of就有出现,事实上一些面试题有时候也喜欢跟这个沾点边。

点击(此处)折叠或打开

  1. #define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)

点击(此处)折叠或打开

  1. #define container_of(ptr, type, member) ({ \
  2.     const typeof(((type *)0)->member)*__mptr = (ptr); \
  3.          (type *)((char *)__mptr - offsetof(type, member)); })
哦,看到没,这里就直接使用了大括号({...})。第一个宏offsetof用来获得一个结构体中某一成员MEMBER与该结构体起始地址的偏移量,这个用法很有创意,0指针,充分利用了编译器的技巧。比如下面的代码,输出的结果是4:

点击(此处)折叠或打开

  1. struct test{
  2.         int a;
  3.         int b;
  4. };
  5. int main(void)
  6. {
  7.         printf("offset_b = %ld\n", (size_t)(&((struct test*)0)->b));
  8.         return 0;
  9. }
typeof是gcc对C标准的一个扩展,用来告诉编译器你打算使用括号里面的变量类型。关于typeof的更详细说明请参考:http://gcc.gnu.org/onlinedocs/gcc/Typeof.html

6. 宏参数的静态检查

下面的宏来自模块参数的定义部分:

点击(此处)折叠或打开

  1. #define __module_param_call(prefix, name, ops, arg, isbool, perm)    \
  2.     /* Default value instead of permissions? */            \
  3.     static int __param_perm_check_##name __attribute__((unused)) =    \
  4.     BUILD_BUG_ON_ZERO((perm) 0 || (perm) > 0777 || ((perm) & 2))    \
  5.     + BUILD_BUG_ON_ZERO(sizeof(""prefix) > MAX_PARAM_PREFIX_LEN);    \
  6.     static const char __param_str_##name[] = prefix #name;        \
  7.     static struct kernel_param __moduleparam_const __param_##name    \
  8.     __used                                \
  9.     __attribute__ ((unused,__section__ ("__param"),aligned(sizeof(void *)))) \
  10.     = { __param_str_##name, ops, perm, isbool ? KPARAM_ISBOOL : 0,    \
  11.      { arg } }
其中第3-5行定义了一个事实上并不会用到的变量__param_perm_check_##name,为了防止编译器产生未使用变量的警告,该变量后面使用了__attribute__((unused))属性。这个并不会使用到的变量在该宏中的唯一作用是对变量参数perm和prefix的大小进行一个静态检查,如果BUILD_BUG_ON_ZERO中的条件为TRUE,那么将会产生一个编译错误。






目录
相关文章
|
16天前
|
Linux C语言
Linux内核队列queue.h
Linux内核队列queue.h
|
1月前
|
存储 Shell Linux
【Shell 命令集合 系统设置 】Linux 生成并更新内核模块的依赖 depmod命令 使用指南
【Shell 命令集合 系统设置 】Linux 生成并更新内核模块的依赖 depmod命令 使用指南
32 0
|
1月前
|
Shell Linux C语言
【Shell 命令集合 系统设置 】⭐Linux 卸载已加载的内核模块rmmod命令 使用指南
【Shell 命令集合 系统设置 】⭐Linux 卸载已加载的内核模块rmmod命令 使用指南
29 1
|
3天前
|
Linux 开发工具 Android开发
Docker系列(1)安装Linux系统编译Android源码
Docker系列(1)安装Linux系统编译Android源码
6 0
|
9天前
|
算法 Linux 调度
深入理解Linux内核的进程调度机制
【4月更文挑战第17天】在多任务操作系统中,进程调度是核心功能之一,它决定了处理机资源的分配。本文旨在剖析Linux操作系统内核的进程调度机制,详细讨论其调度策略、调度算法及实现原理,并探讨了其对系统性能的影响。通过分析CFS(完全公平调度器)和实时调度策略,揭示了Linux如何在保证响应速度与公平性之间取得平衡。文章还将评估最新的调度技术趋势,如容器化和云计算环境下的调度优化。
|
14天前
|
算法 Linux 调度
深度解析:Linux内核的进程调度机制
【4月更文挑战第12天】 在多任务操作系统如Linux中,进程调度机制是系统的核心组成部分之一,它决定了处理器资源如何分配给多个竞争的进程。本文深入探讨了Linux内核中的进程调度策略和相关算法,包括其设计哲学、实现原理及对系统性能的影响。通过分析进程调度器的工作原理,我们能够理解操作系统如何平衡效率、公平性和响应性,进而优化系统表现和用户体验。
20 3
|
21天前
|
负载均衡 算法 Linux
深度解析:Linux内核调度器的演变与优化策略
【4月更文挑战第5天】 在本文中,我们将深入探讨Linux操作系统的核心组成部分——内核调度器。文章将首先回顾Linux内核调度器的发展历程,从早期的简单轮转调度(Round Robin)到现代的完全公平调度器(Completely Fair Scheduler, CFS)。接着,分析当前CFS面临的挑战以及社区提出的各种优化方案,最后提出未来可能的发展趋势和研究方向。通过本文,读者将对Linux调度器的原理、实现及其优化有一个全面的认识。
|
21天前
|
Ubuntu Linux
Linux查看内核版本
在Linux系统中查看内核版本有多种方法:1) 使用`uname -r`命令直接显示版本号;2) 通过`cat /proc/version`查看内核详细信息;3) 利用`dmesg | grep Linux`显示内核版本行;4) 如果支持,使用`lsb_release -a`查看发行版及内核版本。
36 6
|
24天前
|
Linux 内存技术
Linux内核读取spi-nor flash sn
Linux内核读取spi-nor flash sn
18 1
|
1月前
|
存储 网络协议 Linux
【Linux 解惑 】谈谈你对linux内核的理解
【Linux 解惑 】谈谈你对linux内核的理解
24 0