mirror of
https://github.com/dart-lang/sdk
synced 2024-11-02 14:32:24 +00:00
Produce the correct assertion failure expression in the face of inlining
Fixes #27671 BUG= R=asiva@google.com Review URL: https://codereview.chromium.org/2453263002 .
This commit is contained in:
parent
f276f3422f
commit
8007a98b5b
3 changed files with 83 additions and 2 deletions
|
@ -7,9 +7,49 @@
|
||||||
#include "vm/object_store.h"
|
#include "vm/object_store.h"
|
||||||
#include "vm/runtime_entry.h"
|
#include "vm/runtime_entry.h"
|
||||||
#include "vm/stack_frame.h"
|
#include "vm/stack_frame.h"
|
||||||
|
#include "vm/symbols.h"
|
||||||
|
|
||||||
namespace dart {
|
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 RawScript* FindScript(DartFrameIterator* iterator) {
|
||||||
|
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;
|
||||||
|
while (stack_frame != NULL) {
|
||||||
|
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.raw());
|
||||||
|
inlined_iterator.Advance();
|
||||||
|
}
|
||||||
|
} 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.raw());
|
||||||
|
}
|
||||||
|
stack_frame = iterator->NextFrame();
|
||||||
|
}
|
||||||
|
UNREACHABLE();
|
||||||
|
return Script::null();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// Allocate and throw a new AssertionError.
|
// Allocate and throw a new AssertionError.
|
||||||
// Arg0: index of the first token of the failed assertion.
|
// Arg0: index of the first token of the failed assertion.
|
||||||
// Arg1: index of the first token after the failed assertion.
|
// Arg1: index of the first token after the failed assertion.
|
||||||
|
@ -26,8 +66,7 @@ DEFINE_NATIVE_ENTRY(AssertionError_throwNew, 2) {
|
||||||
|
|
||||||
DartFrameIterator iterator;
|
DartFrameIterator iterator;
|
||||||
iterator.NextFrame(); // Skip native call.
|
iterator.NextFrame(); // Skip native call.
|
||||||
iterator.NextFrame(); // Skip _AssertionError._checkAssertion frame
|
const Script& script = Script::Handle(FindScript(&iterator));
|
||||||
const Script& script = Script::Handle(Exceptions::GetCallerScript(&iterator));
|
|
||||||
|
|
||||||
// Initialize argument 'failed_assertion' with source snippet.
|
// Initialize argument 'failed_assertion' with source snippet.
|
||||||
intptr_t from_line, from_column;
|
intptr_t from_line, from_column;
|
||||||
|
|
10
tests/language/vm/regress_27671_other.dart
Normal file
10
tests/language/vm/regress_27671_other.dart
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
// Copyright (c) 2016, 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.
|
||||||
|
|
||||||
|
import 'regress_27671_test.dart';
|
||||||
|
|
||||||
|
@AlwaysInline
|
||||||
|
void check(f, x) {
|
||||||
|
assert(f(x) && true);
|
||||||
|
}
|
32
tests/language/vm/regress_27671_test.dart
Normal file
32
tests/language/vm/regress_27671_test.dart
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
// Copyright (c) 2016, 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.
|
||||||
|
//
|
||||||
|
// VMOptions=--enable_asserts --enable-inlining-annotations --optimization-counter-threshold=10 --no-background-compilation
|
||||||
|
|
||||||
|
import 'package:expect/expect.dart';
|
||||||
|
import 'regress_27671_other.dart';
|
||||||
|
|
||||||
|
const AlwaysInline = "AlwaysInline";
|
||||||
|
const NeverInline = "NeverInline";
|
||||||
|
|
||||||
|
@AlwaysInline
|
||||||
|
bounce(x) {
|
||||||
|
for (int i = 0; i < 10; i++) {
|
||||||
|
check(f, x);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@AlwaysInline
|
||||||
|
bool f(y) => y > 0;
|
||||||
|
|
||||||
|
main() {
|
||||||
|
for (int i = 0; i < 100; i++) {
|
||||||
|
bounce(1);
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
bounce(-1);
|
||||||
|
} catch (e) {
|
||||||
|
Expect.isTrue(e.toString().contains('f(x) && true'));
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue