Reputation: 103
I'm having a really strange inconsistancy.
I'm preparing for reading from binary files on Arduino (for a midi player, in case you were interested).
If I try to combine 4 bytes on Arduino to a long, it gives me a wrong result.
However, if I use the equivalent code on PC, I get the correct value.
Input is: 0x12481248 (0x12, 0x48, 0x12, 0x48) (really a random number).
Arduino gives: 4680.
Code::Blocks gives: 306713160.
4680 is the same as 0x1248, and the result you get when you use an int instead of long on Arduino (2 bytes omitted).
Arduino code:
void setup(){
Serial.begin(57600);
char read1 = 0x12;
char read2 = 0x48;
char read3 = 0x12;
char read4 = 0x48;
unsigned long testint = read1<<24|read2<<16|read3<<8|read4;
unsigned long testint2 = 306713160;
Serial.println(testint);
Serial.println(testint2);
}
void loop(){}
testint2 is to show that it isn't caused by Serial.println(). Serial Monitor output is indeed:
4680
306713160
C++ code:
#include <iostream>
using namespace std;
int main(){
char read1 = 0x12;
char read2 = 0x48;
char read3 = 0x12;
char read4 = 0x48;
unsigned long testint = read1<<24|read2<<16|read3<<8|read4;
cout << testint;
}
Any idea what's going on?
Also, does anyone know a better/prettier way of converting bytes with Arduino/SD library?
Upvotes: 7
Views: 4888
Reputation: 1
Here is another way to create 4 bytes to Long value and the reverse. Includes a simple test for these functions.
byte dataj[4]; //4 byte array output
unsigned long testint;
unsigned long testnum, temp;
unsigned long prev_time, cur_time = 0;
void setup() {
Serial.begin(115200);
Serial.println(F("********** Initalized +++++++"));
// -----4294967295 Max value
testnum = 4294967295L; //Start at maximum value
for (int i = 0; i < 999; i++)
{
Serial.print(F("Test #"));
Serial.print( i + 1);
Serial.print(F(" Initial value = "));
Serial.println(testnum);
prev_time = micros();
Long_2_Four_Bytes(testnum);
testint = Four_Bytes_2_Long(dataj[3], dataj[2], dataj[1], dataj[0]);
cur_time = micros();
Serial.print(F(" final value = "));
Serial.print(testint);
Serial.print(" time (microsec) = "); Serial.print(prev_time - cur_time);
if (testint == testnum)
Serial.println();
else
Serial.println(F(" ERROR <-----"));
testnum--;
delay(50);
}
}
void loop() {}
//---- Long_2_Four_Bytes ---------------------------
// Fills an array 'dataj[]' with 4 bytes representing
// the 'unsigned long val'.
// Input : unsigned long number - range 0 - 4294967295
// Output : dataj[] where dataj[0] = LSB & dataj[3] = MSB
//---------------------------------------------------
void Long_2_Four_Bytes(unsigned long val )
{
unsigned long temp;
byte data1[] = { 0, 0, 0, 0}; //4 byte array
temp = val;
for (byte bytegroup = 0; bytegroup < 4; bytegroup ++)
{
int offset = bytegroup * 8;
for (byte bit_pntr = 0; bit_pntr < 8; bit_pntr++)
{
if ( bitRead(temp, (7 - bit_pntr) + offset) == 1)
bitSet(data1[bytegroup], (7 - bit_pntr));
}
}
for (byte i = 0; i < 4; i++)
dataj[i] = data1[i];
}
//-- Four_Bytes_2_Long -------------------------------------
// Creates an unsigned long from 4 bytes where A = MSB and
// D = LSB. Based upon code by Mehrdad Nazmdar.
// Called by :
// Calls : N/A
// Returns : unsigned long number - range 0 - 4294967295
//------------------------------------------------------------
unsigned long Four_Bytes_2_Long(byte A, byte B, byte C, byte D)
{
uint8_t data[4];
unsigned long testBigint;
data[0] = D; //LSB
data[1] = C;
data[2] = B;
data[3] = A; //MSB
testBigint = *(unsigned long*)(&data);
return testBigint;
}
Upvotes: 0
Reputation: 192
You can also use the following code:
uint8_t data[4];
data[0]=read4;
data[1]=read3;
data[2]=read2;
data[3]=read1;
unsigned long testint =*(unsigned long*)(&data);
Upvotes: 0
Reputation: 4230
On Arduino, int
size is 16 bits.
In this line:
unsigned long testint = read1<<24|read2<<16|read3<<8|read4;
even if the result is stored in a unsigned long
(32 bits), the bitwise operations are done on int
s.
Change this line to:
unsigned long testint = (unsigned long)read1 << 24
| (unsigned long)read2 << 16
| (unsigned long)read3 << 8
| (unsigned long)read4;
Upvotes: 11
Reputation: 5230
I would expect the result 4680 (=0x1248) on any platform where sizeof(int)=2, and I think this is the case for arduino.
That's because (read1 << 24) gets implicitly converted to int (not long), so the upper two bytes get lost. Yout should convert read* to unsigned long first
Upvotes: 4