Commit 3dbefbed authored by whesse@chromium.org's avatar whesse@chromium.org

Add AST analysis that flags expressions that will have ToInt32 applied to them.

Simplify AST analysis of side-effect-free int32 expressions.
Review URL: http://codereview.chromium.org/668256

git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@4056 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
parent b4e0375e
...@@ -237,27 +237,19 @@ class Expression: public AstNode { ...@@ -237,27 +237,19 @@ class Expression: public AstNode {
// side-effect free compiler. // side-effect free compiler.
bool side_effect_free() { return SideEffectFreeField::decode(bitfields_); } bool side_effect_free() { return SideEffectFreeField::decode(bitfields_); }
void set_side_effect_free(bool is_side_effect_free) { void set_side_effect_free(bool is_side_effect_free) {
bitfields_ = (bitfields_ & ~SideEffectFreeField::mask()) | bitfields_ &= ~SideEffectFreeField::mask();
SideEffectFreeField::encode(is_side_effect_free); bitfields_ |= SideEffectFreeField::encode(is_side_effect_free);
} }
// The number of unary and binary operations contained in the expression // Will ToInt32 ECMA 262-3 9.5) or ToUunt32 (ECMA 262-3 9.6)
// rooted at this node. Valid only if side_effect_free() is true. // be applied to the value of this expression?
int expression_size() { return ExpressionSizeField::decode(bitfields_); } // If so, we may be able to optimize the calculation of the value.
void set_expression_size(int expression_size) { bool to_int32() { return ToInt32Field::decode(bitfields_); }
bitfields_ = (bitfields_ & ~ExpressionSizeField::mask()) | void set_to_int32(bool to_int32) {
ExpressionSizeField::encode(Min(expression_size, ExpressionSizeMax)); bitfields_ &= ~ToInt32Field::mask();
bitfields_ |= ToInt32Field::encode(to_int32);
} }
// The number of expression stack slots needed to compute the expression
// rooted at this node. Does not count the slot needed by the value
// computed by this expression. Valid only if side_effect_free() is true.
int stack_height() { return StackHeightField::decode(bitfields_); }
void set_stack_height(int stack_height) {
bitfields_ &= ~StackHeightField::mask();
bitfields_ |=
StackHeightField::encode(Min(stack_height, StackHeightMax));
}
private: private:
uint32_t bitfields_; uint32_t bitfields_;
...@@ -266,30 +258,9 @@ class Expression: public AstNode { ...@@ -266,30 +258,9 @@ class Expression: public AstNode {
DefinitionInfo* def_; DefinitionInfo* def_;
ZoneList<DefinitionInfo*>* defined_vars_; ZoneList<DefinitionInfo*>* defined_vars_;
static const int SideEffectFreeFieldStart = 0; // Using template BitField<type, start, size>.
static const int SideEffectFreeFieldLength = 1; class SideEffectFreeField : public BitField<bool, 0, 1> {};
class SideEffectFreeField: public BitField<bool, class ToInt32Field : public BitField<bool, 1, 1> {};
SideEffectFreeFieldStart,
SideEffectFreeFieldLength> {
};
static const int ExpressionSizeFieldStart =
SideEffectFreeFieldStart + SideEffectFreeFieldLength;
static const int ExpressionSizeFieldLength = 5;
static const int ExpressionSizeMax = (1 << ExpressionSizeFieldLength) - 1;
class ExpressionSizeField: public BitField<int,
ExpressionSizeFieldStart,
ExpressionSizeFieldLength> {
};
static const int StackHeightFieldStart =
ExpressionSizeFieldStart + ExpressionSizeFieldLength;
static const int StackHeightFieldLength = 5;
static const int StackHeightMax = (1 << StackHeightFieldLength) - 1;
class StackHeightField: public BitField<int,
StackHeightFieldStart,
StackHeightFieldLength> {
};
}; };
......
...@@ -153,6 +153,8 @@ DEFINE_bool(always_fast_compiler, false, ...@@ -153,6 +153,8 @@ DEFINE_bool(always_fast_compiler, false,
"try to use the speculative optimizing backend for all code") "try to use the speculative optimizing backend for all code")
DEFINE_bool(trace_bailout, false, DEFINE_bool(trace_bailout, false,
"print reasons for falling back to using the classic V8 backend") "print reasons for falling back to using the classic V8 backend")
DEFINE_bool(safe_int32_compiler, false,
"enable optimized side-effect-free int32 expressions.")
DEFINE_bool(use_flow_graph, false, "perform flow-graph based optimizations") DEFINE_bool(use_flow_graph, false, "perform flow-graph based optimizations")
// compilation-cache.cc // compilation-cache.cc
......
...@@ -245,12 +245,13 @@ void AstOptimizer::VisitVariableProxy(VariableProxy* node) { ...@@ -245,12 +245,13 @@ void AstOptimizer::VisitVariableProxy(VariableProxy* node) {
func_name_inferrer_.PushName(var->name()); func_name_inferrer_.PushName(var->name());
} }
if (var->slot() != NULL) { if (FLAG_safe_int32_compiler) {
Slot* slot = var->slot(); Slot* slot = var->slot();
node->set_side_effect_free( if (slot != NULL) {
(slot->type() == Slot::LOCAL && !slot->is_arguments()) || node->set_side_effect_free(
slot->type() == Slot::PARAMETER); (slot->type() == Slot::LOCAL && !slot->is_arguments()) ||
// stack_height and expression_size remain 0. slot->type() == Slot::PARAMETER);
}
} }
} }
} }
...@@ -261,7 +262,6 @@ void AstOptimizer::VisitLiteral(Literal* node) { ...@@ -261,7 +262,6 @@ void AstOptimizer::VisitLiteral(Literal* node) {
if (literal->IsSmi()) { if (literal->IsSmi()) {
node->type()->SetAsLikelySmi(); node->type()->SetAsLikelySmi();
node->set_side_effect_free(true); node->set_side_effect_free(true);
// stack_height and expression_size remain 0.
} else if (literal->IsString()) { } else if (literal->IsString()) {
Handle<String> lit_str(Handle<String>::cast(literal)); Handle<String> lit_str(Handle<String>::cast(literal));
if (!Heap::prototype_symbol()->Equals(*lit_str)) { if (!Heap::prototype_symbol()->Equals(*lit_str)) {
...@@ -269,7 +269,6 @@ void AstOptimizer::VisitLiteral(Literal* node) { ...@@ -269,7 +269,6 @@ void AstOptimizer::VisitLiteral(Literal* node) {
} }
} else if (literal->IsHeapNumber()) { } else if (literal->IsHeapNumber()) {
node->set_side_effect_free(true); node->set_side_effect_free(true);
// stack_height and expression_size remain 0.
} }
} }
...@@ -427,22 +426,26 @@ void AstOptimizer::VisitCallRuntime(CallRuntime* node) { ...@@ -427,22 +426,26 @@ void AstOptimizer::VisitCallRuntime(CallRuntime* node) {
void AstOptimizer::VisitUnaryOperation(UnaryOperation* node) { void AstOptimizer::VisitUnaryOperation(UnaryOperation* node) {
Visit(node->expression()); Visit(node->expression());
switch (node->op()) { if (FLAG_safe_int32_compiler) {
case Token::ADD: switch (node->op()) {
case Token::SUB: case Token::BIT_NOT:
node->set_side_effect_free(node->expression()->side_effect_free()); node->expression()->set_to_int32(true);
node->set_expression_size(node->expression()->expression_size() + 1); // Fall through.
node->set_stack_height(node->expression()->stack_height()); case Token::ADD:
break; case Token::SUB:
case Token::DELETE: case Token::NOT:
case Token::TYPEOF: node->set_side_effect_free(node->expression()->side_effect_free());
case Token::VOID: break;
case Token::BIT_NOT: case Token::DELETE:
case Token::NOT: case Token::TYPEOF:
break; case Token::VOID:
default: break;
UNREACHABLE(); default:
break; UNREACHABLE();
break;
}
} else if (node->op() == Token::BIT_NOT) {
node->expression()->set_to_int32(true);
} }
} }
...@@ -472,6 +475,8 @@ void AstOptimizer::VisitBinaryOperation(BinaryOperation* node) { ...@@ -472,6 +475,8 @@ void AstOptimizer::VisitBinaryOperation(BinaryOperation* node) {
node->type()->SetAsLikelySmiIfUnknown(); node->type()->SetAsLikelySmiIfUnknown();
node->left()->type()->SetAsLikelySmiIfUnknown(); node->left()->type()->SetAsLikelySmiIfUnknown();
node->right()->type()->SetAsLikelySmiIfUnknown(); node->right()->type()->SetAsLikelySmiIfUnknown();
node->left()->set_to_int32(true);
node->right()->set_to_int32(true);
break; break;
case Token::ADD: case Token::ADD:
case Token::SUB: case Token::SUB:
...@@ -513,32 +518,31 @@ void AstOptimizer::VisitBinaryOperation(BinaryOperation* node) { ...@@ -513,32 +518,31 @@ void AstOptimizer::VisitBinaryOperation(BinaryOperation* node) {
} }
} }
} }
switch (node->op()) {
case Token::COMMA: if (FLAG_safe_int32_compiler) {
case Token::OR: switch (node->op()) {
case Token::AND: case Token::COMMA:
case Token::BIT_OR: case Token::OR:
case Token::BIT_XOR: case Token::AND:
case Token::BIT_AND: break;
case Token::SHL: case Token::BIT_OR:
case Token::SAR: case Token::BIT_XOR:
case Token::SHR: case Token::BIT_AND:
case Token::MOD: case Token::SHL:
break; case Token::SAR:
case Token::ADD: case Token::SHR:
case Token::SUB: case Token::ADD:
case Token::MUL: case Token::SUB:
case Token::DIV: case Token::MUL:
node->set_side_effect_free(node->left()->side_effect_free() && case Token::DIV:
node->right()->side_effect_free()); case Token::MOD:
node->set_expression_size(node->left()->expression_size() + node->set_side_effect_free(node->left()->side_effect_free() &&
node->right()->expression_size() + 1); node->right()->side_effect_free());
node->set_stack_height(Max(node->left()->stack_height(), break;
node->right()->stack_height() + 1)); default:
break; UNREACHABLE();
default: break;
UNREACHABLE(); }
break;
} }
} }
......
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