Jos*_*one 2 c swap endianness bit-fields
我正在使用C和bswap_ {16,32,64}来自byteswap.h的宏从big-endian读取/编写一个小端格式的二进制文件,用于字节交换.
除了40位的位字段外,所有值都被正确读取和写入.
该bswap_40宏不存在,我不知道该怎么做,或有没有更好的解决方案是可能的.
这是一个显示此问题的小代码:
#include <stdio.h>
#include <inttypes.h>
#include <byteswap.h>
#define bswap_40(x) bswap_64(x)
struct tIndex {
uint64_t val_64;
uint64_t val_40:40;
} s1 = { 5294967296, 5294967296 };
int main(void)
{
// write swapped values
struct tIndex s2 = { bswap_64(s1.val_64), bswap_40(s1.val_40) };
FILE *fp = fopen("index.bin", "w");
fwrite(&s2, sizeof(s2), 1, fp);
fclose(fp);
// read swapped values
struct tIndex s3;
fp = fopen("index.bin", "r");
fread(&s3, sizeof(s3), 1, fp);
fclose(fp);
s3.val_64 = bswap_64(s3.val_64);
s3.val_40 = bswap_40(s3.val_40);
printf("val_64: %" PRIu64 " -> %s\n", s3.val_64, (s1.val_64 == s3.val_64 ? "OK" : "Error"));
printf("val_40: %" PRIu64 " -> %s\n", s3.val_40, (s1.val_40 == s3.val_40 ? "OK" : "Error"));
return 0;
}
Run Code Online (Sandbox Code Playgroud)
该代码编译为:
gcc -D_FILE_OFFSET_BITS = 64 -D_LARGEFILE_SOURCE -D_LARGEFILE64_SOURCE swap_40.c -o swap_40
如何定义bswap_40宏来读取和写入40位的字节交换值?
通过定义bswap_40为相同bswap_64,您将交换8个字节而不是5个.所以如果你从这开始:
00 00 00 01 02 03 04 05
Run Code Online (Sandbox Code Playgroud)
你最终得到这个:
05 04 03 02 01 00 00 00
Run Code Online (Sandbox Code Playgroud)
而不是这个:
00 00 00 05 04 03 02 01
Run Code Online (Sandbox Code Playgroud)
处理此问题的最简单方法是获取结果bswap_64并将其右移24:
#define bswap_40(x) (bswap_64(x) >> 24)
Run Code Online (Sandbox Code Playgroud)