Index: src/arm/lithium-codegen-arm.cc |
diff --git a/src/arm/lithium-codegen-arm.cc b/src/arm/lithium-codegen-arm.cc |
index f5d744914977f3d90a008c1f7f6bc42c06a0ebc2..c88a93ae5c5b11af79a02ef338ea69b1d1c364f7 100644 |
--- a/src/arm/lithium-codegen-arm.cc |
+++ b/src/arm/lithium-codegen-arm.cc |
@@ -1918,28 +1918,36 @@ void LCodeGen::EmitClassOfTest(Label* is_true, |
ASSERT(!input.is(temp)); |
ASSERT(!temp.is(temp2)); // But input and temp2 may be the same register. |
__ JumpIfSmi(input, is_false); |
- __ CompareObjectType(input, temp, temp2, FIRST_SPEC_OBJECT_TYPE); |
- __ b(lt, is_false); |
- // Map is now in temp. |
- // Functions have class 'Function'. |
- __ CompareInstanceType(temp, temp2, FIRST_CALLABLE_SPEC_OBJECT_TYPE); |
if (class_name->IsEqualTo(CStrVector("Function"))) { |
- __ b(ge, is_true); |
+ // Assuming the following assertions, we can use the same compares to test |
+ // for both being a function type and being in the object type range. |
+ STATIC_ASSERT(NUM_OF_CALLABLE_SPEC_OBJECT_TYPES == 2); |
+ STATIC_ASSERT(FIRST_NONCALLABLE_SPEC_OBJECT_TYPE == |
+ FIRST_SPEC_OBJECT_TYPE + 1); |
+ STATIC_ASSERT(LAST_NONCALLABLE_SPEC_OBJECT_TYPE == |
+ LAST_SPEC_OBJECT_TYPE - 1); |
+ STATIC_ASSERT(LAST_SPEC_OBJECT_TYPE == LAST_TYPE); |
+ __ CompareObjectType(input, temp, temp2, FIRST_SPEC_OBJECT_TYPE); |
+ __ b(lt, is_false); |
+ __ b(eq, is_true); |
+ __ cmp(temp2, Operand(LAST_SPEC_OBJECT_TYPE)); |
+ __ b(eq, is_true); |
} else { |
- __ b(ge, is_false); |
+ // Faster code path to avoid two compares: subtract lower bound from the |
+ // actual type and do a signed compare with the width of the type range. |
+ __ ldr(temp, FieldMemOperand(input, HeapObject::kMapOffset)); |
+ __ ldrb(temp2, FieldMemOperand(temp, Map::kInstanceTypeOffset)); |
+ __ sub(temp2, temp2, Operand(FIRST_NONCALLABLE_SPEC_OBJECT_TYPE)); |
+ __ cmp(temp2, Operand(LAST_NONCALLABLE_SPEC_OBJECT_TYPE - |
+ FIRST_NONCALLABLE_SPEC_OBJECT_TYPE)); |
+ __ b(gt, is_false); |
} |
+ // Now we are in the FIRST-LAST_NONCALLABLE_SPEC_OBJECT_TYPE range. |
// Check if the constructor in the map is a function. |
__ ldr(temp, FieldMemOperand(temp, Map::kConstructorOffset)); |
- // As long as LAST_CALLABLE_SPEC_OBJECT_TYPE is the last instance type and |
- // FIRST_CALLABLE_SPEC_OBJECT_TYPE comes right after |
- // LAST_NONCALLABLE_SPEC_OBJECT_TYPE, we can avoid checking for the latter. |
- STATIC_ASSERT(LAST_TYPE == LAST_CALLABLE_SPEC_OBJECT_TYPE); |
- STATIC_ASSERT(FIRST_CALLABLE_SPEC_OBJECT_TYPE == |
- LAST_NONCALLABLE_SPEC_OBJECT_TYPE + 1); |
- |
// Objects with a non-function constructor have class 'Object'. |
__ CompareObjectType(temp, temp2, temp2, JS_FUNCTION_TYPE); |
if (class_name->IsEqualTo(CStrVector("Object"))) { |
@@ -4343,10 +4351,12 @@ Condition LCodeGen::EmitTypeofIs(Label* true_label, |
final_branch_condition = ne; |
} else if (type_name->Equals(heap()->function_symbol())) { |
+ STATIC_ASSERT(NUM_OF_CALLABLE_SPEC_OBJECT_TYPES == 2); |
__ JumpIfSmi(input, false_label); |
- __ CompareObjectType(input, input, scratch, |
- FIRST_CALLABLE_SPEC_OBJECT_TYPE); |
- final_branch_condition = ge; |
+ __ CompareObjectType(input, scratch, input, JS_FUNCTION_TYPE); |
+ __ b(eq, true_label); |
+ __ cmp(input, Operand(JS_FUNCTION_PROXY_TYPE)); |
+ final_branch_condition = eq; |
} else if (type_name->Equals(heap()->object_symbol())) { |
__ JumpIfSmi(input, false_label); |