English 中文(简体)
Pointer alignment in libjpeg
原标题:

From jmorecfg.h:

#define PACK_TWO_PIXELS(l,r)   ((r<<16) | l)
#define PACK_NEED_ALIGNMENT(ptr) (((int)(ptr))&3)
#define WRITE_TWO_PIXELS(addr, pixels) do {     
         ((INT16*)(addr))[0] = (pixels);        
         ((INT16*)(addr))[1] = (pixels)>>16;    
    } while(0)
#define WRITE_TWO_ALIGNED_PIXELS(addr, pixels)  ((*(INT32*)(addr)) = pixels

Can someone explain the difference between WRITE_TWO_PIXELS and WRITE_TWO_ALIGNED_PIXELS? If pixels is a stack allocated uint32_t and addr & 3 == 0, shouldn t they be equivalent?

Thanks.

最佳回答

WRITE_TWO_PIXELS and WRITE_TWO_ALIGNED_PIXELS are equivalent for little endian machines but not for big endian architecture.

[Example edited: thanks to Steve Jessop]

Let, pixels = 0x0A0B0C0D

For big endian machines, WRITE_TWO_PIXELS work as follows:

---------------------
| 0B | 0A | 0D | 0C |
---------------------
  3    2    1    0          <--- addr

where WRITE_TWO_ALIGNED_PIXELS will write as follows:

---------------------
| 0D | 0C | 0B | 0A |
---------------------
  3    2    1    0          <--- addr
问题回答

In the two macros, the only alignment that matters is the alignment of addr. As written in the question, they are equivalent if addr is 32-bit aligned (meaning its low two bits are zero), but only if the target architecture is also little-endian.

On a big-endian machine, the upper 16 bits of pixels must be written to (INT16*)(addr))[0] and the lower 16 bits to (INT16*)(addr))[1] for them to be equivalent.

Without checking my copy of the libjpeg source code, I d guess that these definitions are either expected to be modified as part of porting the library, or they are already guarded by a declaration of endianness.

If addr is not 32-bit aligned, then the WRITE_TWO_ALIGNED_PIXELS macro might cause an exception to be thrown on architectures where unaligned access is not permitted. Of course in some cases, unaligned access is permitted, but is much more expensive than two smaller aligned accesses, and on some other architectures, unaligned access is difficult to distinguish from aligned access.

The two macros exist as a reminder to the author of the library to think about alignment, and to standardize the approach to handling misaligned access so that it can be optimized out when building for platforms where it doesn t matter.





相关问题
Undefined reference

I m getting this linker error. I know a way around it, but it s bugging me because another part of the project s linking fine and it s designed almost identically. First, I have namespace LCD. Then I ...

C++ Equivalent of Tidy

Is there an equivalent to tidy for HTML code for C++? I have searched on the internet, but I find nothing but C++ wrappers for tidy, etc... I think the keyword tidy is what has me hung up. I am ...

Template Classes in C++ ... a required skill set?

I m new to C++ and am wondering how much time I should invest in learning how to implement template classes. Are they widely used in industry, or is this something I should move through quickly?

Print possible strings created from a Number

Given a 10 digit Telephone Number, we have to print all possible strings created from that. The mapping of the numbers is the one as exactly on a phone s keypad. i.e. for 1,0-> No Letter for 2->...

typedef ing STL wstring

Why is it when i do the following i get errors when relating to with wchar_t? namespace Foo { typedef std::wstring String; } Now i declare all my strings as Foo::String through out the program, ...

C# Marshal / Pinvoke CBitmap?

I cannot figure out how to marshal a C++ CBitmap to a C# Bitmap or Image class. My import looks like this: [DllImport(@"test.dll", CharSet = CharSet.Unicode)] public static extern IntPtr ...

Window iconification status via Xlib

Is it possible to check with the means of pure X11/Xlib only whether the given window is iconified/minimized, and, if it is, how?

热门标签