| OLD | NEW |
| 1 // Copyright 2013 the V8 project authors. All rights reserved. | 1 // Copyright 2013 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 1568 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1579 __ and_(result, Immediate(Map::kElementsKindMask)); | 1579 __ and_(result, Immediate(Map::kElementsKindMask)); |
| 1580 __ shr(result, Immediate(Map::kElementsKindShift)); | 1580 __ shr(result, Immediate(Map::kElementsKindShift)); |
| 1581 } | 1581 } |
| 1582 | 1582 |
| 1583 | 1583 |
| 1584 void LCodeGen::DoValueOf(LValueOf* instr) { | 1584 void LCodeGen::DoValueOf(LValueOf* instr) { |
| 1585 Register input = ToRegister(instr->value()); | 1585 Register input = ToRegister(instr->value()); |
| 1586 Register result = ToRegister(instr->result()); | 1586 Register result = ToRegister(instr->result()); |
| 1587 ASSERT(input.is(result)); | 1587 ASSERT(input.is(result)); |
| 1588 Label done; | 1588 Label done; |
| 1589 // If the object is a smi return the object. | 1589 |
| 1590 __ JumpIfSmi(input, &done, Label::kNear); | 1590 if (!instr->hydrogen()->value()->IsHeapObject()) { |
| 1591 // If the object is a smi return the object. |
| 1592 __ JumpIfSmi(input, &done, Label::kNear); |
| 1593 } |
| 1591 | 1594 |
| 1592 // If the object is not a value type, return the object. | 1595 // If the object is not a value type, return the object. |
| 1593 __ CmpObjectType(input, JS_VALUE_TYPE, kScratchRegister); | 1596 __ CmpObjectType(input, JS_VALUE_TYPE, kScratchRegister); |
| 1594 __ j(not_equal, &done, Label::kNear); | 1597 __ j(not_equal, &done, Label::kNear); |
| 1595 __ movq(result, FieldOperand(input, JSValue::kValueOffset)); | 1598 __ movq(result, FieldOperand(input, JSValue::kValueOffset)); |
| 1596 | 1599 |
| 1597 __ bind(&done); | 1600 __ bind(&done); |
| 1598 } | 1601 } |
| 1599 | 1602 |
| 1600 | 1603 |
| (...skipping 543 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2144 | 2147 |
| 2145 Condition true_cond = EmitIsObject( | 2148 Condition true_cond = EmitIsObject( |
| 2146 reg, instr->FalseLabel(chunk_), instr->TrueLabel(chunk_)); | 2149 reg, instr->FalseLabel(chunk_), instr->TrueLabel(chunk_)); |
| 2147 | 2150 |
| 2148 EmitBranch(instr, true_cond); | 2151 EmitBranch(instr, true_cond); |
| 2149 } | 2152 } |
| 2150 | 2153 |
| 2151 | 2154 |
| 2152 Condition LCodeGen::EmitIsString(Register input, | 2155 Condition LCodeGen::EmitIsString(Register input, |
| 2153 Register temp1, | 2156 Register temp1, |
| 2154 Label* is_not_string) { | 2157 Label* is_not_string, |
| 2155 __ JumpIfSmi(input, is_not_string); | 2158 SmiCheck check_needed = INLINE_SMI_CHECK) { |
| 2159 if (check_needed == INLINE_SMI_CHECK) { |
| 2160 __ JumpIfSmi(input, is_not_string); |
| 2161 } |
| 2162 |
| 2156 Condition cond = masm_->IsObjectStringType(input, temp1, temp1); | 2163 Condition cond = masm_->IsObjectStringType(input, temp1, temp1); |
| 2157 | 2164 |
| 2158 return cond; | 2165 return cond; |
| 2159 } | 2166 } |
| 2160 | 2167 |
| 2161 | 2168 |
| 2162 void LCodeGen::DoIsStringAndBranch(LIsStringAndBranch* instr) { | 2169 void LCodeGen::DoIsStringAndBranch(LIsStringAndBranch* instr) { |
| 2163 Register reg = ToRegister(instr->value()); | 2170 Register reg = ToRegister(instr->value()); |
| 2164 Register temp = ToRegister(instr->temp()); | 2171 Register temp = ToRegister(instr->temp()); |
| 2165 | 2172 |
| 2166 Condition true_cond = EmitIsString(reg, temp, instr->FalseLabel(chunk_)); | 2173 SmiCheck check_needed = |
| 2174 instr->hydrogen()->value()->IsHeapObject() |
| 2175 ? OMIT_SMI_CHECK : INLINE_SMI_CHECK; |
| 2176 |
| 2177 Condition true_cond = EmitIsString( |
| 2178 reg, temp, instr->FalseLabel(chunk_), check_needed); |
| 2167 | 2179 |
| 2168 EmitBranch(instr, true_cond); | 2180 EmitBranch(instr, true_cond); |
| 2169 } | 2181 } |
| 2170 | 2182 |
| 2171 | 2183 |
| 2172 void LCodeGen::DoIsSmiAndBranch(LIsSmiAndBranch* instr) { | 2184 void LCodeGen::DoIsSmiAndBranch(LIsSmiAndBranch* instr) { |
| 2173 Condition is_smi; | 2185 Condition is_smi; |
| 2174 if (instr->value()->IsRegister()) { | 2186 if (instr->value()->IsRegister()) { |
| 2175 Register input = ToRegister(instr->value()); | 2187 Register input = ToRegister(instr->value()); |
| 2176 is_smi = masm()->CheckSmi(input); | 2188 is_smi = masm()->CheckSmi(input); |
| 2177 } else { | 2189 } else { |
| 2178 Operand input = ToOperand(instr->value()); | 2190 Operand input = ToOperand(instr->value()); |
| 2179 is_smi = masm()->CheckSmi(input); | 2191 is_smi = masm()->CheckSmi(input); |
| 2180 } | 2192 } |
| 2181 EmitBranch(instr, is_smi); | 2193 EmitBranch(instr, is_smi); |
| 2182 } | 2194 } |
| 2183 | 2195 |
| 2184 | 2196 |
| 2185 void LCodeGen::DoIsUndetectableAndBranch(LIsUndetectableAndBranch* instr) { | 2197 void LCodeGen::DoIsUndetectableAndBranch(LIsUndetectableAndBranch* instr) { |
| 2186 Register input = ToRegister(instr->value()); | 2198 Register input = ToRegister(instr->value()); |
| 2187 Register temp = ToRegister(instr->temp()); | 2199 Register temp = ToRegister(instr->temp()); |
| 2188 | 2200 |
| 2189 __ JumpIfSmi(input, instr->FalseLabel(chunk_)); | 2201 if (!instr->hydrogen()->value()->IsHeapObject()) { |
| 2202 __ JumpIfSmi(input, instr->FalseLabel(chunk_)); |
| 2203 } |
| 2190 __ movq(temp, FieldOperand(input, HeapObject::kMapOffset)); | 2204 __ movq(temp, FieldOperand(input, HeapObject::kMapOffset)); |
| 2191 __ testb(FieldOperand(temp, Map::kBitFieldOffset), | 2205 __ testb(FieldOperand(temp, Map::kBitFieldOffset), |
| 2192 Immediate(1 << Map::kIsUndetectable)); | 2206 Immediate(1 << Map::kIsUndetectable)); |
| 2193 EmitBranch(instr, not_zero); | 2207 EmitBranch(instr, not_zero); |
| 2194 } | 2208 } |
| 2195 | 2209 |
| 2196 | 2210 |
| 2197 void LCodeGen::DoStringCompareAndBranch(LStringCompareAndBranch* instr) { | 2211 void LCodeGen::DoStringCompareAndBranch(LStringCompareAndBranch* instr) { |
| 2198 Token::Value op = instr->op(); | 2212 Token::Value op = instr->op(); |
| 2199 | 2213 |
| (...skipping 23 matching lines...) Expand all Loading... |
| 2223 if (to == LAST_TYPE) return above_equal; | 2237 if (to == LAST_TYPE) return above_equal; |
| 2224 if (from == FIRST_TYPE) return below_equal; | 2238 if (from == FIRST_TYPE) return below_equal; |
| 2225 UNREACHABLE(); | 2239 UNREACHABLE(); |
| 2226 return equal; | 2240 return equal; |
| 2227 } | 2241 } |
| 2228 | 2242 |
| 2229 | 2243 |
| 2230 void LCodeGen::DoHasInstanceTypeAndBranch(LHasInstanceTypeAndBranch* instr) { | 2244 void LCodeGen::DoHasInstanceTypeAndBranch(LHasInstanceTypeAndBranch* instr) { |
| 2231 Register input = ToRegister(instr->value()); | 2245 Register input = ToRegister(instr->value()); |
| 2232 | 2246 |
| 2233 __ JumpIfSmi(input, instr->FalseLabel(chunk_)); | 2247 if (!instr->hydrogen()->value()->IsHeapObject()) { |
| 2248 __ JumpIfSmi(input, instr->FalseLabel(chunk_)); |
| 2249 } |
| 2234 | 2250 |
| 2235 __ CmpObjectType(input, TestType(instr->hydrogen()), kScratchRegister); | 2251 __ CmpObjectType(input, TestType(instr->hydrogen()), kScratchRegister); |
| 2236 EmitBranch(instr, BranchCondition(instr->hydrogen())); | 2252 EmitBranch(instr, BranchCondition(instr->hydrogen())); |
| 2237 } | 2253 } |
| 2238 | 2254 |
| 2239 | 2255 |
| 2240 void LCodeGen::DoGetCachedArrayIndex(LGetCachedArrayIndex* instr) { | 2256 void LCodeGen::DoGetCachedArrayIndex(LGetCachedArrayIndex* instr) { |
| 2241 Register input = ToRegister(instr->value()); | 2257 Register input = ToRegister(instr->value()); |
| 2242 Register result = ToRegister(instr->result()); | 2258 Register result = ToRegister(instr->result()); |
| 2243 | 2259 |
| (...skipping 389 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2633 __ CompareRoot(target, Heap::kTheHoleValueRootIndex); | 2649 __ CompareRoot(target, Heap::kTheHoleValueRootIndex); |
| 2634 if (instr->hydrogen()->DeoptimizesOnHole()) { | 2650 if (instr->hydrogen()->DeoptimizesOnHole()) { |
| 2635 DeoptimizeIf(equal, instr->environment()); | 2651 DeoptimizeIf(equal, instr->environment()); |
| 2636 } else { | 2652 } else { |
| 2637 __ j(not_equal, &skip_assignment); | 2653 __ j(not_equal, &skip_assignment); |
| 2638 } | 2654 } |
| 2639 } | 2655 } |
| 2640 __ movq(target, value); | 2656 __ movq(target, value); |
| 2641 | 2657 |
| 2642 if (instr->hydrogen()->NeedsWriteBarrier()) { | 2658 if (instr->hydrogen()->NeedsWriteBarrier()) { |
| 2643 HType type = instr->hydrogen()->value()->type(); | |
| 2644 SmiCheck check_needed = | 2659 SmiCheck check_needed = |
| 2645 type.IsHeapObject() ? OMIT_SMI_CHECK : INLINE_SMI_CHECK; | 2660 instr->hydrogen()->value()->IsHeapObject() |
| 2661 ? OMIT_SMI_CHECK : INLINE_SMI_CHECK; |
| 2646 int offset = Context::SlotOffset(instr->slot_index()); | 2662 int offset = Context::SlotOffset(instr->slot_index()); |
| 2647 Register scratch = ToRegister(instr->temp()); | 2663 Register scratch = ToRegister(instr->temp()); |
| 2648 __ RecordWriteContextSlot(context, | 2664 __ RecordWriteContextSlot(context, |
| 2649 offset, | 2665 offset, |
| 2650 value, | 2666 value, |
| 2651 scratch, | 2667 scratch, |
| 2652 kSaveFPRegs, | 2668 kSaveFPRegs, |
| 2653 EMIT_REMEMBERED_SET, | 2669 EMIT_REMEMBERED_SET, |
| 2654 check_needed); | 2670 check_needed); |
| 2655 } | 2671 } |
| (...skipping 1302 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 3958 HeapObject::kMapOffset, | 3974 HeapObject::kMapOffset, |
| 3959 kScratchRegister, | 3975 kScratchRegister, |
| 3960 temp, | 3976 temp, |
| 3961 kSaveFPRegs, | 3977 kSaveFPRegs, |
| 3962 OMIT_REMEMBERED_SET, | 3978 OMIT_REMEMBERED_SET, |
| 3963 OMIT_SMI_CHECK); | 3979 OMIT_SMI_CHECK); |
| 3964 } | 3980 } |
| 3965 } | 3981 } |
| 3966 | 3982 |
| 3967 // Do the store. | 3983 // Do the store. |
| 3968 HType type = instr->hydrogen()->value()->type(); | |
| 3969 SmiCheck check_needed = | 3984 SmiCheck check_needed = |
| 3970 type.IsHeapObject() ? OMIT_SMI_CHECK : INLINE_SMI_CHECK; | 3985 instr->hydrogen()->value()->IsHeapObject() |
| 3986 ? OMIT_SMI_CHECK : INLINE_SMI_CHECK; |
| 3971 | 3987 |
| 3972 Register write_register = object; | 3988 Register write_register = object; |
| 3973 if (!access.IsInobject()) { | 3989 if (!access.IsInobject()) { |
| 3974 write_register = ToRegister(instr->temp()); | 3990 write_register = ToRegister(instr->temp()); |
| 3975 __ movq(write_register, FieldOperand(object, JSObject::kPropertiesOffset)); | 3991 __ movq(write_register, FieldOperand(object, JSObject::kPropertiesOffset)); |
| 3976 } | 3992 } |
| 3977 | 3993 |
| 3978 if (instr->value()->IsConstantOperand()) { | 3994 if (instr->value()->IsConstantOperand()) { |
| 3979 LConstantOperand* operand_value = LConstantOperand::cast(instr->value()); | 3995 LConstantOperand* operand_value = LConstantOperand::cast(instr->value()); |
| 3980 if (operand_value->IsRegister()) { | 3996 if (operand_value->IsRegister()) { |
| (...skipping 218 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 4199 } else { | 4215 } else { |
| 4200 Handle<Object> handle_value = ToHandle(operand_value); | 4216 Handle<Object> handle_value = ToHandle(operand_value); |
| 4201 __ Move(operand, handle_value); | 4217 __ Move(operand, handle_value); |
| 4202 } | 4218 } |
| 4203 } | 4219 } |
| 4204 | 4220 |
| 4205 if (instr->hydrogen()->NeedsWriteBarrier()) { | 4221 if (instr->hydrogen()->NeedsWriteBarrier()) { |
| 4206 ASSERT(instr->value()->IsRegister()); | 4222 ASSERT(instr->value()->IsRegister()); |
| 4207 Register value = ToRegister(instr->value()); | 4223 Register value = ToRegister(instr->value()); |
| 4208 ASSERT(!instr->key()->IsConstantOperand()); | 4224 ASSERT(!instr->key()->IsConstantOperand()); |
| 4209 HType type = instr->hydrogen()->value()->type(); | |
| 4210 SmiCheck check_needed = | 4225 SmiCheck check_needed = |
| 4211 type.IsHeapObject() ? OMIT_SMI_CHECK : INLINE_SMI_CHECK; | 4226 instr->hydrogen()->value()->IsHeapObject() |
| 4227 ? OMIT_SMI_CHECK : INLINE_SMI_CHECK; |
| 4212 // Compute address of modified element and store it into key register. | 4228 // Compute address of modified element and store it into key register. |
| 4213 Register key_reg(ToRegister(key)); | 4229 Register key_reg(ToRegister(key)); |
| 4214 __ lea(key_reg, operand); | 4230 __ lea(key_reg, operand); |
| 4215 __ RecordWrite(elements, | 4231 __ RecordWrite(elements, |
| 4216 key_reg, | 4232 key_reg, |
| 4217 value, | 4233 value, |
| 4218 kSaveFPRegs, | 4234 kSaveFPRegs, |
| 4219 EMIT_REMEMBERED_SET, | 4235 EMIT_REMEMBERED_SET, |
| 4220 check_needed); | 4236 check_needed); |
| 4221 } | 4237 } |
| (...skipping 645 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 4867 | 4883 |
| 4868 | 4884 |
| 4869 void LCodeGen::DoCheckSmi(LCheckSmi* instr) { | 4885 void LCodeGen::DoCheckSmi(LCheckSmi* instr) { |
| 4870 LOperand* input = instr->value(); | 4886 LOperand* input = instr->value(); |
| 4871 Condition cc = masm()->CheckSmi(ToRegister(input)); | 4887 Condition cc = masm()->CheckSmi(ToRegister(input)); |
| 4872 DeoptimizeIf(NegateCondition(cc), instr->environment()); | 4888 DeoptimizeIf(NegateCondition(cc), instr->environment()); |
| 4873 } | 4889 } |
| 4874 | 4890 |
| 4875 | 4891 |
| 4876 void LCodeGen::DoCheckNonSmi(LCheckNonSmi* instr) { | 4892 void LCodeGen::DoCheckNonSmi(LCheckNonSmi* instr) { |
| 4877 LOperand* input = instr->value(); | 4893 if (!instr->hydrogen()->value()->IsHeapObject()) { |
| 4878 Condition cc = masm()->CheckSmi(ToRegister(input)); | 4894 LOperand* input = instr->value(); |
| 4879 DeoptimizeIf(cc, instr->environment()); | 4895 Condition cc = masm()->CheckSmi(ToRegister(input)); |
| 4896 DeoptimizeIf(cc, instr->environment()); |
| 4897 } |
| 4880 } | 4898 } |
| 4881 | 4899 |
| 4882 | 4900 |
| 4883 void LCodeGen::DoCheckInstanceType(LCheckInstanceType* instr) { | 4901 void LCodeGen::DoCheckInstanceType(LCheckInstanceType* instr) { |
| 4884 Register input = ToRegister(instr->value()); | 4902 Register input = ToRegister(instr->value()); |
| 4885 | 4903 |
| 4886 __ movq(kScratchRegister, FieldOperand(input, HeapObject::kMapOffset)); | 4904 __ movq(kScratchRegister, FieldOperand(input, HeapObject::kMapOffset)); |
| 4887 | 4905 |
| 4888 if (instr->hydrogen()->is_interval_check()) { | 4906 if (instr->hydrogen()->is_interval_check()) { |
| 4889 InstanceType first; | 4907 InstanceType first; |
| (...skipping 742 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 5632 FixedArray::kHeaderSize - kPointerSize)); | 5650 FixedArray::kHeaderSize - kPointerSize)); |
| 5633 __ bind(&done); | 5651 __ bind(&done); |
| 5634 } | 5652 } |
| 5635 | 5653 |
| 5636 | 5654 |
| 5637 #undef __ | 5655 #undef __ |
| 5638 | 5656 |
| 5639 } } // namespace v8::internal | 5657 } } // namespace v8::internal |
| 5640 | 5658 |
| 5641 #endif // V8_TARGET_ARCH_X64 | 5659 #endif // V8_TARGET_ARCH_X64 |
| OLD | NEW |