| OLD | NEW |
| 1 // Copyright 2012 the V8 project authors. All rights reserved. | 1 // Copyright 2012 the V8 project authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #if V8_TARGET_ARCH_X87 | 5 #if V8_TARGET_ARCH_X87 |
| 6 | 6 |
| 7 #include "src/ast/scopes.h" | 7 #include "src/ast/scopes.h" |
| 8 #include "src/code-factory.h" | 8 #include "src/code-factory.h" |
| 9 #include "src/code-stubs.h" | 9 #include "src/code-stubs.h" |
| 10 #include "src/codegen.h" | 10 #include "src/codegen.h" |
| (...skipping 908 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 919 PrepareForBailoutForId(stmt->ExitId(), NO_REGISTERS); | 919 PrepareForBailoutForId(stmt->ExitId(), NO_REGISTERS); |
| 920 } | 920 } |
| 921 | 921 |
| 922 | 922 |
| 923 void FullCodeGenerator::VisitForInStatement(ForInStatement* stmt) { | 923 void FullCodeGenerator::VisitForInStatement(ForInStatement* stmt) { |
| 924 Comment cmnt(masm_, "[ ForInStatement"); | 924 Comment cmnt(masm_, "[ ForInStatement"); |
| 925 SetStatementPosition(stmt, SKIP_BREAK); | 925 SetStatementPosition(stmt, SKIP_BREAK); |
| 926 | 926 |
| 927 FeedbackVectorSlot slot = stmt->ForInFeedbackSlot(); | 927 FeedbackVectorSlot slot = stmt->ForInFeedbackSlot(); |
| 928 | 928 |
| 929 Label loop, exit; | |
| 930 ForIn loop_statement(this, stmt); | |
| 931 increment_loop_depth(); | |
| 932 | |
| 933 // Get the object to enumerate over. | 929 // Get the object to enumerate over. |
| 934 SetExpressionAsStatementPosition(stmt->enumerable()); | 930 SetExpressionAsStatementPosition(stmt->enumerable()); |
| 935 VisitForAccumulatorValue(stmt->enumerable()); | 931 VisitForAccumulatorValue(stmt->enumerable()); |
| 936 OperandStackDepthIncrement(ForIn::kElementCount); | 932 OperandStackDepthIncrement(5); |
| 933 |
| 934 Label loop, exit; |
| 935 Iteration loop_statement(this, stmt); |
| 936 increment_loop_depth(); |
| 937 | 937 |
| 938 // If the object is null or undefined, skip over the loop, otherwise convert | 938 // If the object is null or undefined, skip over the loop, otherwise convert |
| 939 // it to a JS receiver. See ECMA-262 version 5, section 12.6.4. | 939 // it to a JS receiver. See ECMA-262 version 5, section 12.6.4. |
| 940 Label convert, done_convert; | 940 Label convert, done_convert; |
| 941 __ JumpIfSmi(eax, &convert, Label::kNear); | 941 __ JumpIfSmi(eax, &convert, Label::kNear); |
| 942 __ CmpObjectType(eax, FIRST_JS_RECEIVER_TYPE, ecx); | 942 __ CmpObjectType(eax, FIRST_JS_RECEIVER_TYPE, ecx); |
| 943 __ j(above_equal, &done_convert, Label::kNear); | 943 __ j(above_equal, &done_convert, Label::kNear); |
| 944 __ cmp(eax, isolate()->factory()->undefined_value()); | 944 __ cmp(eax, isolate()->factory()->undefined_value()); |
| 945 __ j(equal, &exit); | 945 __ j(equal, &exit); |
| 946 __ cmp(eax, isolate()->factory()->null_value()); | 946 __ cmp(eax, isolate()->factory()->null_value()); |
| (...skipping 127 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1074 // Generate code for going to the next element by incrementing the | 1074 // Generate code for going to the next element by incrementing the |
| 1075 // index (smi) stored on top of the stack. | 1075 // index (smi) stored on top of the stack. |
| 1076 __ bind(loop_statement.continue_label()); | 1076 __ bind(loop_statement.continue_label()); |
| 1077 __ add(Operand(esp, 0 * kPointerSize), Immediate(Smi::FromInt(1))); | 1077 __ add(Operand(esp, 0 * kPointerSize), Immediate(Smi::FromInt(1))); |
| 1078 | 1078 |
| 1079 EmitBackEdgeBookkeeping(stmt, &loop); | 1079 EmitBackEdgeBookkeeping(stmt, &loop); |
| 1080 __ jmp(&loop); | 1080 __ jmp(&loop); |
| 1081 | 1081 |
| 1082 // Remove the pointers stored on the stack. | 1082 // Remove the pointers stored on the stack. |
| 1083 __ bind(loop_statement.break_label()); | 1083 __ bind(loop_statement.break_label()); |
| 1084 __ add(esp, Immediate(5 * kPointerSize)); | 1084 DropOperands(5); |
| 1085 OperandStackDepthDecrement(ForIn::kElementCount); | |
| 1086 | 1085 |
| 1087 // Exit and decrement the loop depth. | 1086 // Exit and decrement the loop depth. |
| 1088 PrepareForBailoutForId(stmt->ExitId(), NO_REGISTERS); | 1087 PrepareForBailoutForId(stmt->ExitId(), NO_REGISTERS); |
| 1089 __ bind(&exit); | 1088 __ bind(&exit); |
| 1090 decrement_loop_depth(); | 1089 decrement_loop_depth(); |
| 1091 } | 1090 } |
| 1092 | 1091 |
| 1093 | 1092 |
| 1094 void FullCodeGenerator::EmitSetHomeObject(Expression* initializer, int offset, | 1093 void FullCodeGenerator::EmitSetHomeObject(Expression* initializer, int offset, |
| 1095 FeedbackVectorSlot slot) { | 1094 FeedbackVectorSlot slot) { |
| (...skipping 669 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1765 SetExpressionPosition(expr); | 1764 SetExpressionPosition(expr); |
| 1766 | 1765 |
| 1767 // Evaluate yielded value first; the initial iterator definition depends on | 1766 // Evaluate yielded value first; the initial iterator definition depends on |
| 1768 // this. It stays on the stack while we update the iterator. | 1767 // this. It stays on the stack while we update the iterator. |
| 1769 VisitForStackValue(expr->expression()); | 1768 VisitForStackValue(expr->expression()); |
| 1770 | 1769 |
| 1771 switch (expr->yield_kind()) { | 1770 switch (expr->yield_kind()) { |
| 1772 case Yield::kSuspend: | 1771 case Yield::kSuspend: |
| 1773 // Pop value from top-of-stack slot; box result into result register. | 1772 // Pop value from top-of-stack slot; box result into result register. |
| 1774 EmitCreateIteratorResult(false); | 1773 EmitCreateIteratorResult(false); |
| 1775 __ push(result_register()); | 1774 PushOperand(result_register()); |
| 1776 // Fall through. | 1775 // Fall through. |
| 1777 case Yield::kInitial: { | 1776 case Yield::kInitial: { |
| 1778 Label suspend, continuation, post_runtime, resume; | 1777 Label suspend, continuation, post_runtime, resume; |
| 1779 | 1778 |
| 1780 __ jmp(&suspend); | 1779 __ jmp(&suspend); |
| 1781 __ bind(&continuation); | 1780 __ bind(&continuation); |
| 1782 // When we arrive here, the stack top is the resume mode and | 1781 // When we arrive here, the stack top is the resume mode and |
| 1783 // result_register() holds the input value (the argument given to the | 1782 // result_register() holds the input value (the argument given to the |
| 1784 // respective resume operation). | 1783 // respective resume operation). |
| 1785 __ RecordGeneratorContinuation(); | 1784 __ RecordGeneratorContinuation(); |
| 1786 __ pop(ebx); | 1785 __ pop(ebx); |
| 1787 __ cmp(ebx, Immediate(Smi::FromInt(JSGeneratorObject::RETURN))); | 1786 __ cmp(ebx, Immediate(Smi::FromInt(JSGeneratorObject::RETURN))); |
| 1788 __ j(not_equal, &resume); | 1787 __ j(not_equal, &resume); |
| 1789 __ push(result_register()); | 1788 __ push(result_register()); |
| 1790 EmitCreateIteratorResult(true); | 1789 EmitCreateIteratorResult(true); |
| 1791 EmitUnwindAndReturn(); | 1790 EmitUnwindAndReturn(); |
| 1792 | 1791 |
| 1793 __ bind(&suspend); | 1792 __ bind(&suspend); |
| 1793 OperandStackDepthIncrement(1); // Not popped on this path. |
| 1794 VisitForAccumulatorValue(expr->generator_object()); | 1794 VisitForAccumulatorValue(expr->generator_object()); |
| 1795 DCHECK(continuation.pos() > 0 && Smi::IsValid(continuation.pos())); | 1795 DCHECK(continuation.pos() > 0 && Smi::IsValid(continuation.pos())); |
| 1796 __ mov(FieldOperand(eax, JSGeneratorObject::kContinuationOffset), | 1796 __ mov(FieldOperand(eax, JSGeneratorObject::kContinuationOffset), |
| 1797 Immediate(Smi::FromInt(continuation.pos()))); | 1797 Immediate(Smi::FromInt(continuation.pos()))); |
| 1798 __ mov(FieldOperand(eax, JSGeneratorObject::kContextOffset), esi); | 1798 __ mov(FieldOperand(eax, JSGeneratorObject::kContextOffset), esi); |
| 1799 __ mov(ecx, esi); | 1799 __ mov(ecx, esi); |
| 1800 __ RecordWriteField(eax, JSGeneratorObject::kContextOffset, ecx, edx, | 1800 __ RecordWriteField(eax, JSGeneratorObject::kContextOffset, ecx, edx, |
| 1801 kDontSaveFPRegs); | 1801 kDontSaveFPRegs); |
| 1802 __ lea(ebx, Operand(ebp, StandardFrameConstants::kExpressionsOffset)); | 1802 __ lea(ebx, Operand(ebp, StandardFrameConstants::kExpressionsOffset)); |
| 1803 __ cmp(esp, ebx); | 1803 __ cmp(esp, ebx); |
| 1804 __ j(equal, &post_runtime); | 1804 __ j(equal, &post_runtime); |
| 1805 __ push(eax); // generator object | 1805 __ push(eax); // generator object |
| 1806 __ CallRuntime(Runtime::kSuspendJSGeneratorObject, 1); | 1806 __ CallRuntime(Runtime::kSuspendJSGeneratorObject, 1); |
| 1807 __ mov(context_register(), | 1807 __ mov(context_register(), |
| 1808 Operand(ebp, StandardFrameConstants::kContextOffset)); | 1808 Operand(ebp, StandardFrameConstants::kContextOffset)); |
| 1809 __ bind(&post_runtime); | 1809 __ bind(&post_runtime); |
| 1810 PopOperand(result_register()); | 1810 PopOperand(result_register()); |
| 1811 EmitReturnSequence(); | 1811 EmitReturnSequence(); |
| 1812 | 1812 |
| 1813 __ bind(&resume); | 1813 __ bind(&resume); |
| 1814 context()->Plug(result_register()); | 1814 context()->Plug(result_register()); |
| 1815 break; | 1815 break; |
| 1816 } | 1816 } |
| 1817 | 1817 |
| 1818 case Yield::kFinal: { | 1818 case Yield::kFinal: { |
| 1819 // Pop value from top-of-stack slot, box result into result register. | 1819 // Pop value from top-of-stack slot, box result into result register. |
| 1820 OperandStackDepthDecrement(1); | |
| 1821 EmitCreateIteratorResult(true); | 1820 EmitCreateIteratorResult(true); |
| 1822 EmitUnwindAndReturn(); | 1821 EmitUnwindAndReturn(); |
| 1823 break; | 1822 break; |
| 1824 } | 1823 } |
| 1825 | 1824 |
| 1826 case Yield::kDelegating: | 1825 case Yield::kDelegating: |
| 1827 UNREACHABLE(); | 1826 UNREACHABLE(); |
| 1828 } | 1827 } |
| 1829 } | 1828 } |
| 1830 | 1829 |
| (...skipping 120 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1951 __ mov(ebx, ContextOperand(ebx, Context::ITERATOR_RESULT_MAP_INDEX)); | 1950 __ mov(ebx, ContextOperand(ebx, Context::ITERATOR_RESULT_MAP_INDEX)); |
| 1952 __ mov(FieldOperand(eax, HeapObject::kMapOffset), ebx); | 1951 __ mov(FieldOperand(eax, HeapObject::kMapOffset), ebx); |
| 1953 __ mov(FieldOperand(eax, JSObject::kPropertiesOffset), | 1952 __ mov(FieldOperand(eax, JSObject::kPropertiesOffset), |
| 1954 isolate()->factory()->empty_fixed_array()); | 1953 isolate()->factory()->empty_fixed_array()); |
| 1955 __ mov(FieldOperand(eax, JSObject::kElementsOffset), | 1954 __ mov(FieldOperand(eax, JSObject::kElementsOffset), |
| 1956 isolate()->factory()->empty_fixed_array()); | 1955 isolate()->factory()->empty_fixed_array()); |
| 1957 __ pop(FieldOperand(eax, JSIteratorResult::kValueOffset)); | 1956 __ pop(FieldOperand(eax, JSIteratorResult::kValueOffset)); |
| 1958 __ mov(FieldOperand(eax, JSIteratorResult::kDoneOffset), | 1957 __ mov(FieldOperand(eax, JSIteratorResult::kDoneOffset), |
| 1959 isolate()->factory()->ToBoolean(done)); | 1958 isolate()->factory()->ToBoolean(done)); |
| 1960 STATIC_ASSERT(JSIteratorResult::kSize == 5 * kPointerSize); | 1959 STATIC_ASSERT(JSIteratorResult::kSize == 5 * kPointerSize); |
| 1960 OperandStackDepthDecrement(1); |
| 1961 } | 1961 } |
| 1962 | 1962 |
| 1963 | 1963 |
| 1964 void FullCodeGenerator::EmitInlineSmiBinaryOp(BinaryOperation* expr, | 1964 void FullCodeGenerator::EmitInlineSmiBinaryOp(BinaryOperation* expr, |
| 1965 Token::Value op, | 1965 Token::Value op, |
| 1966 Expression* left, | 1966 Expression* left, |
| 1967 Expression* right) { | 1967 Expression* right) { |
| 1968 // Do combined smi check of the operands. Left operand is on the | 1968 // Do combined smi check of the operands. Left operand is on the |
| 1969 // stack. Right operand is in eax. | 1969 // stack. Right operand is in eax. |
| 1970 Label smi_case, done, stub_call; | 1970 Label smi_case, done, stub_call; |
| (...skipping 2117 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 4088 Assembler::target_address_at(call_target_address, | 4088 Assembler::target_address_at(call_target_address, |
| 4089 unoptimized_code)); | 4089 unoptimized_code)); |
| 4090 return OSR_AFTER_STACK_CHECK; | 4090 return OSR_AFTER_STACK_CHECK; |
| 4091 } | 4091 } |
| 4092 | 4092 |
| 4093 | 4093 |
| 4094 } // namespace internal | 4094 } // namespace internal |
| 4095 } // namespace v8 | 4095 } // namespace v8 |
| 4096 | 4096 |
| 4097 #endif // V8_TARGET_ARCH_X87 | 4097 #endif // V8_TARGET_ARCH_X87 |
| OLD | NEW |