Index: src/arm/ic-arm.cc |
=================================================================== |
--- src/arm/ic-arm.cc (revision 4872) |
+++ src/arm/ic-arm.cc (working copy) |
@@ -167,17 +167,23 @@ |
Label* miss, |
Register elements, |
Register key, |
+ Register result, |
Register t0, |
Register t1, |
Register t2) { |
// Register use: |
// |
- // elements - holds the slow-case elements of the receiver and is unchanged. |
+ // elements - holds the slow-case elements of the receiver on entry. |
+ // Unchanged unless 'result' is the same register. |
// |
- // key - holds the smi key on entry and is unchanged if a branch is |
- // performed to the miss label. |
- // Holds the result on exit if the load succeeded. |
+ // key - holds the smi key on entry. |
+ // Unchanged unless 'result' is the same register. |
// |
+ // result - holds the result on exit if the load succeeded. |
+ // Allowed to be the same as 'key' or 'result'. |
+ // Unchanged on bailout so 'key' or 'result' can be used |
+ // in further computation. |
+ // |
// Scratch registers: |
// |
// t0 - holds the untagged key on entry and holds the hash once computed. |
@@ -248,7 +254,7 @@ |
// Get the value at the masked, scaled index and return. |
const int kValueOffset = |
NumberDictionary::kElementsStartOffset + kPointerSize; |
- __ ldr(key, FieldMemOperand(t2, kValueOffset)); |
+ __ ldr(result, FieldMemOperand(t2, kValueOffset)); |
} |
@@ -298,22 +304,159 @@ |
} |
+// Checks the receiver for special cases (value type, slow case bits). |
+// Falls through for regular JS object. |
+static void GenerateKeyedLoadReceiverCheck(MacroAssembler* masm, |
+ Register receiver, |
+ Register scratch1, |
+ Register scratch2, |
+ Label* slow) { |
+ // Check that the object isn't a smi. |
+ __ BranchOnSmi(receiver, slow); |
+ // Get the map of the receiver. |
+ __ ldr(scratch1, FieldMemOperand(receiver, HeapObject::kMapOffset)); |
+ // Check bit field. |
+ __ ldrb(scratch2, FieldMemOperand(scratch1, Map::kBitFieldOffset)); |
+ __ tst(scratch2, Operand(KeyedLoadIC::kSlowCaseBitFieldMask)); |
+ __ b(ne, slow); |
+ // Check that the object is some kind of JS object EXCEPT JS Value type. |
+ // In the case that the object is a value-wrapper object, |
+ // we enter the runtime system to make sure that indexing into string |
+ // objects work as intended. |
+ ASSERT(JS_OBJECT_TYPE > JS_VALUE_TYPE); |
+ __ ldrb(scratch1, FieldMemOperand(scratch1, Map::kInstanceTypeOffset)); |
+ __ cmp(scratch1, Operand(JS_OBJECT_TYPE)); |
+ __ b(lt, slow); |
+} |
+ |
+ |
+// Loads an indexed element from a fast case array. |
+static void GenerateFastArrayLoad(MacroAssembler* masm, |
+ Register receiver, |
+ Register key, |
+ Register elements, |
+ Register scratch1, |
+ Register scratch2, |
+ Register result, |
+ Label* not_fast_array, |
+ Label* out_of_range) { |
+ // Register use: |
+ // |
+ // receiver - holds the receiver on entry. |
+ // Unchanged unless 'result' is the same register. |
+ // |
+ // key - holds the smi key on entry. |
+ // Unchanged unless 'result' is the same register. |
+ // |
+ // elements - holds the elements of the receiver on exit. |
+ // |
+ // result - holds the result on exit if the load succeeded. |
+ // Allowed to be the the same as 'receiver' or 'key'. |
+ // Unchanged on bailout so 'receiver' and 'key' can be safely |
+ // used by further computation. |
+ // |
+ // Scratch registers: |
+ // |
+ // scratch1 - used to hold elements map and elements length. |
+ // Holds the elements map if not_fast_array branch is taken. |
+ // |
+ // scratch2 - used to hold the loaded value. |
+ |
+ __ ldr(elements, FieldMemOperand(receiver, JSObject::kElementsOffset)); |
+ // Check that the object is in fast mode (not dictionary). |
+ __ ldr(scratch1, FieldMemOperand(elements, HeapObject::kMapOffset)); |
+ __ LoadRoot(ip, Heap::kFixedArrayMapRootIndex); |
+ __ cmp(scratch1, ip); |
+ __ b(ne, not_fast_array); |
+ // Check that the key (index) is within bounds. |
+ __ ldr(scratch1, FieldMemOperand(elements, FixedArray::kLengthOffset)); |
+ __ cmp(key, Operand(scratch1)); |
+ __ b(hs, out_of_range); |
+ // Fast case: Do the load. |
+ __ add(scratch1, elements, Operand(FixedArray::kHeaderSize - kHeapObjectTag)); |
+ // The key is a smi. |
+ ASSERT(kSmiTag == 0 && kSmiTagSize < kPointerSizeLog2); |
+ __ ldr(scratch2, |
+ MemOperand(scratch1, key, LSL, kPointerSizeLog2 - kSmiTagSize)); |
+ __ LoadRoot(ip, Heap::kTheHoleValueRootIndex); |
+ __ cmp(scratch2, ip); |
+ // In case the loaded value is the_hole we have to consult GetProperty |
+ // to ensure the prototype chain is searched. |
+ __ b(eq, out_of_range); |
+ __ mov(result, scratch2); |
+} |
+ |
+ |
+// Checks whether a key is an array index string or a symbol string. |
+// Falls through if a key is a symbol. |
+static void GenerateKeyStringCheck(MacroAssembler* masm, |
+ Register key, |
+ Register map, |
+ Register hash, |
+ Label* index_string, |
+ Label* not_symbol) { |
+ // The key is not a smi. |
+ // Is it a string? |
+ __ CompareObjectType(key, map, hash, FIRST_NONSTRING_TYPE); |
+ __ b(ge, not_symbol); |
+ |
+ // Is the string an array index, with cached numeric value? |
+ __ ldr(hash, FieldMemOperand(key, String::kHashFieldOffset)); |
+ __ tst(hash, Operand(String::kContainsCachedArrayIndexMask)); |
+ __ b(eq, index_string); |
+ |
+ // Is the string a symbol? |
+ // map: key map |
+ __ ldrb(hash, FieldMemOperand(map, Map::kInstanceTypeOffset)); |
+ ASSERT(kSymbolTag != 0); |
+ __ tst(hash, Operand(kIsSymbolMask)); |
+ __ b(eq, not_symbol); |
+} |
+ |
+ |
+// Picks out an array index from the hash field. |
+static void GenerateIndexFromHash(MacroAssembler* masm, |
+ Register key, |
+ Register hash) { |
+ // Register use: |
+ // key - holds the overwritten key on exit. |
+ // hash - holds the key's hash. Clobbered. |
+ |
+ // If the hash field contains an array index pick it out. The assert checks |
+ // that the constants for the maximum number of digits for an array index |
+ // cached in the hash field and the number of bits reserved for it does not |
+ // conflict. |
+ ASSERT(TenToThe(String::kMaxCachedArrayIndexLength) < |
+ (1 << String::kArrayIndexValueBits)); |
+ // We want the smi-tagged index in key. kArrayIndexValueMask has zeros in |
+ // the low kHashShift bits. |
+ ASSERT(String::kHashShift >= kSmiTagSize); |
+ // Here we actually clobber the key which will be used if calling into |
+ // runtime later. However as the new key is the numeric value of a string key |
+ // there is no difference in using either key. |
+ ASSERT(String::kHashShift >= kSmiTagSize); |
+ __ Ubfx(hash, hash, String::kHashShift, String::kArrayIndexValueBits); |
+ __ mov(key, Operand(hash, LSL, kSmiTagSize)); |
+} |
+ |
+ |
// Defined in ic.cc. |
Object* CallIC_Miss(Arguments args); |
-void CallIC::GenerateMegamorphic(MacroAssembler* masm, int argc) { |
+// The generated code does not accept smi keys. |
+// The generated code falls through if both probes miss. |
+static void GenerateMonomorphicCacheProbe(MacroAssembler* masm, |
+ int argc, |
+ Code::Kind kind) { |
// ----------- S t a t e ------------- |
+ // -- r1 : receiver |
// -- r2 : name |
- // -- lr : return address |
// ----------------------------------- |
Label number, non_number, non_string, boolean, probe, miss; |
- // Get the receiver of the function from the stack into r1. |
- __ ldr(r1, MemOperand(sp, argc * kPointerSize)); |
- |
// Probe the stub cache. |
Code::Flags flags = |
- Code::ComputeFlags(Code::CALL_IC, NOT_IN_LOOP, MONOMORPHIC, NORMAL, argc); |
+ Code::ComputeFlags(kind, NOT_IN_LOOP, MONOMORPHIC, NORMAL, argc); |
StubCache::GenerateProbe(masm, flags, r1, r2, r3, no_reg); |
// If the stub cache probing failed, the receiver might be a value. |
@@ -355,9 +498,7 @@ |
__ bind(&probe); |
StubCache::GenerateProbe(masm, flags, r1, r2, r3, no_reg); |
- // Cache miss: Jump to runtime. |
__ bind(&miss); |
- GenerateMiss(masm, argc); |
} |
@@ -390,7 +531,7 @@ |
} |
-void CallIC::GenerateNormal(MacroAssembler* masm, int argc) { |
+static void GenerateCallNormal(MacroAssembler* masm, int argc) { |
// ----------- S t a t e ------------- |
// -- r2 : name |
// -- lr : return address |
@@ -443,13 +584,11 @@ |
__ CheckAccessGlobalProxy(r1, r0, &miss); |
__ b(&invoke); |
- // Cache miss: Jump to runtime. |
__ bind(&miss); |
- GenerateMiss(masm, argc); |
} |
-void CallIC::GenerateMiss(MacroAssembler* masm, int argc) { |
+static void GenerateCallMiss(MacroAssembler* masm, int argc, IC::UtilityId id) { |
// ----------- S t a t e ------------- |
// -- r2 : name |
// -- lr : return address |
@@ -465,7 +604,7 @@ |
// Call the entry. |
__ mov(r0, Operand(2)); |
- __ mov(r1, Operand(ExternalReference(IC_Utility(kCallIC_Miss)))); |
+ __ mov(r1, Operand(ExternalReference(IC_Utility(id)))); |
CEntryStub stub(1); |
__ CallStub(&stub); |
@@ -496,18 +635,165 @@ |
} |
+void CallIC::GenerateMiss(MacroAssembler* masm, int argc) { |
+ // ----------- S t a t e ------------- |
+ // -- r2 : name |
+ // -- lr : return address |
+ // ----------------------------------- |
+ |
+ GenerateCallMiss(masm, argc, IC::kCallIC_Miss); |
+} |
+ |
+ |
+void CallIC::GenerateMegamorphic(MacroAssembler* masm, int argc) { |
+ // ----------- S t a t e ------------- |
+ // -- r2 : name |
+ // -- lr : return address |
+ // ----------------------------------- |
+ |
+ // Get the receiver of the function from the stack into r1. |
+ __ ldr(r1, MemOperand(sp, argc * kPointerSize)); |
+ GenerateMonomorphicCacheProbe(masm, argc, Code::CALL_IC); |
+ GenerateMiss(masm, argc); |
+} |
+ |
+ |
+void CallIC::GenerateNormal(MacroAssembler* masm, int argc) { |
+ // ----------- S t a t e ------------- |
+ // -- r2 : name |
+ // -- lr : return address |
+ // ----------------------------------- |
+ |
+ GenerateCallNormal(masm, argc); |
+ GenerateMiss(masm, argc); |
+} |
+ |
+ |
void KeyedCallIC::GenerateMiss(MacroAssembler* masm, int argc) { |
- UNREACHABLE(); |
+ // ----------- S t a t e ------------- |
+ // -- r2 : name |
+ // -- lr : return address |
+ // ----------------------------------- |
+ |
+ GenerateCallMiss(masm, argc, IC::kKeyedCallIC_Miss); |
} |
void KeyedCallIC::GenerateMegamorphic(MacroAssembler* masm, int argc) { |
- UNREACHABLE(); |
+ // ----------- S t a t e ------------- |
+ // -- r2 : name |
+ // -- lr : return address |
+ // ----------------------------------- |
+ |
+ // Get the receiver of the function from the stack into r1. |
+ __ ldr(r1, MemOperand(sp, argc * kPointerSize)); |
+ |
+ Label do_call, slow_call, slow_load, slow_reload_receiver; |
+ Label check_number_dictionary, check_string, lookup_monomorphic_cache; |
+ Label index_smi, index_string; |
+ |
+ // Check that the key is a smi. |
+ __ BranchOnNotSmi(r2, &check_string); |
+ __ bind(&index_smi); |
+ // Now the key is known to be a smi. This place is also jumped to from below |
+ // where a numeric string is converted to a smi. |
+ |
+ GenerateKeyedLoadReceiverCheck(masm, r1, r0, r3, &slow_call); |
+ |
+ GenerateFastArrayLoad( |
+ masm, r1, r2, r4, r3, r0, r1, &check_number_dictionary, &slow_load); |
+ __ IncrementCounter(&Counters::keyed_call_generic_smi_fast, 1, r0, r3); |
+ |
+ __ bind(&do_call); |
+ // receiver in r1 is not used after this point. |
+ // r2: key |
+ // r1: function |
+ |
+ // Check that the value in r1 is a JSFunction. |
+ __ BranchOnSmi(r1, &slow_call); |
+ __ CompareObjectType(r1, r0, r0, JS_FUNCTION_TYPE); |
+ __ b(ne, &slow_call); |
+ // Invoke the function. |
+ ParameterCount actual(argc); |
+ __ InvokeFunction(r1, actual, JUMP_FUNCTION); |
+ |
+ __ bind(&check_number_dictionary); |
+ // r2: key |
+ // r3: elements map |
+ // r4: elements |
+ // Check whether the elements is a number dictionary. |
+ __ LoadRoot(ip, Heap::kHashTableMapRootIndex); |
+ __ cmp(r3, ip); |
+ __ b(ne, &slow_load); |
+ __ mov(r0, Operand(r2, ASR, kSmiTagSize)); |
+ // r0: untagged index |
+ GenerateNumberDictionaryLoad(masm, &slow_load, r4, r2, r1, r0, r3, r5); |
+ __ IncrementCounter(&Counters::keyed_call_generic_smi_dict, 1, r0, r3); |
+ __ jmp(&do_call); |
+ |
+ __ bind(&slow_load); |
+ // This branch is taken when calling KeyedCallIC_Miss is neither required |
+ // nor beneficial. |
+ __ IncrementCounter(&Counters::keyed_call_generic_slow_load, 1, r0, r3); |
+ __ EnterInternalFrame(); |
+ __ push(r2); // save the key |
+ __ Push(r1, r2); // pass the receiver and the key |
+ __ CallRuntime(Runtime::kKeyedGetProperty, 2); |
+ __ pop(r2); // restore the key |
+ __ LeaveInternalFrame(); |
+ __ mov(r1, r0); |
+ __ jmp(&do_call); |
+ |
+ __ bind(&check_string); |
+ GenerateKeyStringCheck(masm, r2, r0, r3, &index_string, &slow_call); |
+ |
+ // The key is known to be a symbol. |
+ // If the receiver is a regular JS object with slow properties then do |
+ // a quick inline probe of the receiver's dictionary. |
+ // Otherwise do the monomorphic cache probe. |
+ GenerateKeyedLoadReceiverCheck(masm, r1, r0, r3, &lookup_monomorphic_cache); |
+ |
+ __ ldr(r3, FieldMemOperand(r1, JSObject::kPropertiesOffset)); |
+ __ ldr(r3, FieldMemOperand(r3, HeapObject::kMapOffset)); |
+ __ LoadRoot(ip, Heap::kHashTableMapRootIndex); |
+ __ cmp(r3, ip); |
+ __ b(ne, &lookup_monomorphic_cache); |
+ |
+ GenerateDictionaryLoad( |
+ masm, &slow_load, r1, r2, r1, r0, r3, r4, DICTIONARY_CHECK_DONE); |
+ __ IncrementCounter(&Counters::keyed_call_generic_lookup_dict, 1, r0, r3); |
+ __ jmp(&do_call); |
+ |
+ __ bind(&lookup_monomorphic_cache); |
+ __ IncrementCounter(&Counters::keyed_call_generic_lookup_cache, 1, r0, r3); |
+ GenerateMonomorphicCacheProbe(masm, argc, Code::KEYED_CALL_IC); |
+ // Fall through on miss. |
+ |
+ __ bind(&slow_call); |
+ // This branch is taken if: |
+ // - the receiver requires boxing or access check, |
+ // - the key is neither smi nor symbol, |
+ // - the value loaded is not a function, |
+ // - there is hope that the runtime will create a monomorphic call stub |
+ // that will get fetched next time. |
+ __ IncrementCounter(&Counters::keyed_call_generic_slow, 1, r0, r3); |
+ GenerateMiss(masm, argc); |
+ |
+ __ bind(&index_string); |
+ GenerateIndexFromHash(masm, r2, r3); |
+ // Now jump to the place where smi keys are handled. |
+ __ jmp(&index_smi); |
} |
void KeyedCallIC::GenerateNormal(MacroAssembler* masm, int argc) { |
- UNREACHABLE(); |
+ // ----------- S t a t e ------------- |
+ // -- r2 : name |
+ // -- lr : return address |
+ // ----------------------------------- |
+ |
+ GenerateCallNormal(masm, argc); |
+ GenerateMiss(masm, argc); |
} |
@@ -759,49 +1045,16 @@ |
Register key = r0; |
Register receiver = r1; |
- // Check that the object isn't a smi. |
- __ BranchOnSmi(receiver, &slow); |
- // Get the map of the receiver. |
- __ ldr(r2, FieldMemOperand(receiver, HeapObject::kMapOffset)); |
- // Check bit field. |
- __ ldrb(r3, FieldMemOperand(r2, Map::kBitFieldOffset)); |
- __ tst(r3, Operand(kSlowCaseBitFieldMask)); |
- __ b(ne, &slow); |
- // Check that the object is some kind of JS object EXCEPT JS Value type. |
- // In the case that the object is a value-wrapper object, |
- // we enter the runtime system to make sure that indexing into string |
- // objects work as intended. |
- ASSERT(JS_OBJECT_TYPE > JS_VALUE_TYPE); |
- __ ldrb(r2, FieldMemOperand(r2, Map::kInstanceTypeOffset)); |
- __ cmp(r2, Operand(JS_OBJECT_TYPE)); |
- __ b(lt, &slow); |
+ GenerateKeyedLoadReceiverCheck(masm, receiver, r2, r3, &slow); |
// Check that the key is a smi. |
__ BranchOnNotSmi(key, &check_string); |
__ bind(&index_smi); |
// Now the key is known to be a smi. This place is also jumped to from below |
// where a numeric string is converted to a smi. |
- __ ldr(r4, FieldMemOperand(receiver, JSObject::kElementsOffset)); |
- // Check that the object is in fast mode (not dictionary). |
- __ ldr(r3, FieldMemOperand(r4, HeapObject::kMapOffset)); |
- __ LoadRoot(ip, Heap::kFixedArrayMapRootIndex); |
- __ cmp(r3, ip); |
- __ b(ne, &check_pixel_array); |
- // Check that the key (index) is within bounds. |
- __ ldr(r3, FieldMemOperand(r4, FixedArray::kLengthOffset)); |
- __ cmp(key, Operand(r3)); |
- __ b(hs, &slow); |
- // Fast case: Do the load. |
- __ add(r3, r4, Operand(FixedArray::kHeaderSize - kHeapObjectTag)); |
- // The key is a smi. |
- ASSERT(kSmiTag == 0 && kSmiTagSize < kPointerSizeLog2); |
- __ ldr(r2, MemOperand(r3, key, LSL, kPointerSizeLog2 - kSmiTagSize)); |
- __ LoadRoot(ip, Heap::kTheHoleValueRootIndex); |
- __ cmp(r2, ip); |
- // In case the loaded value is the_hole we have to consult GetProperty |
- // to ensure the prototype chain is searched. |
- __ b(eq, &slow); |
- __ mov(r0, r2); |
+ |
+ GenerateFastArrayLoad( |
+ masm, receiver, key, r4, r3, r2, r0, &check_pixel_array, &slow); |
__ IncrementCounter(&Counters::keyed_load_generic_smi, 1, r2, r3); |
__ Ret(); |
@@ -831,7 +1084,7 @@ |
__ cmp(r3, ip); |
__ b(ne, &slow); |
__ mov(r2, Operand(r0, ASR, kSmiTagSize)); |
- GenerateNumberDictionaryLoad(masm, &slow, r4, r0, r2, r3, r5); |
+ GenerateNumberDictionaryLoad(masm, &slow, r4, r0, r0, r2, r3, r5); |
__ Ret(); |
// Slow case, key and receiver still in r0 and r1. |
@@ -840,25 +1093,8 @@ |
GenerateRuntimeGetProperty(masm); |
__ bind(&check_string); |
- // The key is not a smi. |
- // Is it a string? |
- // r0: key |
- // r1: receiver |
- __ CompareObjectType(r0, r2, r3, FIRST_NONSTRING_TYPE); |
- __ b(ge, &slow); |
+ GenerateKeyStringCheck(masm, key, r2, r3, &index_string, &slow); |
- // Is the string an array index, with cached numeric value? |
- __ ldr(r3, FieldMemOperand(r0, String::kHashFieldOffset)); |
- __ tst(r3, Operand(String::kContainsCachedArrayIndexMask)); |
- __ b(eq, &index_string); |
- |
- // Is the string a symbol? |
- // r2: key map |
- __ ldrb(r3, FieldMemOperand(r2, Map::kInstanceTypeOffset)); |
- ASSERT(kSymbolTag != 0); |
- __ tst(r3, Operand(kIsSymbolMask)); |
- __ b(eq, &slow); |
- |
// If the receiver is a fast-case object, check the keyed lookup |
// cache. Otherwise probe the dictionary. |
__ ldr(r3, FieldMemOperand(r1, JSObject::kPropertiesOffset)); |
@@ -918,25 +1154,8 @@ |
__ IncrementCounter(&Counters::keyed_load_generic_symbol, 1, r2, r3); |
__ Ret(); |
- __ b(&slow); |
- // If the hash field contains an array index pick it out. The assert checks |
- // that the constants for the maximum number of digits for an array index |
- // cached in the hash field and the number of bits reserved for it does not |
- // conflict. |
- ASSERT(TenToThe(String::kMaxCachedArrayIndexLength) < |
- (1 << String::kArrayIndexValueBits)); |
__ bind(&index_string); |
- // r0: key (string) |
- // r1: receiver |
- // r3: hash field |
- // We want the smi-tagged index in r0. kArrayIndexValueMask has zeros in |
- // the low kHashShift bits. |
- ASSERT(String::kHashShift >= kSmiTagSize); |
- __ Ubfx(r3, r3, String::kHashShift, String::kArrayIndexValueBits); |
- // Here we actually clobber the key (r0) which will be used if calling into |
- // runtime later. However as the new key is the numeric value of a string key |
- // there is no difference in using either key. |
- __ mov(r0, Operand(r3, LSL, kSmiTagSize)); |
+ GenerateIndexFromHash(masm, key, r3); |
// Now jump to the place where smi keys are handled. |
__ jmp(&index_smi); |
} |