C++指针算术:对齐异常 (0x801)

C++ pointer arithmetic: alignment exception (0x801)

本文关键字:异常 0x801 对齐 指针 C++      更新时间:2023-10-16

我得到了一个非常奇怪的对齐异常,它只发生在某些硬件组合上。 我已经实现了一个蓝牙音频接收器,它从 unix 文件描述符获取数据。当我组合Macbook Pro(作为蓝牙源(和覆盆子派(作为蓝牙接收器(时,我在以下一点出现对齐异常:

void process(uint8_t* inData, uint32_t size, float* outData)
{
int16_t* from = (int16_t*)inData;
float* to = outData;
for (size_t i = 0; i < size/2; ++i) {
*to = *from/32767.0;
++to;
++from; // Crashes on MacbookPro/RasPi combination
}
}

怎么来了?我的水槽显然不知道我的来源。这适用于其他平台(组合(?

我也尝试了这个片段,但是也没有成功。

int8_t* from = (int8_t*)inData;
float* to = outData;
for (size_t i = 0; i < size/2; ++i) {
int16_t tmp;
std::memcpy(&tmp, from, 2);
*to = tmp/32767.0;
++to;
from += 2; // This crashes
}

我想一个运行的示例在这里无济于事,因为当使用另一个数据(蓝牙(源时,完全相同的代码有效。

您将指向 8 位值的指针视为指向 16 位值的指针。这是未定义的。

此外,通常,8 位值的对齐方式为 1 个字节,而 16 位值的对齐方式为 2 个字节。

正如Eljay在评论中所说,您可以更改inData的对齐方式,但结果仍未定义。

这修复了它:

char* to; // = ...
char* from; // = ...
for (size_t i = 0; i < buffer.size()/size(conf.codec); ++i) {
int16_t tmp;
std::memcpy(&tmp, from+(i*2), 2);
float f = tmp/32767.0;
std::memcpy(to+(i*4), &f, 4);
}