宏BITS_TO_LONGS (数组位图操作) 代码测试

本文介绍了一种使用C语言实现的位操作方法,包括设置、清除及检测特定位的方法,并提供了将整数转换为二进制和十六进制字符串的函数。通过具体的代码示例展示了如何操作位以及进行数值转换。

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

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#define BITS_PER_LONG 32
#define BITS_PER_BYTE       8  
#define BITS_TO_LONGS(nr)       DIV_ROUND_UP(nr, BITS_PER_BYTE * sizeof(long))//nr个位需要多少个long型数据变量来存储
#define DIV_ROUND_UP(n,d) (((n) + (d) - 1) / (d)) 
转载请注明原文地址:http://blog.youkuaiyun.com/oyhb_1992/article/details/77680363
static inline void set_bit(int nr, unsigned long *addr)  
{  
	addr[nr / BITS_PER_LONG] |= 1UL << (nr % BITS_PER_LONG);  
}  

static inline void clear_bit(int nr, unsigned long *addr)  
{  
	addr[nr / BITS_PER_LONG] &= ~(1UL << (nr % BITS_PER_LONG)); 
}  

static __always_inline int test_bit(unsigned int nr, const unsigned long *addr)  
{  
	return ((1UL << (nr % BITS_PER_LONG))&(((unsigned long *)addr)[nr / BITS_PER_LONG])) != 0;    
}  

// 十进制转换为二进制,十进制数的每1bit转换为二进制的1位数字
char *int_to_bin(unsigned long data)
{
	unsigned int i = 0;
	int bit_num = sizeof(unsigned long) * 8;
	char *p_bin = (char *)malloc(sizeof(char)*(bit_num+1));
	//char *p_bin = new char[bit_num+1];
	p_bin[bit_num] = '\0';
	for (i = 0; i < bit_num; ++i)
	{
		//p_bin[i] = data << i >> (bit_num-1);//获取第i位
		//p_bin[i] = data>> (bit_num-1);
		p_bin[i] = data&(1<<i);//等价于上面两条语句
		if (p_bin[i] == 0)
			p_bin[i] = '0';
		else if (p_bin[i] != 0)
			p_bin[i] = '1';
	}
	return p_bin;
}

// 十进制转换为十六进制,十进制数的每4bit转换为十六进制的1位数字
char *int_to_hex(unsigned long data)
{
	unsigned int i = 0;
	int bit_num = sizeof(unsigned long) * 8;
	char *p_hex = (char *)malloc(sizeof(unsigned long)*8/4+3);
	p_hex[0] = '0';
	p_hex[1] = 'x';
	p_hex[bit_num/4+2] = '\0';
	char *p_tmp = p_hex + 2;
	for (i = 0; i < bit_num/4; ++i)
	{
		p_tmp[i] = data << (4*i) >> (bit_num-4);
		if (p_tmp[i] >= 0 && p_tmp[i] <= 9)
			p_tmp[i] += '0';
		else if (p_tmp[i] >= 10 && p_tmp[i] <= 15)
			p_tmp[i] = p_tmp[i] - 10 + 'A';
	}
	return p_hex;
}

int main()
{
	#define EV_MAX			0x1f
	#define EV_CNT			(EV_MAX+1)   //32
	int i=0,j=0;
	unsigned long evbit[BITS_TO_LONGS(EV_CNT)];   //32个位共需要多少个long型数据来存储

	memset(evbit,0,sizeof(evbit));
	//等价于
	/*for(i=0;i<sizeof(evbit)/sizeof(long);i++)
		for(j=0;j<BITS_PER_LONG/BITS_PER_BYTE;j++)
		{
			char *p = (char *)(&evbit[i]);
			p[j] = 0;  
		}*/

	for(i=0;i<sizeof(evbit)/sizeof(unsigned long);i++)
	{
		char *p = int_to_bin(evbit[i]);
		printf("p[%d] = %s\n",i,p++);   
	}

	//第7位设为1
	set_bit(7,evbit);
	//测试第7位是否为1
	int test = test_bit(7,evbit);	
	printf("test_bit nr = 7 value = %d\n",test);
	for(i=0;i<sizeof(evbit)/sizeof(unsigned long);i++)
	{
		char *p = int_to_bin(evbit[i]);
		printf("p[%d] = %s\n",i,p++);   
	}
	return 0;
}



测试结果:

/* SPDX-License-Identifier: GPL-2.0 */ #ifndef _LINUX_TYPES_H #define _LINUX_TYPES_H #define __EXPORTED_HEADERS__ #include <uapi/linux/types.h> #ifndef __ASSEMBLY__ #define DECLARE_BITMAP(name,bits) \ unsigned long name[BITS_TO_LONGS(bits)] typedef u32 __kernel_dev_t; typedef __kernel_fd_set fd_set; typedef __kernel_dev_t dev_t; typedef __kernel_ulong_t ino_t; typedef __kernel_mode_t mode_t; typedef unsigned short umode_t; typedef u32 nlink_t; typedef __kernel_off_t off_t; typedef __kernel_pid_t pid_t; typedef __kernel_daddr_t daddr_t; typedef __kernel_key_t key_t; typedef __kernel_suseconds_t suseconds_t; typedef __kernel_timer_t timer_t; typedef __kernel_clockid_t clockid_t; typedef __kernel_mqd_t mqd_t; typedef _Bool bool; typedef __kernel_uid32_t uid_t; typedef __kernel_gid32_t gid_t; typedef __kernel_uid16_t uid16_t; typedef __kernel_gid16_t gid16_t; typedef unsigned long uintptr_t; #ifdef CONFIG_HAVE_UID16 /* This is defined by include/asm-{arch}/posix_types.h */ typedef __kernel_old_uid_t old_uid_t; typedef __kernel_old_gid_t old_gid_t; #endif /* CONFIG_UID16 */ #if defined(__GNUC__) typedef __kernel_loff_t loff_t; #endif /* * The following typedefs are also protected by individual ifdefs for * historical reasons: */ #ifndef _SIZE_T #define _SIZE_T typedef __kernel_size_t size_t; #endif #ifndef _SSIZE_T #define _SSIZE_T typedef __kernel_ssize_t ssize_t; #endif #ifndef _PTRDIFF_T #define _PTRDIFF_T typedef __kernel_ptrdiff_t ptrdiff_t; #endif #ifndef _CLOCK_T #define _CLOCK_T typedef __kernel_clock_t clock_t; #endif #ifndef _CADDR_T #define _CADDR_T typedef __kernel_caddr_t caddr_t; #endif /* bsd */ typedef unsigned char u_char; typedef unsigned short u_short; typedef unsigned int u_int; typedef unsigned long u_long; /* sysv */ typedef unsigned char unchar; typedef unsigned short ushort; typedef unsigned int uint; typedef unsigned long ulong; #ifndef __BIT_TYPES_DEFINED__ #define __BIT_TYPES_DEFINED__ typedef u8 u_int8_t; typedef s8 int8_t; typedef u16 u_int16_t; typedef s16 int16_t; typedef u32 u_int32_t; typedef s32 int32_t; #endif /* !(__BIT_TYPES_DEFINED__) */ typedef u8 uint8_t; typedef u16 uint16_t; typedef u32 uint32_t; #if defined(__GNUC__) typedef u64 uint64_t; typedef u64 u_int64_t; typedef s64 int64_t; #endif /* this is a special 64bit data type that is 8-byte aligned */ #define aligned_u64 __aligned_u64 #define aligned_be64 __aligned_be64 #define aligned_le64 __aligned_le64 /** * The type used for indexing onto a disc or disc partition. * * Linux always considers sectors to be 512 bytes long independently * of the devices real block size. * * blkcnt_t is the type of the inode's block count. */ typedef u64 sector_t; typedef u64 blkcnt_t; /* * The type of an index into the pagecache. */ #define pgoff_t unsigned long /* * A dma_addr_t can hold any valid DMA address, i.e., any address returned * by the DMA API. * * If the DMA API only uses 32-bit addresses, dma_addr_t need only be 32 * bits wide. Bus addresses, e.g., PCI BARs, may be wider than 32 bits, * but drivers do memory-mapped I/O to ioremapped kernel virtual addresses, * so they don't care about the size of the actual bus addresses. */ #ifdef CONFIG_ARCH_DMA_ADDR_T_64BIT typedef u64 dma_addr_t; #else typedef u32 dma_addr_t; #endif typedef unsigned int __bitwise gfp_t; typedef unsigned int __bitwise slab_flags_t; typedef unsigned int __bitwise fmode_t; #ifdef CONFIG_PHYS_ADDR_T_64BIT typedef u64 phys_addr_t; #else typedef u32 phys_addr_t; #endif typedef phys_addr_t resource_size_t; /* * This type is the placeholder for a hardware interrupt number. It has to be * big enough to enclose whatever representation is used by a given platform. */ typedef unsigned long irq_hw_number_t; typedef struct { int counter; } atomic_t; #define ATOMIC_INIT(i) { (i) } #ifdef CONFIG_64BIT typedef struct { s64 counter; } atomic64_t; #endif struct list_head { struct list_head *next, *prev; }; struct hlist_head { struct hlist_node *first; }; struct hlist_node { struct hlist_node *next, **pprev; }; struct ustat { __kernel_daddr_t f_tfree; #ifdef CONFIG_ARCH_32BIT_USTAT_F_TINODE unsigned int f_tinode; #else unsigned long f_tinode; #endif char f_fname[6]; char f_fpack[6]; }; /** * struct callback_head - callback structure for use with RCU and task_work * @next: next update requests in a list * @func: actual update function to call after the grace period. * * The struct is aligned to size of pointer. On most architectures it happens * naturally due ABI requirements, but some architectures (like CRIS) have * weird ABI and we need to ask it explicitly. * * The alignment is required to guarantee that bit 0 of @next will be * clear under normal conditions -- as long as we use call_rcu() or * call_srcu() to queue the callback. * * This guarantee is important for few reasons: * - future call_rcu_lazy() will make use of lower bits in the pointer; * - the structure shares storage space in struct page with @compound_head, * which encode PageTail() in bit 0. The guarantee is needed to avoid * false-positive PageTail(). */ struct callback_head { struct callback_head *next; void (*func)(struct callback_head *head); } __attribute__((aligned(sizeof(void *)))); #define rcu_head callback_head typedef void (*rcu_callback_t)(struct rcu_head *head); typedef void (*call_rcu_func_t)(struct rcu_head *head, rcu_callback_t func); typedef void (*swap_func_t)(void *a, void *b, int size); typedef int (*cmp_r_func_t)(const void *a, const void *b, const void *priv); typedef int (*cmp_func_t)(const void *a, const void *b); #endif /* __ASSEMBLY__ */ #endif /* _LINUX_TYPES_H */
最新发布
08-09
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值