| OLD | NEW |
| 1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file |
| 2 // for details. All rights reserved. Use of this source code is governed by a | 2 // for details. All rights reserved. Use of this source code is governed by a |
| 3 // BSD-style license that can be found in the LICENSE file. | 3 // BSD-style license that can be found in the LICENSE file. |
| 4 | 4 |
| 5 #include "vm/bootstrap_natives.h" | 5 #include "vm/bootstrap_natives.h" |
| 6 | 6 |
| 7 #include "vm/compiler.h" | 7 #include "vm/compiler.h" |
| 8 #include "vm/dart_entry.h" | 8 #include "vm/dart_entry.h" |
| 9 #include "vm/exceptions.h" | 9 #include "vm/exceptions.h" |
| 10 #include "vm/native_entry.h" | 10 #include "vm/native_entry.h" |
| (...skipping 10 matching lines...) Expand all Loading... |
| 21 fun_arg_names)); | 21 fun_arg_names)); |
| 22 const Object& result = | 22 const Object& result = |
| 23 Object::Handle(DartEntry::InvokeClosure(fun_arguments, fun_args_desc)); | 23 Object::Handle(DartEntry::InvokeClosure(fun_arguments, fun_args_desc)); |
| 24 if (result.IsError()) { | 24 if (result.IsError()) { |
| 25 Exceptions::PropagateError(Error::Cast(result)); | 25 Exceptions::PropagateError(Error::Cast(result)); |
| 26 } | 26 } |
| 27 return result.raw(); | 27 return result.raw(); |
| 28 } | 28 } |
| 29 | 29 |
| 30 | 30 |
| 31 DEFINE_NATIVE_ENTRY(FunctionImpl_equals, 2) { | 31 DEFINE_NATIVE_ENTRY(Closure_equals, 2) { |
| 32 const Instance& receiver = Instance::CheckedHandle( | 32 const Closure& receiver = Closure::CheckedHandle( |
| 33 zone, arguments->NativeArgAt(0)); | 33 zone, arguments->NativeArgAt(0)); |
| 34 ASSERT(receiver.IsClosure()); | |
| 35 GET_NATIVE_ARGUMENT(Instance, other, arguments->NativeArgAt(1)); | 34 GET_NATIVE_ARGUMENT(Instance, other, arguments->NativeArgAt(1)); |
| 36 ASSERT(!other.IsNull()); | 35 ASSERT(!other.IsNull()); |
| 37 if (receiver.raw() == other.raw()) return Bool::True().raw(); | 36 if (receiver.raw() == other.raw()) return Bool::True().raw(); |
| 38 if (other.IsClosure()) { | 37 if (other.IsClosure()) { |
| 39 const Function& func_a = Function::Handle(Closure::function(receiver)); | 38 const Function& func_a = Function::Handle(receiver.function()); |
| 40 const Function& func_b = Function::Handle(Closure::function(other)); | 39 const Function& func_b = Function::Handle(Closure::Cast(other).function()); |
| 41 if (func_a.raw() == func_b.raw()) { | 40 if (func_a.raw() == func_b.raw()) { |
| 42 ASSERT(!func_a.IsImplicitStaticClosureFunction()); | 41 ASSERT(!func_a.IsImplicitStaticClosureFunction()); |
| 43 if (func_a.IsImplicitInstanceClosureFunction()) { | 42 if (func_a.IsImplicitInstanceClosureFunction()) { |
| 44 const Context& context_a = Context::Handle(Closure::context(receiver)); | 43 const Context& context_a = Context::Handle(receiver.context()); |
| 45 const Context& context_b = Context::Handle(Closure::context(other)); | 44 const Context& context_b = Context::Handle( |
| 45 Closure::Cast(other).context()); |
| 46 const Object& receiver_a = Object::Handle(context_a.At(0)); | 46 const Object& receiver_a = Object::Handle(context_a.At(0)); |
| 47 const Object& receiver_b = Object::Handle(context_b.At(0)); | 47 const Object& receiver_b = Object::Handle(context_b.At(0)); |
| 48 if (receiver_a.raw() == receiver_b.raw()) return Bool::True().raw(); | 48 if (receiver_a.raw() == receiver_b.raw()) return Bool::True().raw(); |
| 49 } | 49 } |
| 50 } | 50 } |
| 51 } | 51 } |
| 52 return Bool::False().raw(); | 52 return Bool::False().raw(); |
| 53 } | 53 } |
| 54 | 54 |
| 55 | 55 |
| 56 DEFINE_NATIVE_ENTRY(FunctionImpl_hashCode, 1) { | 56 DEFINE_NATIVE_ENTRY(Closure_hashCode, 1) { |
| 57 const Instance& receiver = Instance::CheckedHandle( | 57 const Closure& receiver = Closure::CheckedHandle( |
| 58 zone, arguments->NativeArgAt(0)); | 58 zone, arguments->NativeArgAt(0)); |
| 59 if (receiver.IsClosure()) { | 59 const Function& func = Function::Handle(receiver.function()); |
| 60 const Function& func = Function::Handle(Closure::function(receiver)); | 60 // Hash together name, class name and signature. |
| 61 // Hash together name, class name and signature. | 61 const Class& cls = Class::Handle(func.Owner()); |
| 62 const Class& cls = Class::Handle(func.Owner()); | 62 intptr_t result = String::Handle(func.name()).Hash(); |
| 63 intptr_t result = String::Handle(func.name()).Hash(); | 63 result += String::Handle(func.Signature()).Hash(); |
| 64 result += String::Handle(func.Signature()).Hash(); | 64 result += String::Handle(cls.Name()).Hash(); |
| 65 result += String::Handle(cls.Name()).Hash(); | 65 // Finalize hash value like for strings so that it fits into a smi. |
| 66 // Finalize hash value like for strings so that it fits into a smi. | 66 result += result << 3; |
| 67 result += result << 3; | 67 result ^= result >> 11; |
| 68 result ^= result >> 11; | 68 result += result << 15; |
| 69 result += result << 15; | 69 result &= ((static_cast<intptr_t>(1) << String::kHashBits) - 1); |
| 70 result &= ((static_cast<intptr_t>(1) << String::kHashBits) - 1); | 70 return Smi::New(result); |
| 71 return Smi::New(result); | |
| 72 } | |
| 73 UNREACHABLE(); | |
| 74 return Object::null(); | |
| 75 } | 71 } |
| 76 | 72 |
| 77 | 73 |
| 78 DEFINE_NATIVE_ENTRY(FunctionImpl_clone, 1) { | 74 DEFINE_NATIVE_ENTRY(Closure_clone, 1) { |
| 79 const Instance& receiver = Instance::CheckedHandle( | 75 const Closure& receiver = Closure::CheckedHandle( |
| 80 zone, arguments->NativeArgAt(0)); | 76 zone, arguments->NativeArgAt(0)); |
| 81 ASSERT(receiver.IsClosure()); | 77 const Function& func = Function::Handle(zone, receiver.function()); |
| 82 if (receiver.IsClosure()) { | 78 const Context& ctx = Context::Handle(zone, receiver.context()); |
| 83 const Function& func = | 79 Context& cloned_ctx = |
| 84 Function::Handle(zone, Closure::function(receiver)); | 80 Context::Handle(zone, Context::New(ctx.num_variables())); |
| 85 const Context& ctx = | 81 cloned_ctx.set_parent(Context::Handle(zone, ctx.parent())); |
| 86 Context::Handle(zone, Closure::context(receiver)); | 82 Object& inst = Object::Handle(zone); |
| 87 Context& cloned_ctx = | 83 for (int i = 0; i < ctx.num_variables(); i++) { |
| 88 Context::Handle(zone, Context::New(ctx.num_variables())); | 84 inst = ctx.At(i); |
| 89 cloned_ctx.set_parent(Context::Handle(zone, ctx.parent())); | 85 cloned_ctx.SetAt(i, inst); |
| 90 Object& inst = Object::Handle(zone); | |
| 91 for (int i = 0; i < ctx.num_variables(); i++) { | |
| 92 inst = ctx.At(i); | |
| 93 cloned_ctx.SetAt(i, inst); | |
| 94 } | |
| 95 return Closure::New(func, cloned_ctx); | |
| 96 } | 86 } |
| 97 return Object::null(); | 87 return Closure::New(func, cloned_ctx); |
| 98 } | 88 } |
| 99 | 89 |
| 100 | 90 |
| 101 } // namespace dart | 91 } // namespace dart |
| OLD | NEW |