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

fix: add safe integer conversion for bitwise operations #1217

Merged

Conversation

thevilledev
Copy link
Contributor

@thevilledev thevilledev commented Mar 3, 2025

Implements proper safety checks for all bitwise operations (<<, >>, &, ^, |, ~) when converting double values to int64_t. This prevents undefined behavior and potential security issues that could arise from integer overflow or the use of non-finite double values (NaN/Infinity) in bitwise contexts.

The implementation:

  • Adds a safeDoubleToInt64 helper function with range validation.
  • Applies this function consistently across all relevant bitwise operations in the interpreter.
  • Limits values to the safe integer range [-2^53, 2^53] before conversion to int64_t.
  • Includes an additional check to prevent left shift (<<) operations from causing int64_t overflow.
  • Provides clear error messages, including source location context, when validation fails.

The safe integer range [-2^53, 2^53] is used because IEEE 754 double-precision floating-point numbers, as used by Jsonnet, can precisely represent all integers within this range. Using doubles outside this range for bitwise operations could lead to unpredictable results due to precision loss.

This change aligns with the Jsonnet specification, which requires bitwise operations to convert operands to signed 64-bit integers, while ensuring the conversion process is safe and mathematically sound within the limits of double precision representation. New test cases have been added to cover boundary conditions and potential errors.

NOTE: This was originally reported through the Google OSS VRP. Conclusion was that this can be reported publicly.

Copy link

google-cla bot commented Mar 3, 2025

Thanks for your pull request! It looks like this may be your first contribution to a Google open source project. Before we can look at your pull request, you'll need to sign a Contributor License Agreement (CLA).

View this failed invocation of the CLA check for more information.

For the most up to date status, view the checks section at the bottom of the pull request.

@thevilledev
Copy link
Contributor Author

Unable to sign the CLA:

A server error occurred, please try your request again.

I will try later.

@thevilledev thevilledev force-pushed the fix/bitwise-double-to-int64 branch from 715e678 to 42e89a4 Compare March 3, 2025 21:10
@thevilledev
Copy link
Contributor Author

CLA signed.

Copy link
Collaborator

@johnbartholomew johnbartholomew left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for the contribution!

It would be good to have some basic test coverage of this.

@johnbartholomew
Copy link
Collaborator

I also wonder if we need to do something extra with left-shifts.

With the current code, 2^53 (or some other large value) can be left-shifted beyond the safe integer range. In fact this does work correctly up to a certain point because left shifting naturally gives a value which is a multiple of 2^x, and those values can be exactly represented even though adjacent integers cannot. However, since the operation is performed via a signed 64-bit integer, left shifting can easily result in undefined behaviour (pre C++20) or defined but confusing behaviour in C++20 and beyond.

See, for example, https://stackoverflow.com/a/60712725/52251

And this correct (perhaps) but rather confusing result:

$ ./jsonnet -e '4503599627370496 << 11'
-9223372036854775808

@thevilledev thevilledev force-pushed the fix/bitwise-double-to-int64 branch 3 times, most recently from af364c5 to 3544891 Compare March 22, 2025 19:12
@thevilledev
Copy link
Contributor Author

Good point! I added explicit checks before left-shifting to ensure the value won’t overflow a signed 64‑bit integer. If the shift would exceed 63 bits, we throw an error. This prevents undefined or confusing behavior and keeps left‑shifts within the safe range.

Tests are in and pass at least locally. Feedback welcome.

Copy link
Collaborator

@johnbartholomew johnbartholomew left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for continuing to work on this :-)

There is some inconsistency in the comments and code about whether the maximum safe value is 2^53 or (2^53)-1. It would be good to make sure this is correct and consistent in the comments and the actual logic.

2^53 can be represented exactly, so I would set that as the maximum (and correspondingly -2^53 as the minimum). This also matches the Jsonnet documentation.

https://en.wikipedia.org/wiki/Double-precision_floating-point_format#Precision_limitations_on_integer_values

(You can also verify it yourself by iterating through adjacent double-precision values, e.g., using nextafter, or by manually working out what the double precision representation is for 2^53-1 (ok), 2^53 (ok), and 2^53+1 (not representable!))

@thevilledev
Copy link
Contributor Author

Thanks @johnbartholomew for the feedback! I guess I misunderstood the previous comment, but I've now aligned comments, code and tests to match what's in the Jsonnet spec.

@thevilledev thevilledev changed the title fix: add safe double-to-int64 conversion for bitwise operations fix: add safe integer conversion for bitwise operations Mar 30, 2025
Copy link
Collaborator

@johnbartholomew johnbartholomew left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks good! Would you mind rebasing onto master? Or I can do it when I merge.

@thevilledev thevilledev force-pushed the fix/bitwise-double-to-int64 branch from 85fb4ed to e7d6e86 Compare March 30, 2025 18:41
Implements proper safety checks for all bitwise operations (<<, >>, &, ^, |)
when converting double values to int64_t. This prevents undefined behavior
and potential security issues from integer overflow or invalid values.

The implementation:
- Adds safeDoubleToInt64 function with proper range validation
- Applies this function to all bitwise operations
- Limits values to the safe integer range (±(2^53-1))
- Provides clear error messages with source location context

The safe integer range limitation is necessary because IEEE 754 doubles
can only precisely represent integers up to 2^53-1. Beyond this range,
precision is lost, which would lead to unpredictable results in bitwise
operations that depend on exact bit patterns.

This change aligns with the Jsonnet specification which requires bitwise
operations to convert operands to signed 64-bit integers before performing
operations, while ensuring mathematical correctness.

Signed-off-by: Ville Vesilehto <[email protected]>
Add safety checks to prevent undefined behavior in bitwise left shift
operations.

These checks detect potential overflow conditions before they occur and throw
appropriate error messages to help users identify problematic code. The error
messages are consistent between runtime and static error handling.

Added tests.

Signed-off-by: Ville Vesilehto <[email protected]>
IEEE 754 double-precision floating-point numbers can precisely
represent all integers in the range [-2^53, 2^53]. The previous
implementation incorrectly used the range [-(2^53 - 1), 2^53 - 1]
when checking if a double could be safely converted to an int64
for bitwise operations.

This commit updates the `safeDoubleToInt64` function and its
associated comments and tests to use the correct safe integer range,
aligning with the IEEE 754 standard and Jsonnet documentation.
Test cases have been adjusted to verify the new boundaries and
fix related assertions.

Signed-off-by: Ville Vesilehto <[email protected]>
Adds the `safe_integer_conversion.jsonnet.fmt.golden` file containing
the expected output from `jsonnetfmt`. This resolves the formatting
test failure for this file in `run_fmt_tests.sh` by providing an
explicit golden output for comparison, accounting for minor spacing
differences introduced by the formatter.

Signed-off-by: Ville Vesilehto <[email protected]>
@thevilledev thevilledev force-pushed the fix/bitwise-double-to-int64 branch from e7d6e86 to eff455a Compare March 30, 2025 18:41
@thevilledev
Copy link
Contributor Author

Awesome! Rebase done

@johnbartholomew johnbartholomew merged commit eff455a into google:master Mar 30, 2025
7 checks passed
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

Successfully merging this pull request may close these issues.

4 participants