Skip to content
New issue

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

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

Already on GitHub? # to your account

Proposal: Change the semantics of bitwise operations to be fully platform-dependent #802

Closed
wants to merge 1 commit into from

Conversation

yoshi-monster
Copy link
Contributor

@yoshi-monster yoshi-monster commented Feb 11, 2025

Hello!!

in my array implementation, I use bitwise operations for internal index calculations. Bitwise operations are generally considered to be quite low-level, but extremely fast in comparison. They can be therefore be thought of a micro-optimisation; for example a left shift a >> b is a hardware implementation for the equivalent

let assert Ok(b2) = int.power(2, b)
float.truncate(float.floor(a /. b2))

From my understanding, number semantics are platform-defined, except that Gleam defines division by zero to equal zero. Floating point numbers have slightly different semantics, since Erlang is not fully compliant with IEEE 754. Integers in Erlang have arbitrary precision, while standard floating point numbers are used on the Javascript target, even though a native BigInt type would exist on that target. It is well-known that while whole integers can be accurately represented for up to 53 bit using 64-bit floating-point numbers, bitwise operations are defined using the semantics of 32-bit integers in the ECMAScript standard.

The current implementation of bitwise operations for Javascript instead tries to support the full range of valid integers, up to 53-bit. To that end, it converts all inputs to BigInts and back, which implies a significant overhead. These are notably the only sets of operations that try to preserve integer semantics beyond what the platform defines; I could not find another use of BigInt for any other operation in the standard library. This does not what I'd have expected for these reasons:

  • I would have expected for bitwise operations to follow the semantics of the platform
  • I would have expected bitwise operations to be useful as a micro-optimisation, and in particular not imply memory allocations.

If this is unacceptable, I think a range check that falls back to a mathematically equivalent expression defined on floating-point numbers (as outlined above) would overall still be faster and easier to optimise for engines.

thanks ~ 💜

(I have left the tests intentionally broken for now)

@lpil
Copy link
Member

lpil commented Feb 12, 2025

We don't accept unplanned changes, please open an issue first 🙏

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

Successfully merging this pull request may close these issues.

2 participants