gpt4 book ai didi

C pread给出不同的结果

转载 作者:太空宇宙 更新时间:2023-11-04 02:47:17 25 4
gpt4 key购买 nike

我有两个系统。一个是 Intel CPU 上的 Ubuntu 14.04 64 位,另一个是 CubieTruck 上的 ARM 版 Ubuntu 14.04。

Intel 系统有一个数据文件存储在 ext4 格式的硬盘上。 CubieTruck 在 NTFS HDD 上有相同的文件,它安装了 NTFS-3G。

我目前在这些系统上遇到 pread() 问题。我从一个文件中读取了一堆字节,并从该 block 中打印出前 64 个字节。稍后,这些字节用于使用 Shabal 计算一些哈希值。

虽然 CubieTruck 上打印的数据与我在 Windows 系统上使用十六进制编辑器打开文件时看到的完全匹配,但 64 位 Ubuntu 上的输出是不同的。看起来像是填满了“FFFFFF”,但也有大体上的区别。更奇怪的是,虽然 CubieTruck 上的输出始终保持不变,但一段时间后它在 64 位 Ubuntu 系统上发生了变化(我没有看到这种情况发生时的模式,我只是不时检查一下)。

但最烦人的是,x64系统好像可以正确计算,而ARM系统是错误的。

我不知道为什么 pread 在这些系统下为同一个文件提供不同的结果,但我希望有人能对此有所启发。

编辑,代码:

int main(int argc, char **argv) {
unsigned int readsize = 16384 * 32 * 2;
char *cache = (char*) malloc(readsize);

int fh = open("/home/user/somefile", O_RDONLY);

if (fh < 0) {
printf("can't open file");
exit(-1);
}

int bytes = 0, b;

do {
b = pread(fh, &cache[bytes], readsize - bytes, bytes);
bytes += b;
} while(bytes < readsize && b > 0);

int i = 0;
for (i=0; i < 64; i++) {
printf("%02X", cache[i]);
}

close(fh);
free(cache);
return 0;
}

两个系统都打开了完全相同的文件。

x64 上的结果:FFFFFF94FFFFFFF16D25FFFFFFC0FFFFFFA3367D010BFFFFFFEF1E12FFFFFF841CFFFFFFBE4C26FFFFFF92FFFFFF80FFFFFF86FFFFFFA822FFFFFF8A26FFFFFF906CFFFFFFAD05FFFFFFE7FFFFFFB124FFFFFFA8FFFFFFF77B16FFFFFFEAFFFFFFACFFFFFF9DFFFFFF9EFFFFFF81FFFFFFC7FFFFFF92FFFFFFCDFFFFFFB0FFFFFFE86270FFFFFFF974FFFFFFA8420C45FFFFFFFC04FFFFFFF9103F2E3A47FFFFFF990F

在 ARM 上的结果:94F16D25C0A3367D010BEF1E12841CBE4C26928086A8228A26906CAD05E7B124A8F77B16EAAC9D9E81C792CDB0E86270F974A8420C45FC04F9103F2E3A47990F

您可以看到,在 x64 上,结果填充有“FFFFFF”,而且看起来,这在以后不知何故需要。但我不明白为什么它在我的系统上有所不同。

最佳答案

C 的一个有趣方面是实现定义行为的数量 - 在本例中,whether char is signed or not .

%x 格式说明符采用 unsigned int 参数,因此在 ARM 情况下转换很简单 - char 是无符号的,因此只需零扩展到 无符号整数。但是对于已签名的 x86,转换可以采用以下两种方式之一:

  • char 符号扩展为 signed int,然后将其转换为 unsigned int
  • 首先转换为 unsigned char,然后零扩展为 unisgned int

看起来转换的 char->int 部分优先于 signed->unsigned 部分* 所以你得到前者(注意字节没有最高位集是明确的,并且在两个实现上打印相同)。我想您的计算会在某个地方进行类似的转换,期望有符号,因此它会在 ARM 上中断。

简而言之,如果您处理的是字符大小的而不是字符总是指定signed charunsigned char 视情况而定,永远不要裸露 char

* 我想我可以挖掘出标准来检查是否确实指定了,但在这一点上它只是一个微不足道的细节

关于C pread给出不同的结果,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25960190/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com