| // Copyright (c) 2019, 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. |
| |
| @JS() |
| library native_version_javascript; |
| |
| import 'package:js/js.dart'; |
| import 'native_version.dart'; |
| |
| const NativeBigIntMethods nativeBigInt = _Methods(); |
| |
| @JS('eval') |
| external Object _eval(String s); |
| |
| @JS('bigint_parse') |
| external Object _parse(String s); |
| |
| @JS('bigint_toString') |
| external String _toStringMethod(Object o); |
| |
| @JS('bigint_bitLength') |
| external int _bitLength(Object o); |
| |
| @JS('bigint_isEven') |
| external bool _isEven(Object o); |
| |
| @JS('bigint_add') |
| external Object _add(Object left, Object right); |
| |
| @JS('bigint_shiftLeft') |
| external Object _shiftLeft(Object o, Object i); |
| |
| @JS('bigint_shiftRight') |
| external Object _shiftRight(Object o, Object i); |
| |
| @JS('bigint_subtract') |
| external Object _subtract(Object left, Object right); |
| |
| @JS('bigint_fromInt') |
| external Object _fromInt(int i); |
| |
| class _Methods implements NativeBigIntMethods { |
| static bool _initialized = false; |
| static bool _enabled = false; |
| |
| const _Methods(); |
| |
| @override |
| bool get enabled { |
| if (!_initialized) { |
| _initialize(); |
| } |
| return _enabled; |
| } |
| |
| void _initialize() { |
| _initialized = true; |
| try { |
| _setup(); |
| _enabled = true; |
| } catch (e) { |
| // We get here if the JavaScript implementation does not have BigInt (or |
| // run in a stand-alone JavaScript implementation without the right |
| // 'preamble'). |
| // |
| // Print so we can see what failed. |
| print(e); |
| } |
| } |
| |
| @override |
| Object parse(String string) => _parse(string); |
| |
| @override |
| String toStringMethod(Object value) => _toStringMethod(value); |
| |
| @override |
| Object fromInt(int i) => _fromInt(i); |
| |
| @override |
| Object get one => _one; |
| |
| @override |
| Object get eight => _eight; |
| |
| @override |
| int bitLength(Object value) => _bitLength(value); |
| |
| @override |
| bool isEven(Object value) => _isEven(value); |
| |
| @override |
| Object add(Object left, Object right) => _add(left, right); |
| |
| @override |
| Object shiftLeft(Object value, Object count) => _shiftLeft(value, count); |
| |
| @override |
| Object shiftRight(Object value, Object count) => _shiftRight(value, count); |
| |
| @override |
| Object subtract(Object left, Object right) => _subtract(left, right); |
| } |
| |
| void _setup() { |
| _one = _eval('1n'); // Throws if JavaScript does not have BigInt. |
| _eight = _eval('8n'); |
| |
| _eval('self.bigint_parse = function parse(s) { return BigInt(s); }'); |
| _eval('self.bigint_toString = function toString(b) { return b.toString(); }'); |
| _eval('self.bigint_add = function add(a, b) { return a + b; }'); |
| _eval('self.bigint_shiftLeft = function shl(v, i) { return v << i; }'); |
| _eval('self.bigint_shiftRight = function shr(v, i) { return v >> i; }'); |
| _eval('self.bigint_subtract = function subtract(a, b) { return a - b; }'); |
| _eval('self.bigint_fromInt = function fromInt(i) { return BigInt(i); }'); |
| |
| _eval('self.bigint_bitLength = function bitLength(b) {' |
| 'return b == 0 ? 0 : (b < 0 ? ~b : b).toString(2).length;' |
| '}'); |
| _eval('self.bigint_isEven = function isEven(b) { return (b & 1n) == 0n; }'); |
| } |
| |
| // `dynamic` to allow null initialization pre- and post- NNBD. |
| dynamic _one; |
| dynamic _eight; |