| OLD | NEW |
| 1 // Copyright 2006-2009 the V8 project authors. All rights reserved. | 1 // Copyright 2006-2009 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 1233 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1244 } else { | 1244 } else { |
| 1245 // Get the elements array of the object. | 1245 // Get the elements array of the object. |
| 1246 __ mov(ebx, FieldOperand(edx, JSArray::kElementsOffset)); | 1246 __ mov(ebx, FieldOperand(edx, JSArray::kElementsOffset)); |
| 1247 | 1247 |
| 1248 // Check that the elements are in fast mode (not dictionary). | 1248 // Check that the elements are in fast mode (not dictionary). |
| 1249 __ cmp(FieldOperand(ebx, HeapObject::kMapOffset), | 1249 __ cmp(FieldOperand(ebx, HeapObject::kMapOffset), |
| 1250 Immediate(Factory::fixed_array_map())); | 1250 Immediate(Factory::fixed_array_map())); |
| 1251 __ j(not_equal, &miss); | 1251 __ j(not_equal, &miss); |
| 1252 | 1252 |
| 1253 if (argc == 1) { // Otherwise fall through to call builtin. | 1253 if (argc == 1) { // Otherwise fall through to call builtin. |
| 1254 Label call_builtin, exit, with_rset_update; | 1254 Label call_builtin, exit, with_rset_update, |
| 1255 attempt_to_grow_elements, finish_push; |
| 1255 | 1256 |
| 1256 // Get the array's length into eax and calculate new length. | 1257 // Get the array's length into eax and calculate new length. |
| 1257 __ mov(eax, FieldOperand(edx, JSArray::kLengthOffset)); | 1258 __ mov(eax, FieldOperand(edx, JSArray::kLengthOffset)); |
| 1258 STATIC_ASSERT(kSmiTagSize == 1); | 1259 STATIC_ASSERT(kSmiTagSize == 1); |
| 1259 STATIC_ASSERT(kSmiTag == 0); | 1260 STATIC_ASSERT(kSmiTag == 0); |
| 1260 __ add(Operand(eax), Immediate(argc << 1)); | 1261 __ add(Operand(eax), Immediate(argc << 1)); |
| 1261 | 1262 |
| 1262 // Get the element's length into ecx. | 1263 // Get the element's length into ecx. |
| 1263 __ mov(ecx, FieldOperand(ebx, FixedArray::kLengthOffset)); | 1264 __ mov(ecx, FieldOperand(ebx, FixedArray::kLengthOffset)); |
| 1264 __ SmiTag(ecx); | 1265 __ SmiTag(ecx); |
| 1265 | 1266 |
| 1266 // Check if we could survive without allocation, go to builtin otherwise. | 1267 // Check if we could survive without allocation. |
| 1267 __ cmp(eax, Operand(ecx)); | 1268 __ cmp(eax, Operand(ecx)); |
| 1268 __ j(greater, &call_builtin); | 1269 __ j(greater, &attempt_to_grow_elements); |
| 1269 | 1270 |
| 1270 // Save new length. | 1271 // Save new length. |
| 1271 __ mov(FieldOperand(edx, JSArray::kLengthOffset), eax); | 1272 __ mov(FieldOperand(edx, JSArray::kLengthOffset), eax); |
| 1272 | 1273 |
| 1273 // Push the element. | 1274 // Push the element. |
| 1274 __ lea(edx, FieldOperand(ebx, | 1275 __ lea(edx, FieldOperand(ebx, |
| 1275 eax, times_half_pointer_size, | 1276 eax, times_half_pointer_size, |
| 1276 FixedArray::kHeaderSize - argc * kPointerSize)); | 1277 FixedArray::kHeaderSize - argc * kPointerSize)); |
| 1277 __ mov(ecx, Operand(esp, argc * kPointerSize)); | 1278 __ mov(ecx, Operand(esp, argc * kPointerSize)); |
| 1278 __ mov(Operand(edx, 0), ecx); | 1279 __ mov(Operand(edx, 0), ecx); |
| 1279 | 1280 |
| 1281 __ bind(&finish_push); |
| 1282 |
| 1280 // Check if value is a smi. | 1283 // Check if value is a smi. |
| 1281 __ test(ecx, Immediate(kSmiTagMask)); | 1284 __ test(ecx, Immediate(kSmiTagMask)); |
| 1282 __ j(not_zero, &with_rset_update); | 1285 __ j(not_zero, &with_rset_update); |
| 1283 | 1286 |
| 1284 __ bind(&exit); | 1287 __ bind(&exit); |
| 1285 __ ret((argc + 1) * kPointerSize); | 1288 __ ret((argc + 1) * kPointerSize); |
| 1286 | 1289 |
| 1287 __ bind(&with_rset_update); | 1290 __ bind(&with_rset_update); |
| 1288 | 1291 |
| 1289 __ InNewSpace(ebx, ecx, equal, &exit); | 1292 __ InNewSpace(ebx, ecx, equal, &exit); |
| 1290 | 1293 |
| 1291 RecordWriteStub stub(ebx, edx, ecx); | 1294 RecordWriteStub stub(ebx, edx, ecx); |
| 1292 __ CallStub(&stub); | 1295 __ CallStub(&stub); |
| 1293 __ ret((argc + 1) * kPointerSize); | 1296 __ ret((argc + 1) * kPointerSize); |
| 1294 | 1297 |
| 1298 __ bind(&attempt_to_grow_elements); |
| 1299 ExternalReference new_space_allocation_top = |
| 1300 ExternalReference::new_space_allocation_top_address(); |
| 1301 ExternalReference new_space_allocation_limit = |
| 1302 ExternalReference::new_space_allocation_limit_address(); |
| 1303 |
| 1304 const int kAllocationDelta = 4; |
| 1305 // Load top. |
| 1306 __ mov(ecx, Operand::StaticVariable(new_space_allocation_top)); |
| 1307 |
| 1308 // Check if it's the end of elements. |
| 1309 __ lea(edx, FieldOperand(ebx, |
| 1310 eax, times_half_pointer_size, |
| 1311 FixedArray::kHeaderSize - argc * kPointerSize)); |
| 1312 __ cmp(edx, Operand(ecx)); |
| 1313 __ j(not_equal, &call_builtin); |
| 1314 __ add(Operand(ecx), Immediate(kAllocationDelta * kPointerSize)); |
| 1315 __ cmp(ecx, Operand::StaticVariable(new_space_allocation_limit)); |
| 1316 __ j(greater, &call_builtin); |
| 1317 |
| 1318 // We fit and could grow elements. |
| 1319 __ mov(Operand::StaticVariable(new_space_allocation_top), ecx); |
| 1320 __ mov(ecx, Operand(esp, argc * kPointerSize)); |
| 1321 __ mov(Operand(edx, 0), ecx); |
| 1322 for (int i = 1; i < kAllocationDelta; i++) { |
| 1323 __ mov(Operand(edx, i * kPointerSize), |
| 1324 Immediate(Factory::undefined_value())); |
| 1325 } |
| 1326 |
| 1327 // Restore receiver to edx as finish sequence assumes it's here. |
| 1328 __ mov(edx, Operand(esp, (argc + 1) * kPointerSize)); |
| 1329 |
| 1330 // Increment element's and array's sizes. |
| 1331 __ add(FieldOperand(ebx, FixedArray::kLengthOffset), |
| 1332 Immediate(kAllocationDelta)); |
| 1333 __ mov(FieldOperand(edx, JSArray::kLengthOffset), eax); |
| 1334 |
| 1335 __ jmp(&finish_push); |
| 1336 |
| 1295 __ bind(&call_builtin); | 1337 __ bind(&call_builtin); |
| 1296 } | 1338 } |
| 1297 | 1339 |
| 1298 __ TailCallExternalReference(ExternalReference(Builtins::c_ArrayPush), | 1340 __ TailCallExternalReference(ExternalReference(Builtins::c_ArrayPush), |
| 1299 argc + 1, | 1341 argc + 1, |
| 1300 1); | 1342 1); |
| 1301 } | 1343 } |
| 1302 | 1344 |
| 1303 __ bind(&miss); | 1345 __ bind(&miss); |
| 1304 | 1346 |
| (...skipping 1077 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2382 __ jmp(generic_construct_stub, RelocInfo::CODE_TARGET); | 2424 __ jmp(generic_construct_stub, RelocInfo::CODE_TARGET); |
| 2383 | 2425 |
| 2384 // Return the generated code. | 2426 // Return the generated code. |
| 2385 return GetCode(); | 2427 return GetCode(); |
| 2386 } | 2428 } |
| 2387 | 2429 |
| 2388 | 2430 |
| 2389 #undef __ | 2431 #undef __ |
| 2390 | 2432 |
| 2391 } } // namespace v8::internal | 2433 } } // namespace v8::internal |
| OLD | NEW |