Index: src/builtins/s390/builtins-s390.cc |
diff --git a/src/builtins/s390/builtins-s390.cc b/src/builtins/s390/builtins-s390.cc |
index edd96ce8de7dd132744362b631639cddf938ada4..306769b1871be6d0d5843df0150411226edad986 100644 |
--- a/src/builtins/s390/builtins-s390.cc |
+++ b/src/builtins/s390/builtins-s390.cc |
@@ -2174,7 +2174,8 @@ void Builtins::Generate_Apply(MacroAssembler* masm) { |
// Create the list of arguments from the array-like argumentsList. |
{ |
- Label create_arguments, create_array, create_runtime, done_create; |
+ Label create_arguments, create_array, create_holey_array, create_runtime, |
+ done_create; |
__ JumpIfSmi(r2, &create_runtime); |
// Load the map of argumentsList into r4. |
@@ -2218,17 +2219,37 @@ void Builtins::Generate_Apply(MacroAssembler* masm) { |
__ LoadRR(r2, r6); |
__ b(&done_create); |
+ // For holey JSArrays we need to check that the array prototype chain |
+ // protector is intact and our prototype is the Array.prototype actually. |
+ __ bind(&create_holey_array); |
+ __ LoadP(r4, FieldMemOperand(r4, Map::kPrototypeOffset)); |
+ __ LoadP(r6, ContextMemOperand(r6, Context::INITIAL_ARRAY_PROTOTYPE_INDEX)); |
+ __ CmpP(r4, r6); |
+ __ bne(&create_runtime); |
+ __ LoadRoot(r6, Heap::kArrayProtectorRootIndex); |
+ __ LoadP(r4, FieldMemOperand(r6, PropertyCell::kValueOffset)); |
+ __ CmpSmiLiteral(r4, Smi::FromInt(Isolate::kProtectorValid), r0); |
+ __ bne(&create_runtime); |
+ __ LoadP(r4, FieldMemOperand(r2, JSArray::kLengthOffset)); |
+ __ LoadP(r2, FieldMemOperand(r2, JSArray::kElementsOffset)); |
+ __ SmiUntag(r4); |
+ __ b(&done_create); |
+ |
// Try to create the list from a JSArray object. |
+ // -- r4 and r6 must be preserved till bne create_holey_array. |
__ bind(&create_array); |
- __ LoadlB(r4, FieldMemOperand(r4, Map::kBitField2Offset)); |
- __ DecodeField<Map::ElementsKindBits>(r4); |
+ __ LoadlB(r7, FieldMemOperand(r4, Map::kBitField2Offset)); |
+ __ DecodeField<Map::ElementsKindBits>(r7); |
STATIC_ASSERT(FAST_SMI_ELEMENTS == 0); |
STATIC_ASSERT(FAST_HOLEY_SMI_ELEMENTS == 1); |
STATIC_ASSERT(FAST_ELEMENTS == 2); |
- __ CmpP(r4, Operand(FAST_ELEMENTS)); |
+ STATIC_ASSERT(FAST_HOLEY_ELEMENTS == 3); |
+ __ CmpP(r7, Operand(FAST_HOLEY_ELEMENTS)); |
__ bgt(&create_runtime); |
- __ CmpP(r4, Operand(FAST_HOLEY_SMI_ELEMENTS)); |
- __ beq(&create_runtime); |
+ // Only FAST_XXX after this point, FAST_HOLEY_XXX are odd values. |
+ __ TestBit(r7, Map::kHasNonInstancePrototype, r0); |
+ __ bne(&create_holey_array); |
+ // FAST_SMI_ELEMENTS or FAST_ELEMENTS after this point. |
__ LoadP(r4, FieldMemOperand(r2, JSArray::kLengthOffset)); |
__ LoadP(r2, FieldMemOperand(r2, JSArray::kElementsOffset)); |
__ SmiUntag(r4); |
@@ -2263,16 +2284,21 @@ void Builtins::Generate_Apply(MacroAssembler* masm) { |
// Push arguments onto the stack (thisArgument is already on the stack). |
{ |
- Label loop, no_args; |
+ __ LoadRoot(r8, Heap::kUndefinedValueRootIndex); |
+ Label loop, no_args, skip; |
__ CmpP(r4, Operand::Zero()); |
__ beq(&no_args); |
__ AddP(r2, r2, |
Operand(FixedArray::kHeaderSize - kHeapObjectTag - kPointerSize)); |
__ LoadRR(r1, r4); |
__ bind(&loop); |
- __ LoadP(r0, MemOperand(r2, kPointerSize)); |
+ __ LoadP(ip, MemOperand(r2, kPointerSize)); |
__ la(r2, MemOperand(r2, kPointerSize)); |
- __ push(r0); |
+ __ CompareRoot(ip, Heap::kTheHoleValueRootIndex); |
+ __ bne(&skip, Label::kNear); |
+ __ LoadRR(ip, r8); |
+ __ bind(&skip); |
+ __ push(ip); |
__ BranchOnCount(r1, &loop); |
__ bind(&no_args); |
__ LoadRR(r2, r4); |