I compiled this code using Visual Studio 2010 (cl.exe /W4
) as a C file:
int main( int argc, char *argv[] )
{
unsigned __int64 a = 0x00000000FFFFFFFF;
void *orig = (void *)0xFFFFFFFF;
unsigned __int64 b = (unsigned __int64)orig;
if( a != b )
printf( " problema: %016I64Xb: %016I64X
", a, b );
return;
}
There are no warnings and the result is:
problem a: 00000000FFFFFFFF b: FFFFFFFFFFFFFFFF
I suppose int orig = (int)0xFFFFFFFF
would be less controversial as I'm not assigning a pointer to an integer. However the result would be the same.
Can someone explain to me where in the C standard it is covered that orig
is sign extended from 0xFFFFFFFF to 0xFFFFFFFFFFFFFFFF?
I had assumed that (unsigned __int64)orig
would become 0x00000000FFFFFFFF. It appears that the conversion is first to the signed __int64 type and then it becomes unsigned?
EDIT: This question has been answered in that pointers are sign extended which is why I see this behavior in gcc and msvc. However I don't understand why when I do something like (unsigned __int64)(int)0xF0000000
it sign extends to 0xFFFFFFFFF0000000 but (unsigned __int64)0xF0000000
does not instead showing what I want which is 0x00000000F0000000.
EDIT: An answer to the above edit. The reason that (unsigned __int64)(int)0xF0000000
is sign extended is because, as noted by user R:
Conversion of a signed type (or any type) to an unsigned type
always takes place via reduction modulo one plus the max value of
the destination type.
And in (unsigned __int64)0xF0000000
0xF0000000 starts off as an unsigned integer type because it cannot fit in an integer type. Next that already unsigned type is converted unsigned __int64
.
So the takeaway from this for me is with a function that's returning a 32-bit or 64-bit pointer as an unsigned __int64
to compare I must first convert the 32-bit pointer in my 32-bit application to an unsigned type before promoting to unsigned __int64
. The resulting code looks like this (but, you know, better):
unsigned __int64 functionidontcontrol( char * );
unsigned __int64 x;
void *y = thisisa32bitaddress;
x = functionidontcontrol(str);
if( x != (uintptr_t)y )
EDIT again:
Here is what I found in the C99 standard:
6.3.1.3 Signed and unsigned integers
- 1 When a value with integer type is converted to another integer
type other than _Bool, if the value can be represented by the new
type, it is unchanged.
- 2 Otherwise, if the new type is unsigned, the value is converted by
repeatedly adding or subtracting one more than the maximum value
that can be represented in the new type until the value is in the
range of the new type.49)
- 3 Otherwise, the new type is signed and the value cannot be
represented in it; either the result is implementation-defined or an
implementation-defined signal is raised.
- 49) The rules describe arithmetic on the mathematical value, not the
value of a given type of expression.
See Question&Answers more detail:
os 与恶龙缠斗过久,自身亦成为恶龙;凝视深渊过久,深渊将回以凝视…