Introduce local function declarations in Crankshaft and fix issue 1647.

We have to emit code for declarations later into the body block
(and not into the start block) so that the environment contains
the correct values.

In order to capture the environment effect of the declarations
that generate code (function declarations) I inserted a separate
AST id and a HSimulate after the declarations are visited.

Also fixes handling deopt in named function expressions:
BUG=v8:1647
TEST=test/mjsunit/regress/regress-fundecl.js, test/mjsunit/regress/regress-1647.js
Review URL: http://codereview.chromium.org/7776009

git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@9083 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
parent 829e174a
...@@ -258,6 +258,7 @@ void FullCodeGenerator::Generate(CompilationInfo* info) { ...@@ -258,6 +258,7 @@ void FullCodeGenerator::Generate(CompilationInfo* info) {
scope()->VisitIllegalRedeclaration(this); scope()->VisitIllegalRedeclaration(this);
} else { } else {
PrepareForBailoutForId(AstNode::kFunctionEntryId, NO_REGISTERS);
{ Comment cmnt(masm_, "[ Declarations"); { Comment cmnt(masm_, "[ Declarations");
// For named function expressions, declare the function name as a // For named function expressions, declare the function name as a
// constant. // constant.
...@@ -268,7 +269,7 @@ void FullCodeGenerator::Generate(CompilationInfo* info) { ...@@ -268,7 +269,7 @@ void FullCodeGenerator::Generate(CompilationInfo* info) {
} }
{ Comment cmnt(masm_, "[ Stack check"); { Comment cmnt(masm_, "[ Stack check");
PrepareForBailoutForId(AstNode::kFunctionEntryId, NO_REGISTERS); PrepareForBailoutForId(AstNode::kDeclarationsId, NO_REGISTERS);
Label ok; Label ok;
__ LoadRoot(ip, Heap::kStackLimitRootIndex); __ LoadRoot(ip, Heap::kStackLimitRootIndex);
__ cmp(sp, Operand(ip)); __ cmp(sp, Operand(ip));
......
...@@ -134,6 +134,10 @@ class AstNode: public ZoneObject { ...@@ -134,6 +134,10 @@ class AstNode: public ZoneObject {
static const int kNoNumber = -1; static const int kNoNumber = -1;
static const int kFunctionEntryId = 2; // Using 0 could disguise errors. static const int kFunctionEntryId = 2; // Using 0 could disguise errors.
// This AST id identifies the point after the declarations have been
// visited. We need it to capture the environment effects of declarations
// that emit code (function declarations).
static const int kDeclarationsId = 3;
// Override ZoneObject's new to count allocated AST nodes. // Override ZoneObject's new to count allocated AST nodes.
void* operator new(size_t size, Zone* zone) { void* operator new(size_t size, Zone* zone) {
......
...@@ -2273,10 +2273,6 @@ HGraph* HGraphBuilder::CreateGraph() { ...@@ -2273,10 +2273,6 @@ HGraph* HGraphBuilder::CreateGraph() {
return NULL; return NULL;
} }
SetupScope(scope); SetupScope(scope);
VisitDeclarations(scope->declarations());
HValue* context = environment()->LookupContext();
AddInstruction(
new(zone()) HStackCheck(context, HStackCheck::kFunctionEntry));
// Add an edge to the body entry. This is warty: the graph's start // Add an edge to the body entry. This is warty: the graph's start
// environment will be used by the Lithium translation as the initial // environment will be used by the Lithium translation as the initial
...@@ -2298,6 +2294,23 @@ HGraph* HGraphBuilder::CreateGraph() { ...@@ -2298,6 +2294,23 @@ HGraph* HGraphBuilder::CreateGraph() {
current_block()->Goto(body_entry); current_block()->Goto(body_entry);
body_entry->SetJoinId(AstNode::kFunctionEntryId); body_entry->SetJoinId(AstNode::kFunctionEntryId);
set_current_block(body_entry); set_current_block(body_entry);
// Handle implicit declaration of the function name in named function
// expressions before other declarations.
if (scope->is_function_scope() && scope->function() != NULL) {
if (!scope->function()->IsStackAllocated()) {
Bailout("unsupported declaration");
return NULL;
}
environment()->Bind(scope->function(), graph()->GetConstantHole());
}
VisitDeclarations(scope->declarations());
AddSimulate(AstNode::kDeclarationsId);
HValue* context = environment()->LookupContext();
AddInstruction(
new(zone()) HStackCheck(context, HStackCheck::kFunctionEntry));
VisitStatements(info()->function()->body()); VisitStatements(info()->function()->body());
if (HasStackOverflow()) return NULL; if (HasStackOverflow()) return NULL;
...@@ -5810,7 +5823,6 @@ void HGraphBuilder::VisitDeclaration(Declaration* decl) { ...@@ -5810,7 +5823,6 @@ void HGraphBuilder::VisitDeclaration(Declaration* decl) {
// We support only declarations that do not require code generation. // We support only declarations that do not require code generation.
Variable* var = decl->proxy()->var(); Variable* var = decl->proxy()->var();
if (!var->IsStackAllocated() || if (!var->IsStackAllocated() ||
decl->fun() != NULL ||
decl->mode() == Variable::LET) { decl->mode() == Variable::LET) {
return Bailout("unsupported declaration"); return Bailout("unsupported declaration");
} }
...@@ -5818,6 +5830,10 @@ void HGraphBuilder::VisitDeclaration(Declaration* decl) { ...@@ -5818,6 +5830,10 @@ void HGraphBuilder::VisitDeclaration(Declaration* decl) {
if (decl->mode() == Variable::CONST) { if (decl->mode() == Variable::CONST) {
ASSERT(var->IsStackAllocated()); ASSERT(var->IsStackAllocated());
environment()->Bind(var, graph()->GetConstantHole()); environment()->Bind(var, graph()->GetConstantHole());
} else if (decl->fun() != NULL) {
VisitForValue(decl->fun());
HValue* function = Pop();
environment()->Bind(var, function);
} }
} }
......
...@@ -255,6 +255,7 @@ void FullCodeGenerator::Generate(CompilationInfo* info) { ...@@ -255,6 +255,7 @@ void FullCodeGenerator::Generate(CompilationInfo* info) {
scope()->VisitIllegalRedeclaration(this); scope()->VisitIllegalRedeclaration(this);
} else { } else {
PrepareForBailoutForId(AstNode::kFunctionEntryId, NO_REGISTERS);
{ Comment cmnt(masm_, "[ Declarations"); { Comment cmnt(masm_, "[ Declarations");
// For named function expressions, declare the function name as a // For named function expressions, declare the function name as a
// constant. // constant.
...@@ -265,7 +266,7 @@ void FullCodeGenerator::Generate(CompilationInfo* info) { ...@@ -265,7 +266,7 @@ void FullCodeGenerator::Generate(CompilationInfo* info) {
} }
{ Comment cmnt(masm_, "[ Stack check"); { Comment cmnt(masm_, "[ Stack check");
PrepareForBailoutForId(AstNode::kFunctionEntryId, NO_REGISTERS); PrepareForBailoutForId(AstNode::kDeclarationsId, NO_REGISTERS);
Label ok; Label ok;
ExternalReference stack_limit = ExternalReference stack_limit =
ExternalReference::address_of_stack_limit(isolate()); ExternalReference::address_of_stack_limit(isolate());
......
...@@ -266,6 +266,7 @@ void FullCodeGenerator::Generate(CompilationInfo* info) { ...@@ -266,6 +266,7 @@ void FullCodeGenerator::Generate(CompilationInfo* info) {
scope()->VisitIllegalRedeclaration(this); scope()->VisitIllegalRedeclaration(this);
} else { } else {
PrepareForBailoutForId(AstNode::kFunctionEntryId, NO_REGISTERS);
{ Comment cmnt(masm_, "[ Declarations"); { Comment cmnt(masm_, "[ Declarations");
// For named function expressions, declare the function name as a // For named function expressions, declare the function name as a
// constant. // constant.
...@@ -276,7 +277,7 @@ void FullCodeGenerator::Generate(CompilationInfo* info) { ...@@ -276,7 +277,7 @@ void FullCodeGenerator::Generate(CompilationInfo* info) {
} }
{ Comment cmnt(masm_, "[ Stack check"); { Comment cmnt(masm_, "[ Stack check");
PrepareForBailoutForId(AstNode::kFunctionEntryId, NO_REGISTERS); PrepareForBailoutForId(AstNode::kDeclarationsId, NO_REGISTERS);
Label ok; Label ok;
__ LoadRoot(t0, Heap::kStackLimitRootIndex); __ LoadRoot(t0, Heap::kStackLimitRootIndex);
__ Branch(&ok, hs, sp, Operand(t0)); __ Branch(&ok, hs, sp, Operand(t0));
......
...@@ -532,7 +532,7 @@ LexicalScope::LexicalScope(Parser* parser, Scope* scope, Isolate* isolate) ...@@ -532,7 +532,7 @@ LexicalScope::LexicalScope(Parser* parser, Scope* scope, Isolate* isolate)
parser->top_scope_ = scope; parser->top_scope_ = scope;
parser->lexical_scope_ = this; parser->lexical_scope_ = this;
parser->with_nesting_level_ = 0; parser->with_nesting_level_ = 0;
isolate->set_ast_node_id(AstNode::kFunctionEntryId + 1); isolate->set_ast_node_id(AstNode::kDeclarationsId + 1);
} }
......
...@@ -245,6 +245,7 @@ void FullCodeGenerator::Generate(CompilationInfo* info) { ...@@ -245,6 +245,7 @@ void FullCodeGenerator::Generate(CompilationInfo* info) {
Comment cmnt(masm_, "[ Declarations"); Comment cmnt(masm_, "[ Declarations");
scope()->VisitIllegalRedeclaration(this); scope()->VisitIllegalRedeclaration(this);
} else { } else {
PrepareForBailoutForId(AstNode::kFunctionEntryId, NO_REGISTERS);
{ Comment cmnt(masm_, "[ Declarations"); { Comment cmnt(masm_, "[ Declarations");
// For named function expressions, declare the function name as a // For named function expressions, declare the function name as a
// constant. // constant.
...@@ -255,7 +256,7 @@ void FullCodeGenerator::Generate(CompilationInfo* info) { ...@@ -255,7 +256,7 @@ void FullCodeGenerator::Generate(CompilationInfo* info) {
} }
{ Comment cmnt(masm_, "[ Stack check"); { Comment cmnt(masm_, "[ Stack check");
PrepareForBailoutForId(AstNode::kFunctionEntryId, NO_REGISTERS); PrepareForBailoutForId(AstNode::kDeclarationsId, NO_REGISTERS);
Label ok; Label ok;
__ CompareRoot(rsp, Heap::kStackLimitRootIndex); __ CompareRoot(rsp, Heap::kStackLimitRootIndex);
__ j(above_equal, &ok, Label::kNear); __ j(above_equal, &ok, Label::kNear);
......
// Copyright 2011 the V8 project authors. All rights reserved.
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are
// met:
//
// * Redistributions of source code must retain the above copyright
// notice, this list of conditions and the following disclaimer.
// * Redistributions in binary form must reproduce the above
// copyright notice, this list of conditions and the following
// disclaimer in the documentation and/or other materials provided
// with the distribution.
// * Neither the name of Google Inc. nor the names of its
// contributors may be used to endorse or promote products derived
// from this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
// Flags: --allow-natives-syntax
// Test for correct deoptimization in named function expressions.
var t = { foo: function() {} };
var f = (function bar() {
t.foo();
assertEquals("function", typeof bar);
});
for (var i = 0; i < 10; i++) f();
%OptimizeFunctionOnNextCall(f);
t.number = 2;
f();
// Copyright 2011 the V8 project authors. All rights reserved.
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are
// met:
//
// * Redistributions of source code must retain the above copyright
// notice, this list of conditions and the following disclaimer.
// * Redistributions in binary form must reproduce the above
// copyright notice, this list of conditions and the following
// disclaimer in the documentation and/or other materials provided
// with the distribution.
// * Neither the name of Google Inc. nor the names of its
// contributors may be used to endorse or promote products derived
// from this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
// Flags: --allow-natives-syntax
// Test hoisting of function declarations in the optimizing
// compiler in case of deoptimization.
function h(a, b) {
var r = a + b;
function X() { return 42; }
return r + X();
}
for (var i = 0; i < 5; i++) h(1,2);
%OptimizeFunctionOnNextCall(h);
assertEquals(45, h(1,2));
assertEquals("foo742", h("foo", 7));
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment