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

How to handle non-continuous ranges? #129

Open
adambadura opened this issue Oct 8, 2022 · 3 comments
Open

How to handle non-continuous ranges? #129

adambadura opened this issue Oct 8, 2022 · 3 comments

Comments

@adambadura
Copy link

Is it possible to somehow include non-continuous ranges? For example. in computation, I'm using an enum (converted to its underlying type). The enum has a few possible values but the values do not form a continuous range. Of course, I could take the least range covering all possible values. But this would include more values than are truly possible and I'm afraid in some cases of complex computations it could impact the resulting range.

@robertramey
Copy link
Member

robertramey commented Oct 8, 2022 via email

@adambadura
Copy link
Author

Well... My first idea was that ranged integer would have to support a list of ranges instead.

One-element list would then be the "special case" corresponding to the current ranged integer. Whereas a list of single values would be the "special case" corresponding to enumeration-like values. (Perhaps, in the future at least, developments in reflection would allow to covert enum value to such list-ranged safe integer within the language itself.)

A list of ranges is needed since further arithmetic could change what was originaly a list of single values into a list of ranges. Connecting overlapping ranges and removing empty ones would have to be handled.

Would it be possible to add something like this to the current implementation of the library?

@adambadura
Copy link
Author

I just realized the problem (lack of non-continuous ranges) exists in the library even without considering enum-based values. Multiplication introduces this problem already! Consider a simple program (godbolt.org):

using delta_safe_t = boost::safe_numerics::safe_signed_range<-1, 1>;

template<int I>
using const_safe_t = boost::safe_numerics::safe_signed_literal<I>;

int main() {
  constexpr auto x = const_safe_t<8>{};
  std::cout << "x = " << safe_format(x) << '\n';
  constexpr auto y = delta_safe_t{1};
  std::cout << "y = " << safe_format(y) << '\n';
  constexpr auto z = y * x;
  std::cout << "z = " << safe_format(z) << '\n';
}

It outputs the following

x = <signed char>[8,8] = 8
y = <signed char>[-1,1] = 1
z = <int>[-8,8] = 8

The problem is visible in the range of z. It is [-8;8] whereas, in fact, it can be only one of: -8, 0, or 8.

Now, I cannot readily provide a realistic example where this "drop of precision" leads to subsequent issues but I expect there might be such cases. For example, where divisibility is a required condition.

Then again, now I'm no longer sure this can be reasonably covered. With a precise approach, a multiplication of a ranged integer by a constant would result in a ranged integer with a list of ranges of length equal to the source range length. This could easily exceed compiler limits for templating. Not to mention multiplying two ranged integers.

However, it would be nice to have it somehow covered at least in simpler cases.

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

2 participants