https://gcc.gnu.org/bugzilla/show_bug.cgi?id=69776
--- Comment #2 from Alexander Cherepanov <ch3root at openwall dot com> --- [CC'ing Martin Sebor and Joseph S. Myers as it's potentially related to bug 65892.] On 2016-02-12 01:20, pinskia at gcc dot gnu.org wrote: >> The last value is wrong, it should be 1. > > Why do you think that? > > If we look at your code: > > void *p = malloc(10); > int *pi = p; > double *pd = p; > > << at this point p has no effective type. Ok > *pi = 1; > printf("*pi = %d\n", *pi); > > << Now it has an effective type of int or a struct containing int Structs are a separate topic (and I thinks it's still unsettled in the standard). Fortunately they are not relevant here. Back to int... C11, 6.5p6: "If a value is stored into an object having no declared type through an lvalue having a type that is not a character type, then the type of the lvalue becomes the effective type of the object for that access and for subsequent accesses that do not modify the stored value." So, yes, it has an effective type of int for this store and for subsequent reads. But not for writes. > int a = *pi; > > << a read Yes, a read with an effective type of int. > *pd = 0; > > << a write to a double, it does not alias int at all so it can be moved past > the next statement This is a write. Hence the previous effective type is not relevant. And effective type for this write and for subsequent reads is double. > *pi = a; > > << store via an int > > Since the order of pi and pd is not specified due to different aliasing of int > and double so either can be done first Sorry, I don't see what the talk about moving stores could be based on. This is another write. So it ignores the previous effective type and sets a new one. Looking at it at a higher level: I think it is generally agreed that allocated memory could be repurposed at will -- once you don't need old data you just overwrite it with new data without any regard to the types. I have seen proposals[1] for forbidding this technique but I don't think it was seriously considered. Maybe I missed it. [1] https://gcc.gnu.org/ml/gcc/2004-12/msg00193.html The program in this bug report is loosely based on the Example 1 in DR 236 and I was surprised that the problematic behavior is present even without functions involved. The variant with a function (closer to the Example 1 in DR 236): // file 1 void f(int *qi, double *qd) { int i = *qi; *qd = 0; *qi = i; } // file 2 #include <stdlib.h> #include <stdio.h> void f(int *qi, double *qd); int main() { void *p = malloc(10); int *pi = p; double *pd = p; *pi = 1; printf("*pi = %d\n", *pi); f(pi, pd); printf("*pi = %d\n", *pi); } I don't know if it should be considered the same case or a separate one. Most discussions about DR 236 (like in bug 65892) are concerned about unions and it's not clear to me if the case of allocated objects is settled or not, in gcc or in general. IMHO the case of allocated objects is easier (no tricky visibility rules) and more general but maybe I'm missing something.