mirror of
https://github.com/dart-lang/sdk
synced 2024-11-02 15:17:07 +00:00
b9b6511ca6
Closes https://github.com/dart-lang/sdk/pull/50918 Co-authored-by: Josh Soref <jsoref@gmail.com> GitOrigin-RevId: 1fd275051c561b63d374fb47e76a22424c4a12a9 Change-Id: I97790d9c79ff659f2c1fa2d2d46d041fe67957cb Reviewed-on: https://dart-review.googlesource.com/c/sdk/+/278530 Reviewed-by: William Hesse <whesse@google.com> Commit-Queue: Alexander Thomas <athom@google.com> Reviewed-by: Tess Strickland <sstrickl@google.com>
83 lines
2.8 KiB
Dart
83 lines
2.8 KiB
Dart
// Copyright (c) 2022, the Dart project authors. Please see the AUTHORS file
|
|
// for details. All rights reserved. Use of this source code is governed by a
|
|
// BSD-style license that can be found in the LICENSE file.
|
|
|
|
// Testing Bigints with and without intrinsics.
|
|
// VMOptions=--intrinsify --no-enable-asserts
|
|
// VMOptions=--intrinsify --enable-asserts
|
|
// VMOptions=--no-intrinsify --enable-asserts
|
|
// VMOptions=--no-intrinsify --no-enable-asserts
|
|
|
|
import "package:expect/expect.dart";
|
|
|
|
const debugPrint = bool.fromEnvironment('debugPrint');
|
|
|
|
void check(int length, BigInt base) {
|
|
assert(length >= 5);
|
|
assert(base > BigInt.zero);
|
|
|
|
// Check with slight adjustments. We choose -3..+3 so that the lowest bit in
|
|
// the 2's-complement representation is both zero and one for both the
|
|
// positive [n] and its negative complement [m] below.
|
|
for (int delta = -3; delta <= 3; delta++) {
|
|
BigInt n = base + BigInt.from(delta);
|
|
assert(n >= BigInt.zero);
|
|
|
|
// Compute the bitLength by shifting the value into a small integer range
|
|
// and adjust the `int.bitLength` value by the shift count.
|
|
int shiftCount = length - 5;
|
|
int shiftedN = (n >> shiftCount).toInt();
|
|
int expectedLength = shiftCount + shiftedN.bitLength;
|
|
|
|
int nLength = n.bitLength;
|
|
Expect.equals(expectedLength, nLength);
|
|
|
|
// Use identity `x.bitLength == (-x-1).bitLength` to check negative values.
|
|
BigInt m = -n - BigInt.one;
|
|
int mLength = m.bitLength;
|
|
|
|
if (debugPrint) {
|
|
final printLength = length + 4;
|
|
final nDigits =
|
|
n.toUnsigned(printLength).toRadixString(2).padLeft(printLength);
|
|
final mDigits = m.toUnsigned(printLength).toRadixString(2);
|
|
print('$nDigits: $nLength');
|
|
print('$mDigits: $mLength');
|
|
}
|
|
|
|
Expect.equals(nLength, mLength);
|
|
}
|
|
}
|
|
|
|
void main() {
|
|
// For small values, [BigInt.bitLength] should be the same as [int.bitLength].
|
|
for (int i = 0; i <= 64; i++) {
|
|
Expect.equals(i.bitLength, BigInt.from(i).bitLength);
|
|
// Note: This is not quite redundant for `i==0` since on the web platform
|
|
// `-i` is negative zero and not the same as `0-i`.
|
|
Expect.equals((-i).bitLength, BigInt.from(-i).bitLength);
|
|
}
|
|
|
|
// Test x.bitLength for a large variety of lengths.
|
|
for (int length = 5; length <= 512; length++) {
|
|
BigInt base = BigInt.one << (length - 1);
|
|
Expect.equals(length, base.bitLength);
|
|
|
|
// Power of two.
|
|
check(length, base);
|
|
|
|
// Two high bits set.
|
|
check(length, base | base >> 1);
|
|
|
|
// Check for values with an additional bit set near a potential internal
|
|
// digit boundary.
|
|
for (int i1 = 16; i1 < length; i1 += 16) {
|
|
for (int i2 = -1; i2 <= 1; i2++) {
|
|
int i = i1 + i2;
|
|
if (i < length - 1) {
|
|
check(length, base | BigInt.one << (i - 1));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|