Linux内核中位图的使用

本文探讨了Linux内核如何利用位图进行高效的数据标记和管理。通过了解位图的基本原理,分析其在内存分配、磁盘块管理等场景的应用,揭示位图在系统性能优化中的关键作用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

使用位图标记

 

#include <stdio.h>


#ifdef CONFIG_64BIT
#define BITS_PER_LONG 64
#else
#define BITS_PER_LONG 32
#endif /* CONFIG_64BIT */

/*
计算在第几个unsigned long数组中的掩码
*/
#define BIT_MASK(nr)		(1UL << ((nr) % BITS_PER_LONG))
/*
第@nr个位在unsigned long数组中的下标
*/
#define BIT_WORD(nr)		((nr) / BITS_PER_LONG)
#define BITS_PER_BYTE		8

#define DIV_ROUND_UP(n,d) (((n) + (d) - 1) / (d))

/*
表示@nr个位需要几个unsigned long数
*/
#define BITS_TO_LONGS(nr)	DIV_ROUND_UP(nr, BITS_PER_BYTE * sizeof(long))

/*
定义位图@name[]
使用unsigned long数组表示@bits个位
*/
#define DECLARE_BITMAP(name,bits) \
	unsigned long name[BITS_TO_LONGS(bits)]


/*
选择通用函数为例
include/asm-generic/bitops/non-atomic.h
*/

static inline void __set_bit(int nr, volatile unsigned long *addr)
{
	unsigned long mask = BIT_MASK(nr);
	unsigned long *p = ((unsigned long *)addr) + BIT_WORD(nr);
	
	*p  |= mask;
}

static inline void __clear_bit(int nr, volatile unsigned long *addr)
{
	unsigned long mask = BIT_MASK(nr);
	unsigned long *p = ((unsigned long *)addr) + BIT_WORD(nr);

	*p &= ~mask;
}

static inline void __change_bit(int nr, volatile unsigned long *addr)
{
	unsigned long mask = BIT_MASK(nr);
	unsigned long *p = ((unsigned long *)addr) + BIT_WORD(nr);

	*p ^= mask;
}

/*
test_and_XX_bit返回之前@nr位的置位情况

1 - 调用函数前已经置位
0 - 调用函数前没有置位
*/

static inline int __test_and_set_bit(int nr, volatile unsigned long *addr)
{
	unsigned long mask = BIT_MASK(nr);
	unsigned long *p = ((unsigned long *)addr) + BIT_WORD(nr);
	unsigned long old;

	old = *p;
	*p = old | mask;

	return (old & mask) != 0;
}

static inline int __test_and_clear_bit(int nr, volatile unsigned long *addr)
{
	unsigned long mask = BIT_MASK(nr);
	unsigned long *p = ((unsigned long *)addr) + BIT_WORD(nr);
	unsigned long old;

	old = *p;
	*p = old & ~mask;

	return (old & mask) != 0;
}

static inline int __test_and_change_bit(int nr, volatile unsigned long *addr)
{
	unsigned long mask = BIT_MASK(nr);
	unsigned long *p = ((unsigned long *)addr) + BIT_WORD(nr);
	unsigned long old;

	old = *p;
	*p = old ^ mask;

	return (old & mask) != 0;
}

#define BITS_NR		50
DECLARE_BITMAP(testbit, BITS_NR);

#define ARRAY_SIZE(__arr)	(sizeof(__arr) / sizeof(__arr[0]))

void
testbit_dump(void)
{
	int size = ARRAY_SIZE(testbit);
	printf("array size: %d\n", size);

	int i;
	for (i = 0; i < size; i++)
	{
		printf("[%d]: 0x%08lx\n", i, testbit[i]);
	}
	printf("\n");
}

int
main(void)
{
	int b;

	testbit_dump();

	printf("__set_bit 0\n");
	__set_bit(0, testbit);
	testbit_dump();

	printf("__test_and_clear_bit 0\n");
	b = __test_and_clear_bit(0, testbit);
	printf("b: %d\n", b);
	testbit_dump();

	printf("__set_bit 19\n");
	__set_bit(19, testbit);
	testbit_dump();

	printf("__clear_bit 0\n");
	__clear_bit(0, testbit);
	testbit_dump();

	printf("__test_and_change_bit 33\n");
	b = __test_and_change_bit(33, testbit);
	printf("b: %d\n", b);
	testbit_dump();

	printf("__test_and_clear_bit 33\n");
	b = __test_and_clear_bit(33, testbit);
	printf("b: %d\n", b);
	testbit_dump();

	return 0;
}





 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值