Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How do you call this binary data type, and how to handle it in C#?

Tags:

c#

file

binary

byte

Let's say we have a binary file that contains 2 bytes that form an integer, in reverse.

So for example, the bytes show like this: (hexadecimal)

EB 03 00 00

Which should be interpreted as this:

00 00 03 EB

Which C# should be able to input as decimal 1003. Is this possible if you have the EB and 03 bytes already in memory in 2 different variables? Is there some math I can apply here to form decimal 1003 from the numbers 235 and 3? Or should I do it completely different?

Thanks in advance!

like image 452
Codecat Avatar asked Dec 01 '22 04:12

Codecat


2 Answers

What you talk about is called Endianness, in particular Little Endian format.

In C#, it's probably easiest to use a BinaryReader or BinaryWriter to read from binary files which wrap the correct conversion of the byte order.

The following sample shows how to use a BinaryReader to get the correct integer interpretation of a number in Little Endian format:

using System.IO;

class EndiannessSample
{
    static void Main(string[] args)
    {
        using (MemoryStream ms = new MemoryStream())
        {
            // write bytes in little-endian format
            ms.WriteByte(0xEB);
            ms.WriteByte(0x03);
            ms.WriteByte(0x00);
            ms.WriteByte(0x00);
            ms.Position = 0;

            using (BinaryReader reader = new BinaryReader(ms))
            {
                int i = reader.ReadInt32(); // decimal value of i is 1003
            }
        }
    }
}

Little endian is the standard on Intel (and Windows) platforms. In case you have to deal with data in Big Endian format (e.g. when importing files created on an old Macintosh) there is no direct support within .NET. You can write a simple utility function for converting endianness using the BitConverter class. In the sample above you could do the following to cope with Big Endian (on a Little Endian platform):

using (MemoryStream ms = new MemoryStream())
{
    // write bytes in big-endian format
    ms.WriteByte(0x00);
    ms.WriteByte(0x00);
    ms.WriteByte(0x03);
    ms.WriteByte(0xEB);
        
    ms.Position = 0;
    using (BinaryReader reader = new BinaryReader(ms))
    {
        byte[] temp = reader.ReadBytes(4);
        if (BitConverter.IsLittleEndian)
        {
            // reverse the byte order only if we are on a little-endian system,
            // because the BitConverter is aware of the endianness of the system
            //
            Array.Reverse(temp);
        }
        int i = BitConverter.ToInt32(temp, 0);
    }
}

LukeH provided a link that further discusses problems related to Endianness, e.g. when targeting Xbox 360 (which happens to be a Big Endian platform):

One Little, Two Little, Three Little Endian... 0x0A Big Endian Bugs

Update

The MiscUtil library provides a binary reader/writer class that can be configured for a specific Endianness:

MiscUtil.IO.EndianBinary{Writer/Reader}
like image 114
Dirk Vollmar Avatar answered Dec 03 '22 16:12

Dirk Vollmar


You should be able to just read the value from the binary file into an unsigned 32 bit integer:

UInt32 myValue;

using this method from your BinaryReader object:

myValue = reader.ReadUInt32();

On a Little Endian system (i.e. Intel), the byte order will reverse automatically.

To get your array of bytes:

byte[] b = BitConverter.GetBytes(myValue);
like image 41
Robert Harvey Avatar answered Dec 03 '22 16:12

Robert Harvey