2012-01-13 20:07:54 +00:00
|
|
|
// Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file
|
2011-10-05 05:20:07 +00:00
|
|
|
// 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.
|
|
|
|
|
|
|
|
#include "vm/bootstrap_natives.h"
|
|
|
|
|
|
|
|
#include "vm/bigint_operations.h"
|
|
|
|
#include "vm/dart_entry.h"
|
2011-11-04 16:30:29 +00:00
|
|
|
#include "vm/exceptions.h"
|
2011-10-05 05:20:07 +00:00
|
|
|
#include "vm/native_entry.h"
|
|
|
|
#include "vm/object.h"
|
2012-11-16 19:42:02 +00:00
|
|
|
#include "vm/symbols.h"
|
2011-10-05 05:20:07 +00:00
|
|
|
|
|
|
|
namespace dart {
|
|
|
|
|
|
|
|
DEFINE_FLAG(bool, trace_intrinsified_natives, false,
|
|
|
|
"Report if any of the intrinsified natives are called");
|
|
|
|
|
|
|
|
// Smi natives.
|
|
|
|
|
|
|
|
// Returns false if integer is in wrong representation, e.g., as is a Bigint
|
|
|
|
// when it could have been a Smi.
|
|
|
|
static bool CheckInteger(const Integer& i) {
|
|
|
|
if (i.IsBigint()) {
|
2013-01-23 20:01:31 +00:00
|
|
|
const Bigint& bigint = Bigint::Cast(i);
|
2011-10-05 05:20:07 +00:00
|
|
|
return !BigintOperations::FitsIntoSmi(bigint) &&
|
2012-02-28 18:36:39 +00:00
|
|
|
!BigintOperations::FitsIntoMint(bigint);
|
2011-10-05 05:20:07 +00:00
|
|
|
}
|
|
|
|
if (i.IsMint()) {
|
2013-01-23 20:01:31 +00:00
|
|
|
const Mint& mint = Mint::Cast(i);
|
2011-10-05 05:20:07 +00:00
|
|
|
return !Smi::IsValid64(mint.value());
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DEFINE_NATIVE_ENTRY(Integer_bitAndFromInteger, 2) {
|
2012-11-15 01:14:01 +00:00
|
|
|
const Integer& right = Integer::CheckedHandle(arguments->NativeArgAt(0));
|
2012-12-10 17:59:40 +00:00
|
|
|
GET_NON_NULL_NATIVE_ARGUMENT(Integer, left, arguments->NativeArgAt(1));
|
2011-10-05 05:20:07 +00:00
|
|
|
ASSERT(CheckInteger(right));
|
|
|
|
ASSERT(CheckInteger(left));
|
|
|
|
if (FLAG_trace_intrinsified_natives) {
|
|
|
|
OS::Print("Integer_bitAndFromInteger %s & %s\n",
|
|
|
|
right.ToCString(), left.ToCString());
|
|
|
|
}
|
2012-11-26 16:09:50 +00:00
|
|
|
const Integer& result =
|
2012-09-24 13:14:54 +00:00
|
|
|
Integer::Handle(left.BitOp(Token::kBIT_AND, right));
|
2012-11-26 16:09:50 +00:00
|
|
|
return result.AsValidInteger();
|
2011-10-05 05:20:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DEFINE_NATIVE_ENTRY(Integer_bitOrFromInteger, 2) {
|
2012-11-15 01:14:01 +00:00
|
|
|
const Integer& right = Integer::CheckedHandle(arguments->NativeArgAt(0));
|
2012-12-10 17:59:40 +00:00
|
|
|
GET_NON_NULL_NATIVE_ARGUMENT(Integer, left, arguments->NativeArgAt(1));
|
2011-10-05 05:20:07 +00:00
|
|
|
ASSERT(CheckInteger(right));
|
|
|
|
ASSERT(CheckInteger(left));
|
|
|
|
if (FLAG_trace_intrinsified_natives) {
|
|
|
|
OS::Print("Integer_bitOrFromInteger %s | %s\n",
|
|
|
|
left.ToCString(), right.ToCString());
|
|
|
|
}
|
2012-11-26 16:09:50 +00:00
|
|
|
const Integer& result =
|
2012-09-24 13:14:54 +00:00
|
|
|
Integer::Handle(left.BitOp(Token::kBIT_OR, right));
|
2012-11-26 16:09:50 +00:00
|
|
|
return result.AsValidInteger();
|
2011-10-05 05:20:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DEFINE_NATIVE_ENTRY(Integer_bitXorFromInteger, 2) {
|
2012-11-15 01:14:01 +00:00
|
|
|
const Integer& right = Integer::CheckedHandle(arguments->NativeArgAt(0));
|
2012-12-10 17:59:40 +00:00
|
|
|
GET_NON_NULL_NATIVE_ARGUMENT(Integer, left, arguments->NativeArgAt(1));
|
2011-10-05 05:20:07 +00:00
|
|
|
ASSERT(CheckInteger(right));
|
|
|
|
ASSERT(CheckInteger(left));
|
|
|
|
if (FLAG_trace_intrinsified_natives) {
|
|
|
|
OS::Print("Integer_bitXorFromInteger %s ^ %s\n",
|
|
|
|
left.ToCString(), right.ToCString());
|
|
|
|
}
|
2012-11-26 16:09:50 +00:00
|
|
|
const Integer& result =
|
2012-09-24 13:14:54 +00:00
|
|
|
Integer::Handle(left.BitOp(Token::kBIT_XOR, right));
|
2012-11-26 16:09:50 +00:00
|
|
|
return result.AsValidInteger();
|
2011-10-05 05:20:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DEFINE_NATIVE_ENTRY(Integer_addFromInteger, 2) {
|
2012-11-15 01:14:01 +00:00
|
|
|
const Integer& right_int = Integer::CheckedHandle(arguments->NativeArgAt(0));
|
2012-12-10 17:59:40 +00:00
|
|
|
GET_NON_NULL_NATIVE_ARGUMENT(Integer, left_int, arguments->NativeArgAt(1));
|
2011-10-05 05:20:07 +00:00
|
|
|
ASSERT(CheckInteger(right_int));
|
|
|
|
ASSERT(CheckInteger(left_int));
|
|
|
|
if (FLAG_trace_intrinsified_natives) {
|
|
|
|
OS::Print("Integer_addFromInteger %s + %s\n",
|
|
|
|
left_int.ToCString(), right_int.ToCString());
|
|
|
|
}
|
2012-11-26 16:09:50 +00:00
|
|
|
const Integer& result =
|
|
|
|
Integer::Handle(left_int.ArithmeticOp(Token::kADD, right_int));
|
|
|
|
return result.AsValidInteger();
|
2011-10-05 05:20:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DEFINE_NATIVE_ENTRY(Integer_subFromInteger, 2) {
|
2012-11-15 01:14:01 +00:00
|
|
|
const Integer& right_int = Integer::CheckedHandle(arguments->NativeArgAt(0));
|
2012-12-10 17:59:40 +00:00
|
|
|
GET_NON_NULL_NATIVE_ARGUMENT(Integer, left_int, arguments->NativeArgAt(1));
|
2011-10-05 05:20:07 +00:00
|
|
|
ASSERT(CheckInteger(right_int));
|
|
|
|
ASSERT(CheckInteger(left_int));
|
|
|
|
if (FLAG_trace_intrinsified_natives) {
|
|
|
|
OS::Print("Integer_subFromInteger %s - %s\n",
|
|
|
|
left_int.ToCString(), right_int.ToCString());
|
|
|
|
}
|
2012-11-26 16:09:50 +00:00
|
|
|
const Integer& result =
|
|
|
|
Integer::Handle(left_int.ArithmeticOp(Token::kSUB, right_int));
|
|
|
|
return result.AsValidInteger();
|
2011-10-05 05:20:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DEFINE_NATIVE_ENTRY(Integer_mulFromInteger, 2) {
|
2012-11-15 01:14:01 +00:00
|
|
|
const Integer& right_int = Integer::CheckedHandle(arguments->NativeArgAt(0));
|
2012-12-10 17:59:40 +00:00
|
|
|
GET_NON_NULL_NATIVE_ARGUMENT(Integer, left_int, arguments->NativeArgAt(1));
|
2011-10-05 05:20:07 +00:00
|
|
|
ASSERT(CheckInteger(right_int));
|
|
|
|
ASSERT(CheckInteger(left_int));
|
|
|
|
if (FLAG_trace_intrinsified_natives) {
|
|
|
|
OS::Print("Integer_mulFromInteger %s * %s\n",
|
|
|
|
left_int.ToCString(), right_int.ToCString());
|
|
|
|
}
|
2012-11-26 16:09:50 +00:00
|
|
|
const Integer& result =
|
|
|
|
Integer::Handle(left_int.ArithmeticOp(Token::kMUL, right_int));
|
|
|
|
return result.AsValidInteger();
|
2011-10-05 05:20:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DEFINE_NATIVE_ENTRY(Integer_truncDivFromInteger, 2) {
|
2012-11-15 01:14:01 +00:00
|
|
|
const Integer& right_int = Integer::CheckedHandle(arguments->NativeArgAt(0));
|
2012-12-10 17:59:40 +00:00
|
|
|
GET_NON_NULL_NATIVE_ARGUMENT(Integer, left_int, arguments->NativeArgAt(1));
|
2011-10-05 05:20:07 +00:00
|
|
|
ASSERT(CheckInteger(right_int));
|
|
|
|
ASSERT(CheckInteger(left_int));
|
|
|
|
ASSERT(!right_int.IsZero());
|
2012-11-26 16:09:50 +00:00
|
|
|
const Integer& result =
|
|
|
|
Integer::Handle(left_int.ArithmeticOp(Token::kTRUNCDIV, right_int));
|
|
|
|
return result.AsValidInteger();
|
2011-10-05 05:20:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DEFINE_NATIVE_ENTRY(Integer_moduloFromInteger, 2) {
|
2012-11-15 01:14:01 +00:00
|
|
|
const Integer& right_int = Integer::CheckedHandle(arguments->NativeArgAt(0));
|
2012-12-10 17:59:40 +00:00
|
|
|
GET_NON_NULL_NATIVE_ARGUMENT(Integer, left_int, arguments->NativeArgAt(1));
|
2011-10-05 05:20:07 +00:00
|
|
|
ASSERT(CheckInteger(right_int));
|
|
|
|
ASSERT(CheckInteger(right_int));
|
|
|
|
if (FLAG_trace_intrinsified_natives) {
|
|
|
|
OS::Print("Integer_moduloFromInteger %s mod %s\n",
|
|
|
|
left_int.ToCString(), right_int.ToCString());
|
|
|
|
}
|
|
|
|
if (right_int.IsZero()) {
|
|
|
|
// Should have been caught before calling into runtime.
|
|
|
|
UNIMPLEMENTED();
|
|
|
|
}
|
2012-11-26 16:09:50 +00:00
|
|
|
const Integer& result =
|
|
|
|
Integer::Handle(left_int.ArithmeticOp(Token::kMOD, right_int));
|
|
|
|
return result.AsValidInteger();
|
2011-10-05 05:20:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DEFINE_NATIVE_ENTRY(Integer_greaterThanFromInteger, 2) {
|
2012-11-15 01:14:01 +00:00
|
|
|
const Integer& right = Integer::CheckedHandle(arguments->NativeArgAt(0));
|
2012-12-10 17:59:40 +00:00
|
|
|
GET_NON_NULL_NATIVE_ARGUMENT(Integer, left, arguments->NativeArgAt(1));
|
2011-10-05 05:20:07 +00:00
|
|
|
ASSERT(CheckInteger(right));
|
|
|
|
ASSERT(CheckInteger(left));
|
|
|
|
if (FLAG_trace_intrinsified_natives) {
|
|
|
|
OS::Print("Integer_greaterThanFromInteger %s > %s\n",
|
|
|
|
left.ToCString(), right.ToCString());
|
|
|
|
}
|
2012-08-30 17:41:19 +00:00
|
|
|
return Bool::Get(left.CompareWith(right) == 1);
|
2011-10-05 05:20:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DEFINE_NATIVE_ENTRY(Integer_equalToInteger, 2) {
|
2012-11-15 01:14:01 +00:00
|
|
|
const Integer& left = Integer::CheckedHandle(arguments->NativeArgAt(0));
|
2012-12-10 17:59:40 +00:00
|
|
|
GET_NON_NULL_NATIVE_ARGUMENT(Integer, right, arguments->NativeArgAt(1));
|
2011-10-05 05:20:07 +00:00
|
|
|
ASSERT(CheckInteger(left));
|
|
|
|
ASSERT(CheckInteger(right));
|
|
|
|
if (FLAG_trace_intrinsified_natives) {
|
|
|
|
OS::Print("Integer_equalToInteger %s == %s\n",
|
|
|
|
left.ToCString(), right.ToCString());
|
|
|
|
}
|
2012-08-30 17:41:19 +00:00
|
|
|
return Bool::Get(left.CompareWith(right) == 0);
|
2011-10-05 05:20:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-11-16 19:42:02 +00:00
|
|
|
DEFINE_NATIVE_ENTRY(Integer_parse, 1) {
|
2012-12-10 17:59:40 +00:00
|
|
|
GET_NON_NULL_NATIVE_ARGUMENT(String, value, arguments->NativeArgAt(0));
|
2013-02-25 18:32:46 +00:00
|
|
|
if (value.IsOneByteString()) {
|
|
|
|
// Quick conversion for unpadded integers in strings.
|
|
|
|
const intptr_t len = value.Length();
|
|
|
|
if (len > 0) {
|
|
|
|
const char* cstr = value.ToCString();
|
|
|
|
ASSERT(cstr != NULL);
|
|
|
|
// Dart differences from strtol:
|
|
|
|
// a) '+5' is not a valid integer (leading plus).
|
|
|
|
if (cstr[0] != '+') {
|
|
|
|
char* p_end = NULL;
|
2013-04-19 20:22:26 +00:00
|
|
|
const int64_t int_value = strtoll(cstr, &p_end, 10);
|
2013-02-25 18:32:46 +00:00
|
|
|
if (p_end == (cstr + len)) {
|
2013-04-19 20:22:26 +00:00
|
|
|
if ((int_value != LLONG_MIN) && (int_value != LLONG_MAX)) {
|
|
|
|
return Integer::New(int_value);
|
2013-02-25 18:32:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-02 19:31:59 +00:00
|
|
|
Scanner scanner(value, Symbols::Empty());
|
2012-11-16 19:42:02 +00:00
|
|
|
const Scanner::GrowableTokenStream& tokens = scanner.GetStream();
|
|
|
|
String* int_string;
|
|
|
|
bool is_positive;
|
|
|
|
if (Scanner::IsValidLiteral(tokens,
|
|
|
|
Token::kINTEGER,
|
|
|
|
&is_positive,
|
|
|
|
&int_string)) {
|
|
|
|
if (is_positive) {
|
|
|
|
return Integer::New(*int_string);
|
|
|
|
}
|
|
|
|
String& temp = String::Handle();
|
2013-01-02 19:31:59 +00:00
|
|
|
temp = String::Concat(Symbols::Dash(), *int_string);
|
2012-11-16 19:42:02 +00:00
|
|
|
return Integer::New(temp);
|
|
|
|
}
|
|
|
|
|
2012-12-18 23:47:35 +00:00
|
|
|
const Array& args = Array::Handle(Array::New(1));
|
|
|
|
args.SetAt(0, value);
|
2012-11-16 19:42:02 +00:00
|
|
|
Exceptions::ThrowByType(Exceptions::kFormat, args);
|
|
|
|
return Object::null();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-10-05 05:20:07 +00:00
|
|
|
static RawInteger* ShiftOperationHelper(Token::Kind kind,
|
|
|
|
const Integer& value,
|
|
|
|
const Smi& amount) {
|
|
|
|
if (amount.Value() < 0) {
|
2012-12-18 23:47:35 +00:00
|
|
|
const Array& args = Array::Handle(Array::New(1));
|
|
|
|
args.SetAt(0, amount);
|
2012-09-25 12:27:52 +00:00
|
|
|
Exceptions::ThrowByType(Exceptions::kArgument, args);
|
2011-10-05 05:20:07 +00:00
|
|
|
}
|
|
|
|
if (value.IsSmi()) {
|
2013-01-23 20:01:31 +00:00
|
|
|
const Smi& smi_value = Smi::Cast(value);
|
2012-09-24 13:14:54 +00:00
|
|
|
return smi_value.ShiftOp(kind, amount);
|
2011-10-05 05:20:07 +00:00
|
|
|
}
|
|
|
|
Bigint& big_value = Bigint::Handle();
|
|
|
|
if (value.IsMint()) {
|
|
|
|
const int64_t mint_value = value.AsInt64Value();
|
2012-09-24 13:14:54 +00:00
|
|
|
const int count = Utils::HighestBit(mint_value);
|
2011-10-05 05:20:07 +00:00
|
|
|
if ((count + amount.Value()) < Mint::kBits) {
|
|
|
|
switch (kind) {
|
|
|
|
case Token::kSHL:
|
|
|
|
return Integer::New(mint_value << amount.Value());
|
2012-01-13 09:10:48 +00:00
|
|
|
case Token::kSHR:
|
2011-10-05 05:20:07 +00:00
|
|
|
return Integer::New(mint_value >> amount.Value());
|
|
|
|
default:
|
|
|
|
UNIMPLEMENTED();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Overflow in shift, use Bigints
|
|
|
|
big_value = BigintOperations::NewFromInt64(mint_value);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ASSERT(value.IsBigint());
|
2013-01-23 20:01:31 +00:00
|
|
|
big_value = Bigint::Cast(value).raw();
|
2011-10-05 05:20:07 +00:00
|
|
|
}
|
|
|
|
switch (kind) {
|
|
|
|
case Token::kSHL:
|
|
|
|
return BigintOperations::ShiftLeft(big_value, amount.Value());
|
2012-01-13 09:10:48 +00:00
|
|
|
case Token::kSHR:
|
2011-10-05 05:20:07 +00:00
|
|
|
return BigintOperations::ShiftRight(big_value, amount.Value());
|
|
|
|
default:
|
|
|
|
UNIMPLEMENTED();
|
|
|
|
}
|
|
|
|
return Integer::null();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-01-13 22:46:32 +00:00
|
|
|
DEFINE_NATIVE_ENTRY(Smi_shrFromInt, 2) {
|
2012-11-15 01:14:01 +00:00
|
|
|
const Smi& amount = Smi::CheckedHandle(arguments->NativeArgAt(0));
|
2012-12-10 17:59:40 +00:00
|
|
|
GET_NON_NULL_NATIVE_ARGUMENT(Integer, value, arguments->NativeArgAt(1));
|
2011-10-05 05:20:07 +00:00
|
|
|
ASSERT(CheckInteger(amount));
|
|
|
|
ASSERT(CheckInteger(value));
|
2012-11-26 16:09:50 +00:00
|
|
|
const Integer& result = Integer::Handle(
|
2012-01-13 09:10:48 +00:00
|
|
|
ShiftOperationHelper(Token::kSHR, value, amount));
|
2012-11-26 16:09:50 +00:00
|
|
|
return result.AsValidInteger();
|
2011-10-05 05:20:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEFINE_NATIVE_ENTRY(Smi_shlFromInt, 2) {
|
2012-11-15 01:14:01 +00:00
|
|
|
const Smi& amount = Smi::CheckedHandle(arguments->NativeArgAt(0));
|
2012-12-10 17:59:40 +00:00
|
|
|
GET_NON_NULL_NATIVE_ARGUMENT(Integer, value, arguments->NativeArgAt(1));
|
2011-10-05 05:20:07 +00:00
|
|
|
ASSERT(CheckInteger(amount));
|
|
|
|
ASSERT(CheckInteger(value));
|
|
|
|
if (FLAG_trace_intrinsified_natives) {
|
|
|
|
OS::Print("Smi_shlFromInt: %s << %s\n",
|
|
|
|
value.ToCString(), amount.ToCString());
|
|
|
|
}
|
2012-11-26 16:09:50 +00:00
|
|
|
const Integer& result = Integer::Handle(
|
2011-10-05 05:20:07 +00:00
|
|
|
ShiftOperationHelper(Token::kSHL, value, amount));
|
2012-11-26 16:09:50 +00:00
|
|
|
return result.AsValidInteger();
|
2011-10-05 05:20:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DEFINE_NATIVE_ENTRY(Smi_bitNegate, 1) {
|
2012-11-15 01:14:01 +00:00
|
|
|
const Smi& operand = Smi::CheckedHandle(arguments->NativeArgAt(0));
|
2011-10-05 05:20:07 +00:00
|
|
|
if (FLAG_trace_intrinsified_natives) {
|
|
|
|
OS::Print("Smi_bitNegate: %s\n", operand.ToCString());
|
|
|
|
}
|
|
|
|
intptr_t result = ~operand.Value();
|
|
|
|
ASSERT(Smi::IsValid(result));
|
2012-08-30 17:41:19 +00:00
|
|
|
return Smi::New(result);
|
2011-10-05 05:20:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Mint natives.
|
|
|
|
|
|
|
|
DEFINE_NATIVE_ENTRY(Mint_bitNegate, 1) {
|
2012-11-15 01:14:01 +00:00
|
|
|
const Mint& operand = Mint::CheckedHandle(arguments->NativeArgAt(0));
|
2011-10-05 05:20:07 +00:00
|
|
|
ASSERT(CheckInteger(operand));
|
|
|
|
if (FLAG_trace_intrinsified_natives) {
|
|
|
|
OS::Print("Mint_bitNegate: %s\n", operand.ToCString());
|
|
|
|
}
|
|
|
|
int64_t result = ~operand.value();
|
2012-08-30 17:41:19 +00:00
|
|
|
return Integer::New(result);
|
2011-10-05 05:20:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Bigint natives.
|
|
|
|
|
|
|
|
DEFINE_NATIVE_ENTRY(Bigint_bitNegate, 1) {
|
2012-11-15 01:14:01 +00:00
|
|
|
const Bigint& value = Bigint::CheckedHandle(arguments->NativeArgAt(0));
|
2011-10-05 05:20:07 +00:00
|
|
|
const Bigint& result = Bigint::Handle(BigintOperations::BitNot(value));
|
|
|
|
ASSERT(CheckInteger(value));
|
|
|
|
ASSERT(CheckInteger(result));
|
2012-11-26 16:09:50 +00:00
|
|
|
return result.AsValidInteger();
|
2011-10-05 05:20:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace dart
|