[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))
相关推荐
Xの哲學2 小时前
Linux epoll 深度剖析: 从设计哲学到底层实现
linux·服务器·网络·算法·边缘计算
求梦8202 小时前
JVM学习
jvm·学习
iYun在学C2 小时前
驱动程序(注册字符设备)
linux·嵌入式硬件
九成宫2 小时前
计算机网络期末复习——第4章:网络层 Part One
网络·笔记·计算机网络·软件工程
星火开发设计2 小时前
C++ multiset 全面解析与实战指南
开发语言·数据结构·c++·学习·set·知识
shuangrenlong2 小时前
音乐app笔记
笔记
am心2 小时前
学习笔记-菜品接口-菜品分页查询
笔记·学习
延延oO2 小时前
zyzyzyzyzy
linux