Forum: Mikrocontroller und Digitale Elektronik 24 Bit in 2er Komplement wandeln (Arduino Uno)


von Absence (Gast)


Lesenswert?

Moin,

Ich nutze einen Arduino Uno und zwei verschiedene ADC einen mit einer 16 
Bit Auflösung und einen mit einer 24 Bit Auflösung, beide geben ihre 
Werte im 2er Komplement aus.
Der Typ "int" beim Uno ist mit 16 Bit definiert, also wandel ich meine 
2x8 Bit wie folgt in einen Zahlenwert um:
int Value = int(Byte1 << 8) + int(Byte2);

Beim 24 Bit wird es schwieriger... ich wollte den Typ "long" verwenden, 
da der mit 32 Bit definiert ist und meine 24 Bit wie folgt in einen Wert 
wandeln:
long Value = long(Byte1 << 16) + int(Byte2 <<8) + int(Byte3);

Ist das der richtige Weg oder gibt es vielleicht einen clevereren Weg?

Vielen Dank!

von Jim M. (turboj)


Lesenswert?

Vorsicht ist bei Shift Operationen geboten, wenn die über den int 
Bereich rausgehen.

Beim Uno ist der nur 15 Bit groß - das oberste Bit im int ist das 
Vorzeichen Bit und da rein shiften ist in C undefiniertes Verhalten. Man 
darf das nur bei vorzeichenlosen Typen machen.

Abhilfe:
1
#include <stdint.h>
2
3
4
uint8_t Byte1, Byte2, Byte3;
5
6
void foo() {
7
8
int16_t Value = int16_t(uint16_t(Byte1) << 8 + Byte2);
9
10
int32_t Value24 = int32_t((uint32_t(Byte1) <<16)+ (uint16_t(Byte2) << 8) + Byte3 + (Byte1 & 0x80 ? 0xFF000000UL:0));
11
12
}

Letzteres Beispiel korrigiert zusätzlich das Vorzeichen, falls der 24 
Bit Wert negativ ist.

Ich bin ein Fan von den expliziten Typen aus stdint.h, auch weil z.B. 
Arduinos mit ARM Architektur 32 Bit int haben.

von Hopperla (Gast)


Lesenswert?

Absence schrieb:
> int(Byte2 <<8)

Wenn du ein Byte um 8 Bit verschiebst kommt immer Null raus.

Wenn schon dann

( ( (uint16_t) Byte2) << 8)

Merkst du den Unterschied?

von Falk B. (falk)


Lesenswert?

Absence schrieb:

> Der Typ "int" beim Uno ist mit 16 Bit definiert, also wandel ich meine
> 2x8 Bit wie folgt in einen Zahlenwert um:
> int Value = int(Byte1 << 8) + int(Byte2);

Ist OK.

> Beim 24 Bit wird es schwieriger... ich wollte den Typ "long" verwenden,
> da der mit 32 Bit definiert ist und meine 24 Bit wie folgt in einen Wert
> wandeln:
> long Value = long(Byte1 << 16) + int(Byte2 <<8) + int(Byte3);

Nicht so ganz OK. Denn je nach der Definition von Byte1 klappt das oder 
auch nicht. Und wenn schon, dann richtig. Erst der Cast, dann schieben.
1
long Value = ((Long)Byte1 << 16) + ((long)Byte2 <<8) + (long)Byte3;
2
if (Byte1 & 0x80) {   // Vorzeichenerweiterung
3
   Value |= 0xFF000000; 
4
}

von Chris (Gast)


Lesenswert?

Und arduino (avr) kennt den int24, __uint24 bzw __int24.
Weiters gibt es auch cast, wobei dann der LSB zuerst gespeichert wird.

von Jim M. (turboj)


Lesenswert?

Hopperla schrieb:
> Wenn du ein Byte um 8 Bit verschiebst kommt immer Null raus.

Nö. Der shift Operator arbeitet mindestens mit int, beim Uno also 
16-bittig.

Wobei man mit dem Vorzeichen Bit dann eventuell aufpassen muss.

von Absence (Gast)


Lesenswert?

Danke für die Aufklärung und die Lösung!
Jetzt klappt es perfekt! :)

von Hopperla (Gast)


Lesenswert?

Jim M. schrieb:
> Nö. Der shift Operator arbeitet mindestens mit int, beim Uno also
> 16-bittig.
1
#include <stdio.h>
2
3
int main()
4
{
5
    unsigned char aa;
6
    unsigned short int bb;
7
8
    aa = 1;
9
    printf("%c %u \n", aa, (unsigned short int)aa);
10
    aa = aa << 8;
11
    printf("%c %u \n", aa, (unsigned short int)aa);
12
13
    bb = 256;
14
    bb = bb + aa;
15
16
    printf("%u \n", bb);
17
18
    return 0;
19
}

von Jemand (Gast)


Lesenswert?

Hopperla schrieb:
> Jim M. schrieb:
> Nö. Der shift Operator arbeitet mindestens mit int, beim Uno also
> 16-bittig.
>
> #include <stdio.h>
>
> [...]

Das widerspricht dem so ungefähr überhaupt nicht.

von Markus F. (mfro)


Lesenswert?

Hopperla schrieb:
> Absence schrieb:
>> int(Byte2 <<8)
>
> Wenn du ein Byte um 8 Bit verschiebst kommt immer Null raus.
>
> Wenn schon dann
>
> ( ( (uint16_t) Byte2) << 8)
>
> Merkst du den Unterschied?

Stichwort: Integer Promotion (nachschlagen!). Der cast ist hier unnötig.

Bitte melde dich an um einen Beitrag zu schreiben. Anmeldung ist kostenlos und dauert nur eine Minute.
Bestehender Account
Schon ein Account bei Google/GoogleMail? Keine Anmeldung erforderlich!
Mit Google-Account einloggen
Noch kein Account? Hier anmelden.