Commit 158dcbc3 authored by lrn@chromium.org's avatar lrn@chromium.org

X64: Extract all smi operations into MacroAssembler macros.

First step in changing Smi representation.

Review URL: http://codereview.chromium.org/196077


git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@2869 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
parent cf37189c
...@@ -852,7 +852,7 @@ class Assembler : public Malloced { ...@@ -852,7 +852,7 @@ class Assembler : public Malloced {
class EnsureSpace BASE_EMBEDDED { class EnsureSpace BASE_EMBEDDED {
public: public:
explicit EnsureSpace(Assembler* assembler) : assembler_(assembler) { explicit EnsureSpace(Assembler* assembler) : assembler_(assembler) {
if (assembler_->overflow()) assembler_->GrowBuffer(); if (assembler_->buffer_overflow()) assembler_->GrowBuffer();
#ifdef DEBUG #ifdef DEBUG
space_before_ = assembler_->available_space(); space_before_ = assembler_->available_space();
#endif #endif
......
...@@ -366,7 +366,7 @@ void Assembler::bind(Label* L) { ...@@ -366,7 +366,7 @@ void Assembler::bind(Label* L) {
void Assembler::GrowBuffer() { void Assembler::GrowBuffer() {
ASSERT(overflow()); // should not call this otherwise ASSERT(buffer_overflow()); // should not call this otherwise
if (!own_buffer_) FATAL("external code buffer is too small"); if (!own_buffer_) FATAL("external code buffer is too small");
// compute new buffer size // compute new buffer size
...@@ -428,7 +428,7 @@ void Assembler::GrowBuffer() { ...@@ -428,7 +428,7 @@ void Assembler::GrowBuffer() {
} }
} }
ASSERT(!overflow()); ASSERT(!buffer_overflow());
} }
...@@ -1410,6 +1410,15 @@ void Assembler::neg(Register dst) { ...@@ -1410,6 +1410,15 @@ void Assembler::neg(Register dst) {
} }
void Assembler::negl(Register dst) {
EnsureSpace ensure_space(this);
last_pc_ = pc_;
emit_optional_rex_32(dst);
emit(0xF7);
emit_modrm(0x3, dst);
}
void Assembler::neg(const Operand& dst) { void Assembler::neg(const Operand& dst) {
EnsureSpace ensure_space(this); EnsureSpace ensure_space(this);
last_pc_ = pc_; last_pc_ = pc_;
......
...@@ -721,6 +721,7 @@ class Assembler : public Malloced { ...@@ -721,6 +721,7 @@ class Assembler : public Malloced {
void neg(Register dst); void neg(Register dst);
void neg(const Operand& dst); void neg(const Operand& dst);
void negl(Register dst);
void not_(Register dst); void not_(Register dst);
void not_(const Operand& dst); void not_(const Operand& dst);
...@@ -729,6 +730,10 @@ class Assembler : public Malloced { ...@@ -729,6 +730,10 @@ class Assembler : public Malloced {
arithmetic_op(0x0B, dst, src); arithmetic_op(0x0B, dst, src);
} }
void orl(Register dst, Register src) {
arithmetic_op_32(0x0B, dst, src);
}
void or_(Register dst, const Operand& src) { void or_(Register dst, const Operand& src) {
arithmetic_op(0x0B, dst, src); arithmetic_op(0x0B, dst, src);
} }
...@@ -860,6 +865,10 @@ class Assembler : public Malloced { ...@@ -860,6 +865,10 @@ class Assembler : public Malloced {
arithmetic_op(0x33, dst, src); arithmetic_op(0x33, dst, src);
} }
void xorl(Register dst, Register src) {
arithmetic_op_32(0x33, dst, src);
}
void xor_(Register dst, const Operand& src) { void xor_(Register dst, const Operand& src) {
arithmetic_op(0x33, dst, src); arithmetic_op(0x33, dst, src);
} }
...@@ -1049,7 +1058,9 @@ class Assembler : public Malloced { ...@@ -1049,7 +1058,9 @@ class Assembler : public Malloced {
// Check if there is less than kGap bytes available in the buffer. // Check if there is less than kGap bytes available in the buffer.
// If this is the case, we need to grow the buffer before emitting // If this is the case, we need to grow the buffer before emitting
// an instruction or relocation information. // an instruction or relocation information.
inline bool overflow() const { return pc_ >= reloc_info_writer.pos() - kGap; } inline bool buffer_overflow() const {
return pc_ >= reloc_info_writer.pos() - kGap;
}
// Get the number of bytes available in the buffer. // Get the number of bytes available in the buffer.
inline int available_space() const { return reloc_info_writer.pos() - pc_; } inline int available_space() const { return reloc_info_writer.pos() - pc_; }
...@@ -1279,7 +1290,7 @@ class Assembler : public Malloced { ...@@ -1279,7 +1290,7 @@ class Assembler : public Malloced {
class EnsureSpace BASE_EMBEDDED { class EnsureSpace BASE_EMBEDDED {
public: public:
explicit EnsureSpace(Assembler* assembler) : assembler_(assembler) { explicit EnsureSpace(Assembler* assembler) : assembler_(assembler) {
if (assembler_->overflow()) assembler_->GrowBuffer(); if (assembler_->buffer_overflow()) assembler_->GrowBuffer();
#ifdef DEBUG #ifdef DEBUG
space_before_ = assembler_->available_space(); space_before_ = assembler_->available_space();
#endif #endif
......
...@@ -61,8 +61,7 @@ static void EnterArgumentsAdaptorFrame(MacroAssembler* masm) { ...@@ -61,8 +61,7 @@ static void EnterArgumentsAdaptorFrame(MacroAssembler* masm) {
// Preserve the number of arguments on the stack. Must preserve both // Preserve the number of arguments on the stack. Must preserve both
// rax and rbx because these registers are used when copying the // rax and rbx because these registers are used when copying the
// arguments and the receiver. // arguments and the receiver.
ASSERT(kSmiTagSize == 1); __ Integer32ToSmi(rcx, rax);
__ lea(rcx, Operand(rax, rax, times_1, kSmiTag));
__ push(rcx); __ push(rcx);
} }
...@@ -77,10 +76,13 @@ static void LeaveArgumentsAdaptorFrame(MacroAssembler* masm) { ...@@ -77,10 +76,13 @@ static void LeaveArgumentsAdaptorFrame(MacroAssembler* masm) {
// Remove caller arguments from the stack. // Remove caller arguments from the stack.
// rbx holds a Smi, so we convery to dword offset by multiplying by 4. // rbx holds a Smi, so we convery to dword offset by multiplying by 4.
// TODO(smi): Find a way to abstract indexing by a smi.
ASSERT_EQ(kSmiTagSize, 1 && kSmiTag == 0); ASSERT_EQ(kSmiTagSize, 1 && kSmiTag == 0);
ASSERT_EQ(kPointerSize, (1 << kSmiTagSize) * 4); ASSERT_EQ(kPointerSize, (1 << kSmiTagSize) * 4);
// TODO(smi): Find way to abstract indexing by a smi.
__ pop(rcx); __ pop(rcx);
__ lea(rsp, Operand(rsp, rbx, times_4, 1 * kPointerSize)); // 1 ~ receiver // 1 * kPointerSize is offset of receiver.
__ lea(rsp, Operand(rsp, rbx, times_half_pointer_size, 1 * kPointerSize));
__ push(rcx); __ push(rcx);
} }
...@@ -192,8 +194,7 @@ void Builtins::Generate_FunctionCall(MacroAssembler* masm) { ...@@ -192,8 +194,7 @@ void Builtins::Generate_FunctionCall(MacroAssembler* masm) {
{ Label done, non_function, function; { Label done, non_function, function;
// The function to call is at position n+1 on the stack. // The function to call is at position n+1 on the stack.
__ movq(rdi, Operand(rsp, rax, times_pointer_size, +1 * kPointerSize)); __ movq(rdi, Operand(rsp, rax, times_pointer_size, +1 * kPointerSize));
__ testl(rdi, Immediate(kSmiTagMask)); __ JumpIfSmi(rdi, &non_function);
__ j(zero, &non_function);
__ CmpObjectType(rdi, JS_FUNCTION_TYPE, rcx); __ CmpObjectType(rdi, JS_FUNCTION_TYPE, rcx);
__ j(equal, &function); __ j(equal, &function);
...@@ -213,8 +214,7 @@ void Builtins::Generate_FunctionCall(MacroAssembler* masm) { ...@@ -213,8 +214,7 @@ void Builtins::Generate_FunctionCall(MacroAssembler* masm) {
{ Label call_to_object, use_global_receiver, patch_receiver, done; { Label call_to_object, use_global_receiver, patch_receiver, done;
__ movq(rbx, Operand(rsp, rax, times_pointer_size, 0)); __ movq(rbx, Operand(rsp, rax, times_pointer_size, 0));
__ testl(rbx, Immediate(kSmiTagMask)); __ JumpIfSmi(rbx, &call_to_object);
__ j(zero, &call_to_object);
__ CompareRoot(rbx, Heap::kNullValueRootIndex); __ CompareRoot(rbx, Heap::kNullValueRootIndex);
__ j(equal, &use_global_receiver); __ j(equal, &use_global_receiver);
...@@ -230,8 +230,7 @@ void Builtins::Generate_FunctionCall(MacroAssembler* masm) { ...@@ -230,8 +230,7 @@ void Builtins::Generate_FunctionCall(MacroAssembler* masm) {
__ EnterInternalFrame(); // preserves rax, rbx, rdi __ EnterInternalFrame(); // preserves rax, rbx, rdi
// Store the arguments count on the stack (smi tagged). // Store the arguments count on the stack (smi tagged).
ASSERT(kSmiTag == 0); __ Integer32ToSmi(rax, rax);
__ shl(rax, Immediate(kSmiTagSize));
__ push(rax); __ push(rax);
__ push(rdi); // save edi across the call __ push(rdi); // save edi across the call
...@@ -242,7 +241,7 @@ void Builtins::Generate_FunctionCall(MacroAssembler* masm) { ...@@ -242,7 +241,7 @@ void Builtins::Generate_FunctionCall(MacroAssembler* masm) {
// Get the arguments count and untag it. // Get the arguments count and untag it.
__ pop(rax); __ pop(rax);
__ shr(rax, Immediate(kSmiTagSize)); __ SmiToInteger32(rax, rax);
__ LeaveInternalFrame(); __ LeaveInternalFrame();
__ jmp(&patch_receiver); __ jmp(&patch_receiver);
...@@ -355,8 +354,7 @@ void Builtins::Generate_FunctionApply(MacroAssembler* masm) { ...@@ -355,8 +354,7 @@ void Builtins::Generate_FunctionApply(MacroAssembler* masm) {
Label okay; Label okay;
// Make rdx the space we need for the array when it is unrolled onto the // Make rdx the space we need for the array when it is unrolled onto the
// stack. // stack.
__ movq(rdx, rax); __ PositiveSmiTimesPowerOfTwoToInteger64(rdx, rax, kPointerSizeLog2);
__ shl(rdx, Immediate(kPointerSizeLog2 - kSmiTagSize));
__ cmpq(rcx, rdx); __ cmpq(rcx, rdx);
__ j(greater, &okay); __ j(greater, &okay);
...@@ -382,8 +380,7 @@ void Builtins::Generate_FunctionApply(MacroAssembler* masm) { ...@@ -382,8 +380,7 @@ void Builtins::Generate_FunctionApply(MacroAssembler* masm) {
// Compute the receiver. // Compute the receiver.
Label call_to_object, use_global_receiver, push_receiver; Label call_to_object, use_global_receiver, push_receiver;
__ movq(rbx, Operand(rbp, kReceiverOffset)); __ movq(rbx, Operand(rbp, kReceiverOffset));
__ testl(rbx, Immediate(kSmiTagMask)); __ JumpIfSmi(rbx, &call_to_object);
__ j(zero, &call_to_object);
__ CompareRoot(rbx, Heap::kNullValueRootIndex); __ CompareRoot(rbx, Heap::kNullValueRootIndex);
__ j(equal, &use_global_receiver); __ j(equal, &use_global_receiver);
__ CompareRoot(rbx, Heap::kUndefinedValueRootIndex); __ CompareRoot(rbx, Heap::kUndefinedValueRootIndex);
...@@ -446,7 +443,7 @@ void Builtins::Generate_FunctionApply(MacroAssembler* masm) { ...@@ -446,7 +443,7 @@ void Builtins::Generate_FunctionApply(MacroAssembler* masm) {
// Invoke the function. // Invoke the function.
ParameterCount actual(rax); ParameterCount actual(rax);
__ shr(rax, Immediate(kSmiTagSize)); __ SmiToInteger32(rax, rax);
__ movq(rdi, Operand(rbp, kFunctionOffset)); __ movq(rdi, Operand(rbp, kFunctionOffset));
__ InvokeFunction(rdi, actual, CALL_FUNCTION); __ InvokeFunction(rdi, actual, CALL_FUNCTION);
...@@ -463,8 +460,7 @@ void Builtins::Generate_JSConstructCall(MacroAssembler* masm) { ...@@ -463,8 +460,7 @@ void Builtins::Generate_JSConstructCall(MacroAssembler* masm) {
Label non_function_call; Label non_function_call;
// Check that function is not a smi. // Check that function is not a smi.
__ testl(rdi, Immediate(kSmiTagMask)); __ JumpIfSmi(rdi, &non_function_call);
__ j(zero, &non_function_call);
// Check that function is a JSFunction. // Check that function is a JSFunction.
__ CmpObjectType(rdi, JS_FUNCTION_TYPE, rcx); __ CmpObjectType(rdi, JS_FUNCTION_TYPE, rcx);
__ j(not_equal, &non_function_call); __ j(not_equal, &non_function_call);
...@@ -492,7 +488,7 @@ void Builtins::Generate_JSConstructStubGeneric(MacroAssembler* masm) { ...@@ -492,7 +488,7 @@ void Builtins::Generate_JSConstructStubGeneric(MacroAssembler* masm) {
__ EnterConstructFrame(); __ EnterConstructFrame();
// Store a smi-tagged arguments count on the stack. // Store a smi-tagged arguments count on the stack.
__ shl(rax, Immediate(kSmiTagSize)); __ Integer32ToSmi(rax, rax);
__ push(rax); __ push(rax);
// Push the function to invoke on the stack. // Push the function to invoke on the stack.
...@@ -517,8 +513,7 @@ void Builtins::Generate_JSConstructStubGeneric(MacroAssembler* masm) { ...@@ -517,8 +513,7 @@ void Builtins::Generate_JSConstructStubGeneric(MacroAssembler* masm) {
// rdi: constructor // rdi: constructor
__ movq(rax, FieldOperand(rdi, JSFunction::kPrototypeOrInitialMapOffset)); __ movq(rax, FieldOperand(rdi, JSFunction::kPrototypeOrInitialMapOffset));
// Will both indicate a NULL and a Smi // Will both indicate a NULL and a Smi
__ testl(rax, Immediate(kSmiTagMask)); __ JumpIfSmi(rax, &rt_call);
__ j(zero, &rt_call);
// rdi: constructor // rdi: constructor
// rax: initial map (if proven valid below) // rax: initial map (if proven valid below)
__ CmpObjectType(rax, MAP_TYPE, rbx); __ CmpObjectType(rax, MAP_TYPE, rbx);
...@@ -668,7 +663,7 @@ void Builtins::Generate_JSConstructStubGeneric(MacroAssembler* masm) { ...@@ -668,7 +663,7 @@ void Builtins::Generate_JSConstructStubGeneric(MacroAssembler* masm) {
// Retrieve smi-tagged arguments count from the stack. // Retrieve smi-tagged arguments count from the stack.
__ movq(rax, Operand(rsp, 0)); __ movq(rax, Operand(rsp, 0));
__ shr(rax, Immediate(kSmiTagSize)); __ SmiToInteger32(rax, rax);
// Push the allocated receiver to the stack. We need two copies // Push the allocated receiver to the stack. We need two copies
// because we may have to return the original one and the calling // because we may have to return the original one and the calling
...@@ -701,8 +696,7 @@ void Builtins::Generate_JSConstructStubGeneric(MacroAssembler* masm) { ...@@ -701,8 +696,7 @@ void Builtins::Generate_JSConstructStubGeneric(MacroAssembler* masm) {
// on page 74. // on page 74.
Label use_receiver, exit; Label use_receiver, exit;
// If the result is a smi, it is *not* an object in the ECMA sense. // If the result is a smi, it is *not* an object in the ECMA sense.
__ testl(rax, Immediate(kSmiTagMask)); __ JumpIfSmi(rax, &use_receiver);
__ j(zero, &use_receiver);
// If the type of the result (stored in its map) is less than // If the type of the result (stored in its map) is less than
// FIRST_JS_OBJECT_TYPE, it is not an object in the ECMA sense. // FIRST_JS_OBJECT_TYPE, it is not an object in the ECMA sense.
...@@ -721,8 +715,10 @@ void Builtins::Generate_JSConstructStubGeneric(MacroAssembler* masm) { ...@@ -721,8 +715,10 @@ void Builtins::Generate_JSConstructStubGeneric(MacroAssembler* masm) {
// Remove caller arguments from the stack and return. // Remove caller arguments from the stack and return.
ASSERT(kSmiTagSize == 1 && kSmiTag == 0); ASSERT(kSmiTagSize == 1 && kSmiTag == 0);
// TODO(smi): Find a way to abstract indexing by a smi.
__ pop(rcx); __ pop(rcx);
__ lea(rsp, Operand(rsp, rbx, times_4, 1 * kPointerSize)); // 1 ~ receiver // 1 * kPointerSize is offset of receiver.
__ lea(rsp, Operand(rsp, rbx, times_half_pointer_size, 1 * kPointerSize));
__ push(rcx); __ push(rcx);
__ IncrementCounter(&Counters::constructed_objects, 1); __ IncrementCounter(&Counters::constructed_objects, 1);
__ ret(0); __ ret(0);
......
This diff is collapsed.
...@@ -95,7 +95,7 @@ static void GenerateDictionaryLoad(MacroAssembler* masm, Label* miss_label, ...@@ -95,7 +95,7 @@ static void GenerateDictionaryLoad(MacroAssembler* masm, Label* miss_label,
StringDictionary::kHeaderSize + StringDictionary::kHeaderSize +
StringDictionary::kCapacityIndex * kPointerSize; StringDictionary::kCapacityIndex * kPointerSize;
__ movq(r2, FieldOperand(r0, kCapacityOffset)); __ movq(r2, FieldOperand(r0, kCapacityOffset));
__ shrl(r2, Immediate(kSmiTagSize)); // convert smi to int __ SmiToInteger32(r2, r2);
__ decl(r2); __ decl(r2);
// Generate an unrolled loop that performs a few probes before // Generate an unrolled loop that performs a few probes before
...@@ -132,7 +132,7 @@ static void GenerateDictionaryLoad(MacroAssembler* masm, Label* miss_label, ...@@ -132,7 +132,7 @@ static void GenerateDictionaryLoad(MacroAssembler* masm, Label* miss_label,
__ bind(&done); __ bind(&done);
const int kDetailsOffset = kElementsStartOffset + 2 * kPointerSize; const int kDetailsOffset = kElementsStartOffset + 2 * kPointerSize;
__ testl(Operand(r0, r1, times_pointer_size, kDetailsOffset - kHeapObjectTag), __ testl(Operand(r0, r1, times_pointer_size, kDetailsOffset - kHeapObjectTag),
Immediate(PropertyDetails::TypeField::mask() << kSmiTagSize)); Immediate(Smi::FromInt(PropertyDetails::TypeField::mask())));
__ j(not_zero, miss_label); __ j(not_zero, miss_label);
// Get the value at the masked, scaled index. // Get the value at the masked, scaled index.
...@@ -148,8 +148,7 @@ static void GenerateCheckNonObjectOrLoaded(MacroAssembler* masm, Label* miss, ...@@ -148,8 +148,7 @@ static void GenerateCheckNonObjectOrLoaded(MacroAssembler* masm, Label* miss,
Register value) { Register value) {
Label done; Label done;
// Check if the value is a Smi. // Check if the value is a Smi.
__ testl(value, Immediate(kSmiTagMask)); __ JumpIfSmi(value, &done);
__ j(zero, &done);
// Check if the object has been loaded. // Check if the object has been loaded.
__ movq(kScratchRegister, FieldOperand(value, JSFunction::kMapOffset)); __ movq(kScratchRegister, FieldOperand(value, JSFunction::kMapOffset));
__ testb(FieldOperand(kScratchRegister, Map::kBitField2Offset), __ testb(FieldOperand(kScratchRegister, Map::kBitField2Offset),
...@@ -265,8 +264,7 @@ void KeyedLoadIC::GenerateGeneric(MacroAssembler* masm) { ...@@ -265,8 +264,7 @@ void KeyedLoadIC::GenerateGeneric(MacroAssembler* masm) {
__ movq(rcx, Operand(rsp, 2 * kPointerSize)); __ movq(rcx, Operand(rsp, 2 * kPointerSize));
// Check that the object isn't a smi. // Check that the object isn't a smi.
__ testl(rcx, Immediate(kSmiTagMask)); __ JumpIfSmi(rcx, &slow);
__ j(zero, &slow);
// Check that the object is some kind of JS object EXCEPT JS Value type. // Check that the object is some kind of JS object EXCEPT JS Value type.
// In the case that the object is a value-wrapper object, // In the case that the object is a value-wrapper object,
...@@ -283,9 +281,8 @@ void KeyedLoadIC::GenerateGeneric(MacroAssembler* masm) { ...@@ -283,9 +281,8 @@ void KeyedLoadIC::GenerateGeneric(MacroAssembler* masm) {
__ j(not_zero, &slow); __ j(not_zero, &slow);
// Check that the key is a smi. // Check that the key is a smi.
__ testl(rax, Immediate(kSmiTagMask)); __ JumpIfNotSmi(rax, &check_string);
__ j(not_zero, &check_string); __ SmiToInteger32(rax, rax);
__ sarl(rax, Immediate(kSmiTagSize));
// Get the elements array of the object. // Get the elements array of the object.
__ bind(&index_int); __ bind(&index_int);
__ movq(rcx, FieldOperand(rcx, JSObject::kElementsOffset)); __ movq(rcx, FieldOperand(rcx, JSObject::kElementsOffset));
...@@ -410,8 +407,7 @@ void KeyedStoreIC::GenerateGeneric(MacroAssembler* masm) { ...@@ -410,8 +407,7 @@ void KeyedStoreIC::GenerateGeneric(MacroAssembler* masm) {
// Get the receiver from the stack. // Get the receiver from the stack.
__ movq(rdx, Operand(rsp, 2 * kPointerSize)); // 2 ~ return address, key __ movq(rdx, Operand(rsp, 2 * kPointerSize)); // 2 ~ return address, key
// Check that the object isn't a smi. // Check that the object isn't a smi.
__ testl(rdx, Immediate(kSmiTagMask)); __ JumpIfSmi(rdx, &slow);
__ j(zero, &slow);
// Get the map from the receiver. // Get the map from the receiver.
__ movq(rcx, FieldOperand(rdx, HeapObject::kMapOffset)); __ movq(rcx, FieldOperand(rdx, HeapObject::kMapOffset));
// Check that the receiver does not require access checks. We need // Check that the receiver does not require access checks. We need
...@@ -422,8 +418,7 @@ void KeyedStoreIC::GenerateGeneric(MacroAssembler* masm) { ...@@ -422,8 +418,7 @@ void KeyedStoreIC::GenerateGeneric(MacroAssembler* masm) {
// Get the key from the stack. // Get the key from the stack.
__ movq(rbx, Operand(rsp, 1 * kPointerSize)); // 1 ~ return address __ movq(rbx, Operand(rsp, 1 * kPointerSize)); // 1 ~ return address
// Check that the key is a smi. // Check that the key is a smi.
__ testl(rbx, Immediate(kSmiTagMask)); __ JumpIfNotSmi(rbx, &slow);
__ j(not_zero, &slow);
// If it is a smi, make sure it is zero-extended, so it can be // If it is a smi, make sure it is zero-extended, so it can be
// used as an index in a memory operand. // used as an index in a memory operand.
__ movl(rbx, rbx); // Clear the high bits of rbx. __ movl(rbx, rbx); // Clear the high bits of rbx.
...@@ -443,8 +438,7 @@ void KeyedStoreIC::GenerateGeneric(MacroAssembler* masm) { ...@@ -443,8 +438,7 @@ void KeyedStoreIC::GenerateGeneric(MacroAssembler* masm) {
__ Cmp(FieldOperand(rcx, HeapObject::kMapOffset), Factory::fixed_array_map()); __ Cmp(FieldOperand(rcx, HeapObject::kMapOffset), Factory::fixed_array_map());
__ j(not_equal, &slow); __ j(not_equal, &slow);
// Untag the key (for checking against untagged length in the fixed array). // Untag the key (for checking against untagged length in the fixed array).
__ movl(rdx, rbx); __ SmiToInteger32(rdx, rbx);
__ sarl(rdx, Immediate(kSmiTagSize));
__ cmpl(rdx, FieldOperand(rcx, Array::kLengthOffset)); __ cmpl(rdx, FieldOperand(rcx, Array::kLengthOffset));
// rax: value // rax: value
// rcx: FixedArray // rcx: FixedArray
...@@ -473,13 +467,13 @@ void KeyedStoreIC::GenerateGeneric(MacroAssembler* masm) { ...@@ -473,13 +467,13 @@ void KeyedStoreIC::GenerateGeneric(MacroAssembler* masm) {
// rbx: index (as a smi) // rbx: index (as a smi)
// flags: compare (rbx, rdx.length()) // flags: compare (rbx, rdx.length())
__ j(not_equal, &slow); // do not leave holes in the array __ j(not_equal, &slow); // do not leave holes in the array
__ sarl(rbx, Immediate(kSmiTagSize)); // untag __ SmiToInteger64(rbx, rbx);
__ cmpl(rbx, FieldOperand(rcx, FixedArray::kLengthOffset)); __ cmpl(rbx, FieldOperand(rcx, FixedArray::kLengthOffset));
__ j(above_equal, &slow); __ j(above_equal, &slow);
// Restore tag and increment. // Increment and restore smi-tag.
__ lea(rbx, Operand(rbx, rbx, times_1, 1 << kSmiTagSize)); __ Integer64AddToSmi(rbx, rbx, 1);
__ movq(FieldOperand(rdx, JSArray::kLengthOffset), rbx); __ movq(FieldOperand(rdx, JSArray::kLengthOffset), rbx);
__ subl(rbx, Immediate(1 << kSmiTagSize)); // decrement rbx again __ SmiSubConstant(rbx, rbx, 1, NULL);
__ jmp(&fast); __ jmp(&fast);
...@@ -544,8 +538,7 @@ void CallIC::Generate(MacroAssembler* masm, ...@@ -544,8 +538,7 @@ void CallIC::Generate(MacroAssembler* masm,
// Check if the receiver is a global object of some sort. // Check if the receiver is a global object of some sort.
Label invoke, global; Label invoke, global;
__ movq(rdx, Operand(rsp, (argc + 1) * kPointerSize)); // receiver __ movq(rdx, Operand(rsp, (argc + 1) * kPointerSize)); // receiver
__ testl(rdx, Immediate(kSmiTagMask)); __ JumpIfSmi(rdx, &invoke);
__ j(zero, &invoke);
__ CmpObjectType(rdx, JS_GLOBAL_OBJECT_TYPE, rcx); __ CmpObjectType(rdx, JS_GLOBAL_OBJECT_TYPE, rcx);
__ j(equal, &global); __ j(equal, &global);
__ CmpInstanceType(rcx, JS_BUILTINS_OBJECT_TYPE); __ CmpInstanceType(rcx, JS_BUILTINS_OBJECT_TYPE);
...@@ -594,8 +587,7 @@ void CallIC::GenerateMegamorphic(MacroAssembler* masm, int argc) { ...@@ -594,8 +587,7 @@ void CallIC::GenerateMegamorphic(MacroAssembler* masm, int argc) {
// to probe. // to probe.
// //
// Check for number. // Check for number.
__ testl(rdx, Immediate(kSmiTagMask)); __ JumpIfSmi(rdx, &number);
__ j(zero, &number);
__ CmpObjectType(rdx, HEAP_NUMBER_TYPE, rbx); __ CmpObjectType(rdx, HEAP_NUMBER_TYPE, rbx);
__ j(not_equal, &non_number); __ j(not_equal, &non_number);
__ bind(&number); __ bind(&number);
...@@ -640,8 +632,7 @@ static void GenerateNormalHelper(MacroAssembler* masm, ...@@ -640,8 +632,7 @@ static void GenerateNormalHelper(MacroAssembler* masm,
// Move the result to register rdi and check that it isn't a smi. // Move the result to register rdi and check that it isn't a smi.
__ movq(rdi, rdx); __ movq(rdi, rdx);
__ testl(rdx, Immediate(kSmiTagMask)); __ JumpIfSmi(rdx, miss);
__ j(zero, miss);
// Check that the value is a JavaScript function. // Check that the value is a JavaScript function.
__ CmpObjectType(rdx, JS_FUNCTION_TYPE, rdx); __ CmpObjectType(rdx, JS_FUNCTION_TYPE, rdx);
...@@ -683,8 +674,7 @@ void CallIC::GenerateNormal(MacroAssembler* masm, int argc) { ...@@ -683,8 +674,7 @@ void CallIC::GenerateNormal(MacroAssembler* masm, int argc) {
__ movq(rcx, Operand(rsp, (argc + 2) * kPointerSize)); __ movq(rcx, Operand(rsp, (argc + 2) * kPointerSize));
// Check that the receiver isn't a smi. // Check that the receiver isn't a smi.
__ testl(rdx, Immediate(kSmiTagMask)); __ JumpIfSmi(rdx, &miss);
__ j(zero, &miss);
// Check that the receiver is a valid JS object. // Check that the receiver is a valid JS object.
// Because there are so many map checks and type checks, do not // Because there are so many map checks and type checks, do not
...@@ -844,8 +834,7 @@ void LoadIC::GenerateNormal(MacroAssembler* masm) { ...@@ -844,8 +834,7 @@ void LoadIC::GenerateNormal(MacroAssembler* masm) {
__ movq(rax, Operand(rsp, kPointerSize)); __ movq(rax, Operand(rsp, kPointerSize));
// Check that the receiver isn't a smi. // Check that the receiver isn't a smi.
__ testl(rax, Immediate(kSmiTagMask)); __ JumpIfSmi(rax, &miss);
__ j(zero, &miss);
// Check that the receiver is a valid JS object. // Check that the receiver is a valid JS object.
__ CmpObjectType(rax, FIRST_JS_OBJECT_TYPE, rbx); __ CmpObjectType(rax, FIRST_JS_OBJECT_TYPE, rbx);
......
This diff is collapsed.
...@@ -126,6 +126,215 @@ class MacroAssembler: public Assembler { ...@@ -126,6 +126,215 @@ class MacroAssembler: public Assembler {
// Store the code object for the given builtin in the target register. // Store the code object for the given builtin in the target register.
void GetBuiltinEntry(Register target, Builtins::JavaScript id); void GetBuiltinEntry(Register target, Builtins::JavaScript id);
// ---------------------------------------------------------------------------
// Smi tagging, untagging and operations on tagged smis.
// Conversions between tagged smi values and non-tagged integer values.
// Tag an integer value. The result must be known to be a valid smi value.
// Only uses the low 32 bits of the src register.
void Integer32ToSmi(Register dst, Register src);
// Tag an integer value if possible, or jump the integer value cannot be
// represented as a smi. Only uses the low 32 bit of the src registers.
void Integer32ToSmi(Register dst, Register src, Label* on_overflow);
// Adds constant to src and tags the result as a smi.
// Result must be a valid smi.
void Integer64AddToSmi(Register dst, Register src, int constant);
// Convert smi to 32-bit integer. I.e., not sign extended into
// high 32 bits of destination.
void SmiToInteger32(Register dst, Register src);
// Convert smi to 64-bit integer (sign extended if necessary).
void SmiToInteger64(Register dst, Register src);
// Multiply a positive smi's integer value by a power of two.
// Provides result as 64-bit integer value.
void PositiveSmiTimesPowerOfTwoToInteger64(Register dst,
Register src,
int power);
// Functions performing a check on a known or potential smi. Returns
// a condition that is satisfied if the check is successful.
// Is the value a tagged smi.
Condition CheckSmi(Register src);
// Is the value not a tagged smi.
Condition CheckNotSmi(Register src);
// Is the value a positive tagged smi.
Condition CheckPositiveSmi(Register src);
// Is the value not a positive tagged smi.
Condition CheckNotPositiveSmi(Register src);
// Are both values are tagged smis.
Condition CheckBothSmi(Register first, Register second);
// Is one of the values not a tagged smi.
Condition CheckNotBothSmi(Register first, Register second);
// Is the value the minimum smi value (since we are using
// two's complement numbers, negating the value is known to yield
// a non-smi value).
Condition CheckIsMinSmi(Register src);
// Check whether a tagged smi is equal to a constant.
Condition CheckSmiEqualsConstant(Register src, int constant);
// Checks whether an 32-bit integer value is a valid for conversion
// to a smi.
Condition CheckInteger32ValidSmiValue(Register src);
// Test-and-jump functions. Typically combines a check function
// above with a conditional jump.
// Jump if the value cannot be represented by a smi.
void JumpIfNotValidSmiValue(Register src, Label* on_invalid);
// Jump to label if the value is a tagged smi.
void JumpIfSmi(Register src, Label* on_smi);
// Jump to label if the value is not a tagged smi.
void JumpIfNotSmi(Register src, Label* on_not_smi);
// Jump to label if the value is not a positive tagged smi.
void JumpIfNotPositiveSmi(Register src, Label* on_not_smi);
// Jump to label if the value is a tagged smi with value equal
// to the constant.
void JumpIfSmiEqualsConstant(Register src, int constant, Label* on_equals);
// Jump if either or both register are not smi values.
void JumpIfNotBothSmi(Register src1, Register src2, Label* on_not_both_smi);
// Operations on tagged smi values.
// Smis represent a subset of integers. The subset is always equivalent to
// a two's complement interpretation of a fixed number of bits.
// Optimistically adds an integer constant to a supposed smi.
// If the src is not a smi, or the result is not a smi, jump to
// the label.
void SmiTryAddConstant(Register dst,
Register src,
int32_t constant,
Label* on_not_smi_result);
// Add an integer constant to a tagged smi, giving a tagged smi as result,
// or jumping to a label if the result cannot be represented by a smi.
// If the label is NULL, no testing on the result is done.
void SmiAddConstant(Register dst,
Register src,
int32_t constant,
Label* on_not_smi_result);
// Subtract an integer constant from a tagged smi, giving a tagged smi as
// result, or jumping to a label if the result cannot be represented by a smi.
// If the label is NULL, no testing on the result is done.
void SmiSubConstant(Register dst,
Register src,
int32_t constant,
Label* on_not_smi_result);
// Negating a smi can give a negative zero or too larget positive value.
void SmiNeg(Register dst,
Register src,
Label* on_not_smi_result);
// Adds smi values and return the result as a smi.
// If dst is src1, then src1 will be destroyed, even if
// the operation is unsuccessful.
void SmiAdd(Register dst,
Register src1,
Register src2,
Label* on_not_smi_result);
// Subtracts smi values and return the result as a smi.
// If dst is src1, then src1 will be destroyed, even if
// the operation is unsuccessful.
void SmiSub(Register dst,
Register src1,
Register src2,
Label* on_not_smi_result);
// Multiplies smi values and return the result as a smi,
// if possible.
// If dst is src1, then src1 will be destroyed, even if
// the operation is unsuccessful.
void SmiMul(Register dst,
Register src1,
Register src2,
Label* on_not_smi_result);
// Divides one smi by another and returns the quotient.
// Clobbers rax and rdx registers.
void SmiDiv(Register dst,
Register src1,
Register src2,
Label* on_not_smi_result);
// Divides one smi by another and returns the remainder.
// Clobbers rax and rdx registers.
void SmiMod(Register dst,
Register src1,
Register src2,
Label* on_not_smi_result);
// Bitwise operations.
void SmiNot(Register dst, Register src);
void SmiAnd(Register dst, Register src1, Register src2);
void SmiOr(Register dst, Register src1, Register src2);
void SmiXor(Register dst, Register src1, Register src2);
void SmiAndConstant(Register dst, Register src1, int constant);
void SmiOrConstant(Register dst, Register src1, int constant);
void SmiXorConstant(Register dst, Register src1, int constant);
void SmiShiftLeftConstant(Register dst,
Register src,
int shift_value,
Label* on_not_smi_result);
void SmiShiftLogicalRightConstant(Register dst,
Register src,
int shift_value,
Label* on_not_smi_result);
void SmiShiftArithmeticRightConstant(Register dst,
Register src,
int shift_value);
// Shifts a smi value to the left, and returns the result if that is a smi.
// Uses and clobbers rcx, so dst may not be rcx.
void SmiShiftLeft(Register dst,
Register src1,
Register src2,
Label* on_not_smi_result);
// Shifts a smi value to the right, shifting in zero bits at the top, and
// returns the unsigned intepretation of the result if that is a smi.
// Uses and clobbers rcx, so dst may not be rcx.
void SmiShiftLogicalRight(Register dst,
Register src1,
Register src2,
Label* on_not_smi_result);
// Shifts a smi value to the right, sign extending the top, and
// returns the signed intepretation of the result. That will always
// be a valid smi value, since it's numerically smaller than the
// original.
// Uses and clobbers rcx, so dst may not be rcx.
void SmiShiftArithmeticRight(Register dst,
Register src1,
Register src2);
// Specialized operations
// Select the non-smi register of two registers where exactly one is a
// smi. If neither are smis, jump to the failure label.
void SelectNonSmi(Register dst,
Register src1,
Register src2,
Label* on_not_smis);
// --------------------------------------------------------------------------- // ---------------------------------------------------------------------------
// Macro instructions // Macro instructions
......
...@@ -163,8 +163,7 @@ void StubCache::GenerateProbe(MacroAssembler* masm, ...@@ -163,8 +163,7 @@ void StubCache::GenerateProbe(MacroAssembler* masm,
ASSERT(!scratch.is(name)); ASSERT(!scratch.is(name));
// Check that the receiver isn't a smi. // Check that the receiver isn't a smi.
__ testl(receiver, Immediate(kSmiTagMask)); __ JumpIfSmi(receiver, &miss);
__ j(zero, &miss);
// Get the map of the receiver and compute the hash. // Get the map of the receiver and compute the hash.
__ movl(scratch, FieldOperand(name, String::kLengthOffset)); __ movl(scratch, FieldOperand(name, String::kLengthOffset));
...@@ -204,8 +203,7 @@ void StubCompiler::GenerateStoreField(MacroAssembler* masm, ...@@ -204,8 +203,7 @@ void StubCompiler::GenerateStoreField(MacroAssembler* masm,
Register scratch, Register scratch,
Label* miss_label) { Label* miss_label) {
// Check that the object isn't a smi. // Check that the object isn't a smi.
__ testl(receiver_reg, Immediate(kSmiTagMask)); __ JumpIfSmi(receiver_reg, miss_label);
__ j(zero, miss_label);
// Check that the map of the object hasn't changed. // Check that the map of the object hasn't changed.
__ Cmp(FieldOperand(receiver_reg, HeapObject::kMapOffset), __ Cmp(FieldOperand(receiver_reg, HeapObject::kMapOffset),
...@@ -275,8 +273,7 @@ void StubCompiler::GenerateLoadArrayLength(MacroAssembler* masm, ...@@ -275,8 +273,7 @@ void StubCompiler::GenerateLoadArrayLength(MacroAssembler* masm,
Register scratch, Register scratch,
Label* miss_label) { Label* miss_label) {
// Check that the receiver isn't a smi. // Check that the receiver isn't a smi.
__ testl(receiver, Immediate(kSmiTagMask)); __ JumpIfSmi(receiver, miss_label);
__ j(zero, miss_label);
// Check that the object is a JS array. // Check that the object is a JS array.
__ CmpObjectType(receiver, JS_ARRAY_TYPE, scratch); __ CmpObjectType(receiver, JS_ARRAY_TYPE, scratch);
...@@ -296,8 +293,7 @@ static void GenerateStringCheck(MacroAssembler* masm, ...@@ -296,8 +293,7 @@ static void GenerateStringCheck(MacroAssembler* masm,
Label* smi, Label* smi,
Label* non_string_object) { Label* non_string_object) {
// Check that the object isn't a smi. // Check that the object isn't a smi.
__ testl(receiver, Immediate(kSmiTagMask)); __ JumpIfSmi(receiver, smi);
__ j(zero, smi);
// Check that the object is a string. // Check that the object is a string.
__ movq(scratch, FieldOperand(receiver, HeapObject::kMapOffset)); __ movq(scratch, FieldOperand(receiver, HeapObject::kMapOffset));
...@@ -325,7 +321,7 @@ void StubCompiler::GenerateLoadStringLength(MacroAssembler* masm, ...@@ -325,7 +321,7 @@ void StubCompiler::GenerateLoadStringLength(MacroAssembler* masm,
// rcx is also the receiver. // rcx is also the receiver.
__ lea(rcx, Operand(scratch, String::kLongLengthShift)); __ lea(rcx, Operand(scratch, String::kLongLengthShift));
__ shr(rax); // rcx is implicit shift register. __ shr(rax); // rcx is implicit shift register.
__ shl(rax, Immediate(kSmiTagSize)); __ Integer32ToSmi(rax, rax);
__ ret(0); __ ret(0);
// Check if the object is a JSValue wrapper. // Check if the object is a JSValue wrapper.
...@@ -535,8 +531,7 @@ static void CompileLoadInterceptor(Compiler* compiler, ...@@ -535,8 +531,7 @@ static void CompileLoadInterceptor(Compiler* compiler,
ASSERT(!holder->GetNamedInterceptor()->getter()->IsUndefined()); ASSERT(!holder->GetNamedInterceptor()->getter()->IsUndefined());
// Check that the receiver isn't a smi. // Check that the receiver isn't a smi.
__ testl(receiver, Immediate(kSmiTagMask)); __ JumpIfSmi(receiver, miss);
__ j(zero, miss);
// Check that the maps haven't changed. // Check that the maps haven't changed.
Register reg = Register reg =
...@@ -701,8 +696,7 @@ Object* CallStubCompiler::CompileCallConstant(Object* object, ...@@ -701,8 +696,7 @@ Object* CallStubCompiler::CompileCallConstant(Object* object,
// Check that the receiver isn't a smi. // Check that the receiver isn't a smi.
if (check != NUMBER_CHECK) { if (check != NUMBER_CHECK) {
__ testl(rdx, Immediate(kSmiTagMask)); __ JumpIfSmi(rdx, &miss);
__ j(zero, &miss);
} }
// Make sure that it's okay not to patch the on stack receiver // Make sure that it's okay not to patch the on stack receiver
...@@ -738,8 +732,7 @@ Object* CallStubCompiler::CompileCallConstant(Object* object, ...@@ -738,8 +732,7 @@ Object* CallStubCompiler::CompileCallConstant(Object* object,
case NUMBER_CHECK: { case NUMBER_CHECK: {
Label fast; Label fast;
// Check that the object is a smi or a heap number. // Check that the object is a smi or a heap number.
__ testl(rdx, Immediate(kSmiTagMask)); __ JumpIfSmi(rdx, &fast);
__ j(zero, &fast);
__ CmpObjectType(rdx, HEAP_NUMBER_TYPE, rcx); __ CmpObjectType(rdx, HEAP_NUMBER_TYPE, rcx);
__ j(not_equal, &miss); __ j(not_equal, &miss);
__ bind(&fast); __ bind(&fast);
...@@ -830,8 +823,7 @@ Object* CallStubCompiler::CompileCallField(Object* object, ...@@ -830,8 +823,7 @@ Object* CallStubCompiler::CompileCallField(Object* object,
__ movq(rdx, Operand(rsp, (argc + 1) * kPointerSize)); __ movq(rdx, Operand(rsp, (argc + 1) * kPointerSize));
// Check that the receiver isn't a smi. // Check that the receiver isn't a smi.
__ testl(rdx, Immediate(kSmiTagMask)); __ JumpIfSmi(rdx, &miss);
__ j(zero, &miss);
// Do the right check and compute the holder register. // Do the right check and compute the holder register.
Register reg = Register reg =
...@@ -841,8 +833,7 @@ Object* CallStubCompiler::CompileCallField(Object* object, ...@@ -841,8 +833,7 @@ Object* CallStubCompiler::CompileCallField(Object* object,
GenerateFastPropertyLoad(masm(), rdi, reg, holder, index); GenerateFastPropertyLoad(masm(), rdi, reg, holder, index);
// Check that the function really is a function. // Check that the function really is a function.
__ testl(rdi, Immediate(kSmiTagMask)); __ JumpIfSmi(rdi, &miss);
__ j(zero, &miss);
__ CmpObjectType(rdi, JS_FUNCTION_TYPE, rbx); __ CmpObjectType(rdi, JS_FUNCTION_TYPE, rbx);
__ j(not_equal, &miss); __ j(not_equal, &miss);
...@@ -899,8 +890,7 @@ Object* CallStubCompiler::CompileCallInterceptor(Object* object, ...@@ -899,8 +890,7 @@ Object* CallStubCompiler::CompileCallInterceptor(Object* object,
__ movq(rdx, Operand(rsp, (argc + 1) * kPointerSize)); __ movq(rdx, Operand(rsp, (argc + 1) * kPointerSize));
// Check that the function really is a function. // Check that the function really is a function.
__ testl(rax, Immediate(kSmiTagMask)); __ JumpIfSmi(rax, &miss);
__ j(zero, &miss);
__ CmpObjectType(rax, JS_FUNCTION_TYPE, rbx); __ CmpObjectType(rax, JS_FUNCTION_TYPE, rbx);
__ j(not_equal, &miss); __ j(not_equal, &miss);
...@@ -952,8 +942,7 @@ Object* CallStubCompiler::CompileCallGlobal(JSObject* object, ...@@ -952,8 +942,7 @@ Object* CallStubCompiler::CompileCallGlobal(JSObject* object,
// object which can only happen for contextual calls. In this case, // object which can only happen for contextual calls. In this case,
// the receiver cannot be a smi. // the receiver cannot be a smi.
if (object != holder) { if (object != holder) {
__ testl(rdx, Immediate(kSmiTagMask)); __ JumpIfSmi(rdx, &miss);
__ j(zero, &miss);
} }
// Check that the maps haven't changed. // Check that the maps haven't changed.
...@@ -1112,8 +1101,7 @@ Object* LoadStubCompiler::CompileLoadGlobal(JSObject* object, ...@@ -1112,8 +1101,7 @@ Object* LoadStubCompiler::CompileLoadGlobal(JSObject* object,
// object which can only happen for contextual loads. In this case, // object which can only happen for contextual loads. In this case,
// the receiver cannot be a smi. // the receiver cannot be a smi.
if (object != holder) { if (object != holder) {
__ testl(rax, Immediate(kSmiTagMask)); __ JumpIfSmi(rax, &miss);
__ j(zero, &miss);
} }
// Check that the maps haven't changed. // Check that the maps haven't changed.
...@@ -1335,8 +1323,7 @@ Object* StoreStubCompiler::CompileStoreCallback(JSObject* object, ...@@ -1335,8 +1323,7 @@ Object* StoreStubCompiler::CompileStoreCallback(JSObject* object,
__ movq(rbx, Operand(rsp, 1 * kPointerSize)); __ movq(rbx, Operand(rsp, 1 * kPointerSize));
// Check that the object isn't a smi. // Check that the object isn't a smi.
__ testl(rbx, Immediate(kSmiTagMask)); __ JumpIfSmi(rbx, &miss);
__ j(zero, &miss);
// Check that the map of the object hasn't changed. // Check that the map of the object hasn't changed.
__ Cmp(FieldOperand(rbx, HeapObject::kMapOffset), __ Cmp(FieldOperand(rbx, HeapObject::kMapOffset),
...@@ -1424,8 +1411,7 @@ Object* StoreStubCompiler::CompileStoreInterceptor(JSObject* receiver, ...@@ -1424,8 +1411,7 @@ Object* StoreStubCompiler::CompileStoreInterceptor(JSObject* receiver,
__ movq(rbx, Operand(rsp, 1 * kPointerSize)); __ movq(rbx, Operand(rsp, 1 * kPointerSize));
// Check that the object isn't a smi. // Check that the object isn't a smi.
__ testl(rbx, Immediate(kSmiTagMask)); __ JumpIfSmi(rbx, &miss);
__ j(zero, &miss);
// Check that the map of the object hasn't changed. // Check that the map of the object hasn't changed.
__ Cmp(FieldOperand(rbx, HeapObject::kMapOffset), __ Cmp(FieldOperand(rbx, HeapObject::kMapOffset),
...@@ -1631,8 +1617,7 @@ void StubCompiler::GenerateLoadCallback(JSObject* object, ...@@ -1631,8 +1617,7 @@ void StubCompiler::GenerateLoadCallback(JSObject* object,
String* name, String* name,
Label* miss) { Label* miss) {
// Check that the receiver isn't a smi. // Check that the receiver isn't a smi.
__ testl(receiver, Immediate(kSmiTagMask)); __ JumpIfSmi(receiver, miss);
__ j(zero, miss);
// Check that the maps haven't changed. // Check that the maps haven't changed.
Register reg = Register reg =
...@@ -1701,8 +1686,7 @@ void StubCompiler::GenerateLoadField(JSObject* object, ...@@ -1701,8 +1686,7 @@ void StubCompiler::GenerateLoadField(JSObject* object,
String* name, String* name,
Label* miss) { Label* miss) {
// Check that the receiver isn't a smi. // Check that the receiver isn't a smi.
__ testl(receiver, Immediate(kSmiTagMask)); __ JumpIfSmi(receiver, miss);
__ j(zero, miss);
// Check the prototype chain. // Check the prototype chain.
Register reg = Register reg =
...@@ -1724,8 +1708,7 @@ void StubCompiler::GenerateLoadConstant(JSObject* object, ...@@ -1724,8 +1708,7 @@ void StubCompiler::GenerateLoadConstant(JSObject* object,
String* name, String* name,
Label* miss) { Label* miss) {
// Check that the receiver isn't a smi. // Check that the receiver isn't a smi.
__ testl(receiver, Immediate(kSmiTagMask)); __ JumpIfSmi(receiver, miss);
__ j(zero, miss);
// Check that the maps haven't changed. // Check that the maps haven't changed.
Register reg = Register reg =
...@@ -1766,8 +1749,7 @@ Object* ConstructStubCompiler::CompileConstructStub( ...@@ -1766,8 +1749,7 @@ Object* ConstructStubCompiler::CompileConstructStub(
// Load the initial map and verify that it is in fact a map. // Load the initial map and verify that it is in fact a map.
__ movq(rbx, FieldOperand(rdi, JSFunction::kPrototypeOrInitialMapOffset)); __ movq(rbx, FieldOperand(rdi, JSFunction::kPrototypeOrInitialMapOffset));
// Will both indicate a NULL and a Smi. // Will both indicate a NULL and a Smi.
__ testq(rbx, Immediate(kSmiTagMask)); __ JumpIfSmi(rbx, &generic_stub_call);
__ j(zero, &generic_stub_call);
__ CmpObjectType(rbx, MAP_TYPE, rcx); __ CmpObjectType(rbx, MAP_TYPE, rcx);
__ j(not_equal, &generic_stub_call); __ j(not_equal, &generic_stub_call);
......
...@@ -65,8 +65,8 @@ void VirtualFrame::Enter() { ...@@ -65,8 +65,8 @@ void VirtualFrame::Enter() {
#ifdef DEBUG #ifdef DEBUG
// Verify that rdi contains a JS function. The following code // Verify that rdi contains a JS function. The following code
// relies on rax being available for use. // relies on rax being available for use.
__ testl(rdi, Immediate(kSmiTagMask)); Condition not_smi = masm()->CheckNotSmi(rdi);
__ Check(not_zero, __ Check(not_smi,
"VirtualFrame::Enter - rdi is not a function (smi check)."); "VirtualFrame::Enter - rdi is not a function (smi check).");
__ CmpObjectType(rdi, JS_FUNCTION_TYPE, rax); __ CmpObjectType(rdi, JS_FUNCTION_TYPE, rax);
__ Check(equal, __ Check(equal,
......
...@@ -47,40 +47,40 @@ assertEquals(one / (minus_one * minus_one), 1, "one / 1"); ...@@ -47,40 +47,40 @@ assertEquals(one / (minus_one * minus_one), 1, "one / 1");
assertEquals(one / (zero / minus_one), -Infinity, "one / -0 III"); assertEquals(one / (zero / minus_one), -Infinity, "one / -0 III");
assertEquals(one / (zero / one), Infinity, "one / 0 II"); assertEquals(one / (zero / one), Infinity, "one / 0 II");
assertEquals(one / (minus_four % two), -Infinity, "foo"); assertEquals(one / (minus_four % two), -Infinity, "foo1");
assertEquals(one / (minus_four % minus_two), -Infinity, "foo"); assertEquals(one / (minus_four % minus_two), -Infinity, "foo2");
assertEquals(one / (four % two), Infinity, "foo"); assertEquals(one / (four % two), Infinity, "foo3");
assertEquals(one / (four % minus_two), Infinity, "foo"); assertEquals(one / (four % minus_two), Infinity, "foo4");
// literal op variable // literal op variable
assertEquals(one / (0 * minus_one), -Infinity, "bar"); assertEquals(one / (0 * minus_one), -Infinity, "bar1");
assertEquals(one / (-1 * zero), -Infinity, "bar"); assertEquals(one / (-1 * zero), -Infinity, "bar2");
assertEquals(one / (0 * zero), Infinity, "bar"); assertEquals(one / (0 * zero), Infinity, "bar3");
assertEquals(one / (-1 * minus_one), 1, "bar"); assertEquals(one / (-1 * minus_one), 1, "bar4");
assertEquals(one / (0 / minus_one), -Infinity, "baz"); assertEquals(one / (0 / minus_one), -Infinity, "baz1");
assertEquals(one / (0 / one), Infinity, "baz"); assertEquals(one / (0 / one), Infinity, "baz2");
assertEquals(one / (-4 % two), -Infinity, "baz"); assertEquals(one / (-4 % two), -Infinity, "baz3");
assertEquals(one / (-4 % minus_two), -Infinity, "baz"); assertEquals(one / (-4 % minus_two), -Infinity, "baz4");
assertEquals(one / (4 % two), Infinity, "baz"); assertEquals(one / (4 % two), Infinity, "baz5");
assertEquals(one / (4 % minus_two), Infinity, "baz"); assertEquals(one / (4 % minus_two), Infinity, "baz6");
// variable op literal // variable op literal
assertEquals(one / (zero * -1), -Infinity, "fizz"); assertEquals(one / (zero * -1), -Infinity, "fizz1");
assertEquals(one / (minus_one * 0), -Infinity, "fizz"); assertEquals(one / (minus_one * 0), -Infinity, "fizz2");
assertEquals(one / (zero * 0), Infinity, "fizz"); assertEquals(one / (zero * 0), Infinity, "fizz3");
assertEquals(one / (minus_one * -1), 1, "fizz"); assertEquals(one / (minus_one * -1), 1, "fizz4");
assertEquals(one / (zero / -1), -Infinity, "buzz"); assertEquals(one / (zero / -1), -Infinity, "buzz1");
assertEquals(one / (zero / 1), Infinity, "buzz"); assertEquals(one / (zero / 1), Infinity, "buzz2");
assertEquals(one / (minus_four % 2), -Infinity, "buzz"); assertEquals(one / (minus_four % 2), -Infinity, "buzz3");
assertEquals(one / (minus_four % -2), -Infinity, "buzz"); assertEquals(one / (minus_four % -2), -Infinity, "buzz4");
assertEquals(one / (four % 2), Infinity, "buzz"); assertEquals(one / (four % 2), Infinity, "buzz5");
assertEquals(one / (four % -2), Infinity, "buzz"); assertEquals(one / (four % -2), Infinity, "buzz6");
// literal op literal // literal op literal
...@@ -91,10 +91,10 @@ assertEquals(one / (-1 * 0), -Infinity, "fisk3"); ...@@ -91,10 +91,10 @@ assertEquals(one / (-1 * 0), -Infinity, "fisk3");
assertEquals(one / (0 * 0), Infinity, "fisk4"); assertEquals(one / (0 * 0), Infinity, "fisk4");
assertEquals(one / (-1 * -1), 1, "fisk5"); assertEquals(one / (-1 * -1), 1, "fisk5");
assertEquals(one / (0 / -1), -Infinity, "hest"); assertEquals(one / (0 / -1), -Infinity, "hest1");
assertEquals(one / (0 / 1), Infinity, "hest"); assertEquals(one / (0 / 1), Infinity, "hest2");
assertEquals(one / (-4 % 2), -Infinity, "fiskhest"); assertEquals(one / (-4 % 2), -Infinity, "fiskhest1");
assertEquals(one / (-4 % -2), -Infinity, "fiskhest"); assertEquals(one / (-4 % -2), -Infinity, "fiskhest2");
assertEquals(one / (4 % 2), Infinity, "fiskhest"); assertEquals(one / (4 % 2), Infinity, "fiskhest3");
assertEquals(one / (4 % -2), Infinity, "fiskhest"); assertEquals(one / (4 % -2), Infinity, "fiskhest4");
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment