| Index: src/hydrogen.cc | 
| =================================================================== | 
| --- src/hydrogen.cc	(revision 9600) | 
| +++ src/hydrogen.cc	(working copy) | 
| @@ -5791,38 +5791,68 @@ | 
|  | 
|  | 
| void HGraphBuilder::HandleLiteralCompareTypeof(CompareOperation* expr, | 
| -                                               Expression* sub_expr, | 
| +                                               HTypeof* typeof_expr, | 
| Handle<String> check) { | 
| -  CHECK_ALIVE(VisitForTypeOf(sub_expr)); | 
| -  HValue* value = Pop(); | 
| +  HValue* value = typeof_expr->value(); | 
| HTypeofIsAndBranch* instr = new(zone()) HTypeofIsAndBranch(value, check); | 
| instr->set_position(expr->position()); | 
| -  return ast_context()->ReturnControl(instr, expr->id()); | 
| +  ast_context()->ReturnControl(instr, expr->id()); | 
| +  typeof_expr->DeleteAndReplaceWith(NULL); | 
| } | 
|  | 
|  | 
| -bool HGraphBuilder::TryLiteralCompare(CompareOperation* expr) { | 
| -  Expression *sub_expr; | 
| -  Handle<String> check; | 
| -  if (expr->IsLiteralCompareTypeof(&sub_expr, &check)) { | 
| -    HandleLiteralCompareTypeof(expr, sub_expr, check); | 
| +static bool MatchLiteralCompareNil(HValue* left, | 
| +                                   Token::Value op, | 
| +                                   HValue* right, | 
| +                                   Handle<Object> nil, | 
| +                                   HValue** expr) { | 
| +  if (left->IsConstant() && | 
| +      HConstant::cast(left)->handle().is_identical_to(nil) && | 
| +      Token::IsEqualityOp(op)) { | 
| +    *expr = right; | 
| return true; | 
| } | 
| +  return false; | 
| +} | 
|  | 
| -  if (expr->IsLiteralCompareUndefined(&sub_expr)) { | 
| -    HandleLiteralCompareNil(expr, sub_expr, kUndefinedValue); | 
| -    return true; | 
| -  } | 
|  | 
| -  if (expr->IsLiteralCompareNull(&sub_expr)) { | 
| -    HandleLiteralCompareNil(expr, sub_expr, kNullValue); | 
| +static bool MatchLiteralCompareTypeof(HValue* left, | 
| +                                      Token::Value op, | 
| +                                      HValue* right, | 
| +                                      HTypeof** typeof_expr, | 
| +                                      Handle<String>* check) { | 
| +  if (left->IsTypeof() && | 
| +      Token::IsEqualityOp(op) && | 
| +      right->IsConstant() && | 
| +      HConstant::cast(right)->HasStringValue()) { | 
| +    *typeof_expr = HTypeof::cast(left); | 
| +    *check = Handle<String>::cast(HConstant::cast(right)->handle()); | 
| return true; | 
| } | 
| - | 
| return false; | 
| } | 
|  | 
|  | 
| +static bool IsLiteralCompareTypeof(HValue* left, | 
| +                                   Token::Value op, | 
| +                                   HValue* right, | 
| +                                   HTypeof** typeof_expr, | 
| +                                   Handle<String>* check) { | 
| +  return MatchLiteralCompareTypeof(left, op, right, typeof_expr, check) || | 
| +      MatchLiteralCompareTypeof(right, op, left, typeof_expr, check); | 
| +} | 
| + | 
| + | 
| +static bool IsLiteralCompareNil(HValue* left, | 
| +                                Token::Value op, | 
| +                                HValue* right, | 
| +                                Handle<Object> nil, | 
| +                                HValue** expr) { | 
| +  return MatchLiteralCompareNil(left, op, right, nil, expr) || | 
| +      MatchLiteralCompareNil(right, op, left, nil, expr); | 
| +} | 
| + | 
| + | 
| void HGraphBuilder::VisitCompareOperation(CompareOperation* expr) { | 
| ASSERT(!HasStackOverflow()); | 
| ASSERT(current_block() != NULL); | 
| @@ -5840,11 +5870,9 @@ | 
| return ast_context()->ReturnControl(instr, expr->id()); | 
| } | 
|  | 
| -  // Check for special cases that compare against literals. | 
| -  if (TryLiteralCompare(expr)) return; | 
| - | 
| TypeInfo type_info = oracle()->CompareType(expr); | 
| // Check if this expression was ever executed according to type feedback. | 
| +  // Note that for the special typeof/null/undefined cases we get unknown here. | 
| if (type_info.IsUninitialized()) { | 
| AddInstruction(new(zone()) HSoftDeoptimize); | 
| current_block()->MarkAsDeoptimizing(); | 
| @@ -5859,6 +5887,20 @@ | 
| HValue* left = Pop(); | 
| Token::Value op = expr->op(); | 
|  | 
| +  HTypeof* typeof_expr = NULL; | 
| +  Handle<String> check; | 
| +  if (IsLiteralCompareTypeof(left, op, right, &typeof_expr, &check)) { | 
| +    return HandleLiteralCompareTypeof(expr, typeof_expr, check); | 
| +  } | 
| +  HValue* sub_expr = NULL; | 
| +  Factory* f = graph()->isolate()->factory(); | 
| +  if (IsLiteralCompareNil(left, op, right, f->undefined_value(), &sub_expr)) { | 
| +    return HandleLiteralCompareNil(expr, sub_expr, kUndefinedValue); | 
| +  } | 
| +  if (IsLiteralCompareNil(left, op, right, f->null_value(), &sub_expr)) { | 
| +    return HandleLiteralCompareNil(expr, sub_expr, kNullValue); | 
| +  } | 
| + | 
| if (op == Token::INSTANCEOF) { | 
| // Check to see if the rhs of the instanceof is a global function not | 
| // residing in new space. If it is we assume that the function will stay the | 
| @@ -5947,13 +5989,11 @@ | 
|  | 
|  | 
| void HGraphBuilder::HandleLiteralCompareNil(CompareOperation* expr, | 
| -                                            Expression* sub_expr, | 
| +                                            HValue* value, | 
| NilValue nil) { | 
| ASSERT(!HasStackOverflow()); | 
| ASSERT(current_block() != NULL); | 
| ASSERT(current_block()->HasPredecessor()); | 
| -  CHECK_ALIVE(VisitForValue(sub_expr)); | 
| -  HValue* value = Pop(); | 
| EqualityKind kind = | 
| expr->op() == Token::EQ_STRICT ? kStrictEquality : kNonStrictEquality; | 
| HIsNilAndBranch* instr = new(zone()) HIsNilAndBranch(value, kind, nil); | 
|  |