I have variable uint16_t value
, I want to copy it to uint8_t buffer[3]
. Is it possible to do (Little endian):
*buffer=*(uint8_t *)&value;
Instead of:
buffer[0] = highByte(value);
buffer[1] = lowByte(value);
Since this replacement cause stm32f7 I2C to not working correctly. Is there any correct casting?
STM32 is little endian so you get the lowest significant byte first:
uint8_t* ptr = (uint8_t*)&value;
uint8_t low = ptr[0];
uint8_t high = ptr[1];
Doing casts and de-referencing like this is fine for character types only. The above code is assuming that uint8_t
is a character type, which is very likely the case (on gcc and other mainstream compilers).
For more info see What is CPU endianness?
EDIT:
If you simply wish to copy a 16 bit number into an array of bytes, the correct solution is this:
memcpy(buffer, &value, sizeof(uint16_t)).
We cannot do *(uint16_t*) buffer=value;
because it invokes undefined behavior. buffer
could be misaligned and it's also a strict aliasing violation. And this is why I wrote with emphasis above "this is fine for character types only".
Is there any correct casting?
No
*buffer = <expression>
will always and only write to buffer[0]
and never to buffer[1]
.
One thing you could do - but I strongly urge you NOT to is stuff like this:
uint16_t *ptr = (uint16_t*)buffer; // Ugh, danger
*ptr = value;
This will be problematic if you run it on machines with different endianess. And it's quite possible that there are other reasons to not do this too. As soon as you ask yourself if you can solve something via casting, you should really take a step back. Here are two answers I have written about that:
https://stackoverflow.com/a/62563330/6699433
https://stackoverflow.com/a/63773195/6699433
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With