Sunday, August 2, 2009

In C, without copying out individual bytes, what's the best way to extract a 16 bit value from a byte array?

example:





unsigned char foo[5] = {0x01, 0x02, 0x03, 0x04, 0x05};





short bar;





//bar should be the 2 byte value spanning foo[1] and foo[2];





bar = (short)(foo[1] ) + (short)(foo[2] * 256); //this is correct.





I can do it this way but i was wondering if there was a casting mechanism to accomplish this without explicity addressing the two separate elements.





//Something like this...


//will only copy the 8 bit value in foo[1].





bar = (short)(*(foo + 1));








suggestions?

In C, without copying out individual bytes, what's the best way to extract a 16 bit value from a byte array?
The best way is the way you did it, accessing the two values separately ((short)(foo[1] ) + (short)(foo[2] * 256)), or you can use "%26lt;%26lt;8" instead of "*256" but the optimizer will probably produce the same code in either case. It actually will run quite efficiently. And it's the safest thing to do.





In "(short)(*(foo + 1))", the expression in parentheses beside the typecast, "*(foo + 1)", references the second character only, because foo is a char *. Casting that to a short AFTER dereferencing it doesn't cause the additional byte to get picked up.





You'd want "*((short *)(foo + 1))", I think. Add 1 to foo while it is still a char *, so that it adds only 1 byte offset. Then cast to a short * to point to 2 bytes, and then dereference.





That being said...





My recollection is that this code will not work, because foo[1..2] won't be aligned on an even byte boundary. While the C language syntax technically allows it, I wouldn't be surprised if it didn't execute. I think it will only work if you use an even-numbered offset in the character string.





For example, Microsoft Developer Network says:


"Access to data at unaligned addresses can affect correct program operation as well as processing efficiency. Some CPU architectures, such as the Itanium, cannot properly reference numeric data unless it is properly aligned in memory. When running a program on the Itanium, access to data which is not properly aligned will generate an EXCEPTION_ DATATYPE_ MISALIGNMENT exception which, if not handled, will cause the application to hang or crash."


http://msdn2.microsoft.com/en-us/library...
Reply:Arrays are contiguous memory, so if you want to bytes that are side by side, just make the short bar address = to address of the first byte from the array that you want.





If they are not contiguous, you must do by individual bytes, which is the most robust way to do it to begin with. It is fast too.
Reply:short bar = ((short)(foo[1] %26lt;%26lt; 256)) | (short)(foo[0]);





with shift and or :)

garden sheds

No comments:

Post a Comment