[Linux]学习笔记系列 -- bits


title: bits

categories:

  • linux
  • include
    tags:
  • linux
  • include
    abbrlink: 75328c4f
    date: 2025-10-03 09:01:49

https://github.com/wdfk-prog/linux-study

文章目录

  • include/linux/bits.h
    • [BIT_WORD 位图的字数](#BIT_WORD 位图的字数)
  • include/linux/bitmap.h
    • [bitmap_weight 位图的权重](#bitmap_weight 位图的权重)
  • include/linux/bitops.h
    • bitop
    • [test_bit 用于测试某个位是否被设置(即是否为 1)](#test_bit 用于测试某个位是否被设置(即是否为 1))
    • [assign_bit 为内存中的位分配值](#assign_bit 为内存中的位分配值)
    • fls_long
    • [BITS_TO_LONGS 位图的字数](#BITS_TO_LONGS 位图的字数)

include/linux/bits.h

BIT_WORD 位图的字数

c 复制代码
//include/asm-generic/bitsperlong.h
#ifdef CONFIG_64BIT
#define BITS_PER_LONG 64
#else
#define BITS_PER_LONG 32
#endif /* CONFIG_64BIT */

#define BIT_WORD(nr)		((nr) / BITS_PER_LONG)

include/linux/bitmap.h

bitmap_weight 位图的权重

  • 用于计算位图(bitmap)中设置为 1 的位的数量(即权重)
c 复制代码
static __always_inline
unsigned int bitmap_weight(const unsigned long *src, unsigned int nbits)
{
	if (small_const_nbits(nbits))
		return hweight_long(*src & BITMAP_LAST_WORD_MASK(nbits));
	return __bitmap_weight(src, nbits);
}

include/linux/bitops.h

bitop

c 复制代码
/* 许多特定于体系结构的非原子 Bitops 包含内联 asm 代码,因此编译器无法将它们优化为编译时表达式或常量。
相反,generic_*() 辅助函数是用纯 C 语言定义的,编译器对它们进行了优化。
为了在某些情况下(如参数是编译时常量时)利用这种优化,bitop 宏会选择通用的 C 实现(如 const##op),而在其他情况下则使用默认的实现(如 op)。这种设计使得代码在性能敏感的场景中既能保持高效,又能在编译时优化常量表达式。
从 .bss 或 .data 传递位图时,需要强制转换到"uintptr_t"以缓解"-Waddress"警告 (-> '!!addr' 始终为 true)。
 */
#define bitop(op, nr, addr)						\
	((__builtin_constant_p(nr) &&					\
	  __builtin_constant_p((uintptr_t)(addr) != (uintptr_t)NULL) &&	\	//检查 addr 是否是一个非空的编译时常量地址
	  (uintptr_t)(addr) != (uintptr_t)NULL &&			\				//检查 addr 是否为非空指针
	  __builtin_constant_p(*(const unsigned long *)(addr))) ?	\		//检查 addr 指向的值是否是一个编译时常量
	 const##op(nr, addr) : op(nr, addr))								//如果所有条件都满足(即 nr、addr 和 *addr 都是编译时常量),宏会调用 const##op(nr, addr)。否则使用通用实现

test_bit 用于测试某个位是否被设置(即是否为 1)

c 复制代码
#define test_bit(nr, addr)		bitop(_test_bit, nr, addr)

assign_bit 为内存中的位分配值

c 复制代码
/**
 * assign_bit - Assign value to a bit in memory
 * @nr: the bit to set
 * @addr: the address to start counting from
 * @value: the value to assign
 */
#define assign_bit(nr, addr, value)					\
	((value) ? set_bit((nr), (addr)) : clear_bit((nr), (addr)))

#define __assign_bit(nr, addr, value)					\
	((value) ? __set_bit((nr), (addr)) : __clear_bit((nr), (addr)))

fls_long

c 复制代码
/**
 * fls - 查找最后(最高有效)位集
 * @x:要搜索的词
 *
 * 此定义的定义方式与 ffs 相同。
 * 注意 fls(0) = 0, fls(1) = 1, fls(0x80000000) = 32。
 */
static __always_inline int fls(unsigned int x)
{
	return x ? sizeof(x) * 8 - __builtin_clz(x) : 0;
}


static inline unsigned int fls_long(unsigned long l)
{
	if (sizeof(l) == 4)
		return fls(l);
	return fls64(l);
}

BITS_TO_LONGS 位图的字数

  • BITS_TO_LONGS 是一个宏,用于计算给定位数所需的长整型数目。它使用了 __KERNEL_DIV_ROUND_UP 来确保结果向上取整,以便为所有位分配足够的空间。这个宏通常用于处理位图或位数组时,计算所需的存储空间。
  • 例如,如果你有一个位图需要存储 100 位数据,BITS_TO_LONGS(100) 将返回 2,因为在 64 位系统上,一个长整型可以存储 64 位,因此需要两个长整型来存储 100 位数据。
c 复制代码
#define BITS_PER_BYTE		8

#define BITS_PER_TYPE(type)	(sizeof(type) * BITS_PER_BYTE)
#define BITS_TO_LONGS(nr)	__KERNEL_DIV_ROUND_UP(nr, BITS_PER_TYPE(long))
#define BITS_TO_U64(nr)		__KERNEL_DIV_ROUND_UP(nr, BITS_PER_TYPE(u64))
#define BITS_TO_U32(nr)		__KERNEL_DIV_ROUND_UP(nr, BITS_PER_TYPE(u32))
#define BITS_TO_BYTES(nr)	__KERNEL_DIV_ROUND_UP(nr, BITS_PER_TYPE(char))
相关推荐
Coder个人博客11 小时前
Linux6.19-ARM64 mm mmu子模块深入分析
大数据·linux·车载系统·系统架构·系统安全·鸿蒙系统
羊群智妍14 小时前
2026 AI搜索流量密码:免费GEO监测工具,优化效果看得见
笔记·百度·微信·facebook·新浪微博
Doro再努力14 小时前
Vim 快速上手实操手册:从入门到生产环境实战
linux·编辑器·vim
wypywyp14 小时前
8. ubuntu 虚拟机 linux 服务器 TCP/IP 概念辨析
linux·服务器·ubuntu
阿蒙Amon14 小时前
TypeScript学习-第10章:模块与命名空间
学习·ubuntu·typescript
AI绘画哇哒哒14 小时前
【干货收藏】深度解析AI Agent框架:设计原理+主流选型+项目实操,一站式学习指南
人工智能·学习·ai·程序员·大模型·产品经理·转行
Doro再努力14 小时前
【Linux操作系统10】Makefile深度解析:从依赖推导到有效编译
android·linux·运维·服务器·编辑器·vim
senijusene14 小时前
Linux软件编程:IO编程,标准IO(1)
linux·运维·服务器
忧郁的橙子.15 小时前
02-本地部署Ollama、Python
linux·运维·服务器
醇氧15 小时前
【linux】查看发行版信息
linux·运维·服务器