blob: 1332aef7d21dca29cd05e8883534fd9af6bcb838 [file] [log] [blame]
// Copyright (c) 2012, 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.
// Patch file for dart:core classes.
import "dart:_internal" hide Symbol, LinkedList, LinkedListEntry;
import "dart:_internal" as _symbol_dev;
import 'dart:_interceptors';
import 'dart:_js_helper'
show
assertUnreachable,
boolConversionCheck,
checkInt,
Closure,
ConstantMap,
convertDartClosureToJS,
getRuntimeType,
JsLinkedHashMap,
jsonEncodeNative,
JSSyntaxRegExp,
objectHashCode,
patch,
Primitives,
quoteStringForRegExp,
getTraceFromException,
RuntimeError,
wrapException,
wrapZoneUnaryCallback;
import 'dart:_foreign_helper' show JS;
import 'dart:_native_typed_data' show NativeUint8List;
import 'dart:_rti' show getRuntimeType;
import "dart:convert" show Encoding, utf8;
import 'dart:typed_data' show Endian, Uint8List, Uint16List;
String _symbolToString(Symbol symbol) =>
_symbol_dev.Symbol.getName(symbol as _symbol_dev.Symbol);
Map<String, dynamic>? _symbolMapToStringMap(Map<Symbol, dynamic>? map) {
if (map == null) return null;
var result = new Map<String, dynamic>();
map.forEach((Symbol key, value) {
result[_symbolToString(key)] = value;
});
return result;
}
@patch
int identityHashCode(Object? object) => objectHashCode(object);
// Patch for Object implementation.
@patch
class Object {
@patch
bool operator ==(Object other) => identical(this, other);
@patch
int get hashCode => Primitives.objectHashCode(this);
@patch
String toString() => Primitives.objectToHumanReadableString(this);
@patch
dynamic noSuchMethod(Invocation invocation) {
throw new NoSuchMethodError(this, invocation.memberName,
invocation.positionalArguments, invocation.namedArguments);
}
@patch
Type get runtimeType => getRuntimeType(this);
}
@patch
class Null {
@patch
int get hashCode => super.hashCode;
}
// Patch for Function implementation.
@patch
class Function {
@patch
static apply(Function function, List<dynamic>? positionalArguments,
[Map<Symbol, dynamic>? namedArguments]) {
return Primitives.applyFunction(
function,
positionalArguments,
// Use this form so that if namedArguments is always null, we can
// tree-shake _symbolMapToStringMap.
namedArguments == null ? null : _symbolMapToStringMap(namedArguments));
}
}
// Patch for Expando implementation.
@patch
class Expando<T extends Object> {
final Object _jsWeakMap = JS('=Object', 'new WeakMap()');
@patch
Expando([this.name]);
@patch
T? operator [](Object object) {
_checkType(object); // WeakMap doesn't check on reading, only writing.
return JS('', '#.get(#)', _jsWeakMap, object);
}
@patch
void operator []=(Object object, T? value) {
JS('void', '#.set(#, #)', _jsWeakMap, object, value);
}
static _checkType(object) {
if (object == null || object is bool || object is num || object is String) {
throw new ArgumentError.value(object,
"Expandos are not allowed on strings, numbers, booleans or null");
}
}
}
// Patch for WeakReference implementation.
@patch
class WeakReference<T extends Object> {
@patch
factory WeakReference(T object) {
return _WeakReferenceWrapper<T>(object);
}
}
class _WeakReferenceWrapper<T extends Object> implements WeakReference<T> {
final Object _weakRef;
_WeakReferenceWrapper(T object) : _weakRef = JS('', 'new WeakRef(#)', object);
T? get target => JS('', '#.deref()', _weakRef);
}
// Patch for Finalizer implementation.
@patch
class Finalizer<T> {
@patch
factory Finalizer(void Function(T) object) {
return _FinalizationRegistryWrapper<T>(object);
}
}
class _FinalizationRegistryWrapper<T> implements Finalizer<T> {
final Object _registry;
_FinalizationRegistryWrapper(void Function(T) callback)
: _registry = JS('', 'new FinalizationRegistry(#)',
convertDartClosureToJS(wrapZoneUnaryCallback(callback), 1));
void attach(Object value, T token, {Object? detach}) {
if (detach != null) {
JS('', '#.register(#, #, #)', _registry, value, token, detach);
} else {
JS('', '#.register(#, #)', _registry, value, token);
}
}
void detach(Object detachToken) {
JS('', '#.unregister(#)', _registry, detachToken);
}
}
@patch
class int {
@patch
static int parse(String source,
{int? radix, @deprecated int onError(String source)?}) {
int? value = tryParse(source, radix: radix);
if (value != null) return value;
if (onError != null) return onError(source);
throw new FormatException(source);
}
@patch
static int? tryParse(String source, {int? radix}) {
return Primitives.parseInt(source, radix);
}
}
@patch
class double {
@patch
static double parse(String source,
[@deprecated double onError(String source)?]) {
double? value = tryParse(source);
if (value != null) return value;
if (onError != null) return onError(source);
throw new FormatException('Invalid double', source);
}
@patch
static double? tryParse(String source) {
return Primitives.parseDouble(source);
}
}
@patch
class BigInt implements Comparable<BigInt> {
@patch
static BigInt get zero => _BigIntImpl.zero;
@patch
static BigInt get one => _BigIntImpl.one;
@patch
static BigInt get two => _BigIntImpl.two;
@patch
static BigInt parse(String source, {int? radix}) =>
_BigIntImpl.parse(source, radix: radix);
@patch
static BigInt? tryParse(String source, {int? radix}) =>
_BigIntImpl._tryParse(source, radix: radix);
@patch
factory BigInt.from(num value) = _BigIntImpl.from;
}
@patch
class Error {
@patch
static String _objectToString(Object object) {
// Closures all have useful and safe toString methods.
if (object is Closure) return object.toString();
return Primitives.objectToHumanReadableString(object);
}
@patch
static String _stringToSafeString(String string) {
return jsonEncodeNative(string);
}
@patch
StackTrace? get stackTrace => Primitives.extractStackTrace(this);
@patch
static Never _throw(Object error, StackTrace stackTrace) {
error = wrapException(error);
JS('void', '#.stack = #', error, stackTrace.toString());
JS('', 'throw #', error);
throw "unreachable";
}
}
@patch
class FallThroughError {
@patch
FallThroughError._create(String url, int line);
@patch
String toString() => super.toString();
}
@patch
class AbstractClassInstantiationError {
@patch
String toString() => "Cannot instantiate abstract class: '$_className'";
}
// Patch for DateTime implementation.
@patch
class DateTime {
@patch
DateTime.fromMillisecondsSinceEpoch(int millisecondsSinceEpoch,
{bool isUtc: false})
// `0 + millisecondsSinceEpoch` forces the inferred result to be non-null.
: this._withValue(0 + millisecondsSinceEpoch, isUtc: isUtc);
@patch
DateTime.fromMicrosecondsSinceEpoch(int microsecondsSinceEpoch,
{bool isUtc: false})
: this._withValue(
_microsecondInRoundedMilliseconds(microsecondsSinceEpoch),
isUtc: isUtc);
@patch
DateTime._internal(int year, int month, int day, int hour, int minute,
int second, int millisecond, int microsecond, bool isUtc)
// checkBool is manually inlined here because dart2js doesn't inline it
// and [isUtc] is usually a constant.
: this.isUtc = isUtc is bool
? isUtc
: throw new ArgumentError.value(isUtc, 'isUtc'),
_value = checkInt(Primitives.valueFromDecomposedDate(
year,
month,
day,
hour,
minute,
second,
millisecond + _microsecondInRoundedMilliseconds(microsecond),
isUtc));
@patch
DateTime._now()
: isUtc = false,
_value = Primitives.dateNow();
/// Rounds the given [microsecond] to the nearest milliseconds value.
///
/// For example, invoked with argument `2600` returns `3`.
static int _microsecondInRoundedMilliseconds(int microsecond) {
return (microsecond / 1000).round();
}
@patch
static int? _brokenDownDateToValue(int year, int month, int day, int hour,
int minute, int second, int millisecond, int microsecond, bool isUtc) {
return Primitives.valueFromDecomposedDate(
year,
month,
day,
hour,
minute,
second,
millisecond + _microsecondInRoundedMilliseconds(microsecond),
isUtc);
}
@patch
String get timeZoneName {
if (isUtc) return "UTC";
return Primitives.getTimeZoneName(this);
}
@patch
Duration get timeZoneOffset {
if (isUtc) return new Duration();
return new Duration(minutes: Primitives.getTimeZoneOffsetInMinutes(this));
}
@patch
DateTime add(Duration duration) {
return new DateTime._withValue(_value + duration.inMilliseconds,
isUtc: isUtc);
}
@patch
DateTime subtract(Duration duration) {
return new DateTime._withValue(_value - duration.inMilliseconds,
isUtc: isUtc);
}
@patch
Duration difference(DateTime other) {
return new Duration(milliseconds: _value - other._value);
}
@patch
int get millisecondsSinceEpoch => _value;
@patch
int get microsecondsSinceEpoch => 1000 * _value;
@patch
int get year => Primitives.getYear(this);
@patch
int get month => Primitives.getMonth(this);
@patch
int get day => Primitives.getDay(this);
@patch
int get hour => Primitives.getHours(this);
@patch
int get minute => Primitives.getMinutes(this);
@patch
int get second => Primitives.getSeconds(this);
@patch
int get millisecond => Primitives.getMilliseconds(this);
@patch
int get microsecond => 0;
@patch
int get weekday => Primitives.getWeekday(this);
@patch
bool operator ==(Object other) =>
other is DateTime &&
_value == other.millisecondsSinceEpoch &&
isUtc == other.isUtc;
@patch
bool isBefore(DateTime other) => _value < other.millisecondsSinceEpoch;
@patch
bool isAfter(DateTime other) => _value > other.millisecondsSinceEpoch;
@patch
bool isAtSameMomentAs(DateTime other) =>
_value == other.millisecondsSinceEpoch;
@patch
int compareTo(DateTime other) =>
_value.compareTo(other.millisecondsSinceEpoch);
}
// Patch for Stopwatch implementation.
@patch
class Stopwatch {
@patch
static int _initTicker() {
Primitives.initTicker();
return Primitives.timerFrequency;
}
@patch
static int _now() => Primitives.timerTicks();
@patch
int get elapsedMicroseconds {
int ticks = elapsedTicks;
if (_frequency == 1000000) return ticks;
assert(_frequency == 1000);
return ticks * 1000;
}
@patch
int get elapsedMilliseconds {
int ticks = elapsedTicks;
if (_frequency == 1000) return ticks;
assert(_frequency == 1000000);
return ticks ~/ 1000;
}
}
// Patch for List implementation.
@patch
class List<E> {
@patch
factory List([int? length]) = JSArray<E>.list;
@patch
factory List.filled(int length, E fill, {bool growable: false}) {
var result = growable
? new JSArray<E>.growable(length)
: new JSArray<E>.fixed(length);
if (length != 0 && fill != null) {
// TODO(sra): Consider using `Array.fill`.
for (int i = 0; i < result.length; i++) {
// Unchecked assignment equivalent to `result[i] = fill`;
// `fill` is checked statically at call site.
JS('', '#[#] = #', result, i, fill);
}
}
return result;
}
@patch
factory List.empty({bool growable: false}) {
return growable ? new JSArray<E>.growable(0) : new JSArray<E>.fixed(0);
}
@patch
factory List.from(Iterable elements, {bool growable: true}) {
List<E> list = <E>[];
for (E e in elements) {
list.add(e);
}
if (growable) return list;
return makeListFixedLength(list);
}
@patch
factory List.of(Iterable<E> elements, {bool growable: true}) {
if (growable == true) return List._of(elements);
if (growable == false) return List._fixedOf(elements);
// [growable] may be `null` in legacy mode. Fail with the same error as if
// [growable] was used in a condition position in spec mode.
boolConversionCheck(growable);
assertUnreachable();
}
factory List._ofArray(Iterable<E> elements) {
return JSArray<E>.markGrowable(
JS('effects:none;depends:no-static', '#.slice(0)', elements));
}
factory List._of(Iterable<E> elements) {
if (elements is JSArray) return List._ofArray(elements);
// This is essentially `<E>[]..addAll(elements)`, but without a check for
// modifiability or ConcurrentModificationError on the receiver.
List<E> list = <E>[];
for (final e in elements) {
list.add(e);
}
return list;
}
factory List._fixedOf(Iterable<E> elements) {
return makeListFixedLength(List._of(elements));
}
@patch
factory List.generate(int length, E generator(int index),
{bool growable = true}) {
final result = growable
? new JSArray<E>.growable(length)
: new JSArray<E>.fixed(length);
for (int i = 0; i < length; i++) {
result[i] = generator(i);
}
return result;
}
@patch
factory List.unmodifiable(Iterable elements) {
var result = List<E>.from(elements, growable: false);
return makeFixedListUnmodifiable(result);
}
}
@patch
class Map<K, V> {
@patch
factory Map.unmodifiable(Map other) = ConstantMap<K, V>.from;
@patch
factory Map() = JsLinkedHashMap<K, V>;
}
@patch
class String {
@patch
factory String.fromCharCodes(Iterable<int> charCodes,
[int start = 0, int? end]) {
if (charCodes is JSArray) {
// Type promotion doesn't work unless the check is `is JSArray<int>`,
// which is more expensive.
// TODO(41383): Optimize `is JSArray<int>` rather than do weird 'casts'.
JSArray array = JS('JSArray', '#', charCodes);
return _stringFromJSArray(array, start, end);
}
if (charCodes is NativeUint8List) {
return _stringFromUint8List(charCodes, start, end);
}
return _stringFromIterable(charCodes, start, end);
}
@patch
factory String.fromCharCode(int charCode) {
return Primitives.stringFromCharCode(charCode);
}
static String _stringFromJSArray(JSArray list, int start, int? endOrNull) {
int len = list.length;
int end = RangeError.checkValidRange(start, endOrNull, len);
if (start > 0 || end < len) {
list = JS('JSArray', '#.slice(#, #)', list, start, end);
}
return Primitives.stringFromCharCodes(list);
}
static String _stringFromUint8List(
NativeUint8List charCodes, int start, int? endOrNull) {
int len = charCodes.length;
int end = RangeError.checkValidRange(start, endOrNull, len);
return Primitives.stringFromNativeUint8List(charCodes, start, end);
}
static String _stringFromIterable(
Iterable<int> charCodes, int start, int? end) {
if (start < 0) throw new RangeError.range(start, 0, charCodes.length);
if (end != null && end < start) {
throw new RangeError.range(end, start, charCodes.length);
}
var it = charCodes.iterator;
for (int i = 0; i < start; i++) {
if (!it.moveNext()) {
throw new RangeError.range(start, 0, i);
}
}
var list = [];
if (end == null) {
while (it.moveNext()) list.add(it.current);
} else {
for (int i = start; i < end; i++) {
if (!it.moveNext()) {
throw new RangeError.range(end, start, i);
}
list.add(it.current);
}
}
return Primitives.stringFromCharCodes(list);
}
}
@patch
class bool {
@patch
int get hashCode => super.hashCode;
}
@patch
class RegExp {
@pragma('dart2js:noInline')
@patch
factory RegExp(String source,
{bool multiLine: false,
bool caseSensitive: true,
bool unicode: false,
bool dotAll: false}) =>
new JSSyntaxRegExp(source,
multiLine: multiLine,
caseSensitive: caseSensitive,
unicode: unicode,
dotAll: dotAll);
@patch
static String escape(String text) => quoteStringForRegExp(text);
}
// Patch for 'identical' function.
@pragma(
'dart2js:noInline') // No inlining since we recognize the call in optimizer.
@patch
bool identical(Object? a, Object? b) {
return JS('bool', '(# == null ? # == null : # === #)', a, b, a, b);
}
@patch
class StringBuffer {
String _contents;
@patch
StringBuffer([Object content = ""]) : _contents = '$content';
@patch
int get length => _contents.length;
@patch
void write(Object? obj) {
_writeString('$obj');
}
@patch
void writeCharCode(int charCode) {
_writeString(new String.fromCharCode(charCode));
}
@patch
void writeAll(Iterable<dynamic> objects, [String separator = ""]) {
_contents = _writeAll(_contents, objects, separator);
}
@patch
void writeln([Object? obj = ""]) {
_writeString('$obj\n');
}
@patch
void clear() {
_contents = "";
}
@patch
String toString() => Primitives.flattenString(_contents);
void _writeString(String str) {
_contents = Primitives.stringConcatUnchecked(_contents, str);
}
static String _writeAll(String string, Iterable objects, String separator) {
Iterator iterator = objects.iterator;
if (!iterator.moveNext()) return string;
if (separator.isEmpty) {
do {
string = _writeOne(string, iterator.current);
} while (iterator.moveNext());
} else {
string = _writeOne(string, iterator.current);
while (iterator.moveNext()) {
string = _writeOne(string, separator);
string = _writeOne(string, iterator.current);
}
}
return string;
}
static String _writeOne(String string, Object? obj) {
return Primitives.stringConcatUnchecked(string, '$obj');
}
}
@patch
class NoSuchMethodError {
final Object? _receiver;
final Symbol _memberName;
final List? _arguments;
final Map<Symbol, dynamic>? _namedArguments;
final List? _existingArgumentNames;
@patch
factory NoSuchMethodError.withInvocation(
Object? receiver, Invocation invocation) =>
NoSuchMethodError(receiver, invocation.memberName,
invocation.positionalArguments, invocation.namedArguments);
@patch
NoSuchMethodError(Object? receiver, Symbol memberName,
List? positionalArguments, Map<Symbol, dynamic>? namedArguments,
[List? existingArgumentNames = null])
: _receiver = receiver,
_memberName = memberName,
_arguments = positionalArguments,
_namedArguments = namedArguments,
_existingArgumentNames = existingArgumentNames;
@patch
String toString() {
StringBuffer sb = StringBuffer('');
String comma = '';
var arguments = _arguments;
if (arguments != null) {
for (var argument in arguments) {
sb.write(comma);
sb.write(Error.safeToString(argument));
comma = ', ';
}
}
var namedArguments = _namedArguments;
if (namedArguments != null) {
namedArguments.forEach((Symbol key, var value) {
sb.write(comma);
sb.write(_symbolToString(key));
sb.write(": ");
sb.write(Error.safeToString(value));
comma = ', ';
});
}
String memberName = _symbolToString(_memberName);
String receiverText = Error.safeToString(_receiver);
String actualParameters = '$sb';
var existingArgumentNames = _existingArgumentNames;
if (existingArgumentNames == null) {
return "NoSuchMethodError: method not found: '$memberName'\n"
"Receiver: ${receiverText}\n"
"Arguments: [$actualParameters]";
} else {
String formalParameters = existingArgumentNames.join(', ');
return "NoSuchMethodError: incorrect number of arguments passed to "
"method named '$memberName'\n"
"Receiver: ${receiverText}\n"
"Tried calling: $memberName($actualParameters)\n"
"Found: $memberName($formalParameters)";
}
}
}
class _CompileTimeError extends Error {
final String _errorMsg;
// TODO(sigmund): consider calling `JS('', 'debugger')`.
_CompileTimeError(this._errorMsg);
String toString() => _errorMsg;
}
@patch
class Uri {
@patch
static Uri get base {
String? uri = Primitives.currentUri();
if (uri != null) return Uri.parse(uri);
throw new UnsupportedError("'Uri.base' is not supported");
}
}
@patch
class _Uri {
@patch
static bool get _isWindows => _isWindowsCached;
static final bool _isWindowsCached = JS(
'bool',
'typeof process != "undefined" && '
'Object.prototype.toString.call(process) == "[object process]" && '
'process.platform == "win32"');
// Matches a String that _uriEncodes to itself regardless of the kind of
// component. This corresponds to [_unreservedTable], i.e. characters that
// are not encoded by any encoding table.
static final RegExp _needsNoEncoding = new RegExp(r'^[\-\.0-9A-Z_a-z~]*$');
/// This is the internal implementation of JavaScript's encodeURI function.
/// It encodes all characters in the string [text] except for those
/// that appear in [canonicalTable], and returns the escaped string.
@patch
static String _uriEncode(List<int> canonicalTable, String text,
Encoding encoding, bool spaceToPlus) {
if (identical(encoding, utf8) && _needsNoEncoding.hasMatch(text)) {
return text;
}
// Encode the string into bytes then generate an ASCII only string
// by percent encoding selected bytes.
StringBuffer result = new StringBuffer('');
var bytes = encoding.encode(text);
for (int i = 0; i < bytes.length; i++) {
int byte = bytes[i];
if (byte < 128 &&
((canonicalTable[byte >> 4] & (1 << (byte & 0x0f))) != 0)) {
result.writeCharCode(byte);
} else if (spaceToPlus && byte == _SPACE) {
result.write('+');
} else {
const String hexDigits = '0123456789ABCDEF';
result.write('%');
result.write(hexDigits[(byte >> 4) & 0x0f]);
result.write(hexDigits[byte & 0x0f]);
}
}
return result.toString();
}
}
bool _hasErrorStackProperty = JS('bool', 'new Error().stack != void 0');
@patch
class StackTrace {
@patch
@pragma('dart2js:noInline')
static StackTrace get current {
if (_hasErrorStackProperty) {
return getTraceFromException(JS('', 'new Error()'));
}
// Fallback if new Error().stack does not exist.
// Currently only required for IE 11.
try {
throw '';
} catch (_, stackTrace) {
return stackTrace;
}
}
}
// Called from kernel generated code.
_malformedTypeError(message) => new RuntimeError(message);
// Called from kernel generated code.
_genericNoSuchMethod(receiver, memberName, positionalArguments, namedArguments,
existingArguments) {
return new NoSuchMethodError(
receiver, memberName, positionalArguments, namedArguments);
}
// Called from kernel generated code.
_unresolvedConstructorError(receiver, memberName, positionalArguments,
namedArguments, existingArguments) {
// TODO(sra): Generate an error that reads:
//
// No constructor '$memberName' declared in class '$receiver'.
return new NoSuchMethodError(
receiver, memberName, positionalArguments, namedArguments);
}
// Called from kernel generated code.
_unresolvedStaticGetterError(receiver, memberName, positionalArguments,
namedArguments, existingArguments) {
// TODO(sra): Generate customized message.
return new NoSuchMethodError(
receiver, memberName, positionalArguments, namedArguments);
}
// Called from kernel generated code.
_unresolvedStaticSetterError(receiver, memberName, positionalArguments,
namedArguments, existingArguments) {
// TODO(sra): Generate customized message.
return new NoSuchMethodError(
receiver, memberName, positionalArguments, namedArguments);
}
// Called from kernel generated code.
_unresolvedStaticMethodError(receiver, memberName, positionalArguments,
namedArguments, existingArguments) {
// TODO(sra): Generate customized message.
return new NoSuchMethodError(
receiver, memberName, positionalArguments, namedArguments);
}
// Called from kernel generated code.
_unresolvedTopLevelGetterError(receiver, memberName, positionalArguments,
namedArguments, existingArguments) {
// TODO(sra): Generate customized message.
return new NoSuchMethodError(
receiver, memberName, positionalArguments, namedArguments);
}
// Called from kernel generated code.
_unresolvedTopLevelSetterError(receiver, memberName, positionalArguments,
namedArguments, existingArguments) {
// TODO(sra): Generate customized message.
return new NoSuchMethodError(
receiver, memberName, positionalArguments, namedArguments);
}
// Called from kernel generated code.
_unresolvedTopLevelMethodError(receiver, memberName, positionalArguments,
namedArguments, existingArguments) {
// TODO(sra): Generate customized message.
return new NoSuchMethodError(
receiver, memberName, positionalArguments, namedArguments);
}
/// Used by Fasta to report a runtime error when a final field with an
/// initializer is also initialized in a generative constructor.
///
/// Note: in strong mode, this is a compile-time error and this class becomes
/// obsolete.
class _DuplicatedFieldInitializerError extends Error {
final String _name;
_DuplicatedFieldInitializerError(this._name);
toString() => "Error: field '$_name' is already initialized.";
}
// TODO(sra): The rest of this core_patch.dart source should reside in an
// included part file instead of being inlined. However, part files are not
// properly supported here.
// Copyright (c) 2017, 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.
// part of dart.core;
int _max(int a, int b) => a > b ? a : b;
int _min(int a, int b) => a < b ? a : b;
/// Empty list used as an initializer for local variables in the `_BigIntImpl`.
final _dummyList = new Uint16List(0);
// Copyright 2009 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
/*
* Copyright (c) 2003-2005 Tom Wu
* Copyright (c) 2012 Adam Singer (adam@solvr.io)
* All Rights Reserved.
*
* Permission is hereby granted, free of charge, to any person obtaining
* a copy of this software and associated documentation files (the
* "Software"), to deal in the Software without restriction, including
* without limitation the rights to use, copy, modify, merge, publish,
* distribute, sublicense, and/or sell copies of the Software, and to
* permit persons to whom the Software is furnished to do so, subject to
* the following conditions:
*
* The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS-IS" AND WITHOUT WARRANTY OF ANY KIND,
* EXPRESS, IMPLIED OR OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY
* WARRANTY OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.
*
* IN NO EVENT SHALL TOM WU BE LIABLE FOR ANY SPECIAL, INCIDENTAL,
* INDIRECT OR CONSEQUENTIAL DAMAGES OF ANY KIND, OR ANY DAMAGES WHATSOEVER
* RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER OR NOT ADVISED OF
* THE POSSIBILITY OF DAMAGE, AND ON ANY THEORY OF LIABILITY, ARISING OUT
* OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*
* In addition, the following condition applies:
*
* All redistributions must retain an intact copy of this copyright notice
* and disclaimer.
*/
/// An implementation for the arbitrarily large integer.
///
/// The integer number is represented by a sign, an array of 16-bit unsigned
/// integers in little endian format, and a number of used digits in that array.
class _BigIntImpl implements BigInt {
// Bits per digit.
static const int _digitBits = 16;
static const int _digitBase = 1 << _digitBits;
static const int _digitMask = (1 << _digitBits) - 1;
static final _BigIntImpl zero = new _BigIntImpl._fromInt(0);
static final _BigIntImpl one = new _BigIntImpl._fromInt(1);
static final _BigIntImpl two = new _BigIntImpl._fromInt(2);
static final _BigIntImpl _minusOne = -one;
static final _BigIntImpl _bigInt10000 = new _BigIntImpl._fromInt(10000);
// Result cache for last _divRem call.
// Result cache for last _divRem call.
static Uint16List? _lastDividendDigits;
static int? _lastDividendUsed;
static Uint16List? _lastDivisorDigits;
static int? _lastDivisorUsed;
static late Uint16List _lastQuoRemDigits;
static late int _lastQuoRemUsed;
static late int _lastRemUsed;
static late int _lastRem_nsh;
/// Whether this bigint is negative.
final bool _isNegative;
/// The unsigned digits of this bigint.
///
/// The least significant digit is in slot 0.
/// The list may have more digits than needed. That is, `_digits.length` may
/// be strictly greater than `_used`.
final Uint16List _digits;
/// The number of used entries in [_digits].
///
/// To avoid reallocating [Uint16List]s, lists that are too big are not
/// replaced.
final int _used;
/// Parses [source] as a, possibly signed, integer literal and returns its
/// value.
///
/// The [source] must be a non-empty sequence of base-[radix] digits,
/// optionally prefixed with a minus or plus sign ('-' or '+').
///
/// The [radix] must be in the range 2..36. The digits used are
/// first the decimal digits 0..9, and then the letters 'a'..'z' with
/// values 10 through 35. Also accepts upper-case letters with the same
/// values as the lower-case ones.
///
/// If no [radix] is given then it defaults to 10. In this case, the [source]
/// digits may also start with `0x`, in which case the number is interpreted
/// as a hexadecimal literal, which effectively means that the `0x` is ignored
/// and the radix is instead set to 16.
///
/// For any int `n` and radix `r`, it is guaranteed that
/// `n == int.parse(n.toRadixString(r), radix: r)`.
///
/// Throws a [FormatException] if the [source] is not a valid integer literal,
/// optionally prefixed by a sign.
static _BigIntImpl parse(String source, {int? radix}) {
var result = _tryParse(source, radix: radix);
if (result == null) {
throw new FormatException("Could not parse BigInt", source);
}
return result;
}
/// Parses a decimal bigint literal.
///
/// The [source] must not contain leading or trailing whitespace.
static _BigIntImpl _parseDecimal(String source, bool isNegative) {
const _0 = 48;
int part = 0;
_BigIntImpl result = zero;
// Read in the source 4 digits at a time.
// The first part may have a few leading virtual '0's to make the remaining
// parts all have exactly 4 digits.
var digitInPartCount = 4 - source.length.remainder(4);
if (digitInPartCount == 4) digitInPartCount = 0;
for (int i = 0; i < source.length; i++) {
part = part * 10 + source.codeUnitAt(i) - _0;
if (++digitInPartCount == 4) {
result = result * _bigInt10000 + new _BigIntImpl._fromInt(part);
part = 0;
digitInPartCount = 0;
}
}
if (isNegative) return -result;
return result;
}
/// Returns the value of a given source digit.
///
/// Source digits between "0" and "9" (inclusive) return their decimal value.
///
/// Source digits between "a" and "z", or "A" and "Z" (inclusive) return
/// 10 + their position in the ASCII alphabet.
///
/// The incoming [codeUnit] must be an ASCII code-unit.
static int _codeUnitToRadixValue(int codeUnit) {
// We know that the characters must be ASCII as otherwise the
// regexp wouldn't have matched. Lowercasing by doing `| 0x20` is thus
// guaranteed to be a safe operation, since it preserves digits
// and lower-cases ASCII letters.
const int _0 = 48;
const int _9 = 57;
const int _a = 97;
if (_0 <= codeUnit && codeUnit <= _9) return codeUnit - _0;
codeUnit |= 0x20;
var result = codeUnit - _a + 10;
return result;
}
/// Parses the given [source] string, starting at [startPos], as a hex
/// literal.
///
/// If [isNegative] is true, negates the result before returning it.
///
/// The [source] (substring) must be a valid hex literal.
static _BigIntImpl? _parseHex(String source, int startPos, bool isNegative) {
int hexDigitsPerChunk = _digitBits ~/ 4;
int sourceLength = source.length - startPos;
int chunkCount = (sourceLength / hexDigitsPerChunk).ceil();
var digits = new Uint16List(chunkCount);
int lastDigitLength = sourceLength - (chunkCount - 1) * hexDigitsPerChunk;
int digitIndex = digits.length - 1;
int i = startPos;
int chunk = 0;
for (int j = 0; j < lastDigitLength; j++) {
var digitValue = _codeUnitToRadixValue(source.codeUnitAt(i++));
if (digitValue >= 16) return null;
chunk = chunk * 16 + digitValue;
}
digits[digitIndex--] = chunk;
while (i < source.length) {
chunk = 0;
for (int j = 0; j < hexDigitsPerChunk; j++) {
var digitValue = _codeUnitToRadixValue(source.codeUnitAt(i++));
if (digitValue >= 16) return null;
chunk = chunk * 16 + digitValue;
}
digits[digitIndex--] = chunk;
}
if (digits.length == 1 && digits[0] == 0) return zero;
return new _BigIntImpl._(isNegative, digits.length, digits);
}
/// Parses the given [source] as a [radix] literal.
///
/// The [source] will be checked for invalid characters. If it is invalid,
/// this function returns `null`.
static _BigIntImpl? _parseRadix(String source, int radix, bool isNegative) {
var result = zero;
var base = new _BigIntImpl._fromInt(radix);
for (int i = 0; i < source.length; i++) {
var digitValue = _codeUnitToRadixValue(source.codeUnitAt(i));
if (digitValue >= radix) return null;
result = result * base + new _BigIntImpl._fromInt(digitValue);
}
if (isNegative) return -result;
return result;
}
/// Tries to parse the given [source] as a [radix] literal.
///
/// Returns the parsed big integer, or `null` if it failed.
///
/// If the [radix] is `null` accepts decimal literals or `0x` hex literals.
static _BigIntImpl? _tryParse(String source, {int? radix}) {
if (source == "") return null;
var match = _parseRE.firstMatch(source);
int signIndex = 1;
int hexIndex = 3;
int decimalIndex = 4;
int nonDecimalHexIndex = 5;
if (match == null) return null;
bool isNegative = match[signIndex] == "-";
String? decimalMatch = match[decimalIndex];
String? hexMatch = match[hexIndex];
String? nonDecimalMatch = match[nonDecimalHexIndex];
if (radix == null) {
if (decimalMatch != null) {
// Cannot fail because we know that the digits are all decimal.
return _parseDecimal(decimalMatch, isNegative);
}
if (hexMatch != null) {
// Cannot fail because we know that the digits are all hex.
return _parseHex(hexMatch, 2, isNegative);
}
return null;
}
if (radix is! int) {
throw new ArgumentError.value(radix, 'radix', 'is not an integer');
}
if (radix < 2 || radix > 36) {
throw new RangeError.range(radix, 2, 36, 'radix');
}
if (radix == 10 && decimalMatch != null) {
return _parseDecimal(decimalMatch, isNegative);
}
if (radix == 16 && (decimalMatch != null || nonDecimalMatch != null)) {
return _parseHex(decimalMatch ?? nonDecimalMatch!, 0, isNegative);
}
return _parseRadix(
decimalMatch ?? nonDecimalMatch ?? hexMatch!, radix, isNegative);
}
static RegExp _parseRE = RegExp(
r'^\s*([+-]?)((0x[a-f0-9]+)|(\d+)|([a-z0-9]+))\s*$',
caseSensitive: false);
/// Finds the amount significant digits in the provided [digits] array.
static int _normalize(int used, Uint16List digits) {
while (used > 0 && digits[used - 1] == 0) used--;
return 0 + used; // force inferred result to be non-null.
}
/// Factory returning an instance initialized with the given field values.
/// If the [digits] array contains leading 0s, the [used] value is adjusted
/// accordingly. The [digits] array is not modified.
_BigIntImpl._(bool isNegative, int used, Uint16List digits)
: this._normalized(isNegative, _normalize(used, digits), digits);
_BigIntImpl._normalized(bool isNegative, this._used, this._digits)
: _isNegative = _used == 0 ? false : isNegative;
/// Whether this big integer is zero.
bool get _isZero => _used == 0;
/// Allocates an array of the given [length] and copies the [digits] in the
/// range [from] to [to-1], starting at index 0, followed by leading zero
/// digits.
static Uint16List _cloneDigits(
Uint16List digits, int from, int to, int length) {
var resultDigits = new Uint16List(length);
var n = to - from;
for (var i = 0; i < n; i++) {
resultDigits[i] = digits[from + i];
}
return resultDigits;
}
/// Allocates a big integer from the provided [value] number.
factory _BigIntImpl.from(num value) {
if (value == 0) return zero;
if (value == 1) return one;
if (value == 2) return two;
// Given this order dart2js will use the `_fromInt` for smaller value and
// then use the bit-manipulating `_fromDouble` for all other values.
if (value.abs() < 0x100000000)
return new _BigIntImpl._fromInt(value.toInt());
if (value is double) return new _BigIntImpl._fromDouble(value);
return new _BigIntImpl._fromInt(value as int);
}
factory _BigIntImpl._fromInt(int value) {
bool isNegative = value < 0;
assert(_digitBits == 16);
if (isNegative) {
// Handle the min 64-bit value differently, since its negation is not
// positive.
const int minInt64 = -0x80000000 * 0x100000000;
if (value == minInt64) {
var digits = new Uint16List(4);
digits[3] = 0x8000;
return new _BigIntImpl._(true, 4, digits);
}
value = -value;
}
if (value < _digitBase) {
var digits = new Uint16List(1);
digits[0] = value;
return new _BigIntImpl._(isNegative, 1, digits);
}
if (value <= 0xFFFFFFFF) {
var digits = new Uint16List(2);
digits[0] = value & _digitMask;
digits[1] = value >> _digitBits;
return new _BigIntImpl._(isNegative, 2, digits);
}
var bits = value.bitLength;
var digits = new Uint16List((bits - 1) ~/ _digitBits + 1);
var i = 0;
while (value != 0) {
digits[i++] = value & _digitMask;
value = value ~/ _digitBase;
}
return new _BigIntImpl._(isNegative, digits.length, digits);
}
/// An 8-byte Uint8List we can reuse for [_fromDouble] to avoid generating
/// garbage.
static final Uint8List _bitsForFromDouble = new Uint8List(8);
factory _BigIntImpl._fromDouble(double value) {
const int exponentBias = 1075;
if (value.isNaN || value.isInfinite) {
throw new ArgumentError("Value must be finite: $value");
}
bool isNegative = value < 0;
if (isNegative) value = -value;
value = value.floorToDouble();
if (value == 0) return zero;
var bits = _bitsForFromDouble;
for (int i = 0; i < 8; i++) {
bits[i] = 0;
}
bits.buffer.asByteData().setFloat64(0, value, Endian.little);
// The exponent is in bits 53..63.
var biasedExponent = (bits[7] << 4) + (bits[6] >> 4);
var exponent = biasedExponent - exponentBias;
assert(_digitBits == 16);
// The significant bits are in 0 .. 52.
var unshiftedDigits = new Uint16List(4);
unshiftedDigits[0] = (bits[1] << 8) + bits[0];
unshiftedDigits[1] = (bits[3] << 8) + bits[2];
unshiftedDigits[2] = (bits[5] << 8) + bits[4];
// Don't forget to add the hidden bit.
unshiftedDigits[3] = 0x10 | (bits[6] & 0xF);
var unshiftedBig = new _BigIntImpl._normalized(false, 4, unshiftedDigits);
_BigIntImpl absResult = unshiftedBig;
if (exponent < 0) {
absResult = unshiftedBig >> -exponent;
} else if (exponent > 0) {
absResult = unshiftedBig << exponent;
}
if (isNegative) return -absResult;
return absResult;
}
/// Return the negative value of this integer.
///
/// The result of negating an integer always has the opposite sign, except
/// for zero, which is its own negation.
_BigIntImpl operator -() {
if (_used == 0) return this;
return new _BigIntImpl._(!_isNegative, _used, _digits);
}
/// Returns the absolute value of this integer.
///
/// For any integer `x`, the result is the same as `x < 0 ? -x : x`.
_BigIntImpl abs() => _isNegative ? -this : this;
/// Returns this << n *_DIGIT_BITS.
_BigIntImpl _dlShift(int n) {
final used = _used;
if (used == 0) {
return zero;
}
final resultUsed = used + n;
final digits = _digits;
final resultDigits = new Uint16List(resultUsed);
for (int i = used - 1; i >= 0; i--) {
resultDigits[i + n] = digits[i];
}
return new _BigIntImpl._(_isNegative, resultUsed, resultDigits);
}
/// Same as [_dlShift] but works on the decomposed big integers.
///
/// Returns `resultUsed`.
///
/// `resultDigits[0..resultUsed-1] = xDigits[0..xUsed-1] << n*_DIGIT_BITS`.
static int _dlShiftDigits(
Uint16List xDigits, int xUsed, int n, Uint16List resultDigits) {
if (xUsed == 0) {
return 0;
}
if (n == 0 && identical(resultDigits, xDigits)) {
return xUsed;
}
final resultUsed = xUsed + n;
for (int i = xUsed - 1; i >= 0; i--) {
resultDigits[i + n] = xDigits[i];
}
for (int i = n - 1; i >= 0; i--) {
resultDigits[i] = 0;
}
return resultUsed;
}
/// Returns `this >> n*_DIGIT_BITS`.
_BigIntImpl _drShift(int n) {
final used = _used;
if (used == 0) {
return zero;
}
final resultUsed = used - n;
if (resultUsed <= 0) {
return _isNegative ? _minusOne : zero;
}
final digits = _digits;
final resultDigits = new Uint16List(resultUsed);
for (var i = n; i < used; i++) {
resultDigits[i - n] = digits[i];
}
final result = new _BigIntImpl._(_isNegative, resultUsed, resultDigits);
if (_isNegative) {
// Round down if any bit was shifted out.
for (var i = 0; i < n; i++) {
if (digits[i] != 0) {
return result - one;
}
}
}
return result;
}
/// Shifts the digits of [xDigits] into the right place in [resultDigits].
///
/// `resultDigits[ds..xUsed+ds] = xDigits[0..xUsed-1] << (n % _DIGIT_BITS)`
/// where `ds = n ~/ _DIGIT_BITS`
///
/// Does *not* clear digits below ds.
static void _lsh(
Uint16List xDigits, int xUsed, int n, Uint16List resultDigits) {
assert(xUsed > 0);
final digitShift = n ~/ _digitBits;
final bitShift = n % _digitBits;
final carryBitShift = _digitBits - bitShift;
final bitMask = (1 << carryBitShift) - 1;
var carry = 0;
for (int i = xUsed - 1; i >= 0; i--) {
final digit = xDigits[i];
resultDigits[i + digitShift + 1] = (digit >> carryBitShift) | carry;
carry = (digit & bitMask) << bitShift;
}
resultDigits[digitShift] = carry;
}
/// Shift the bits of this integer to the left by [shiftAmount].
///
/// Shifting to the left makes the number larger, effectively multiplying
/// the number by `pow(2, shiftIndex)`.
///
/// There is no limit on the size of the result. It may be relevant to
/// limit intermediate values by using the "and" operator with a suitable
/// mask.
///
/// It is an error if [shiftAmount] is negative.
_BigIntImpl operator <<(int shiftAmount) {
if (shiftAmount < 0) {
throw new ArgumentError("shift-amount must be posititve $shiftAmount");
}
if (_isZero) return this;
final digitShift = shiftAmount ~/ _digitBits;
final bitShift = shiftAmount % _digitBits;
if (bitShift == 0) {
return _dlShift(digitShift);
}
var resultUsed = _used + digitShift + 1;
var resultDigits = new Uint16List(resultUsed);
_lsh(_digits, _used, shiftAmount, resultDigits);
return new _BigIntImpl._(_isNegative, resultUsed, resultDigits);
}
// resultDigits[0..resultUsed-1] = xDigits[0..xUsed-1] << n.
// Returns resultUsed.
static int _lShiftDigits(
Uint16List xDigits, int xUsed, int n, Uint16List resultDigits) {
final digitsShift = n ~/ _digitBits;
final bitShift = n % _digitBits;
if (bitShift == 0) {
return _dlShiftDigits(xDigits, xUsed, digitsShift, resultDigits);
}
var resultUsed = xUsed + digitsShift + 1;
_lsh(xDigits, xUsed, n, resultDigits);
var i = digitsShift;
while (--i >= 0) {
resultDigits[i] = 0;
}
if (resultDigits[resultUsed - 1] == 0) {
resultUsed--; // Clamp result.
}
return resultUsed;
}
// resultDigits[0..resultUsed-1] = xDigits[0..xUsed-1] >> n.
static void _rsh(
Uint16List xDigits, int xUsed, int n, Uint16List resultDigits) {
assert(xUsed > 0);
final digitsShift = n ~/ _digitBits;
final bitShift = n % _digitBits;
final carryBitShift = _digitBits - bitShift;
final bitMask = (1 << bitShift) - 1;
var carry = xDigits[digitsShift] >> bitShift;
final last = xUsed - digitsShift - 1;
for (var i = 0; i < last; i++) {
final digit = xDigits[i + digitsShift + 1];
resultDigits[i] = ((digit & bitMask) << carryBitShift) | carry;
carry = digit >> bitShift;
}
resultDigits[last] = carry;
}
/// Shift the bits of this integer to the right by [shiftAmount].
///
/// Shifting to the right makes the number smaller and drops the least
/// significant bits, effectively doing an integer division by
/// `pow(2, shiftIndex)`.
///
/// It is an error if [shiftAmount] is negative.
_BigIntImpl operator >>(int shiftAmount) {
if (shiftAmount < 0) {
throw new ArgumentError("shift-amount must be posititve $shiftAmount");
}
if (_isZero) return this;
final digitShift = shiftAmount ~/ _digitBits;
final bitShift = shiftAmount % _digitBits;
if (bitShift == 0) {
return _drShift(digitShift);
}
final used = _used;
final resultUsed = used - digitShift;
if (resultUsed <= 0) {
return _isNegative ? _minusOne : zero;
}
final digits = _digits;
final resultDigits = new Uint16List(resultUsed);
_rsh(digits, used, shiftAmount, resultDigits);
final result = new _BigIntImpl._(_isNegative, resultUsed, resultDigits);
if (_isNegative) {
// Round down if any bit was shifted out.
if ((digits[digitShift] & ((1 << bitShift) - 1)) != 0) {
return result - one;
}
for (var i = 0; i < digitShift; i++) {
if (digits[i] != 0) {
return result - one;
}
}
}
return result;
}
/// Compares this to [other] taking the absolute value of both operands.
///
/// Returns 0 if abs(this) == abs(other); a positive number if
/// abs(this) > abs(other); and a negative number if abs(this) < abs(other).
int _absCompare(_BigIntImpl other) {
return _compareDigits(_digits, _used, other._digits, other._used);
}
/// Compares this to `other`.
///
/// Returns a negative number if `this` is less than `other`, zero if they are
/// equal, and a positive number if `this` is greater than `other`.
int compareTo(covariant _BigIntImpl other) {
if (_isNegative == other._isNegative) {
var result = _absCompare(other);
// Use 0 - result to avoid negative zero in JavaScript.
return _isNegative ? 0 - result : result;
}
return _isNegative ? -1 : 1;
}
/// Compares `digits[0..used-1]` with `otherDigits[0..otherUsed-1]`.
///
/// Returns 0 if equal; a positive number if larger;
/// and a negative number if smaller.
static int _compareDigits(
Uint16List digits, int used, Uint16List otherDigits, int otherUsed) {
var result = used - otherUsed;
if (result == 0) {
for (int i = used - 1; i >= 0; i--) {
result = digits[i] - otherDigits[i];
if (result != 0) return result;
}
}
return result;
}
// resultDigits[0..used] = digits[0..used-1] + otherDigits[0..otherUsed-1].
// used >= otherUsed > 0.
static void _absAdd(Uint16List digits, int used, Uint16List otherDigits,
int otherUsed, Uint16List resultDigits) {
assert(used >= otherUsed && otherUsed > 0);
var carry = 0;
for (var i = 0; i < otherUsed; i++) {
carry += digits[i] + otherDigits[i];
resultDigits[i] = carry & _digitMask;
carry >>= _digitBits;
}
for (var i = otherUsed; i < used; i++) {
carry += digits[i];
resultDigits[i] = carry & _digitMask;
carry >>= _digitBits;
}
resultDigits[used] = carry;
}
// resultDigits[0..used-1] = digits[0..used-1] - otherDigits[0..otherUsed-1].
// used >= otherUsed > 0.
static void _absSub(Uint16List digits, int used, Uint16List otherDigits,
int otherUsed, Uint16List resultDigits) {
assert(used >= otherUsed && otherUsed > 0);
var carry = 0;
for (var i = 0; i < otherUsed; i++) {
carry += digits[i] - otherDigits[i];
resultDigits[i] = carry & _digitMask;
// Dart2js only supports unsigned shifts.
// Since the carry can only be -1 or 0 use this hack.
carry = 0 - ((carry >> _digitBits) & 1);
}
for (var i = otherUsed; i < used; i++) {
carry += digits[i];
resultDigits[i] = carry & _digitMask;
// Dart2js only supports unsigned shifts.
// Since the carry can only be -1 or 0 use this hack.
carry = 0 - ((carry >> _digitBits) & 1);
}
}
/// Returns `abs(this) + abs(other)` with sign set according to [isNegative].
_BigIntImpl _absAddSetSign(_BigIntImpl other, bool isNegative) {
var used = _used;
var otherUsed = other._used;
if (used < otherUsed) {
return other._absAddSetSign(this, isNegative);
}
if (used == 0) {
assert(!isNegative);
return zero;
}
if (otherUsed == 0) {
return _isNegative == isNegative ? this : -this;
}
var resultUsed = used + 1;
var resultDigits = new Uint16List(resultUsed);
_absAdd(_digits, used, other._digits, otherUsed, resultDigits);
return new _BigIntImpl._(isNegative, resultUsed, resultDigits);
}
/// Returns `abs(this) - abs(other)` with sign set according to [isNegative].
///
/// Requirement: `abs(this) >= abs(other)`.
_BigIntImpl _absSubSetSign(_BigIntImpl other, bool isNegative) {
assert(_absCompare(other) >= 0);
var used = _used;
if (used == 0) {
assert(!isNegative);
return zero;
}
var otherUsed = other._used;
if (otherUsed == 0) {
return _isNegative == isNegative ? this : -this;
}
var resultDigits = new Uint16List(used);
_absSub(_digits, used, other._digits, otherUsed, resultDigits);
return new _BigIntImpl._(isNegative, used, resultDigits);
}
/// Returns `abs(this) & abs(other)` with sign set according to [isNegative].
_BigIntImpl _absAndSetSign(_BigIntImpl other, bool isNegative) {
var resultUsed = _min(_used, other._used);
var digits = _digits;
var otherDigits = other._digits;
var resultDigits = new Uint16List(resultUsed);
for (var i = 0; i < resultUsed; i++) {
resultDigits[i] = digits[i] & otherDigits[i];
}
return new _BigIntImpl._(isNegative, resultUsed, resultDigits);
}
/// Returns `abs(this) &~ abs(other)` with sign set according to [isNegative].
_BigIntImpl _absAndNotSetSign(_BigIntImpl other, bool isNegative) {
var resultUsed = _used;
var digits = _digits;
var otherDigits = other._digits;
var resultDigits = new Uint16List(resultUsed);
var m = _min(resultUsed, other._used);
for (var i = 0; i < m; i++) {
resultDigits[i] = digits[i] & ~otherDigits[i];
}
for (var i = m; i < resultUsed; i++) {
resultDigits[i] = digits[i];
}
return new _BigIntImpl._(isNegative, resultUsed, resultDigits);
}
/// Returns `abs(this) | abs(other)` with sign set according to [isNegative].
_BigIntImpl _absOrSetSign(_BigIntImpl other, bool isNegative) {
var used = _used;
var otherUsed = other._used;
var resultUsed = _max(used, otherUsed);
var digits = _digits;
var otherDigits = other._digits;
var resultDigits = new Uint16List(resultUsed);
_BigIntImpl l;
int m;
if (used < otherUsed) {
l = other;
m = used;
} else {
l = this;
m = otherUsed;
}
for (var i = 0; i < m; i++) {
resultDigits[i] = digits[i] | otherDigits[i];
}
var lDigits = l._digits;
for (var i = m; i < resultUsed; i++) {
resultDigits[i] = lDigits[i];
}
return new _BigIntImpl._(isNegative, resultUsed, resultDigits);
}
/// Returns `abs(this) ^ abs(other)` with sign set according to [isNegative].
_BigIntImpl _absXorSetSign(_BigIntImpl other, bool isNegative) {
var used = _used;
var otherUsed = other._used;
var resultUsed = _max(used, otherUsed);
var digits = _digits;
var otherDigits = other._digits;
var resultDigits = new Uint16List(resultUsed);
_BigIntImpl l;
int m;
if (used < otherUsed) {
l = other;
m = used;
} else {
l = this;
m = otherUsed;
}
for (var i = 0; i < m; i++) {
resultDigits[i] = digits[i] ^ otherDigits[i];
}
var lDigits = l._digits;
for (var i = m; i < resultUsed; i++) {
resultDigits[i] = lDigits[i];
}
return new _BigIntImpl._(isNegative, resultUsed, resultDigits);
}
/// Bit-wise and operator.
///
/// Treating both `this` and [other] as sufficiently large two's component
/// integers, the result is a number with only the bits set that are set in
/// both `this` and [other]
///
/// Of both operands are negative, the result is negative, otherwise
/// the result is non-negative.
_BigIntImpl operator &(covariant _BigIntImpl other) {
if (_isZero || other._isZero) return zero;
if (_isNegative == other._isNegative) {
if (_isNegative) {
// (-this) & (-other) == ~(this-1) & ~(other-1)
// == ~((this-1) | (other-1))
// == -(((this-1) | (other-1)) + 1)
_BigIntImpl this1 = _absSubSetSign(one, true);
_BigIntImpl other1 = other._absSubSetSign(one, true);
// Result cannot be zero if this and other are negative.
return this1._absOrSetSign(other1, true)._absAddSetSign(one, true);
}
return _absAndSetSign(other, false);
}
// _isNegative != other._isNegative
_BigIntImpl p, n;
if (_isNegative) {
p = other;
n = this;
} else {
// & is symmetric.
p = this;
n = other;
}
// p & (-n) == p & ~(n-1) == p &~ (n-1)
var n1 = n._absSubSetSign(one, false);
return p._absAndNotSetSign(n1, false);
}
/// Bit-wise or operator.
///
/// Treating both `this` and [other] as sufficiently large two's component
/// integers, the result is a number with the bits set that are set in either
/// of `this` and [other]
///
/// If both operands are non-negative, the result is non-negative,
/// otherwise the result us negative.
_BigIntImpl operator |(covariant _BigIntImpl other) {
if (_isZero) return other;
if (other._isZero) return this;
if (_isNegative == other._isNegative) {
if (_isNegative) {
// (-this) | (-other) == ~(this-1) | ~(other-1)
// == ~((this-1) & (other-1))
// == -(((this-1) & (other-1)) + 1)
var this1 = _absSubSetSign(one, true);
var other1 = other._absSubSetSign(one, true);
// Result cannot be zero if this and a are negative.
return this1._absAndSetSign(other1, true)._absAddSetSign(one, true);
}
return _absOrSetSign(other, false);
}
// _neg != a._neg
_BigIntImpl p, n;
if (_isNegative) {
p = other;
n = this;
} else {
// | is symmetric.
p = this;
n = other;
}
// p | (-n) == p | ~(n-1) == ~((n-1) &~ p) == -(~((n-1) &~ p) + 1)
var n1 = n._absSubSetSign(one, true);
// Result cannot be zero if only one of this or a is negative.
return n1._absAndNotSetSign(p, true)._absAddSetSign(one, true);
}
/// Bit-wise exclusive-or operator.
///
/// Treating both `this` and [other] as sufficiently large two's component
/// integers, the result is a number with the bits set that are set in one,
/// but not both, of `this` and [other]
///
/// If the operands have the same sign, the result is non-negative,
/// otherwise the result is negative.
_BigIntImpl operator ^(covariant _BigIntImpl other) {
if (_isZero) return other;
if (other._isZero) return this;
if (_isNegative == other._isNegative) {
if (_isNegative) {
// (-this) ^ (-other) == ~(this-1) ^ ~(other-1) == (this-1) ^ (other-1)
var this1 = _absSubSetSign(one, true);
var other1 = other._absSubSetSign(one, true);
return this1._absXorSetSign(other1, false);
}
return _absXorSetSign(other, false);
}
// _isNegative != a._isNegative
_BigIntImpl p, n;
if (_isNegative) {
p = other;
n = this;
} else {
// ^ is symmetric.
p = this;
n = other;
}
// p ^ (-n) == p ^ ~(n-1) == ~(p ^ (n-1)) == -((p ^ (n-1)) + 1)
var n1 = n._absSubSetSign(one, true);
// Result cannot be zero if only one of this or a is negative.
return p._absXorSetSign(n1, true)._absAddSetSign(one, true);
}
/// The bit-wise negate operator.
///
/// Treating `this` as a sufficiently large two's component integer,
/// the result is a number with the opposite bits set.
///
/// This maps any integer `x` to `-x - 1`.
_BigIntImpl operator ~() {
if (_isZero) return _minusOne;
if (_isNegative) {
// ~(-this) == ~(~(this-1)) == this-1
return _absSubSetSign(one, false);
}
// ~this == -this-1 == -(this+1)
// Result cannot be zero if this is positive.
return _absAddSetSign(one, true);
}
/// Addition operator.
_BigIntImpl operator +(covariant _BigIntImpl other) {
if (_isZero) return other;
if (other._isZero) return this;
var isNegative = _isNegative;
if (isNegative == other._isNegative) {
// this + other == this + other
// (-this) + (-other) == -(this + other)
return _absAddSetSign(other, isNegative);
}
// this + (-other) == this - other == -(this - other)
// (-this) + other == other - this == -(this - other)
if (_absCompare(other) >= 0) {
return _absSubSetSign(other, isNegative);
}
return other._absSubSetSign(this, !isNegative);
}
/// Subtraction operator.
_BigIntImpl operator -(covariant _BigIntImpl other) {
if (_isZero) return -other;
if (other._isZero) return this;
var isNegative = _isNegative;
if (isNegative != other._isNegative) {
// this - (-other) == this + other
// (-this) - other == -(this + other)
return _absAddSetSign(other, isNegative);
}
// this - other == this - a == -(this - other)
// (-this) - (-other) == other - this == -(this - other)
if (_absCompare(other) >= 0) {
return _absSubSetSign(other, isNegative);
}
return other._absSubSetSign(this, !isNegative);
}
/// Multiplies [x] with [multiplicandDigits] and adds the result to
/// [accumulatorDigits].
///
/// The [multiplicandDigits] in the range [i] to [i]+[n]-1 are the
/// multiplicand digits.
///
/// The [acculumatorDigits] in the range [j] to [j]+[n]-1 are the accumulator
/// digits.
///
/// Adds the result of the multiplicand-digits * [x] to the accumulator.
///
/// Concretely: `accumulatorDigits[j..j+n] += x * m_digits[i..i+n-1]`.
static void _mulAdd(int x, Uint16List multiplicandDigits, int i,
Uint16List accumulatorDigits, int j, int n) {
if (x == 0) {
// No-op if x is 0.
return;
}
int c = 0;
while (--n >= 0) {
int product = x * multiplicandDigits[i++];
int combined = product + accumulatorDigits[j] + c;
accumulatorDigits[j++] = combined & _digitMask;
// Note that this works with 53 bits, as the division will not lose
// bits.
c = combined ~/ _digitBase;
}
while (c != 0) {
int l = accumulatorDigits[j] + c;
accumulatorDigits[j++] = l & _digitMask;
c = l ~/ _digitBase;
}
}
/// Multiplication operator.
_BigIntImpl operator *(covariant _BigIntImpl other) {
var used = _used;
var otherUsed = other._used;
if (used == 0 || otherUsed == 0) {
return zero;
}
var resultUsed = used + otherUsed;
var digits = _digits;
var otherDigits = other._digits;
var resultDigits = new Uint16List(resultUsed);
var i = 0;
while (i < otherUsed) {
_mulAdd(otherDigits[i], digits, 0, resultDigits, i, used);
i++;
}
return new _BigIntImpl._(
_isNegative != other._isNegative, resultUsed, resultDigits);
}
// r_digits[0..rUsed-1] = xDigits[0..xUsed-1]*otherDigits[0..otherUsed-1].
// Return resultUsed = xUsed + otherUsed.
static int _mulDigits(Uint16List xDigits, int xUsed, Uint16List otherDigits,
int otherUsed, Uint16List resultDigits) {
var resultUsed = xUsed + otherUsed;
var i = resultUsed;
assert(resultDigits.length >= i);
while (--i >= 0) {
resultDigits[i] = 0;
}
i = 0;
while (i < otherUsed) {
_mulAdd(otherDigits[i], xDigits, 0, resultDigits, i, xUsed);
i++;
}
return resultUsed;
}
/// Returns an estimate of `digits[i-1..i] ~/ topDigitDivisor`.
static int _estimateQuotientDigit(
int topDigitDivisor, Uint16List digits, int i) {
if (digits[i] == topDigitDivisor) return _digitMask;
var quotientDigit =
(digits[i] << _digitBits | digits[i - 1]) ~/ topDigitDivisor;
if (quotientDigit > _digitMask) return _digitMask;
return quotientDigit;
}
/// Returns `trunc(this / other)`, with `other != 0`.
_BigIntImpl _div(_BigIntImpl other) {
assert(other._used > 0);
if (_used < other._used) {
return zero;
}
_divRem(other);
// Return quotient, i.e.
// _lastQuoRem_digits[_lastRem_used.._lastQuoRem_used-1] with proper sign.
var lastQuo_used = _lastQuoRemUsed - _lastRemUsed;
var quo_digits = _cloneDigits(
_lastQuoRemDigits, _lastRemUsed, _lastQuoRemUsed, lastQuo_used);
var quo = new _BigIntImpl._(false, lastQuo_used, quo_digits);
if ((_isNegative != other._isNegative) && (quo._used > 0)) {
quo = -quo;
}
return quo;
}
/// Returns `this - other * trunc(this / other)`, with `other != 0`.
_BigIntImpl _rem(_BigIntImpl other) {
assert(other._used > 0);
if (_used < other._used) {
return this;
}
_divRem(other);
// Return remainder, i.e.
// denormalized _lastQuoRem_digits[0.._lastRem_used-1] with proper sign.
var remDigits =
_cloneDigits(_lastQuoRemDigits, 0, _lastRemUsed, _lastRemUsed);
var rem = new _BigIntImpl._(false, _lastRemUsed, remDigits);
if (_lastRem_nsh > 0) {
rem = rem >> _lastRem_nsh; // Denormalize remainder.
}
if (_isNegative && (rem._used > 0)) {
rem = -rem;
}
return rem;
}
/// Computes this ~/ other and this.remainder(other).
///
/// Stores the result in [_lastQuoRemDigits], [_lastQuoRemUsed] and
/// [_lastRemUsed]. The [_lastQuoRemDigits] contains the digits of *both*, the
/// quotient and the remainder.
///
/// Caches the input to avoid doing the work again when users write
/// `a ~/ b` followed by a `a % b`.
void _divRem(_BigIntImpl other) {
// Check if result is already cached.
if ((this._used == _lastDividendUsed) &&
(other._used == _lastDivisorUsed) &&
identical(this._digits, _lastDividendDigits) &&
identical(other._digits, _lastDivisorDigits)) {
return;
}
assert(_used >= other._used);
var nsh = _digitBits - other._digits[other._used - 1].bitLength;
// Concatenated positive quotient and normalized positive remainder.
// The resultDigits can have at most one more digit than the dividend.
Uint16List resultDigits;
int resultUsed;
// Normalized positive divisor.
// The normalized divisor has the most-significant bit of its most
// significant digit set.
// This makes estimating the quotient easier.
Uint16List yDigits;
int yUsed;
if (nsh > 0) {
yDigits = new Uint16List(other._used + 5);
yUsed = _lShiftDigits(other._digits, other._used, nsh, yDigits);
resultDigits = new Uint16List(_used + 5);
resultUsed = _lShiftDigits(_digits, _used, nsh, resultDigits);
} else {
yDigits = other._digits;
yUsed = other._used;
resultDigits = _cloneDigits(_digits, 0, _used, _used + 2);
resultUsed = _used;
}
var topDigitDivisor = yDigits[yUsed - 1];
var i = resultUsed;
var j = i - yUsed;
// tmpDigits is a temporary array of i (resultUsed) digits.
var tmpDigits = new Uint16List(i);
var tmpUsed = _dlShiftDigits(yDigits, yUsed, j, tmpDigits);
// Explicit first division step in case normalized dividend is larger or
// equal to shifted normalized divisor.
if (_compareDigits(resultDigits, resultUsed, tmpDigits, tmpUsed) >= 0) {
assert(i == resultUsed);
resultDigits[resultUsed++] = 1; // Quotient = 1.
// Subtract divisor from remainder.
_absSub(resultDigits, resultUsed, tmpDigits, tmpUsed, resultDigits);
} else {
// Account for possible carry in _mulAdd step.
resultDigits[resultUsed++] = 0;
}
// Negate y so we can later use _mulAdd instead of non-existent _mulSub.
var nyDigits = new Uint16List(yUsed + 2);
nyDigits[yUsed] = 1;
_absSub(nyDigits, yUsed + 1, yDigits, yUsed, nyDigits);
// nyDigits is read-only and has yUsed digits (possibly including several
// leading zeros).
// resultDigits is modified during iteration.
// resultDigits[0..yUsed-1] is the current remainder.
// resultDigits[yUsed..resultUsed-1] is the current quotient.
--i;
while (j > 0) {
var estimatedQuotientDigit =
_estimateQuotientDigit(topDigitDivisor, resultDigits, i);
j--;
_mulAdd(estimatedQuotientDigit, nyDigits, 0, resultDigits, j, yUsed);
if (resultDigits[i] < estimatedQuotientDigit) {
// Reusing the already existing tmpDigits array.
var tmpUsed = _dlShiftDigits(nyDigits, yUsed, j, tmpDigits);
_absSub(resultDigits, resultUsed, tmpDigits, tmpUsed, resultDigits);
while (resultDigits[i] < --estimatedQuotientDigit) {
_absSub(resultDigits, resultUsed, tmpDigits, tmpUsed, resultDigits);
}
}
i--;
}
// Cache result.
_lastDividendDigits = _digits;
_lastDividendUsed = _used;
_lastDivisorDigits = other._digits;
_lastDivisorUsed = other._used;
_lastQuoRemDigits = resultDigits;
_lastQuoRemUsed = resultUsed;
_lastRemUsed = yUsed;
_lastRem_nsh = nsh;
}
int get hashCode {
// This is the [Jenkins hash function][1] but using masking to keep
// values in SMI range.
//
// [1]: http://en.wikipedia.org/wiki/Jenkins_hash_function
int combine(int hash, int value) {
hash = 0x1fffffff & (hash + value);
hash = 0x1fffffff & (hash + ((0x0007ffff & hash) << 10));
return hash ^ (hash >> 6);
}
int finish(int hash) {
hash = 0x1fffffff & (hash + ((0x03ffffff & hash) << 3));
hash = hash ^ (hash >> 11);
return 0x1fffffff & (hash + ((0x00003fff & hash) << 15));
}
if (_isZero) return 6707; // Just a random number.
var hash = _isNegative ? 83585 : 429689; // Also random.
for (int i = 0; i < _used; i++) {
hash = combine(hash, _digits[i]);
}
return finish(hash);
}
/// Test whether this value is numerically equal to `other`.
///
/// If [other] is a [_BigIntImpl] returns whether the two operands have the
/// same value.
///
/// Returns false if `other` is not a [_BigIntImpl].
bool operator ==(Object other) =>
other is _BigIntImpl && compareTo(other) == 0;
/// Returns the minimum number of bits required to store this big integer.
///
/// The number of bits excludes the sign bit, which gives the natural length
/// for non-negative (unsigned) values. Negative values are complemented to
/// return the bit position of the first bit that differs from the sign bit.
///
/// To find the number of bits needed to store the value as a signed value,
/// add one, i.e. use `x.bitLength + 1`.
///
/// ```
/// x.bitLength == (-x-1).bitLength
///
/// new BigInt.from(3).bitLength == 2; // 00000011
/// new BigInt.from(2).bitLength == 2; // 00000010
/// new BigInt.from(1).bitLength == 1; // 00000001
/// new BigInt.from(0).bitLength == 0; // 00000000
/// new BigInt.from(-1).bitLength == 0; // 11111111
/// new BigInt.from(-2).bitLength == 1; // 11111110
/// new BigInt.from(-3).bitLength == 2; // 11111101
/// new BigInt.from(-4).bitLength == 2; // 11111100
/// ```
int get bitLength {
if (_used == 0) return 0;
if (_isNegative) return (~this).bitLength;
return _digitBits * (_used - 1) + _digits[_used - 1].bitLength;
}
/// Truncating division operator.
///
/// Performs a truncating integer division, where the remainder is discarded.
///
/// The remainder can be computed using the [remainder] method.
///
/// Examples:
/// ```
/// var seven = new BigInt.from(7);
/// var three = new BigInt.from(3);
/// seven ~/ three; // => 2
/// (-seven) ~/ three; // => -2
/// seven ~/ -three; // => -2
/// seven.remainder(three); // => 1
/// (-seven).remainder(three); // => -1
/// seven.remainder(-three); // => 1
/// ```
_BigIntImpl operator ~/(covariant _BigIntImpl other) {
if (other._used == 0) {
throw const IntegerDivisionByZeroException();
}
return _div(other);
}
/// Returns the remainder of the truncating division of `this` by [other].
///
/// The result `r` of this operation satisfies:
/// `this == (this ~/ other) * other + r`.
/// As a consequence the remainder `r` has the same sign as the divider
/// `this`.
_BigIntImpl remainder(covariant _BigIntImpl other) {
if (other._used == 0) {
throw const IntegerDivisionByZeroException();
}
return _rem(other);
}
/// Division operator.
double operator /(BigInt other) => this.toDouble() / other.toDouble();
/// Relational less than operator.
bool operator <(covariant _BigIntImpl other) => compareTo(other) < 0;
/// Relational less than or equal operator.
bool operator <=(covariant _BigIntImpl other) => compareTo(other) <= 0;
/// Relational greater than operator.
bool operator >(covariant _BigIntImpl other) => compareTo(other) > 0;
/// Relational greater than or equal operator.
bool operator >=(covariant _BigIntImpl other) => compareTo(other) >= 0;
/// Euclidean modulo operator.
///
/// Returns the remainder of the Euclidean division. The Euclidean division of
/// two integers `a` and `b` yields two integers `q` and `r` such that
/// `a == b * q + r` and `0 <= r < b.abs()`.
///
/// The sign of the returned value `r` is always positive.
///
/// See [remainder] for the remainder of the truncating division.
_BigIntImpl operator %(covariant _BigIntImpl other) {
if (other._used == 0) {
throw const IntegerDivisionByZeroException();
}
var result = _rem(other);
if (result._isNegative) {
if (other._isNegative) {
result = result - other;
} else {
result = result + other;
}
}
return result;
}
/// Returns the sign of this big integer.
///
/// Returns 0 for zero, -1 for values less than zero and
/// +1 for values greater than zero.
int get sign {
if (_used == 0) return 0;
return _isNegative ? -1 : 1;
}
/// Whether this big integer is even.
bool get isEven => _used == 0 || (_digits[0] & 1) == 0;
/// Whether this big integer is odd.
bool get isOdd => !isEven;
/// Whether this number is negative.
bool get isNegative => _isNegative;
_BigIntImpl pow(int exponent) {
if (exponent < 0) {
throw new ArgumentError("Exponent must not be negative: $exponent");
}
if (exponent == 0) return one;
// Exponentiation by squaring.
var result = one;
var base = this;
while (exponent != 0) {
if ((exponent & 1) == 1) {
result *= base;
}
exponent >>= 1;
// Skip unnecessary operation.
if (exponent != 0) {
base *= base;
}
}
return result;
}
/// Returns this integer to the power of [exponent] modulo [modulus].
///
/// The [exponent] must be non-negative and [modulus] must be
/// positive.
_BigIntImpl modPow(
covariant _BigIntImpl exponent, covariant _BigIntImpl modulus) {
if (exponent._isNegative) {
throw new ArgumentError("exponent must be positive: $exponent");
}
if (modulus <= zero) {
throw new ArgumentError("modulus must be strictly positive: $modulus");
}
if (exponent._isZero) return one;
final modulusUsed = modulus._used;
final modulusUsed2p4 = 2 * modulusUsed + 4;
final exponentBitlen = exponent.bitLength;
if (exponentBitlen <= 0) return one;
_BigIntReduction z = new _BigIntClassic(modulus);
var resultDigits = new Uint16List(modulusUsed2p4);
var result2Digits = new Uint16List(modulusUsed2p4);
var gDigits = new Uint16List(modulusUsed);
var gUsed = z.convert(this, gDigits);
// Initialize result with g.
// Copy leading zero if any.
for (int j = gUsed - 1; j >= 0; j--) {
resultDigits[j] = gDigits[j];
}
var resultUsed = gUsed;
int result2Used;
for (int i = exponentBitlen - 2; i >= 0; i--) {
result2Used = z.sqr(resultDigits, resultUsed, result2Digits);
if (!(exponent & (one << i))._isZero) {
resultUsed =
z.mul(result2Digits, result2Used, gDigits, gUsed, resultDigits);
} else {
// Swap result and result2.
var tmpDigits = resultDigits;
var tmpUsed = resultUsed;
resultDigits = result2Digits;
resultUsed = result2Used;
result2Digits = tmpDigits;
result2Used = tmpUsed;
}
}
return z.revert(resultDigits, resultUsed);
}
// If inv is false, returns gcd(x, y).
// If inv is true and gcd(x, y) = 1, returns d, so that c*x + d*y = 1.
// If inv is true and gcd(x, y) != 1, throws Exception("Not coprime").
static _BigIntImpl _binaryGcd(_BigIntImpl x, _BigIntImpl y, bool inv) {
var xDigits = x._digits;
var yDigits = y._digits;
var xUsed = x._used;
var yUsed = y._used;
var maxUsed = xUsed > yUsed ? xUsed : yUsed;
var unshiftedMaxUsed = maxUsed; // Keep
xDigits = _cloneDigits(xDigits, 0, xUsed, maxUsed);
yDigits = _cloneDigits(yDigits, 0, yUsed, maxUsed);
int shiftAmount = 0;
if (inv) {
if ((yUsed == 1) && (yDigits[0] == 1)) return one;
if ((yUsed == 0) || (yDigits[0].isEven && xDigits[0].isEven)) {
throw new Exception("Not coprime");
}
} else {
if (x._isZero) {
throw new ArgumentError.value(0, "this", "must not be zero");
}
if (y._isZero) {
throw new ArgumentError.value(0, "other", "must not be zero");
}
if (((xUsed == 1) && (xDigits[0] == 1)) ||
((yUsed == 1) && (yDigits[0] == 1))) return one;
while (((xDigits[0] & 1) == 0) && ((yDigits[0] & 1) == 0)) {
_rsh(xDigits, xUsed, 1, xDigits);
_rsh(yDigits, yUsed, 1, yDigits);
shiftAmount++;
}
if (shiftAmount >= _digitBits) {
var digitShiftAmount = shiftAmount ~/ _digitBits;
xUsed -= digitShiftAmount;
yUsed -= digitShiftAmount;
maxUsed -= digitShiftAmount;
}
if ((yDigits[0] & 1) == 1) {
// Swap x and y.
var tmpDigits = xDigits;
var tmpUsed = xUsed;
xDigits = yDigits;
xUsed = yUsed;
yDigits = tmpDigits;
yUsed = tmpUsed;
}
}
var uDigits = _cloneDigits(xDigits, 0, xUsed, unshiftedMaxUsed);
var vDigits =
_cloneDigits(yDigits, 0, yUsed, unshiftedMaxUsed + 2); // +2 for lsh.
final bool ac = (xDigits[0] & 1) == 0;
// Variables a, b, c, and d require one more digit.
final abcdUsed = maxUsed + 1;
final abcdLen = abcdUsed + 2; // +2 to satisfy _absAdd.
var aDigits = _dummyList;
var aIsNegative = false;
var cDigits = _dummyList;
var cIsNegative = false;
if (ac) {
aDigits = new Uint16List(abcdLen);
aDigits[0] = 1;
cDigits = new Uint16List(abcdLen);
}
var bDigits = new Uint16List(abcdLen);
var bIsNegative = false;
var dDigits = new Uint16List(abcdLen);
var dIsNegative = false;
dDigits[0] = 1;
while (true) {
while ((uDigits[0] & 1) == 0) {
_rsh(uDigits, maxUsed, 1, uDigits);
if (ac) {
if (((aDigits[0] & 1) == 1) || ((bDigits[0] & 1) == 1)) {
// a += y
if (aIsNegative) {
if ((aDigits[maxUsed] != 0) ||
(_compareDigits(aDigits, maxUsed, yDigits, maxUsed)) > 0) {
_absSub(aDigits, abcdUsed, yDigits, maxUsed, aDigits);
} else {
_absSub(yDigits, maxUsed, aDigits, maxUsed, aDigits);
aIsNegative = false;
}
} else {
_absAdd(aDigits, abcdUsed, yDigits, maxUsed, aDigits);
}
// b -= x
if (bIsNegative) {
_absAdd(bDigits, abcdUsed, xDigits, maxUsed, bDigits);
} else if ((bDigits[maxUsed] != 0) ||
(_compareDigits(bDigits, maxUsed, xDigits, maxUsed) > 0)) {
_absSub(bDigits, abcdUsed, xDigits, maxUsed, bDigits);
} else {
_absSub(xDigits, maxUsed, bDigits, maxUsed, bDigits);
bIsNegative = true;
}
}
_rsh(aDigits, abcdUsed, 1, aDigits);
} else if ((bDigits[0] & 1) == 1) {
// b -= x
if (bIsNegative) {
_absAdd(bDigits, abcdUsed, xDigits, maxUsed, bDigits);
} else if ((bDigits[maxUsed] != 0) ||
(_compareDigits(bDigits, maxUsed, xDigits, maxUsed) > 0)) {
_absSub(bDigits, abcdUsed, xDigits, maxUsed, bDigits);
} else {
_absSub(xDigits, maxUsed, bDigits, maxUsed, bDigits);
bIsNegative = true;
}
}
_rsh(bDigits, abcdUsed, 1, bDigits);
}
while ((vDigits[0] & 1) == 0) {
_rsh(vDigits, maxUsed, 1, vDigits);
if (ac) {
if (((cDigits[0] & 1) == 1) || ((dDigits[0] & 1) == 1)) {
// c += y
if (cIsNegative) {
if ((cDigits[maxUsed] != 0) ||
(_compareDigits(cDigits, maxUsed, yDigits, maxUsed) > 0)) {
_absSub(cDigits, abcdUsed, yDigits, maxUsed, cDigits);
} else {
_absSub(yDigits, maxUsed, cDigits, maxUsed, cDigits);
cIsNegative = false;
}
} else {
_absAdd(cDigits, abcdUsed, yDigits, maxUsed, cDigits);
}
// d -= x
if (dIsNegative) {
_absAdd(dDigits, abcdUsed, xDigits, maxUsed, dDigits);
} else if ((dDigits[maxUsed] != 0) ||
(_compareDigits(dDigits, maxUsed, xDigits, maxUsed) > 0)) {
_absSub(dDigits, abcdUsed, xDigits, maxUsed, dDigits);
} else {
_absSub(xDigits, maxUsed, dDigits, maxUsed, dDigits);
dIsNegative = true;
}
}
_rsh(cDigits, abcdUsed, 1, cDigits);
} else if ((dDigits[0] & 1) == 1) {
// d -= x
if (dIsNegative) {
_absAdd(dDigits, abcdUsed, xDigits, maxUsed, dDigits);
} else if ((dDigits[maxUsed] != 0) ||
(_compareDigits(dDigits, maxUsed, xDigits, maxUsed) > 0)) {
_absSub(dDigits, abcdUsed, xDigits, maxUsed, dDigits);
} else {
_absSub(xDigits, maxUsed, dDigits, maxUsed, dDigits);
dIsNegative = true;
}
}
_rsh(dDigits, abcdUsed, 1, dDigits);
}
if (_compareDigits(uDigits, maxUsed, vDigits, maxUsed) >= 0) {
// u -= v
_absSub(uDigits, maxUsed, vDigits, maxUsed, uDigits);
if (ac) {
// a -= c
if (aIsNegative == cIsNegative) {
var a_cmp_c = _compareDigits(aDigits, abcdUsed, cDigits, abcdUsed);
if (a_cmp_c > 0) {
_absSub(aDigits, abcdUsed, cDigits, abcdUsed, aDigits);
} else {
_absSub(cDigits, abcdUsed, aDigits, abcdUsed, aDigits);
aIsNegative = !aIsNegative && (a_cmp_c != 0);
}
} else {
_absAdd(aDigits, abcdUsed, cDigits, abcdUsed, aDigits);
}
}
// b -= d
if (bIsNegative == dIsNegative) {
var b_cmp_d = _compareDigits(bDigits, abcdUsed, dDigits, abcdUsed);
if (b_cmp_d > 0) {
_absSub(bDigits, abcdUsed, dDigits, abcdUsed, bDigits);
} else {
_absSub(dDigits, abcdUsed, bDigits, abcdUsed, bDigits);
bIsNegative = !bIsNegative && (b_cmp_d != 0);
}
} else {
_absAdd(bDigits, abcdUsed, dDigits, abcdUsed, bDigits);
}
} else {
// v -= u
_absSub(vDigits, maxUsed, uDigits, maxUsed, vDigits);
if (ac) {
// c -= a
if (cIsNegative == aIsNegative) {
var c_cmp_a = _compareDigits(cDigits, abcdUsed, aDigits, abcdUsed);
if (c_cmp_a > 0) {
_absSub(cDigits, abcdUsed, aDigits, abcdUsed, cDigits);
} else {
_absSub(aDigits, abcdUsed, cDigits, abcdUsed, cDigits);
cIsNegative = !cIsNegative && (c_cmp_a != 0);
}
} else {
_absAdd(cDigits, abcdUsed, aDigits, abcdUsed, cDigits);
}
}
// d -= b
if (dIsNegative == bIsNegative) {
var d_cmp_b = _compareDigits(dDigits, abcdUsed, bDigits, abcdUsed);
if (d_cmp_b > 0) {
_absSub(dDigits, abcdUsed, bDigits, abcdUsed, dDigits);
} else {
_absSub(bDigits, abcdUsed, dDigits, abcdUsed, dDigits);
dIsNegative = !dIsNegative && (d_cmp_b != 0);
}
} else {
_absAdd(dDigits, abcdUsed, bDigits, abcdUsed, dDigits);
}
}
// Exit loop if u == 0.
var i = maxUsed;
while ((i > 0) && (uDigits[i - 1] == 0)) --i;
if (i == 0) break;
}
if (!inv) {
if (shiftAmount > 0) {
maxUsed = _lShiftDigits(vDigits, maxUsed, shiftAmount, vDigits);
}
return new _BigIntImpl._(false, maxUsed, vDigits);
}
// No inverse if v != 1.
var i = maxUsed - 1;
while ((i > 0) && (vDigits[i] == 0)) --i;
if ((i != 0) || (vDigits[0] != 1)) {
throw new Exception("Not coprime");
}
if (dIsNegative) {
while ((dDigits[maxUsed] != 0) ||
(_compareDigits(dDigits, maxUsed, xDigits, maxUsed) > 0)) {
// d += x, d still negative
_absSub(dDigits, abcdUsed, xDigits, maxUsed, dDigits);
}
// d += x
_absSub(xDigits, maxUsed, dDigits, maxUsed, dDigits);
dIsNegative = false;
} else {
while ((dDigits[maxUsed] != 0) ||
(_compareDigits(dDigits, maxUsed, xDigits, maxUsed) >= 0)) {
// d -= x
_absSub(dDigits, abcdUsed, xDigits, maxUsed, dDigits);
}
}
return new _BigIntImpl._(false, maxUsed, dDigits);
}
/// Returns the modular multiplicative inverse of this big integer
/// modulo [modulus].
///
/// The [modulus] must be positive.
///
/// It is an error if no modular inverse exists.
// Returns 1/this % modulus, with modulus > 0.
_BigIntImpl modInverse(covariant _BigIntImpl modulus) {
if (modulus <= zero) {
throw new ArgumentError("Modulus must be strictly positive: $modulus");
}
if (modulus == one) return zero;
var tmp = this;
if (tmp._isNegative || (tmp._absCompare(modulus) >= 0)) {
tmp %= modulus;
}
return _binaryGcd(modulus, tmp, true);
}
/// Returns the greatest common divisor of this big integer and [other].
///
/// If either number is non-zero, the result is the numerically greatest
/// integer dividing both `this` and `other`.
///
/// The greatest common divisor is independent of the order,
/// so `x.gcd(y)` is always the same as `y.gcd(x)`.
///
/// For any integer `x`, `x.gcd(x)` is `x.abs()`.
///
/// If both `this` and `other` is zero, the result is also zero.
_BigIntImpl gcd(covariant _BigIntImpl other) {
if (_isZero) return other.abs();
if (other._isZero) return this.abs();
return _binaryGcd(this, other, false);
}
/// Returns the least significant [width] bits of this big integer as a
/// non-negative number (i.e. unsigned representation). The returned value
/// has zeros in all bit positions higher than [width].
///
/// ```
/// new BigInt.from(-1).toUnsigned(5) == 31 // 11111111 -> 00011111
/// ```
///
/// This operation can be used to simulate arithmetic from low level
/// languages. For example, to increment an 8 bit quantity:
///
/// ```
/// q = (q + 1).toUnsigned(8);
/// ```
///
/// `q` will count from `0` up to `255` and then wrap around to `0`.
///
/// If the input fits in [width] bits without truncation, the result is the
/// same as the input. The minimum width needed to avoid truncation of `x` is
/// given by `x.bitLength`, i.e.
///
/// ```
/// x == x.toUnsigned(x.bitLength);
/// ```
_BigIntImpl toUnsigned(int width) {
return this & ((one << width) - one);
}
/// Returns the least significant [width] bits of this integer, extending the
/// highest retained bit to the sign. This is the same as truncating the
/// value to fit in [width] bits using an signed 2-s complement
/// representation. The returned value has the same bit value in all
/// positions higher than [width].
///
/// ```
/// var big15 = new BigInt.from(15);
/// var big16 = new BigInt.from(16);
/// var big239 = new BigInt.from(239);
/// V--sign bit-V
/// big16.toSigned(5) == -big16 // 00010000 -> 11110000
/// big239.toSigned(5) == big15 // 11101111 -> 00001111
/// ^ ^
/// ```
///
/// This operation can be used to simulate arithmetic from low level
/// languages. For example, to increment an 8 bit signed quantity:
///
/// ```
/// q = (q + 1).toSigned(8);
/// ```
///
/// `q` will count from `0` up to `127`, wrap to `-128` and count back up to
/// `127`.
///
/// If the input value fits in [width] bits without truncation, the result is
/// the same as the input. The minimum width needed to avoid truncation of
/// `x` is `x.bitLength + 1`, i.e.
///
/// ```
/// x == x.toSigned(x.bitLength + 1);
/// ```
_BigIntImpl toSigned(int width) {
// The value of binary number weights each bit by a power of two. The
// twos-complement value weights the sign bit negatively. We compute the
// value of the negative weighting by isolating the sign bit with the
// correct power of two weighting and subtracting it from the value of the
// lower bits.
var signMask = one << (width - 1);
return (this & (signMask - one)) - (this & signMask);
}
// Maximum number of digits that always fit in mantissa.
static const _simpleValidIntDigits = 53 ~/ _digitBits;
bool get isValidInt {
if (_used <= _simpleValidIntDigits) return true;
var asInt = toInt();
if (!asInt.toDouble().isFinite) return false;
return this == new _BigIntImpl._fromInt(asInt);
}
int toInt() {
var result = 0;
for (int i = _used - 1; i >= 0; i--) {
result = result * _digitBase + _digits[i];
}
return _isNegative ? -result : result;
}
/// Returns this [_BigIntImpl] as a [double].
///
/// If the number is not representable as a [double], an
/// approximation is returned. For numerically large integers, the
/// approximation may be infinite.
double toDouble() {
const int exponentBias = 1075;
// There are 11 bits for the exponent.
// 2047 (all bits set to 1) is reserved for infinity and NaN.
// When storing the exponent in the 11 bits, it is biased by exponentBias
// to support negative exponents.
const int maxDoubleExponent = 2046 - exponentBias;
if (_isZero) return 0.0;
// We fill the 53 bits little-endian.
var resultBits = new Uint8List(8);
var length = _digitBits * (_used - 1) + _digits[_used - 1].bitLength;
if (length > maxDoubleExponent + 53) {
return _isNegative ? double.negativeInfinity : double.infinity;
}
// The most significant bit is for the sign.
if (_isNegative) resultBits[7] = 0x80;
// Write the exponent into bits 1..12:
var biasedExponent = length - 53 + exponentBias;
resultBits[6] = (biasedExponent & 0xF) << 4;
resultBits[7] |= biasedExponent >> 4;
int cachedBits = 0;
int cachedBitsLength = 0;
int digitIndex = _used - 1;
int readBits(int n) {
// Ensure that we have enough bits in [cachedBits].
while (cachedBitsLength < n) {
int nextDigit;
int nextDigitLength = _digitBits; // May get updated.
if (digitIndex < 0) {
nextDigit = 0;
digitIndex--;
} else {
nextDigit = _digits[digitIndex];
if (digitIndex == _used - 1) nextDigitLength = nextDigit.bitLength;
digitIndex--;
}
cachedBits = (cachedBits << nextDigitLength) + nextDigit;
cachedBitsLength += nextDigitLength;
}
// Read the top [n] bits.
var result = cachedBits >> (cachedBitsLength - n);
// Remove the bits from the cache.
cachedBits -= result << (cachedBitsLength - n);
cachedBitsLength -= n;
return result;
}
// The first leading 1 bit is implicit in the double-representation and can
// be discarded.
var leadingBits = readBits(5) & 0xF;
resultBits[6] |= leadingBits;
for (int i = 5; i >= 0; i--) {
// Get the remaining 48 bits.
resultBits[i] = readBits(8);
}
void roundUp() {
// Simply consists of adding 1 to the whole 64 bit "number".
// It will update the exponent, if necessary.
// It might even round up to infinity (which is what we want).
var carry = 1;
for (int i = 0; i < 8; i++) {
if (carry == 0) break;
var sum = resultBits[i] + carry;
resultBits[i] = sum & 0xFF;
carry = sum >> 8;
}
}
if (readBits(1) == 1) {
if (resultBits[0].isOdd) {
// Rounds to even all the time.
roundUp();
} else {
// Round up, if there is at least one other digit that is not 0.
if (cachedBits != 0) {
// There is already one in the cachedBits.
roundUp();
} else {
for (int i = digitIndex; i >= 0; i--) {
if (_digits[i] != 0) {
roundUp();
break;
}
}
}
}
}
return resultBits.buffer.asByteData().getFloat64(0, Endian.little);
}
/// Returns a String-representation of this integer.
///
/// The returned string is parsable by [parse].