Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

ppci-cc: handling of big constants with L specifier #94

Open
tstreiff opened this issue Jun 23, 2020 · 1 comment
Open

ppci-cc: handling of big constants with L specifier #94

tstreiff opened this issue Jun 23, 2020 · 1 comment

Comments

@tstreiff
Copy link
Contributor

In x86_64, ints are 32bit and longs are 64bit.

long l1 = 9223372036854775807; // LONG_MAX
long l2 = -9223372036854775808; // LONG_MIN
long l3 = -9223372036854775808L; // same value but...
^ Integer value too big for type (9223372036854775807)

The value is OK for an "int" constant (with no type specifier) but is too big for a "long" constant (L type specifier)

@tstreiff tstreiff changed the title ppci-cc: handling of LONG_MAX and LONG_MIN with L specifier ppci-cc: handling of big constants with L specifier Jun 23, 2020
@tstreiff
Copy link
Contributor Author

tstreiff commented Jul 1, 2020

This is checked in semantics.py/on_number()
The subtle point is that integer constants are always positive, so this function does not know if a sign has been found just before.
And the maximum absolute value is not the same for negative et positive values.

  • case l2 works because since there are no type specifiers, the compiler chooses the smallest integer type able to cope with the (positive) value. This requires unsigned 64bit integer (unsigned long for x86_64)
  • case l3 does not work because the type "signed long" is deduced from the L specifier.

According to the C standard, the "L" specifiers implies "long int" or "long long int" (the 1st of the two large enough to hold the value)
A possible fix for integers would be:

  1. We let the compiler find the shortest type suitable for the value
  2. If a specifier is given, we "upgrade" the type found in 1)
    I am not sure step 2) is really necessary because type of integer constants is rarely used (their type is adjusted on demand)
    An exception is passing a constant to a "old-style" (ie K&R) function: specifying the type of a constant in a function call can help the compiler choose how to put it on the stack (on an architecture where ints are 16bit and longs are 32bit, passing 0 requires one level of stack, whereas passing 0L requires two)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

1 participant