Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(683)

Side by Side Diff: src/arm/codegen-arm.cc

Issue 6576024: (early draft) Strict mode - throw exception on assignment to read only property. (Closed) Base URL: https://v8.googlecode.com/svn/branches/bleeding_edge
Patch Set: Created 9 years, 10 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
1 // Copyright 2010 the V8 project authors. All rights reserved. 1 // Copyright 2010 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 3269 matching lines...) Expand 10 before | Expand all | Expand 10 after
3280 // context slot declaration, but we cannot initialize it at the 3280 // context slot declaration, but we cannot initialize it at the
3281 // same time, because the const declaration may be at the end of 3281 // same time, because the const declaration may be at the end of
3282 // the eval code (sigh...) and the const variable may have been 3282 // the eval code (sigh...) and the const variable may have been
3283 // used before (where its value is 'undefined'). Thus, we can only 3283 // used before (where its value is 'undefined'). Thus, we can only
3284 // do the initialization when we actually encounter the expression 3284 // do the initialization when we actually encounter the expression
3285 // and when the expression operands are defined and valid, and 3285 // and when the expression operands are defined and valid, and
3286 // thus we need the split into 2 operations: declaration of the 3286 // thus we need the split into 2 operations: declaration of the
3287 // context slot followed by initialization. 3287 // context slot followed by initialization.
3288 frame_->CallRuntime(Runtime::kInitializeConstContextSlot, 3); 3288 frame_->CallRuntime(Runtime::kInitializeConstContextSlot, 3);
3289 } else { 3289 } else {
3290 frame_->CallRuntime(Runtime::kStoreContextSlot, 3); 3290 frame_->EmitPush(Operand(Smi::FromInt(strict_mode_flag())));
3291 frame_->CallRuntime(Runtime::kStoreContextSlot, 4);
3291 } 3292 }
3292 // Storing a variable must keep the (new) value on the expression 3293 // Storing a variable must keep the (new) value on the expression
3293 // stack. This is necessary for compiling assignment expressions. 3294 // stack. This is necessary for compiling assignment expressions.
3294 frame_->EmitPush(r0); 3295 frame_->EmitPush(r0);
3295 3296
3296 } else { 3297 } else {
3297 ASSERT(!slot->var()->is_dynamic()); 3298 ASSERT(!slot->var()->is_dynamic());
3298 Register scratch = VirtualFrame::scratch0(); 3299 Register scratch = VirtualFrame::scratch0();
3299 Register scratch2 = VirtualFrame::scratch1(); 3300 Register scratch2 = VirtualFrame::scratch1();
3300 3301
(...skipping 329 matching lines...) Expand 10 before | Expand all | Expand 10 after
3630 frame_->Drop(); 3631 frame_->Drop();
3631 } 3632 }
3632 break; 3633 break;
3633 } 3634 }
3634 // else fall through 3635 // else fall through
3635 case ObjectLiteral::Property::PROTOTYPE: { 3636 case ObjectLiteral::Property::PROTOTYPE: {
3636 frame_->Dup(); 3637 frame_->Dup();
3637 Load(key); 3638 Load(key);
3638 Load(value); 3639 Load(value);
3639 if (property->emit_store()) { 3640 if (property->emit_store()) {
3640 frame_->CallRuntime(Runtime::kSetProperty, 3); 3641 frame_->EmitPush(Operand(Smi::FromInt(NONE))); // PropertyAttributes
3642 frame_->CallRuntime(Runtime::kSetProperty, 4);
3641 } else { 3643 } else {
3642 frame_->Drop(3); 3644 frame_->Drop(3);
3643 } 3645 }
3644 break; 3646 break;
3645 } 3647 }
3646 case ObjectLiteral::Property::SETTER: { 3648 case ObjectLiteral::Property::SETTER: {
3647 frame_->Dup(); 3649 frame_->Dup();
3648 Load(key); 3650 Load(key);
3649 frame_->EmitPush(Operand(Smi::FromInt(1))); 3651 frame_->EmitPush(Operand(Smi::FromInt(1)));
3650 Load(value); 3652 Load(value);
(...skipping 3016 matching lines...) Expand 10 before | Expand all | Expand 10 after
6667 // deferred code. 6669 // deferred code.
6668 __ BlockConstPoolFor(1); 6670 __ BlockConstPoolFor(1);
6669 } 6671 }
6670 } 6672 }
6671 6673
6672 6674
6673 class DeferredReferenceSetKeyedValue: public DeferredCode { 6675 class DeferredReferenceSetKeyedValue: public DeferredCode {
6674 public: 6676 public:
6675 DeferredReferenceSetKeyedValue(Register value, 6677 DeferredReferenceSetKeyedValue(Register value,
6676 Register key, 6678 Register key,
6677 Register receiver) 6679 Register receiver,
6678 : value_(value), key_(key), receiver_(receiver) { 6680 StrictModeFlag strict)
6681 : value_(value), key_(key), receiver_(receiver), strict_(strict) {
6679 set_comment("[ DeferredReferenceSetKeyedValue"); 6682 set_comment("[ DeferredReferenceSetKeyedValue");
6680 } 6683 }
6681 6684
6682 virtual void Generate(); 6685 virtual void Generate();
6683 6686
6684 private: 6687 private:
6685 Register value_; 6688 Register value_;
6686 Register key_; 6689 Register key_;
6687 Register receiver_; 6690 Register receiver_;
6691 StrictModeFlag strict_;
6688 }; 6692 };
6689 6693
6690 6694
6691 void DeferredReferenceSetKeyedValue::Generate() { 6695 void DeferredReferenceSetKeyedValue::Generate() {
6692 Register scratch1 = VirtualFrame::scratch0(); 6696 Register scratch1 = VirtualFrame::scratch0();
6693 Register scratch2 = VirtualFrame::scratch1(); 6697 Register scratch2 = VirtualFrame::scratch1();
6694 __ DecrementCounter(&Counters::keyed_store_inline, 1, scratch1, scratch2); 6698 __ DecrementCounter(&Counters::keyed_store_inline, 1, scratch1, scratch2);
6695 __ IncrementCounter( 6699 __ IncrementCounter(
6696 &Counters::keyed_store_inline_miss, 1, scratch1, scratch2); 6700 &Counters::keyed_store_inline_miss, 1, scratch1, scratch2);
6697 6701
6698 // Ensure value in r0, key in r1 and receiver in r2 to match keyed store ic 6702 // Ensure value in r0, key in r1 and receiver in r2 to match keyed store ic
6699 // calling convention. 6703 // calling convention.
6700 if (value_.is(r1)) { 6704 if (value_.is(r1)) {
6701 __ Swap(r0, r1, ip); 6705 __ Swap(r0, r1, ip);
6702 } 6706 }
6703 ASSERT(receiver_.is(r2)); 6707 ASSERT(receiver_.is(r2));
6704 6708
6705 // The rest of the instructions in the deferred code must be together. 6709 // The rest of the instructions in the deferred code must be together.
6706 { Assembler::BlockConstPoolScope block_const_pool(masm_); 6710 { Assembler::BlockConstPoolScope block_const_pool(masm_);
6707 // Call keyed store IC. It has the arguments value, key and receiver in r0, 6711 // Call keyed store IC. It has the arguments value, key and receiver in r0,
6708 // r1 and r2. 6712 // r1 and r2.
6709 Handle<Code> ic(Builtins::builtin(Builtins::KeyedStoreIC_Initialize)); 6713 Handle<Code> ic(Builtins::builtin(
6714 strict_ == kStrictMode ? Builtins::KeyedStoreIC_Initialize_Strict
6715 : Builtins::KeyedStoreIC_Initialize));
6710 __ Call(ic, RelocInfo::CODE_TARGET); 6716 __ Call(ic, RelocInfo::CODE_TARGET);
6711 // The call must be followed by a nop instruction to indicate that the 6717 // The call must be followed by a nop instruction to indicate that the
6712 // keyed store has been inlined. 6718 // keyed store has been inlined.
6713 __ MarkCode(MacroAssembler::PROPERTY_ACCESS_INLINED); 6719 __ MarkCode(MacroAssembler::PROPERTY_ACCESS_INLINED);
6714 6720
6715 // Block the constant pool for one more instruction after leaving this 6721 // Block the constant pool for one more instruction after leaving this
6716 // constant pool block scope to include the branch instruction ending the 6722 // constant pool block scope to include the branch instruction ending the
6717 // deferred code. 6723 // deferred code.
6718 __ BlockConstPoolFor(1); 6724 __ BlockConstPoolFor(1);
6719 } 6725 }
6720 } 6726 }
6721 6727
6722 6728
6723 class DeferredReferenceSetNamedValue: public DeferredCode { 6729 class DeferredReferenceSetNamedValue: public DeferredCode {
6724 public: 6730 public:
6725 DeferredReferenceSetNamedValue(Register value, 6731 DeferredReferenceSetNamedValue(Register value,
6726 Register receiver, 6732 Register receiver,
6727 Handle<String> name) 6733 Handle<String> name,
6728 : value_(value), receiver_(receiver), name_(name) { 6734 StrictModeFlag strict)
6735 : value_(value), receiver_(receiver), name_(name), strict_(strict) {
6729 set_comment("[ DeferredReferenceSetNamedValue"); 6736 set_comment("[ DeferredReferenceSetNamedValue");
6730 } 6737 }
6731 6738
6732 virtual void Generate(); 6739 virtual void Generate();
6733 6740
6734 private: 6741 private:
6735 Register value_; 6742 Register value_;
6736 Register receiver_; 6743 Register receiver_;
6737 Handle<String> name_; 6744 Handle<String> name_;
6745 StrictModeFlag strict_;
6738 }; 6746 };
6739 6747
6740 6748
6741 // Takes value in r0, receiver in r1 and returns the result (the 6749 // Takes value in r0, receiver in r1 and returns the result (the
6742 // value) in r0. 6750 // value) in r0.
6743 void DeferredReferenceSetNamedValue::Generate() { 6751 void DeferredReferenceSetNamedValue::Generate() {
6744 // Record the entry frame and spill. 6752 // Record the entry frame and spill.
6745 VirtualFrame copied_frame(*frame_state()->frame()); 6753 VirtualFrame copied_frame(*frame_state()->frame());
6746 copied_frame.SpillAll(); 6754 copied_frame.SpillAll();
6747 6755
6748 // Ensure value in r0, receiver in r1 to match store ic calling 6756 // Ensure value in r0, receiver in r1 to match store ic calling
6749 // convention. 6757 // convention.
6750 ASSERT(value_.is(r0) && receiver_.is(r1)); 6758 ASSERT(value_.is(r0) && receiver_.is(r1));
6751 __ mov(r2, Operand(name_)); 6759 __ mov(r2, Operand(name_));
6752 6760
6753 // The rest of the instructions in the deferred code must be together. 6761 // The rest of the instructions in the deferred code must be together.
6754 { Assembler::BlockConstPoolScope block_const_pool(masm_); 6762 { Assembler::BlockConstPoolScope block_const_pool(masm_);
6755 // Call keyed store IC. It has the arguments value, key and receiver in r0, 6763 // Call keyed store IC. It has the arguments value, key and receiver in r0,
6756 // r1 and r2. 6764 // r1 and r2.
6757 Handle<Code> ic(Builtins::builtin(Builtins::StoreIC_Initialize)); 6765 Handle<Code> ic(Builtins::builtin(
6766 strict_ == kStrictMode ? Builtins::StoreIC_Initialize_Strict
6767 : Builtins::StoreIC_Initialize));
6758 __ Call(ic, RelocInfo::CODE_TARGET); 6768 __ Call(ic, RelocInfo::CODE_TARGET);
6759 // The call must be followed by a nop instruction to indicate that the 6769 // The call must be followed by a nop instruction to indicate that the
6760 // named store has been inlined. 6770 // named store has been inlined.
6761 __ MarkCode(MacroAssembler::PROPERTY_ACCESS_INLINED); 6771 __ MarkCode(MacroAssembler::PROPERTY_ACCESS_INLINED);
6762 6772
6763 // Go back to the frame we entered with. The instructions 6773 // Go back to the frame we entered with. The instructions
6764 // generated by this merge are skipped over by the inline store 6774 // generated by this merge are skipped over by the inline store
6765 // patching mechanism when looking for the branch instruction that 6775 // patching mechanism when looking for the branch instruction that
6766 // tells it where the code to patch is. 6776 // tells it where the code to patch is.
6767 copied_frame.MergeTo(frame_state()->frame()); 6777 copied_frame.MergeTo(frame_state()->frame());
(...skipping 168 matching lines...) Expand 10 before | Expand all | Expand 10 after
6936 // Inline the in-object property case. 6946 // Inline the in-object property case.
6937 JumpTarget slow, done; 6947 JumpTarget slow, done;
6938 6948
6939 // Get the value and receiver from the stack. 6949 // Get the value and receiver from the stack.
6940 frame()->PopToR0(); 6950 frame()->PopToR0();
6941 Register value = r0; 6951 Register value = r0;
6942 frame()->PopToR1(); 6952 frame()->PopToR1();
6943 Register receiver = r1; 6953 Register receiver = r1;
6944 6954
6945 DeferredReferenceSetNamedValue* deferred = 6955 DeferredReferenceSetNamedValue* deferred =
6946 new DeferredReferenceSetNamedValue(value, receiver, name); 6956 new DeferredReferenceSetNamedValue(
6957 value, receiver, name, strict_mode_flag());
6947 6958
6948 // Check that the receiver is a heap object. 6959 // Check that the receiver is a heap object.
6949 __ tst(receiver, Operand(kSmiTagMask)); 6960 __ tst(receiver, Operand(kSmiTagMask));
6950 deferred->Branch(eq); 6961 deferred->Branch(eq);
6951 6962
6952 // The following instructions are the part of the inlined 6963 // The following instructions are the part of the inlined
6953 // in-object property store code which can be patched. Therefore 6964 // in-object property store code which can be patched. Therefore
6954 // the exact number of instructions generated must be fixed, so 6965 // the exact number of instructions generated must be fixed, so
6955 // the constant pool is blocked while generating this code. 6966 // the constant pool is blocked while generating this code.
6956 { Assembler::BlockConstPoolScope block_const_pool(masm_); 6967 { Assembler::BlockConstPoolScope block_const_pool(masm_);
(...skipping 165 matching lines...) Expand 10 before | Expand all | Expand 10 after
7122 VirtualFrame::SpilledScope spilled(frame_); 7133 VirtualFrame::SpilledScope spilled(frame_);
7123 Register receiver = r2; 7134 Register receiver = r2;
7124 frame_->EmitPop(receiver); 7135 frame_->EmitPop(receiver);
7125 7136
7126 #ifdef DEBUG 7137 #ifdef DEBUG
7127 bool we_remembered_the_write_barrier = value_is_harmless; 7138 bool we_remembered_the_write_barrier = value_is_harmless;
7128 #endif 7139 #endif
7129 7140
7130 // The deferred code expects value, key and receiver in registers. 7141 // The deferred code expects value, key and receiver in registers.
7131 DeferredReferenceSetKeyedValue* deferred = 7142 DeferredReferenceSetKeyedValue* deferred =
7132 new DeferredReferenceSetKeyedValue(value, key, receiver); 7143 new DeferredReferenceSetKeyedValue(
7144 value, key, receiver, strict_mode_flag());
7133 7145
7134 // Check that the value is a smi. As this inlined code does not set the 7146 // Check that the value is a smi. As this inlined code does not set the
7135 // write barrier it is only possible to store smi values. 7147 // write barrier it is only possible to store smi values.
7136 if (!value_is_harmless) { 7148 if (!value_is_harmless) {
7137 // If the value is not likely to be a Smi then let's test the fixed array 7149 // If the value is not likely to be a Smi then let's test the fixed array
7138 // for new space instead. See below. 7150 // for new space instead. See below.
7139 if (wb_info == LIKELY_SMI) { 7151 if (wb_info == LIKELY_SMI) {
7140 __ tst(value, Operand(kSmiTagMask)); 7152 __ tst(value, Operand(kSmiTagMask));
7141 deferred->Branch(ne); 7153 deferred->Branch(ne);
7142 #ifdef DEBUG 7154 #ifdef DEBUG
(...skipping 64 matching lines...) Expand 10 before | Expand all | Expand 10 after
7207 7219
7208 // Make sure that the expected number of instructions are generated. 7220 // Make sure that the expected number of instructions are generated.
7209 ASSERT_EQ(kInlinedKeyedStoreInstructionsAfterPatch, 7221 ASSERT_EQ(kInlinedKeyedStoreInstructionsAfterPatch,
7210 masm_->InstructionsGeneratedSince(&check_inlined_codesize)); 7222 masm_->InstructionsGeneratedSince(&check_inlined_codesize));
7211 } 7223 }
7212 7224
7213 ASSERT(we_remembered_the_write_barrier); 7225 ASSERT(we_remembered_the_write_barrier);
7214 7226
7215 deferred->BindExit(); 7227 deferred->BindExit();
7216 } else { 7228 } else {
7217 frame()->CallKeyedStoreIC(); 7229 frame()->CallKeyedStoreIC(strict_mode_flag());
7218 } 7230 }
7219 } 7231 }
7220 7232
7221 7233
7222 #ifdef DEBUG 7234 #ifdef DEBUG
7223 bool CodeGenerator::HasValidEntryRegisters() { return true; } 7235 bool CodeGenerator::HasValidEntryRegisters() { return true; }
7224 #endif 7236 #endif
7225 7237
7226 7238
7227 #undef __ 7239 #undef __
(...skipping 147 matching lines...) Expand 10 before | Expand all | Expand 10 after
7375 BinaryOpIC::GetName(runtime_operands_type_)); 7387 BinaryOpIC::GetName(runtime_operands_type_));
7376 return name_; 7388 return name_;
7377 } 7389 }
7378 7390
7379 7391
7380 #undef __ 7392 #undef __
7381 7393
7382 } } // namespace v8::internal 7394 } } // namespace v8::internal
7383 7395
7384 #endif // V8_TARGET_ARCH_ARM 7396 #endif // V8_TARGET_ARCH_ARM
OLDNEW
« no previous file with comments | « src/api.cc ('k') | src/arm/full-codegen-arm.cc » ('j') | src/arm/ic-arm.cc » ('J')

Powered by Google App Engine
This is Rietveld 408576698