blob: de4abdb04ce4c2fdc1b30ead70bc56797c37c2b9 [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.
// Dart core library.
// VM implementation of int.
patch class int {
static int _parse(String str) native "Integer_parse";
static int _throwFormatException(String source) {
throw new FormatException(source);
}
/* patch */ static int parse(String source,
{ int radix,
int onError(String str) }) {
if ((radix == null) && (onError == null)) return _parse(source);
return _slowParse(source, radix, onError);
}
static int _slowParse(String source, int radix, int onError(String str)) {
if (source is! String) throw new ArgumentError(source);
if (radix == null) {
assert(onError != null);
try {
return _parse(source);
} on FormatException {
return onError(source);
}
}
if (radix is! int) throw new ArgumentError("Radix is not an integer");
if (radix < 2 || radix > 36) {
throw new RangeError("Radix $radix not in range 2..36");
}
if (onError == null) {
onError = _throwFormatException;
}
// Remove leading and trailing white space.
source = source.trim();
if (source.isEmpty) return onError(source);
bool negative = false;
int result = 0;
// The value 99 is used to represent a non-digit. It is too large to be
// a digit value in any of the used bases.
const NA = 99;
const List<int> digits = const <int>[
00, 01, 02, 03, 04, 05, 06, 07, 08, 09, NA, NA, NA, NA, NA, NA, // 0x30
NA, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, // 0x40
25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, NA, NA, NA, NA, NA, // 0x50
NA, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, // 0x60
25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, NA, NA, NA, NA, NA, // 0x70
];
int i = 0;
int code = source.codeUnitAt(i);
if (code == 0x2d || code == 0x2b) { // Starts with a plus or minus-sign.
negative = (code == 0x2d);
if (source.length == 1) return onError(source);
i = 1;
code = source.codeUnitAt(i);
}
do {
if (code < 0x30 || code > 0x7f) return onError(source);
int digit = digits[code - 0x30];
if (digit >= radix) return onError(source);
result = result * radix + digit;
i++;
if (i == source.length) break;
code = source.codeUnitAt(i);
} while (true);
return negative ? -result : result;
}
}