2014-05-13 10:09:08 zwmyxfbenet 阅读数 1075


#define __ALIGN_MASK(x,mask)	(((x)+(mask))&~(mask))
#define ALIGN(x,a)		__ALIGN_MASK(x,(typeof(x))(a)-1)

写简单点,宏ALIGN实际上是这样定义的:
#define ALIGN(x, a)   (x + (a - 1)) & ~(a - 1)
并且在计算的过程中将a强制转换成x的类型
该宏的作用:将x按a的值来对齐
比如
ALIGN(9,4)=12
ALIGN(10,4)=12
ALIGN(11,4)=12
ALIGN(12,4)=12
ALIGN(13,4)=16
ALIGN(14,4)=16
ALIGN(15,4)=16
ALIGN(16,4)=16

计算ALIGN宏的结果可以用下面这个简单的公式来计算:

if (x % a == 0)     值: x

else      值: ((x / a)  + 1) * a


#define PAGE_SHIFT		12
#define PAGE_SIZE		(_AC(1,UL) << PAGE_SHIFT)
#define PAGE_ALIGN(addr) ALIGN(addr, PAGE_SIZE)
该宏实际上等于
#define PAGE_ALIGN(addr) ALIGN(addr, 4096)
比如:
ALIGN(9,4096)=4096
ALIGN(10,4096)=4096
ALIGN(11,4096)=4096
ALIGN(12,4096)=4096
ALIGN(13,4096)=4096
ALIGN(14,4096)=4096
ALIGN(15,4096)=4096
ALIGN(16,4096)=4096
ALIGN(5000,4096)=8192
ALIGN(5001,4096)=8192
ALIGN(5002,4096)=8192
ALIGN(5003,4096)=8192


2019-09-18 02:43:49 u012088909 阅读数 22

今天遇到了一个BUG,找了半天才定位到是 PAGE_ALIGN 宏导致的。

这个宏在 Windows 上和 Linux 上的定义不同,才得以引发了这次BUG的发生。

 

PAGE_ALIGN 的用处是对齐一个页面地址。

先来看看定义:

// Windows
#define PAGE_ALIGN(Va)      ((PVOID)((ULONG_PTR)(Va) & ~(PAGE_SIZE - 1)))

// Linux
#define PAGE_MASK           (~(PAGE_SIZE - 1))
#define PAGE_ALIGN(addr)    (((addr) + PAGE_SIZE - 1) & PAGE_MASK)

写个例子测试:

#define PAGE_ALIGN(Va)      ((PVOID)((ULONG_PTR)(Va) & ~(PAGE_SIZE - 1)))
	printf("Windows: 0x%llx\n", PAGE_ALIGN(0x1234));
#undef PAGE_ALIGN

#define PAGE_MASK           (~(PAGE_SIZE - 1))
#define PAGE_ALIGN(addr)    (((addr) + PAGE_SIZE - 1) & PAGE_MASK)
	printf("Linux  : 0x%llx\n", PAGE_ALIGN(0x1234));
#undef PAGE_ALIGN

得到结果:

Windows: 0x1000
Linux  : 0x2000

解决方法:

1、在 Windows 下,需要对齐的值在微软的 API 中是会被自动向高对齐的。所以一般不需要手动对齐。

2、如果实在要手动向高对齐,直接复制一下 Linux 中的 PAGE_ALIGN 定义,然后 Rename 一下就行了。

2017-03-26 23:17:12 cuijian12921 阅读数 1167

#define PAGE_ALIGN(addr) ALIGN(addr, PAGE_SIZE)  (include/linux/mm.h)

|->#define ALIGN(x,a)__ALIGN_MASK(x,(typeof(x))(a)-1)(include/linux/kernel.h)

|->#define __ALIGN_MASK(x,mask)(((x)+(mask))&~(mask))


PAGE_ALIGN宏表示将物理地址addr修整为页边界地址(页的上边界)

page.h

/* PAGE_SHIFT determines the page size */
#define PAGE_SHIFT 12
#define PAGE_SIZE (1UL << PAGE_SHIFT)
#define PAGE_MASK (~(PAGE_SIZE-1))


 PAGE_ALIGN(addr) ==> ALIGN(addr, 1<<12)

==>__ALIGN_MASK(addr,(typeof(addr))(1<<12)-1)

==>(((addr)+((1<<12)-1))&~((1<<12)-1))

==>(((addr)+(0xffff)&~(0xffff)

==>(((addr)+(0xffff)&0xffff0000)


也就是以PAGE_SIZE页对其,向上取整

2019-02-18 14:16:40 clam_zxf 阅读数 193

在Linux内核中,经常会遇到类似于如下的代码

#define __DATA_ALIGN(end, align) (((unsigned int)end+align-1)&(~(align-1)))
#define __MSG_ALIGNED(end)       __DATA_ALIGN(end, 0x10)

光看字面意思,可以知道是字节align大小对齐的作用。原理上是怎么一回事,转载了一篇文章:https://blog.csdn.net/supperwangli/article/details/5142956

1. 引子 
int a;   
int size = 8;
    
如果让a为(size =8)的整数倍表示成二进制应是什么样子呢?那就是让这个数表示成二进制时的最后三位为0.
而要达到这一目标,只要下面这个数与它进行与运算就可以了:
  11111111 11111111 11111111 11111000   
而上面这个数实际下就是 ~ (size - 1),可以将该数称为size的对齐掩码size_mask.   
    
可这样做求出的是比a小的那个最大的8的倍数. 如果要求出比a大的是不是需要加上8就可以了? 
可是如果a本身就是8的倍数, 这样加8不就错了吗, 所以在a基础上加上 size - 1, 然后与size的对齐掩码进行与运算. 

这样, 我们可以定义下面的宏, 用于计算一个数a以size为倍数的前后两个值:
#define alignment_down(a, size) (a & (~(size-1)) )
#define alignment_up(a, size) ((a+size-1) & (~ (size-1))) 
  
例如: 
a=0, size=8,  则alignment_down(a,size)=0, alignment_up(a,size)=0.
a=6, size=8,  则alignment_down(a,size)=0, alignment_up(a,size)=8.
a=8, size=8,  则alignment_down(a,size)=8, alignment_up(a,size)=8.
a=14, size=8, 则alignment_down(a,size)=8, alignment_up(a,size)=16. 
  
注意:size应当为2的n次方, 即2, 4, 8, 16, 32, 64, 128, 256, 1024, 2048, 4096..... 
  
2. 在linux中的应用
上面的计算方法在linux等代码中也常常可以看到,下面给出几个例子. 
  
例1:当分配地址addr时, 要将该地址以size为倍数对齐, 而且要得到是比addr大的值, 则使用_ALIGN宏:
/* align addr on a size boundary - adjust address up if needed -- Cort */
#define _ALIGN(addr,size)   (((addr)+(size)-1)&(~((size)-1))) 

例2:与页面对齐相关的宏
#define PAGE_SIZE               4096
#define PAGE_MASK               (~(PAGE_SIZE-1)) 
/* to align the pointer to the (next) page boundary */
#define PAGE_ALIGN(addr)        (((addr)+PAGE_SIZE-1) & PAGE_MASK) 
  
例3:与skb分配时对齐相关的宏
#define SKB_DATA_ALIGN(X)   (((X) + (SMP_CACHE_BYTES - 1)) & ~(SMP_CACHE_BYTES - 1)) 
  
struct sk_buff *__alloc_skb(unsigned int size, gfp_t gfp_mask, int fclone)
{
    ......
 
    /* Get the HEAD */
    skb = kmem_cache_alloc(cache, gfp_mask & ~__GFP_DMA);
    if (!skb)
         goto out; 
  
    /* Get the DATA. Size must match skb_add_mtu(). */
    size = SKB_DATA_ALIGN(size);
    data = kmalloc_track_caller(size + sizeof(struct skb_shared_info),  gfp_mask);
    if (!data)
         goto nodata; 
    ......
}

2010-01-06 14:58:00 supperwangli 阅读数 4208

1. 引子
int a;  
int size = 8;
   
如果让a为(size =8)的整数倍表示成二进制应是什么样子呢?那就是让这个数表示成二进制时的最后三位为0.
而要达到这一目标,只要下面这个数与它进行与运算就可以了:
  11111111 11111111 11111111 11111000  
而上面这个数实际下就是 ~ (size - 1),可以将该数称为size的对齐掩码size_mask.  
   
可这样做求出的是比a小的那个最大的8的倍数. 如果要求出比a大的是不是需要加上8就可以了?
可是如果a本身就是8的倍数, 这样加8不就错了吗, 所以在a基础上加上 size - 1, 然后与size的对齐掩码进行与运算.

这样, 我们可以定义下面的宏, 用于计算一个数a以size为倍数的前后两个值:
#define alignment_down(a, size) (a & (~(size-1)) )
#define alignment_up(a, size) ((a+size-1) & (~ (size-1)))
 
例如:
a=0, size=8,  则alignment_down(a,size)=0, alignment_up(a,size)=0.
a=6, size=8,  则alignment_down(a,size)=0, alignment_up(a,size)=8.
a=8, size=8,  则alignment_down(a,size)=8, alignment_up(a,size)=8.
a=14, size=8, 则alignment_down(a,size)=8, alignment_up(a,size)=16.
 
注意:size应当为2的n次方, 即2, 4, 8, 16, 32, 64, 128, 256, 1024, 2048, 4096.....
 
2. 在linux中的应用
上面的计算方法在linux等代码中也常常可以看到,下面给出几个例子.
 
例1:当分配地址addr时, 要将该地址以size为倍数对齐, 而且要得到是比addr大的值, 则使用_ALIGN宏:
/* align addr on a size boundary - adjust address up if needed -- Cort */
#define _ALIGN(addr,size)   (((addr)+(size)-1)&(~((size)-1)))

例2:与页面对齐相关的宏
#define PAGE_SIZE               4096
#define PAGE_MASK               (~(PAGE_SIZE-1))
/* to align the pointer to the (next) page boundary */
#define PAGE_ALIGN(addr)        (((addr)+PAGE_SIZE-1) & PAGE_MASK)
 
例3:与skb分配时对齐相关的宏
#define SKB_DATA_ALIGN(X)   (((X) + (SMP_CACHE_BYTES - 1)) & ~(SMP_CACHE_BYTES - 1))
 
struct sk_buff *__alloc_skb(unsigned int size, gfp_t gfp_mask, int fclone)
{
    ......
 
    /* Get the HEAD */
    skb = kmem_cache_alloc(cache, gfp_mask & ~__GFP_DMA);
    if (!skb)
         goto out;
 
    /* Get the DATA. Size must match skb_add_mtu(). */
    size = SKB_DATA_ALIGN(size);
    data = kmalloc_track_caller(size + sizeof(struct skb_shared_info),  gfp_mask);
    if (!data)
         goto nodata;
    ......
}

linux sem信号量使用

阅读数 19499

Linux安装分区推荐

阅读数 288

没有更多推荐了,返回首页