Do not check for interrupt when allocating stack locals.
R=dcarney@chromium.org BUG=357137 LOG=N Review URL: https://codereview.chromium.org/219373004 git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@20357 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
This commit is contained in:
parent
989683d479
commit
c0fa861726
@ -118,10 +118,14 @@ static void EmitStackCheck(MacroAssembler* masm_,
|
|||||||
Isolate* isolate = masm_->isolate();
|
Isolate* isolate = masm_->isolate();
|
||||||
Label ok;
|
Label ok;
|
||||||
ASSERT(scratch.is(sp) == (pointers == 0));
|
ASSERT(scratch.is(sp) == (pointers == 0));
|
||||||
|
Heap::RootListIndex index;
|
||||||
if (pointers != 0) {
|
if (pointers != 0) {
|
||||||
__ sub(scratch, sp, Operand(pointers * kPointerSize));
|
__ sub(scratch, sp, Operand(pointers * kPointerSize));
|
||||||
|
index = Heap::kRealStackLimitRootIndex;
|
||||||
|
} else {
|
||||||
|
index = Heap::kStackLimitRootIndex;
|
||||||
}
|
}
|
||||||
__ LoadRoot(stack_limit_scratch, Heap::kStackLimitRootIndex);
|
__ LoadRoot(stack_limit_scratch, index);
|
||||||
__ cmp(scratch, Operand(stack_limit_scratch));
|
__ cmp(scratch, Operand(stack_limit_scratch));
|
||||||
__ b(hs, &ok);
|
__ b(hs, &ok);
|
||||||
PredictableCodeSizeScope predictable(masm_, 2 * Assembler::kInstrSize);
|
PredictableCodeSizeScope predictable(masm_, 2 * Assembler::kInstrSize);
|
||||||
|
@ -117,10 +117,14 @@ static void EmitStackCheck(MacroAssembler* masm_,
|
|||||||
Label ok;
|
Label ok;
|
||||||
ASSERT(jssp.Is(__ StackPointer()));
|
ASSERT(jssp.Is(__ StackPointer()));
|
||||||
ASSERT(scratch.Is(jssp) == (pointers == 0));
|
ASSERT(scratch.Is(jssp) == (pointers == 0));
|
||||||
|
Heap::RootListIndex index;
|
||||||
if (pointers != 0) {
|
if (pointers != 0) {
|
||||||
__ Sub(scratch, jssp, pointers * kPointerSize);
|
__ Sub(scratch, jssp, pointers * kPointerSize);
|
||||||
|
index = Heap::kRealStackLimitRootIndex;
|
||||||
|
} else {
|
||||||
|
index = Heap::kStackLimitRootIndex;
|
||||||
}
|
}
|
||||||
__ CompareRoot(scratch, Heap::kStackLimitRootIndex);
|
__ CompareRoot(scratch, index);
|
||||||
__ B(hs, &ok);
|
__ B(hs, &ok);
|
||||||
PredictableCodeSizeScope predictable(masm_,
|
PredictableCodeSizeScope predictable(masm_,
|
||||||
Assembler::kCallSizeWithRelocation);
|
Assembler::kCallSizeWithRelocation);
|
||||||
|
@ -106,12 +106,14 @@ static void EmitStackCheck(MacroAssembler* masm_,
|
|||||||
Register scratch = esp) {
|
Register scratch = esp) {
|
||||||
Label ok;
|
Label ok;
|
||||||
Isolate* isolate = masm_->isolate();
|
Isolate* isolate = masm_->isolate();
|
||||||
ExternalReference stack_limit =
|
|
||||||
ExternalReference::address_of_stack_limit(isolate);
|
|
||||||
ASSERT(scratch.is(esp) == (pointers == 0));
|
ASSERT(scratch.is(esp) == (pointers == 0));
|
||||||
|
ExternalReference stack_limit;
|
||||||
if (pointers != 0) {
|
if (pointers != 0) {
|
||||||
__ mov(scratch, esp);
|
__ mov(scratch, esp);
|
||||||
__ sub(scratch, Immediate(pointers * kPointerSize));
|
__ sub(scratch, Immediate(pointers * kPointerSize));
|
||||||
|
stack_limit = ExternalReference::address_of_real_stack_limit(isolate);
|
||||||
|
} else {
|
||||||
|
stack_limit = ExternalReference::address_of_stack_limit(isolate);
|
||||||
}
|
}
|
||||||
__ cmp(scratch, Operand::StaticVariable(stack_limit));
|
__ cmp(scratch, Operand::StaticVariable(stack_limit));
|
||||||
__ j(above_equal, &ok, Label::kNear);
|
__ j(above_equal, &ok, Label::kNear);
|
||||||
|
@ -127,10 +127,14 @@ static void EmitStackCheck(MacroAssembler* masm_,
|
|||||||
Isolate* isolate = masm_->isolate();
|
Isolate* isolate = masm_->isolate();
|
||||||
Label ok;
|
Label ok;
|
||||||
ASSERT(scratch.is(sp) == (pointers == 0));
|
ASSERT(scratch.is(sp) == (pointers == 0));
|
||||||
|
Heap::RootListIndex index;
|
||||||
if (pointers != 0) {
|
if (pointers != 0) {
|
||||||
__ Subu(scratch, sp, Operand(pointers * kPointerSize));
|
__ Subu(scratch, sp, Operand(pointers * kPointerSize));
|
||||||
|
index = Heap::kRealStackLimitRootIndex;
|
||||||
|
} else {
|
||||||
|
index = Heap::kStackLimitRootIndex;
|
||||||
}
|
}
|
||||||
__ LoadRoot(stack_limit_scratch, Heap::kStackLimitRootIndex);
|
__ LoadRoot(stack_limit_scratch, index);
|
||||||
__ Branch(&ok, hs, scratch, Operand(stack_limit_scratch));
|
__ Branch(&ok, hs, scratch, Operand(stack_limit_scratch));
|
||||||
PredictableCodeSizeScope predictable(masm_, 4 * Assembler::kInstrSize);
|
PredictableCodeSizeScope predictable(masm_, 4 * Assembler::kInstrSize);
|
||||||
__ Call(isolate->builtins()->StackCheck(), RelocInfo::CODE_TARGET);
|
__ Call(isolate->builtins()->StackCheck(), RelocInfo::CODE_TARGET);
|
||||||
|
@ -107,11 +107,15 @@ static void EmitStackCheck(MacroAssembler* masm_,
|
|||||||
Isolate* isolate = masm_->isolate();
|
Isolate* isolate = masm_->isolate();
|
||||||
Label ok;
|
Label ok;
|
||||||
ASSERT(scratch.is(rsp) == (pointers == 0));
|
ASSERT(scratch.is(rsp) == (pointers == 0));
|
||||||
|
Heap::RootListIndex index;
|
||||||
if (pointers != 0) {
|
if (pointers != 0) {
|
||||||
__ movp(scratch, rsp);
|
__ movp(scratch, rsp);
|
||||||
__ subp(scratch, Immediate(pointers * kPointerSize));
|
__ subp(scratch, Immediate(pointers * kPointerSize));
|
||||||
|
index = Heap::kRealStackLimitRootIndex;
|
||||||
|
} else {
|
||||||
|
index = Heap::kStackLimitRootIndex;
|
||||||
}
|
}
|
||||||
__ CompareRoot(scratch, Heap::kStackLimitRootIndex);
|
__ CompareRoot(scratch, index);
|
||||||
__ j(above_equal, &ok, Label::kNear);
|
__ j(above_equal, &ok, Label::kNear);
|
||||||
__ call(isolate->builtins()->StackCheck(), RelocInfo::CODE_TARGET);
|
__ call(isolate->builtins()->StackCheck(), RelocInfo::CODE_TARGET);
|
||||||
__ bind(&ok);
|
__ bind(&ok);
|
||||||
|
@ -3913,3 +3913,32 @@ TEST(CEntryStubOOM) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#endif // DEBUG
|
#endif // DEBUG
|
||||||
|
|
||||||
|
|
||||||
|
static void InterruptCallback357137(v8::Isolate* isolate, void* data) { }
|
||||||
|
|
||||||
|
|
||||||
|
static void RequestInterrupt(const v8::FunctionCallbackInfo<v8::Value>& args) {
|
||||||
|
CcTest::isolate()->RequestInterrupt(&InterruptCallback357137, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
TEST(Regress357137) {
|
||||||
|
CcTest::InitializeVM();
|
||||||
|
v8::Isolate* isolate = CcTest::isolate();
|
||||||
|
v8::HandleScope hscope(isolate);
|
||||||
|
v8::Handle<v8::ObjectTemplate> global =v8::ObjectTemplate::New(isolate);
|
||||||
|
global->Set(v8::String::NewFromUtf8(isolate, "interrupt"),
|
||||||
|
v8::FunctionTemplate::New(isolate, RequestInterrupt));
|
||||||
|
v8::Local<v8::Context> context = v8::Context::New(isolate, NULL, global);
|
||||||
|
ASSERT(!context.IsEmpty());
|
||||||
|
v8::Context::Scope cscope(context);
|
||||||
|
|
||||||
|
v8::Local<v8::Value> result = CompileRun(
|
||||||
|
"var locals = '';"
|
||||||
|
"for (var i = 0; i < 512; i++) locals += 'var v' + i + '= 42;';"
|
||||||
|
"eval('function f() {' + locals + 'return function() { return v0; }; }');"
|
||||||
|
"interrupt();" // This triggers a fake stack overflow in f.
|
||||||
|
"f()()");
|
||||||
|
CHECK_EQ(42.0, result->ToNumber()->Value());
|
||||||
|
}
|
||||||
|
8
test/mjsunit/regress/regress-crbug-357137.js
Normal file
8
test/mjsunit/regress/regress-crbug-357137.js
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
// Copyright 2014 the V8 project authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style license that can be
|
||||||
|
// found in the LICENSE file.
|
||||||
|
|
||||||
|
var locals = "";
|
||||||
|
for (var i = 0; i < 1024; i++) locals += "var v" + i + ";";
|
||||||
|
eval("function f() {" + locals + "f();}");
|
||||||
|
assertThrows("f()", RangeError);
|
Loading…
Reference in New Issue
Block a user