OLD | NEW |
1 // Copyright 2012 the V8 project authors. All rights reserved. | 1 // Copyright 2012 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 419 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
430 ASSERT(cell->value()->IsTheHole()); | 430 ASSERT(cell->value()->IsTheHole()); |
431 __ mov(scratch, Operand(cell)); | 431 __ mov(scratch, Operand(cell)); |
432 __ ldr(scratch, | 432 __ ldr(scratch, |
433 FieldMemOperand(scratch, JSGlobalPropertyCell::kValueOffset)); | 433 FieldMemOperand(scratch, JSGlobalPropertyCell::kValueOffset)); |
434 __ LoadRoot(ip, Heap::kTheHoleValueRootIndex); | 434 __ LoadRoot(ip, Heap::kTheHoleValueRootIndex); |
435 __ cmp(scratch, ip); | 435 __ cmp(scratch, ip); |
436 __ b(ne, miss); | 436 __ b(ne, miss); |
437 } | 437 } |
438 | 438 |
439 | 439 |
440 // Generate StoreField code, value is passed in r0 register. | 440 // Generate StoreTransition code, value is passed in r0 register. |
441 // When leaving generated code after success, the receiver_reg and name_reg | 441 // When leaving generated code after success, the receiver_reg and name_reg |
442 // may be clobbered. Upon branch to miss_label, the receiver and name | 442 // may be clobbered. Upon branch to miss_label, the receiver and name |
443 // registers have their original values. | 443 // registers have their original values. |
444 void StubCompiler::GenerateStoreField(MacroAssembler* masm, | 444 void StubCompiler::GenerateStoreTransition(MacroAssembler* masm, |
445 Handle<JSObject> object, | 445 Handle<JSObject> object, |
446 LookupResult* lookup, | 446 LookupResult* lookup, |
447 Handle<Map> transition, | 447 Handle<Map> transition, |
448 Handle<Name> name, | 448 Handle<Name> name, |
449 Register receiver_reg, | 449 Register receiver_reg, |
450 Register name_reg, | 450 Register name_reg, |
451 Register value_reg, | 451 Register value_reg, |
452 Register scratch1, | 452 Register scratch1, |
453 Register scratch2, | 453 Register scratch2, |
454 Label* miss_label, | 454 Label* miss_label, |
455 Label* miss_restore_name) { | 455 Label* miss_restore_name) { |
456 // r0 : value | 456 // r0 : value |
457 Label exit; | 457 Label exit; |
458 | 458 |
459 // Check that the map of the object hasn't changed. | 459 // Check that the map of the object hasn't changed. |
460 CompareMapMode mode = transition.is_null() ? ALLOW_ELEMENT_TRANSITION_MAPS | |
461 : REQUIRE_EXACT_MAP; | |
462 __ CheckMap(receiver_reg, scratch1, Handle<Map>(object->map()), miss_label, | 460 __ CheckMap(receiver_reg, scratch1, Handle<Map>(object->map()), miss_label, |
463 DO_SMI_CHECK, mode); | 461 DO_SMI_CHECK, REQUIRE_EXACT_MAP); |
464 | 462 |
465 // Perform global security token check if needed. | 463 // Perform global security token check if needed. |
466 if (object->IsJSGlobalProxy()) { | 464 if (object->IsJSGlobalProxy()) { |
467 __ CheckAccessGlobalProxy(receiver_reg, scratch1, miss_label); | 465 __ CheckAccessGlobalProxy(receiver_reg, scratch1, miss_label); |
468 } | 466 } |
469 | 467 |
470 // Check that we are allowed to write this. | 468 // Check that we are allowed to write this. |
471 if (!transition.is_null() && object->GetPrototype()->IsJSObject()) { | 469 if (object->GetPrototype()->IsJSObject()) { |
472 JSObject* holder; | 470 JSObject* holder; |
473 // holder == object indicates that no property was found. | 471 // holder == object indicates that no property was found. |
474 if (lookup->holder() != *object) { | 472 if (lookup->holder() != *object) { |
475 holder = lookup->holder(); | 473 holder = lookup->holder(); |
476 } else { | 474 } else { |
477 // Find the top object. | 475 // Find the top object. |
478 holder = *object; | 476 holder = *object; |
479 do { | 477 do { |
480 holder = JSObject::cast(holder->GetPrototype()); | 478 holder = JSObject::cast(holder->GetPrototype()); |
481 } while (holder->GetPrototype()->IsJSObject()); | 479 } while (holder->GetPrototype()->IsJSObject()); |
(...skipping 17 matching lines...) Expand all Loading... |
499 masm, miss_restore_name, holder_reg, name, scratch1, scratch2); | 497 masm, miss_restore_name, holder_reg, name, scratch1, scratch2); |
500 } | 498 } |
501 } | 499 } |
502 } | 500 } |
503 | 501 |
504 // Stub never generated for non-global objects that require access | 502 // Stub never generated for non-global objects that require access |
505 // checks. | 503 // checks. |
506 ASSERT(object->IsJSGlobalProxy() || !object->IsAccessCheckNeeded()); | 504 ASSERT(object->IsJSGlobalProxy() || !object->IsAccessCheckNeeded()); |
507 | 505 |
508 // Perform map transition for the receiver if necessary. | 506 // Perform map transition for the receiver if necessary. |
509 if (!transition.is_null() && (object->map()->unused_property_fields() == 0)) { | 507 if (object->map()->unused_property_fields() == 0) { |
510 // The properties must be extended before we can store the value. | 508 // The properties must be extended before we can store the value. |
511 // We jump to a runtime call that extends the properties array. | 509 // We jump to a runtime call that extends the properties array. |
512 __ push(receiver_reg); | 510 __ push(receiver_reg); |
513 __ mov(r2, Operand(transition)); | 511 __ mov(r2, Operand(transition)); |
514 __ Push(r2, r0); | 512 __ Push(r2, r0); |
515 __ TailCallExternalReference( | 513 __ TailCallExternalReference( |
516 ExternalReference(IC_Utility(IC::kSharedStoreIC_ExtendStorage), | 514 ExternalReference(IC_Utility(IC::kSharedStoreIC_ExtendStorage), |
517 masm->isolate()), | 515 masm->isolate()), |
518 3, | 516 3, |
519 1); | 517 1); |
520 return; | 518 return; |
521 } | 519 } |
522 | 520 |
523 int index; | 521 // Update the map of the object. |
524 if (!transition.is_null()) { | 522 __ mov(scratch1, Operand(transition)); |
525 // Update the map of the object. | 523 __ str(scratch1, FieldMemOperand(receiver_reg, HeapObject::kMapOffset)); |
526 __ mov(scratch1, Operand(transition)); | |
527 __ str(scratch1, FieldMemOperand(receiver_reg, HeapObject::kMapOffset)); | |
528 | 524 |
529 // Update the write barrier for the map field and pass the now unused | 525 // Update the write barrier for the map field and pass the now unused |
530 // name_reg as scratch register. | 526 // name_reg as scratch register. |
531 __ RecordWriteField(receiver_reg, | 527 __ RecordWriteField(receiver_reg, |
532 HeapObject::kMapOffset, | 528 HeapObject::kMapOffset, |
533 scratch1, | 529 scratch1, |
534 name_reg, | 530 name_reg, |
535 kLRHasNotBeenSaved, | 531 kLRHasNotBeenSaved, |
536 kDontSaveFPRegs, | 532 kDontSaveFPRegs, |
537 OMIT_REMEMBERED_SET, | 533 OMIT_REMEMBERED_SET, |
538 OMIT_SMI_CHECK); | 534 OMIT_SMI_CHECK); |
539 index = transition->instance_descriptors()->GetFieldIndex( | 535 |
540 transition->LastAdded()); | 536 int index = transition->instance_descriptors()->GetFieldIndex( |
541 } else { | 537 transition->LastAdded()); |
542 index = lookup->GetFieldIndex().field_index(); | |
543 } | |
544 | 538 |
545 // Adjust for the number of properties stored in the object. Even in the | 539 // Adjust for the number of properties stored in the object. Even in the |
546 // face of a transition we can use the old map here because the size of the | 540 // face of a transition we can use the old map here because the size of the |
547 // object and the number of in-object properties is not going to change. | 541 // object and the number of in-object properties is not going to change. |
548 index -= object->map()->inobject_properties(); | 542 index -= object->map()->inobject_properties(); |
549 | 543 |
| 544 // TODO(verwaest): Share this code as a code stub. |
550 if (index < 0) { | 545 if (index < 0) { |
551 // Set the property straight into the object. | 546 // Set the property straight into the object. |
552 int offset = object->map()->instance_size() + (index * kPointerSize); | 547 int offset = object->map()->instance_size() + (index * kPointerSize); |
| 548 __ str(value_reg, FieldMemOperand(receiver_reg, offset)); |
| 549 |
| 550 // Skip updating write barrier if storing a smi. |
| 551 __ JumpIfSmi(value_reg, &exit); |
| 552 |
| 553 // Update the write barrier for the array address. |
| 554 // Pass the now unused name_reg as a scratch register. |
| 555 __ mov(name_reg, value_reg); |
| 556 __ RecordWriteField(receiver_reg, |
| 557 offset, |
| 558 name_reg, |
| 559 scratch1, |
| 560 kLRHasNotBeenSaved, |
| 561 kDontSaveFPRegs); |
| 562 } else { |
| 563 // Write to the properties array. |
| 564 int offset = index * kPointerSize + FixedArray::kHeaderSize; |
| 565 // Get the properties array |
| 566 __ ldr(scratch1, |
| 567 FieldMemOperand(receiver_reg, JSObject::kPropertiesOffset)); |
| 568 __ str(value_reg, FieldMemOperand(scratch1, offset)); |
| 569 |
| 570 // Skip updating write barrier if storing a smi. |
| 571 __ JumpIfSmi(value_reg, &exit); |
| 572 |
| 573 // Update the write barrier for the array address. |
| 574 // Ok to clobber receiver_reg and name_reg, since we return. |
| 575 __ mov(name_reg, value_reg); |
| 576 __ RecordWriteField(scratch1, |
| 577 offset, |
| 578 name_reg, |
| 579 receiver_reg, |
| 580 kLRHasNotBeenSaved, |
| 581 kDontSaveFPRegs); |
| 582 } |
| 583 |
| 584 // Return the value (register r0). |
| 585 ASSERT(value_reg.is(r0)); |
| 586 __ bind(&exit); |
| 587 __ Ret(); |
| 588 } |
| 589 |
| 590 |
| 591 // Generate StoreField code, value is passed in r0 register. |
| 592 // When leaving generated code after success, the receiver_reg and name_reg |
| 593 // may be clobbered. Upon branch to miss_label, the receiver and name |
| 594 // registers have their original values. |
| 595 void StubCompiler::GenerateStoreField(MacroAssembler* masm, |
| 596 Handle<JSObject> object, |
| 597 LookupResult* lookup, |
| 598 Register receiver_reg, |
| 599 Register name_reg, |
| 600 Register value_reg, |
| 601 Register scratch1, |
| 602 Register scratch2, |
| 603 Label* miss_label) { |
| 604 // r0 : value |
| 605 Label exit; |
| 606 |
| 607 // Check that the map of the object hasn't changed. |
| 608 __ CheckMap(receiver_reg, scratch1, Handle<Map>(object->map()), miss_label, |
| 609 DO_SMI_CHECK, ALLOW_ELEMENT_TRANSITION_MAPS); |
| 610 |
| 611 // Perform global security token check if needed. |
| 612 if (object->IsJSGlobalProxy()) { |
| 613 __ CheckAccessGlobalProxy(receiver_reg, scratch1, miss_label); |
| 614 } |
| 615 |
| 616 // Stub never generated for non-global objects that require access |
| 617 // checks. |
| 618 ASSERT(object->IsJSGlobalProxy() || !object->IsAccessCheckNeeded()); |
| 619 |
| 620 int index = lookup->GetFieldIndex().field_index(); |
| 621 |
| 622 // Adjust for the number of properties stored in the object. Even in the |
| 623 // face of a transition we can use the old map here because the size of the |
| 624 // object and the number of in-object properties is not going to change. |
| 625 index -= object->map()->inobject_properties(); |
| 626 |
| 627 // TODO(verwaest): Share this code as a code stub. |
| 628 if (index < 0) { |
| 629 // Set the property straight into the object. |
| 630 int offset = object->map()->instance_size() + (index * kPointerSize); |
553 __ str(value_reg, FieldMemOperand(receiver_reg, offset)); | 631 __ str(value_reg, FieldMemOperand(receiver_reg, offset)); |
554 | 632 |
555 // Skip updating write barrier if storing a smi. | 633 // Skip updating write barrier if storing a smi. |
556 __ JumpIfSmi(value_reg, &exit); | 634 __ JumpIfSmi(value_reg, &exit); |
557 | 635 |
558 // Update the write barrier for the array address. | 636 // Update the write barrier for the array address. |
559 // Pass the now unused name_reg as a scratch register. | 637 // Pass the now unused name_reg as a scratch register. |
560 __ mov(name_reg, value_reg); | 638 __ mov(name_reg, value_reg); |
561 __ RecordWriteField(receiver_reg, | 639 __ RecordWriteField(receiver_reg, |
562 offset, | 640 offset, |
(...skipping 3043 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
3606 TailCallBuiltin(masm, Builtins::kKeyedStoreIC_Slow); | 3684 TailCallBuiltin(masm, Builtins::kKeyedStoreIC_Slow); |
3607 } | 3685 } |
3608 } | 3686 } |
3609 | 3687 |
3610 | 3688 |
3611 #undef __ | 3689 #undef __ |
3612 | 3690 |
3613 } } // namespace v8::internal | 3691 } } // namespace v8::internal |
3614 | 3692 |
3615 #endif // V8_TARGET_ARCH_ARM | 3693 #endif // V8_TARGET_ARCH_ARM |
OLD | NEW |