| OLD | NEW |
| 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 7565 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 7576 // Unary plus has no effect on int32 values. | 7576 // Unary plus has no effect on int32 values. |
| 7577 break; | 7577 break; |
| 7578 } | 7578 } |
| 7579 default: | 7579 default: |
| 7580 UNREACHABLE(); | 7580 UNREACHABLE(); |
| 7581 break; | 7581 break; |
| 7582 } | 7582 } |
| 7583 frame_->Push(&value); | 7583 frame_->Push(&value); |
| 7584 } else { | 7584 } else { |
| 7585 Load(node->expression()); | 7585 Load(node->expression()); |
| 7586 bool overwrite = | 7586 bool can_overwrite = |
| 7587 (node->expression()->AsBinaryOperation() != NULL && | 7587 (node->expression()->AsBinaryOperation() != NULL && |
| 7588 node->expression()->AsBinaryOperation()->ResultOverwriteAllowed()); | 7588 node->expression()->AsBinaryOperation()->ResultOverwriteAllowed()); |
| 7589 UnaryOverwriteMode overwrite = |
| 7590 can_overwrite ? UNARY_OVERWRITE : UNARY_NO_OVERWRITE; |
| 7591 bool no_negative_zero = node->expression()->no_negative_zero(); |
| 7589 switch (op) { | 7592 switch (op) { |
| 7590 case Token::NOT: | 7593 case Token::NOT: |
| 7591 case Token::DELETE: | 7594 case Token::DELETE: |
| 7592 case Token::TYPEOF: | 7595 case Token::TYPEOF: |
| 7593 UNREACHABLE(); // handled above | 7596 UNREACHABLE(); // handled above |
| 7594 break; | 7597 break; |
| 7595 | 7598 |
| 7596 case Token::SUB: { | 7599 case Token::SUB: { |
| 7597 GenericUnaryOpStub stub(Token::SUB, overwrite); | 7600 GenericUnaryOpStub stub( |
| 7601 Token::SUB, |
| 7602 overwrite, |
| 7603 no_negative_zero ? kIgnoreNegativeZero : kStrictNegativeZero); |
| 7598 Result operand = frame_->Pop(); | 7604 Result operand = frame_->Pop(); |
| 7599 Result answer = frame_->CallStub(&stub, &operand); | 7605 Result answer = frame_->CallStub(&stub, &operand); |
| 7600 answer.set_type_info(TypeInfo::Number()); | 7606 answer.set_type_info(TypeInfo::Number()); |
| 7601 frame_->Push(&answer); | 7607 frame_->Push(&answer); |
| 7602 break; | 7608 break; |
| 7603 } | 7609 } |
| 7604 case Token::BIT_NOT: { | 7610 case Token::BIT_NOT: { |
| 7605 // Smi check. | 7611 // Smi check. |
| 7606 JumpTarget smi_label; | 7612 JumpTarget smi_label; |
| 7607 JumpTarget continue_label; | 7613 JumpTarget continue_label; |
| (...skipping 3319 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 10927 | 10933 |
| 10928 void GenericUnaryOpStub::Generate(MacroAssembler* masm) { | 10934 void GenericUnaryOpStub::Generate(MacroAssembler* masm) { |
| 10929 Label slow, done; | 10935 Label slow, done; |
| 10930 | 10936 |
| 10931 if (op_ == Token::SUB) { | 10937 if (op_ == Token::SUB) { |
| 10932 // Check whether the value is a smi. | 10938 // Check whether the value is a smi. |
| 10933 Label try_float; | 10939 Label try_float; |
| 10934 __ test(eax, Immediate(kSmiTagMask)); | 10940 __ test(eax, Immediate(kSmiTagMask)); |
| 10935 __ j(not_zero, &try_float, not_taken); | 10941 __ j(not_zero, &try_float, not_taken); |
| 10936 | 10942 |
| 10937 // Go slow case if the value of the expression is zero | 10943 if (negative_zero_ == kStrictNegativeZero) { |
| 10938 // to make sure that we switch between 0 and -0. | 10944 // Go slow case if the value of the expression is zero |
| 10939 __ test(eax, Operand(eax)); | 10945 // to make sure that we switch between 0 and -0. |
| 10940 __ j(zero, &slow, not_taken); | 10946 __ test(eax, Operand(eax)); |
| 10947 __ j(zero, &slow, not_taken); |
| 10948 } |
| 10941 | 10949 |
| 10942 // The value of the expression is a smi that is not zero. Try | 10950 // The value of the expression is a smi that is not zero. Try |
| 10943 // optimistic subtraction '0 - value'. | 10951 // optimistic subtraction '0 - value'. |
| 10944 Label undo; | 10952 Label undo; |
| 10945 __ mov(edx, Operand(eax)); | 10953 __ mov(edx, Operand(eax)); |
| 10946 __ Set(eax, Immediate(0)); | 10954 __ Set(eax, Immediate(0)); |
| 10947 __ sub(eax, Operand(edx)); | 10955 __ sub(eax, Operand(edx)); |
| 10948 __ j(overflow, &undo, not_taken); | 10956 __ j(no_overflow, &done, taken); |
| 10949 | |
| 10950 // If result is a smi we are done. | |
| 10951 __ test(eax, Immediate(kSmiTagMask)); | |
| 10952 __ j(zero, &done, taken); | |
| 10953 | 10957 |
| 10954 // Restore eax and go slow case. | 10958 // Restore eax and go slow case. |
| 10955 __ bind(&undo); | 10959 __ bind(&undo); |
| 10956 __ mov(eax, Operand(edx)); | 10960 __ mov(eax, Operand(edx)); |
| 10957 __ jmp(&slow); | 10961 __ jmp(&slow); |
| 10958 | 10962 |
| 10959 // Try floating point case. | 10963 // Try floating point case. |
| 10960 __ bind(&try_float); | 10964 __ bind(&try_float); |
| 10961 __ mov(edx, FieldOperand(eax, HeapObject::kMapOffset)); | 10965 __ mov(edx, FieldOperand(eax, HeapObject::kMapOffset)); |
| 10962 __ cmp(edx, Factory::heap_number_map()); | 10966 __ cmp(edx, Factory::heap_number_map()); |
| 10963 __ j(not_equal, &slow); | 10967 __ j(not_equal, &slow); |
| 10964 if (overwrite_) { | 10968 if (overwrite_ == UNARY_OVERWRITE) { |
| 10965 __ mov(edx, FieldOperand(eax, HeapNumber::kExponentOffset)); | 10969 __ mov(edx, FieldOperand(eax, HeapNumber::kExponentOffset)); |
| 10966 __ xor_(edx, HeapNumber::kSignMask); // Flip sign. | 10970 __ xor_(edx, HeapNumber::kSignMask); // Flip sign. |
| 10967 __ mov(FieldOperand(eax, HeapNumber::kExponentOffset), edx); | 10971 __ mov(FieldOperand(eax, HeapNumber::kExponentOffset), edx); |
| 10968 } else { | 10972 } else { |
| 10969 __ mov(edx, Operand(eax)); | 10973 __ mov(edx, Operand(eax)); |
| 10970 // edx: operand | 10974 // edx: operand |
| 10971 __ AllocateHeapNumber(eax, ebx, ecx, &undo); | 10975 __ AllocateHeapNumber(eax, ebx, ecx, &undo); |
| 10972 // eax: allocated 'empty' number | 10976 // eax: allocated 'empty' number |
| 10973 __ mov(ecx, FieldOperand(edx, HeapNumber::kExponentOffset)); | 10977 __ mov(ecx, FieldOperand(edx, HeapNumber::kExponentOffset)); |
| 10974 __ xor_(ecx, HeapNumber::kSignMask); // Flip sign. | 10978 __ xor_(ecx, HeapNumber::kSignMask); // Flip sign. |
| (...skipping 20 matching lines...) Expand all Loading... |
| 10995 __ cmp(ecx, 0xc0000000); | 10999 __ cmp(ecx, 0xc0000000); |
| 10996 __ j(sign, &try_float, not_taken); | 11000 __ j(sign, &try_float, not_taken); |
| 10997 | 11001 |
| 10998 // Tag the result as a smi and we're done. | 11002 // Tag the result as a smi and we're done. |
| 10999 ASSERT(kSmiTagSize == 1); | 11003 ASSERT(kSmiTagSize == 1); |
| 11000 __ lea(eax, Operand(ecx, times_2, kSmiTag)); | 11004 __ lea(eax, Operand(ecx, times_2, kSmiTag)); |
| 11001 __ jmp(&done); | 11005 __ jmp(&done); |
| 11002 | 11006 |
| 11003 // Try to store the result in a heap number. | 11007 // Try to store the result in a heap number. |
| 11004 __ bind(&try_float); | 11008 __ bind(&try_float); |
| 11005 if (!overwrite_) { | 11009 if (overwrite_ == UNARY_NO_OVERWRITE) { |
| 11006 // Allocate a fresh heap number, but don't overwrite eax until | 11010 // Allocate a fresh heap number, but don't overwrite eax until |
| 11007 // we're sure we can do it without going through the slow case | 11011 // we're sure we can do it without going through the slow case |
| 11008 // that needs the value in eax. | 11012 // that needs the value in eax. |
| 11009 __ AllocateHeapNumber(ebx, edx, edi, &slow); | 11013 __ AllocateHeapNumber(ebx, edx, edi, &slow); |
| 11010 __ mov(eax, Operand(ebx)); | 11014 __ mov(eax, Operand(ebx)); |
| 11011 } | 11015 } |
| 11012 if (CpuFeatures::IsSupported(SSE2)) { | 11016 if (CpuFeatures::IsSupported(SSE2)) { |
| 11013 CpuFeatures::Scope use_sse2(SSE2); | 11017 CpuFeatures::Scope use_sse2(SSE2); |
| 11014 __ cvtsi2sd(xmm0, Operand(ecx)); | 11018 __ cvtsi2sd(xmm0, Operand(ecx)); |
| 11015 __ movdbl(FieldOperand(eax, HeapNumber::kValueOffset), xmm0); | 11019 __ movdbl(FieldOperand(eax, HeapNumber::kValueOffset), xmm0); |
| (...skipping 2754 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 13770 masm.GetCode(&desc); | 13774 masm.GetCode(&desc); |
| 13771 // Call the function from C++. | 13775 // Call the function from C++. |
| 13772 return FUNCTION_CAST<MemCopyFunction>(buffer); | 13776 return FUNCTION_CAST<MemCopyFunction>(buffer); |
| 13773 } | 13777 } |
| 13774 | 13778 |
| 13775 #undef __ | 13779 #undef __ |
| 13776 | 13780 |
| 13777 } } // namespace v8::internal | 13781 } } // namespace v8::internal |
| 13778 | 13782 |
| 13779 #endif // V8_TARGET_ARCH_IA32 | 13783 #endif // V8_TARGET_ARCH_IA32 |
| OLD | NEW |