| OLD | NEW | 
|---|
| 1 // Copyright 2011 the V8 project authors. All rights reserved. | 1 // Copyright 2011 the V8 project authors. All rights reserved. | 
| 2 // Redistribution and use in source and binary forms, with or without | 2 // Redistribution and use in source and binary forms, with or without | 
| 3 // modification, are permitted provided that the following conditions are | 3 // modification, are permitted provided that the following conditions are | 
| 4 // met: | 4 // met: | 
| 5 // | 5 // | 
| 6 //     * Redistributions of source code must retain the above copyright | 6 //     * Redistributions of source code must retain the above copyright | 
| 7 //       notice, this list of conditions and the following disclaimer. | 7 //       notice, this list of conditions and the following disclaimer. | 
| 8 //     * Redistributions in binary form must reproduce the above | 8 //     * Redistributions in binary form must reproduce the above | 
| 9 //       copyright notice, this list of conditions and the following | 9 //       copyright notice, this list of conditions and the following | 
| 10 //       disclaimer in the documentation and/or other materials provided | 10 //       disclaimer in the documentation and/or other materials provided | 
| (...skipping 716 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 727 | 727 | 
| 728 | 728 | 
| 729 void KeyedStoreIC::GenerateGeneric(MacroAssembler* masm, | 729 void KeyedStoreIC::GenerateGeneric(MacroAssembler* masm, | 
| 730                                    StrictModeFlag strict_mode) { | 730                                    StrictModeFlag strict_mode) { | 
| 731   // ----------- S t a t e ------------- | 731   // ----------- S t a t e ------------- | 
| 732   //  -- eax    : value | 732   //  -- eax    : value | 
| 733   //  -- ecx    : key | 733   //  -- ecx    : key | 
| 734   //  -- edx    : receiver | 734   //  -- edx    : receiver | 
| 735   //  -- esp[0] : return address | 735   //  -- esp[0] : return address | 
| 736   // ----------------------------------- | 736   // ----------------------------------- | 
| 737   Label slow, fast, array, extra; | 737   Label slow, fast_object_with_map_check, fast_object_without_map_check; | 
|  | 738   Label fast_double_with_map_check, fast_double_without_map_check; | 
|  | 739   Label check_if_double_array, array, extra; | 
| 738 | 740 | 
| 739   // Check that the object isn't a smi. | 741   // Check that the object isn't a smi. | 
| 740   __ JumpIfSmi(edx, &slow); | 742   __ JumpIfSmi(edx, &slow); | 
| 741   // Get the map from the receiver. | 743   // Get the map from the receiver. | 
| 742   __ mov(edi, FieldOperand(edx, HeapObject::kMapOffset)); | 744   __ mov(edi, FieldOperand(edx, HeapObject::kMapOffset)); | 
| 743   // Check that the receiver does not require access checks.  We need | 745   // Check that the receiver does not require access checks.  We need | 
| 744   // to do this because this generic stub does not perform map checks. | 746   // to do this because this generic stub does not perform map checks. | 
| 745   __ test_b(FieldOperand(edi, Map::kBitFieldOffset), | 747   __ test_b(FieldOperand(edi, Map::kBitFieldOffset), | 
| 746             1 << Map::kIsAccessCheckNeeded); | 748             1 << Map::kIsAccessCheckNeeded); | 
| 747   __ j(not_zero, &slow); | 749   __ j(not_zero, &slow); | 
| 748   // Check that the key is a smi. | 750   // Check that the key is a smi. | 
| 749   __ JumpIfNotSmi(ecx, &slow); | 751   __ JumpIfNotSmi(ecx, &slow); | 
| 750   __ CmpInstanceType(edi, JS_ARRAY_TYPE); | 752   __ CmpInstanceType(edi, JS_ARRAY_TYPE); | 
| 751   __ j(equal, &array); | 753   __ j(equal, &array); | 
| 752   // Check that the object is some kind of JSObject. | 754   // Check that the object is some kind of JSObject. | 
| 753   __ CmpInstanceType(edi, FIRST_JS_RECEIVER_TYPE); | 755   __ CmpInstanceType(edi, FIRST_JS_RECEIVER_TYPE); | 
| 754   __ j(below, &slow); | 756   __ j(below, &slow); | 
| 755   __ CmpInstanceType(edi, JS_PROXY_TYPE); | 757   __ CmpInstanceType(edi, JS_PROXY_TYPE); | 
| 756   __ j(equal, &slow); | 758   __ j(equal, &slow); | 
| 757   __ CmpInstanceType(edi, JS_FUNCTION_PROXY_TYPE); | 759   __ CmpInstanceType(edi, JS_FUNCTION_PROXY_TYPE); | 
| 758   __ j(equal, &slow); | 760   __ j(equal, &slow); | 
| 759 | 761 | 
| 760   // Object case: Check key against length in the elements array. | 762   // Object case: Check key against length in the elements array. | 
| 761   // eax: value | 763   // eax: value | 
| 762   // edx: JSObject | 764   // edx: JSObject | 
| 763   // ecx: key (a smi) | 765   // ecx: key (a smi) | 
| 764   __ mov(edi, FieldOperand(edx, JSObject::kElementsOffset)); | 766   // edi: receiver map | 
|  | 767   __ mov(ebx, FieldOperand(edx, JSObject::kElementsOffset)); | 
| 765   // Check that the object is in fast mode and writable. | 768   // Check that the object is in fast mode and writable. | 
| 766   __ CheckMap(edi, FACTORY->fixed_array_map(), &slow, DONT_DO_SMI_CHECK); | 769   __ cmp(ecx, FieldOperand(ebx, FixedArray::kLengthOffset)); | 
| 767   __ cmp(ecx, FieldOperand(edi, FixedArray::kLengthOffset)); | 770   __ j(below, &fast_object_with_map_check); | 
| 768   __ j(below, &fast); |  | 
| 769 | 771 | 
| 770   // Slow case: call runtime. | 772   // Slow case: call runtime. | 
| 771   __ bind(&slow); | 773   __ bind(&slow); | 
| 772   GenerateRuntimeSetProperty(masm, strict_mode); | 774   GenerateRuntimeSetProperty(masm, strict_mode); | 
| 773 | 775 | 
| 774   // Extra capacity case: Check if there is extra capacity to | 776   // Extra capacity case: Check if there is extra capacity to | 
| 775   // perform the store and update the length. Used for adding one | 777   // perform the store and update the length. Used for adding one | 
| 776   // element to the array by writing to array[array.length]. | 778   // element to the array by writing to array[array.length]. | 
| 777   __ bind(&extra); | 779   __ bind(&extra); | 
| 778   // eax: value | 780   // eax: value | 
| 779   // edx: receiver, a JSArray | 781   // edx: receiver, a JSArray | 
| 780   // ecx: key, a smi. | 782   // ecx: key, a smi. | 
| 781   // edi: receiver->elements, a FixedArray | 783   // ebx: receiver->elements, a FixedArray | 
|  | 784   // edi: receiver map | 
| 782   // flags: compare (ecx, edx.length()) | 785   // flags: compare (ecx, edx.length()) | 
| 783   // do not leave holes in the array: | 786   // do not leave holes in the array: | 
| 784   __ j(not_equal, &slow); | 787   __ j(not_equal, &slow); | 
| 785   __ cmp(ecx, FieldOperand(edi, FixedArray::kLengthOffset)); | 788   __ cmp(ecx, FieldOperand(ebx, FixedArray::kLengthOffset)); | 
| 786   __ j(above_equal, &slow); | 789   __ j(above_equal, &slow); | 
| 787   // Add 1 to receiver->length, and go to fast array write. | 790   __ CheckMap(ebx, FACTORY->fixed_array_map(), | 
|  | 791               &check_if_double_array, DONT_DO_SMI_CHECK); | 
|  | 792   // Add 1 to receiver->length, and go to common element store code for Objects. | 
| 788   __ add(FieldOperand(edx, JSArray::kLengthOffset), | 793   __ add(FieldOperand(edx, JSArray::kLengthOffset), | 
| 789          Immediate(Smi::FromInt(1))); | 794          Immediate(Smi::FromInt(1))); | 
| 790   __ jmp(&fast); | 795   __ jmp(&fast_object_without_map_check); | 
|  | 796 | 
|  | 797   __ bind(&check_if_double_array); | 
|  | 798   __ CheckMap(ebx, FACTORY->fixed_double_array_map(), &slow, DONT_DO_SMI_CHECK); | 
|  | 799   // Add 1 to receiver->length, and go to common element store code for doubles. | 
|  | 800   __ add(FieldOperand(edx, JSArray::kLengthOffset), | 
|  | 801          Immediate(Smi::FromInt(1))); | 
|  | 802   __ jmp(&fast_double_without_map_check); | 
| 791 | 803 | 
| 792   // Array case: Get the length and the elements array from the JS | 804   // Array case: Get the length and the elements array from the JS | 
| 793   // array. Check that the array is in fast mode (and writable); if it | 805   // array. Check that the array is in fast mode (and writable); if it | 
| 794   // is the length is always a smi. | 806   // is the length is always a smi. | 
| 795   __ bind(&array); | 807   __ bind(&array); | 
| 796   // eax: value | 808   // eax: value | 
| 797   // edx: receiver, a JSArray | 809   // edx: receiver, a JSArray | 
| 798   // ecx: key, a smi. | 810   // ecx: key, a smi. | 
| 799   __ mov(edi, FieldOperand(edx, JSObject::kElementsOffset)); | 811   // edi: receiver map | 
| 800   __ CheckMap(edi, FACTORY->fixed_array_map(), &slow, DONT_DO_SMI_CHECK); | 812   __ mov(ebx, FieldOperand(edx, JSObject::kElementsOffset)); | 
| 801 | 813 | 
| 802   // Check the key against the length in the array, compute the | 814   // Check the key against the length in the array and fall through to the | 
| 803   // address to store into and fall through to fast case. | 815   // common store code. | 
| 804   __ cmp(ecx, FieldOperand(edx, JSArray::kLengthOffset));  // Compare smis. | 816   __ cmp(ecx, FieldOperand(edx, JSArray::kLengthOffset));  // Compare smis. | 
| 805   __ j(above_equal, &extra); | 817   __ j(above_equal, &extra); | 
| 806 | 818 | 
| 807   // Fast case: Do the store. | 819   // Fast case: Do the store, could either Object or double. | 
| 808   __ bind(&fast); | 820   __ bind(&fast_object_with_map_check); | 
| 809   // eax: value | 821   // eax: value | 
| 810   // ecx: key (a smi) | 822   // ecx: key (a smi) | 
| 811   // edx: receiver | 823   // edx: receiver | 
| 812   // edi: FixedArray receiver->elements | 824   // ebx: FixedArray receiver->elements | 
| 813 | 825   // edi: receiver map | 
|  | 826   __ CheckMap(ebx, FACTORY->fixed_array_map(), | 
|  | 827               &fast_double_with_map_check, DONT_DO_SMI_CHECK); | 
|  | 828   __ bind(&fast_object_without_map_check); | 
|  | 829   // Smi stores don't require further checks. | 
| 814   Label non_smi_value; | 830   Label non_smi_value; | 
| 815   __ JumpIfNotSmi(eax, &non_smi_value); | 831   __ JumpIfNotSmi(eax, &non_smi_value); | 
| 816   // It's irrelevant whether array is smi-only or not when writing a smi. | 832   // It's irrelevant whether array is smi-only or not when writing a smi. | 
| 817   __ mov(CodeGenerator::FixedArrayElementOperand(edi, ecx), eax); | 833   __ mov(CodeGenerator::FixedArrayElementOperand(ebx, ecx), eax); | 
| 818   __ ret(0); | 834   __ ret(0); | 
| 819 | 835 | 
| 820   __ bind(&non_smi_value); | 836   __ bind(&non_smi_value); | 
| 821   if (FLAG_smi_only_arrays) { | 837   if (FLAG_smi_only_arrays) { | 
| 822     // Escape to slow case when writing non-smi into smi-only array. | 838     // Escape to slow case when writing non-smi into smi-only array. | 
| 823     __ mov(ebx, FieldOperand(edx, HeapObject::kMapOffset)); | 839     __ CheckFastObjectElements(edi, &slow, Label::kNear); | 
| 824     __ CheckFastObjectElements(ebx, &slow, Label::kNear); |  | 
| 825   } | 840   } | 
|  | 841 | 
| 826   // Fast elements array, store the value to the elements backing store. | 842   // Fast elements array, store the value to the elements backing store. | 
| 827   __ mov(CodeGenerator::FixedArrayElementOperand(edi, ecx), eax); | 843   __ mov(CodeGenerator::FixedArrayElementOperand(ebx, ecx), eax); | 
| 828   // Update write barrier for the elements array address. | 844   // Update write barrier for the elements array address. | 
| 829   __ mov(edx, Operand(eax));  // Preserve the value which is returned. | 845   __ mov(edx, Operand(eax));  // Preserve the value which is returned. | 
| 830   __ RecordWriteArray( | 846   __ RecordWriteArray( | 
| 831       edi, edx, ecx, kDontSaveFPRegs, EMIT_REMEMBERED_SET, OMIT_SMI_CHECK); | 847       ebx, edx, ecx, kDontSaveFPRegs, EMIT_REMEMBERED_SET, OMIT_SMI_CHECK); | 
|  | 848   __ ret(0); | 
|  | 849 | 
|  | 850   __ bind(&fast_double_with_map_check); | 
|  | 851   // Check for fast double array case. If this fails, call through to the | 
|  | 852   // runtime. | 
|  | 853   __ CheckMap(ebx, FACTORY->fixed_double_array_map(), &slow, DONT_DO_SMI_CHECK); | 
|  | 854   __ bind(&fast_double_without_map_check); | 
|  | 855   // If the value is a number, store it as a double in the FastDoubleElements | 
|  | 856   // array. | 
|  | 857   __ StoreNumberToDoubleElements(eax, | 
|  | 858                                  ebx, | 
|  | 859                                  ecx, | 
|  | 860                                  edx, | 
|  | 861                                  xmm0, | 
|  | 862                                  &slow, | 
|  | 863                                  false); | 
| 832   __ ret(0); | 864   __ ret(0); | 
| 833 } | 865 } | 
| 834 | 866 | 
| 835 | 867 | 
| 836 // The generated code does not accept smi keys. | 868 // The generated code does not accept smi keys. | 
| 837 // The generated code falls through if both probes miss. | 869 // The generated code falls through if both probes miss. | 
| 838 static void GenerateMonomorphicCacheProbe(MacroAssembler* masm, | 870 static void GenerateMonomorphicCacheProbe(MacroAssembler* masm, | 
| 839                                           int argc, | 871                                           int argc, | 
| 840                                           Code::Kind kind, | 872                                           Code::Kind kind, | 
| 841                                           Code::ExtraICState extra_ic_state) { | 873                                           Code::ExtraICState extra_ic_state) { | 
| (...skipping 809 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 1651   Condition cc = *jmp_address == Assembler::kJncShortOpcode | 1683   Condition cc = *jmp_address == Assembler::kJncShortOpcode | 
| 1652       ? not_zero | 1684       ? not_zero | 
| 1653       : zero; | 1685       : zero; | 
| 1654   *jmp_address = static_cast<byte>(Assembler::kJccShortPrefix | cc); | 1686   *jmp_address = static_cast<byte>(Assembler::kJccShortPrefix | cc); | 
| 1655 } | 1687 } | 
| 1656 | 1688 | 
| 1657 | 1689 | 
| 1658 } }  // namespace v8::internal | 1690 } }  // namespace v8::internal | 
| 1659 | 1691 | 
| 1660 #endif  // V8_TARGET_ARCH_IA32 | 1692 #endif  // V8_TARGET_ARCH_IA32 | 
| OLD | NEW | 
|---|