C-用于算术的不兼容指针是否违反严格别名?
这个问题是我之前问过的问题的延伸。但是,经过一段时间后,我发现我的一些关于两个指针之间的转换行为的概念仍然很模糊。
为了便于讨论,我首先对主机实现做如下假设:
- malloc:8 对齐
sizeof(int): 4,_Alignof(int): 4sizeof(double): 8,_Alignof(double): 8
问题一:
void *ptr = malloc(4096); // (A)
*(int *) ptr = 10; // (B)
/*
* Does the following line have undefined behavior
* or violate strict aliasing rules?
*/
*(((double *) ptr) + 2) = 1.618; // (C)
// now, can still read integer value with (*(int *) ptr)
以我目前的理解,答案是否定的。
根据 C11 的 [6.3.2.3 #7]:
指向对象类型的指针可以转换为指向不同对象类型的指针。如果结果指针未针对引用类型正确对齐,则行为未定义。...
和 C11 的 [6.5 #7]:
对象的存储值只能由具有以下类型之一的左值表达式访问:
- 与对象的有效类型兼容的类型,
- ...
因此,据我所知,
- 在 (A) 行之后,我分配了一个没有声明类型并且还没有有效类型的对象。
- 在 (B) 行之后,已分配对象的前 4 个字节已经具有有效类型:
int。 - 对于 (C) 行,类型
ptr正确对齐double,指针转换和指针算术是合法的。因为它没有访问前 4 个字节,所以它没有违反 6.5 #7 规则。
我对上面提到的内容有什么误解吗?
问题二:
void *ptr = malloc(4096); // (A)
*(int *) ptr = 10; // (B)
/*
* Does the following line have undefined behavior
* or violate strict aliasing rules?
*/
*(double *) ptr = 1.618; // (C)
// now, shall not read value with (*(int *) ptr)
以我目前的理解,答案也是No。
根据 C11 的 [6.5 #6]:
如果一个值通过一个类型不是字符类型的左值存储到一个没有声明类型的对象中,那么左值的类型将成为该访问的对象的有效类型,并且对于不修改储值。
因此,据我所知,第 (C) 行是修改存储值并将前 8 个字节的有效类型更新为的后续访问double。我对上面提到的内容有什么误解吗?
主要的困惑是不确定是否违反了 [6.5 #7] 规则:
对象的存储值只能由具有以下类型之一的左值表达式访问:
- 与对象的有效类型兼容的类型,
- ...