|  | // 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. | 
|  |  | 
|  | #include "vm/bootstrap_natives.h" | 
|  | #include "vm/exceptions.h" | 
|  | #include "vm/object_store.h" | 
|  | #include "vm/runtime_entry.h" | 
|  | #include "vm/stack_frame.h" | 
|  | #include "vm/symbols.h" | 
|  |  | 
|  | namespace dart { | 
|  |  | 
|  | // Scan the stack until we hit the first function in the _AssertionError | 
|  | // class. We then return the next frame's script taking inlining into account. | 
|  | static ScriptPtr FindScript(DartFrameIterator* iterator) { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | // The precompiled runtime faces two issues in recovering the correct | 
|  | // assertion text. First, the precompiled runtime does not include | 
|  | // the inlining meta-data so we cannot walk the inline-aware stack trace. | 
|  | // Second, the script text itself is missing so whatever script is returned | 
|  | // from here will be missing the assertion expression text. | 
|  | iterator->NextFrame();  // Skip _AssertionError._evaluateAssertion frame | 
|  | return Exceptions::GetCallerScript(iterator); | 
|  | #else | 
|  | StackFrame* stack_frame = iterator->NextFrame(); | 
|  | Code& code = Code::Handle(); | 
|  | Function& func = Function::Handle(); | 
|  | const Class& assert_error_class = | 
|  | Class::Handle(Library::LookupCoreClass(Symbols::AssertionError())); | 
|  | ASSERT(!assert_error_class.IsNull()); | 
|  | bool hit_assertion_error = false; | 
|  | for (; stack_frame != nullptr; stack_frame = iterator->NextFrame()) { | 
|  | if (stack_frame->is_interpreted()) { | 
|  | func = stack_frame->LookupDartFunction(); | 
|  | } else { | 
|  | code = stack_frame->LookupDartCode(); | 
|  | if (code.is_optimized()) { | 
|  | InlinedFunctionsIterator inlined_iterator(code, stack_frame->pc()); | 
|  | while (!inlined_iterator.Done()) { | 
|  | func = inlined_iterator.function(); | 
|  | if (hit_assertion_error) { | 
|  | return func.script(); | 
|  | } | 
|  | ASSERT(!hit_assertion_error); | 
|  | hit_assertion_error = (func.Owner() == assert_error_class.ptr()); | 
|  | inlined_iterator.Advance(); | 
|  | } | 
|  | continue; | 
|  | } else { | 
|  | func = code.function(); | 
|  | } | 
|  | } | 
|  | ASSERT(!func.IsNull()); | 
|  | if (hit_assertion_error) { | 
|  | return func.script(); | 
|  | } | 
|  | ASSERT(!hit_assertion_error); | 
|  | hit_assertion_error = (func.Owner() == assert_error_class.ptr()); | 
|  | } | 
|  | UNREACHABLE(); | 
|  | return Script::null(); | 
|  | #endif  // defined(DART_PRECOMPILED_RUNTIME) | 
|  | } | 
|  |  | 
|  | // Allocate and throw a new AssertionError. | 
|  | // Arg0: index of the first token of the failed assertion. | 
|  | // Arg1: index of the first token after the failed assertion. | 
|  | // Arg2: Message object or null. | 
|  | // Return value: none, throws an exception. | 
|  | DEFINE_NATIVE_ENTRY(AssertionError_throwNew, 0, 3) { | 
|  | // No need to type check the arguments. This function can only be called | 
|  | // internally from the VM. | 
|  | const TokenPosition assertion_start = TokenPosition::Deserialize( | 
|  | Smi::CheckedHandle(zone, arguments->NativeArgAt(0)).Value()); | 
|  | const TokenPosition assertion_end = TokenPosition::Deserialize( | 
|  | Smi::CheckedHandle(zone, arguments->NativeArgAt(1)).Value()); | 
|  |  | 
|  | const Instance& message = | 
|  | Instance::CheckedHandle(zone, arguments->NativeArgAt(2)); | 
|  | const Array& args = Array::Handle(zone, Array::New(5)); | 
|  |  | 
|  | DartFrameIterator iterator(thread, | 
|  | StackFrameIterator::kNoCrossThreadIteration); | 
|  | iterator.NextFrame();  // Skip native call. | 
|  | const Script& script = Script::Handle(FindScript(&iterator)); | 
|  |  | 
|  | // Initialize argument 'failed_assertion' with source snippet. | 
|  | auto& condition_text = String::Handle(); | 
|  | // Extract the assertion condition text (if source is available). | 
|  | intptr_t from_line = -1, from_column = -1; | 
|  | String& url = String::Handle(); | 
|  | if (!script.IsNull()) { | 
|  | if (script.GetTokenLocation(assertion_start, &from_line, &from_column)) { | 
|  | // Extract the assertion condition text (if source is available). | 
|  | intptr_t to_line, to_column; | 
|  | script.GetTokenLocation(assertion_end, &to_line, &to_column); | 
|  | condition_text = | 
|  | script.GetSnippet(from_line, from_column, to_line, to_column); | 
|  | } | 
|  | url = script.url(); | 
|  | } | 
|  | if (condition_text.IsNull()) { | 
|  | condition_text = Symbols::OptimizedOut().ptr(); | 
|  | } | 
|  | args.SetAt(0, condition_text); | 
|  |  | 
|  | // Initialize location arguments starting at position 1. | 
|  | args.SetAt(1, url); | 
|  | args.SetAt(2, Smi::Handle(Smi::New(from_line))); | 
|  | args.SetAt(3, Smi::Handle(Smi::New(from_column))); | 
|  | args.SetAt(4, message); | 
|  |  | 
|  | Exceptions::ThrowByType(Exceptions::kAssertion, args); | 
|  | UNREACHABLE(); | 
|  | return Object::null(); | 
|  | } | 
|  |  | 
|  | // Allocate and throw a new AssertionError. | 
|  | // Arg0: Source code snippet of failed assertion. | 
|  | // Arg1: Script url string. | 
|  | // Arg2: Line number. | 
|  | // Arg3: Column number. | 
|  | // Arg4: Message object or null. | 
|  | // Return value: none, throws an exception. | 
|  | DEFINE_NATIVE_ENTRY(AssertionError_throwNewSource, 0, 5) { | 
|  | // No need to type check the arguments. This function can only be called | 
|  | // internally from the VM. | 
|  | const String& failed_assertion = | 
|  | String::CheckedHandle(zone, arguments->NativeArgAt(0)); | 
|  | const String& script_url = | 
|  | String::CheckedHandle(zone, arguments->NativeArgAt(1)); | 
|  | const intptr_t line = | 
|  | Smi::CheckedHandle(zone, arguments->NativeArgAt(2)).Value(); | 
|  | const intptr_t column = | 
|  | Smi::CheckedHandle(zone, arguments->NativeArgAt(3)).Value(); | 
|  | const Instance& message = | 
|  | Instance::CheckedHandle(zone, arguments->NativeArgAt(4)); | 
|  |  | 
|  | const Array& args = Array::Handle(zone, Array::New(5)); | 
|  |  | 
|  | args.SetAt(0, failed_assertion); | 
|  | args.SetAt(1, script_url); | 
|  | args.SetAt(2, Smi::Handle(zone, Smi::New(line))); | 
|  | args.SetAt(3, Smi::Handle(zone, Smi::New(column))); | 
|  | args.SetAt(4, message); | 
|  |  | 
|  | Exceptions::ThrowByType(Exceptions::kAssertion, args); | 
|  | UNREACHABLE(); | 
|  | return Object::null(); | 
|  | } | 
|  |  | 
|  | // Allocate and throw a new TypeError. | 
|  | // Arg0: index of the token of the failed type check. | 
|  | // Arg1: src value. | 
|  | // Arg2: dst type. | 
|  | // Arg3: dst name. | 
|  | // Return value: none, throws an exception. | 
|  | DEFINE_NATIVE_ENTRY(TypeError_throwNew, 0, 4) { | 
|  | // No need to type check the arguments. This function can only be called | 
|  | // internally from the VM. | 
|  | const TokenPosition location = TokenPosition::Deserialize( | 
|  | Smi::CheckedHandle(zone, arguments->NativeArgAt(0)).Value()); | 
|  | const Instance& src_value = | 
|  | Instance::CheckedHandle(zone, arguments->NativeArgAt(1)); | 
|  | const AbstractType& dst_type = | 
|  | AbstractType::CheckedHandle(zone, arguments->NativeArgAt(2)); | 
|  | const String& dst_name = | 
|  | String::CheckedHandle(zone, arguments->NativeArgAt(3)); | 
|  | const AbstractType& src_type = | 
|  | AbstractType::Handle(src_value.GetType(Heap::kNew)); | 
|  | Exceptions::CreateAndThrowTypeError(location, src_type, dst_type, dst_name); | 
|  | UNREACHABLE(); | 
|  | return Object::null(); | 
|  | } | 
|  |  | 
|  | // Rethrow an error with a stacktrace. | 
|  | DEFINE_NATIVE_ENTRY(Error_throwWithStackTrace, 0, 2) { | 
|  | GET_NON_NULL_NATIVE_ARGUMENT(Instance, error, arguments->NativeArgAt(0)); | 
|  | GET_NON_NULL_NATIVE_ARGUMENT(Instance, stacktrace, arguments->NativeArgAt(1)); | 
|  | Exceptions::ThrowWithStackTrace(thread, error, stacktrace); | 
|  | return Object::null(); | 
|  | } | 
|  |  | 
|  | // Sets stack trace on object, if it's an `Error` with no `Error.stackTrace`. | 
|  | DEFINE_NATIVE_ENTRY(Error_trySetStackTrace, 0, 2) { | 
|  | GET_NON_NULL_NATIVE_ARGUMENT(Instance, error, arguments->NativeArgAt(0)); | 
|  | GET_NON_NULL_NATIVE_ARGUMENT(Instance, stacktrace, arguments->NativeArgAt(1)); | 
|  | Exceptions::TrySetStackTrace(zone, error, stacktrace); | 
|  | return Object::null(); | 
|  | } | 
|  |  | 
|  | }  // namespace dart |