Support register arguments in more cases.

1. MUL and DIV on SMIs.
2. When calling GenericBinaryOpStub from a virtual frame.
3. When generating code for a loop counter.
Overall performance gain is about 0.6%.
Review URL: http://codereview.chromium.org/555098

git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@3708 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
This commit is contained in:
kaznacheev@chromium.org 2010-01-26 10:27:27 +00:00
parent 2f0ad64db9
commit c33dfa5a1d
3 changed files with 54 additions and 75 deletions

View File

@ -763,10 +763,8 @@ class FloatingPointHelper : public AllStatic {
ArgLocation arg_location = ARGS_ON_STACK); ArgLocation arg_location = ARGS_ON_STACK);
// Similar to LoadFloatOperand but assumes that both operands are smis. // Similar to LoadFloatOperand but assumes that both operands are smis.
// Accepts operands on stack or in eax, ebx. // Accepts operands in eax, ebx.
static void LoadFloatSmis(MacroAssembler* masm, static void LoadFloatSmis(MacroAssembler* masm, Register scratch);
Register scratch,
ArgLocation arg_location);
// Test if operands are smi or number objects (fp). Requirements: // Test if operands are smi or number objects (fp). Requirements:
// operand_1 in eax, operand_2 in edx; falls through on float // operand_1 in eax, operand_2 in edx; falls through on float
@ -786,10 +784,8 @@ class FloatingPointHelper : public AllStatic {
static void LoadSse2Operands(MacroAssembler* masm, Label* not_numbers); static void LoadSse2Operands(MacroAssembler* masm, Label* not_numbers);
// Similar to LoadSse2Operands but assumes that both operands are smis. // Similar to LoadSse2Operands but assumes that both operands are smis.
// Accepts operands on stack or in eax, ebx. // Accepts operands in eax, ebx.
static void LoadSse2Smis(MacroAssembler* masm, static void LoadSse2Smis(MacroAssembler* masm, Register scratch);
Register scratch,
ArgLocation arg_location);
}; };
@ -978,10 +974,8 @@ void CodeGenerator::GenericBinaryOperation(Token::Value op,
Result answer; Result answer;
if (left_is_non_smi || right_is_non_smi) { if (left_is_non_smi || right_is_non_smi) {
// Go straight to the slow case, with no smi code. // Go straight to the slow case, with no smi code.
frame_->Push(&left);
frame_->Push(&right);
GenericBinaryOpStub stub(op, overwrite_mode, NO_SMI_CODE_IN_STUB); GenericBinaryOpStub stub(op, overwrite_mode, NO_SMI_CODE_IN_STUB);
answer = frame_->CallStub(&stub, 2); answer = stub.GenerateCall(masm_, frame_, &left, &right);
} else if (right_is_smi) { } else if (right_is_smi) {
answer = ConstantSmiBinaryOperation(op, &left, right.handle(), answer = ConstantSmiBinaryOperation(op, &left, right.handle(),
type, false, overwrite_mode); type, false, overwrite_mode);
@ -997,10 +991,8 @@ void CodeGenerator::GenericBinaryOperation(Token::Value op,
if (loop_nesting() > 0 && (Token::IsBitOp(op) || type->IsLikelySmi())) { if (loop_nesting() > 0 && (Token::IsBitOp(op) || type->IsLikelySmi())) {
answer = LikelySmiBinaryOperation(op, &left, &right, overwrite_mode); answer = LikelySmiBinaryOperation(op, &left, &right, overwrite_mode);
} else { } else {
frame_->Push(&left);
frame_->Push(&right);
GenericBinaryOpStub stub(op, overwrite_mode, NO_GENERIC_BINARY_FLAGS); GenericBinaryOpStub stub(op, overwrite_mode, NO_GENERIC_BINARY_FLAGS);
answer = frame_->CallStub(&stub, 2); answer = stub.GenerateCall(masm_, frame_, &left, &right);
} }
} }
frame_->Push(&answer); frame_->Push(&answer);
@ -7076,6 +7068,21 @@ void GenericBinaryOpStub::GenerateCall(
} }
Result GenericBinaryOpStub::GenerateCall(MacroAssembler* masm,
VirtualFrame* frame,
Result* left,
Result* right) {
if (ArgsInRegistersSupported()) {
SetArgsInRegisters();
return frame->CallStub(this, left, right);
} else {
frame->Push(left);
frame->Push(right);
return frame->CallStub(this, 2);
}
}
void GenericBinaryOpStub::GenerateSmiCode(MacroAssembler* masm, Label* slow) { void GenericBinaryOpStub::GenerateSmiCode(MacroAssembler* masm, Label* slow) {
if (HasArgsInRegisters()) { if (HasArgsInRegisters()) {
__ mov(ebx, eax); __ mov(ebx, eax);
@ -7107,7 +7114,13 @@ void GenericBinaryOpStub::GenerateSmiCode(MacroAssembler* masm, Label* slow) {
__ j(overflow, &not_smis_or_overflow, not_taken); __ j(overflow, &not_smis_or_overflow, not_taken);
break; break;
case Token::MUL:
__ mov(edi, eax); // Backup the left operand.
break;
case Token::DIV: case Token::DIV:
__ mov(edi, eax); // Backup the left operand.
// Fall through.
case Token::MOD: case Token::MOD:
// Sign extend eax into edx:eax. // Sign extend eax into edx:eax.
__ cdq(); __ cdq();
@ -7243,28 +7256,18 @@ void GenericBinaryOpStub::GenerateSmiCode(MacroAssembler* masm, Label* slow) {
__ bind(&use_fp_on_smis); __ bind(&use_fp_on_smis);
// Both operands are known to be SMIs but the result does not fit into a SMI. // Both operands are known to be SMIs but the result does not fit into a SMI.
switch (op_) { switch (op_) {
case Token::ADD:
case Token::SUB:
case Token::MUL: case Token::MUL:
case Token::DIV: { case Token::DIV:
__ mov(eax, edi); // Restore the left operand.
// Fall through.
case Token::ADD:
case Token::SUB: {
Label after_alloc_failure; Label after_alloc_failure;
__ AllocateHeapNumber(edx, ecx, no_reg, &after_alloc_failure);
FloatingPointHelper::ArgLocation arg_location =
(op_ == Token::ADD || op_ == Token::SUB) ?
FloatingPointHelper::ARGS_IN_REGISTERS :
FloatingPointHelper::ARGS_ON_STACK;
__ AllocateHeapNumber(
edx,
ecx,
no_reg,
arg_location == FloatingPointHelper::ARGS_IN_REGISTERS ?
&after_alloc_failure :
slow);
if (CpuFeatures::IsSupported(SSE2)) { if (CpuFeatures::IsSupported(SSE2)) {
CpuFeatures::Scope use_sse2(SSE2); CpuFeatures::Scope use_sse2(SSE2);
FloatingPointHelper::LoadSse2Smis(masm, ecx, arg_location); FloatingPointHelper::LoadSse2Smis(masm, ecx);
switch (op_) { switch (op_) {
case Token::ADD: __ addsd(xmm0, xmm1); break; case Token::ADD: __ addsd(xmm0, xmm1); break;
case Token::SUB: __ subsd(xmm0, xmm1); break; case Token::SUB: __ subsd(xmm0, xmm1); break;
@ -7274,7 +7277,7 @@ void GenericBinaryOpStub::GenerateSmiCode(MacroAssembler* masm, Label* slow) {
} }
__ movdbl(FieldOperand(edx, HeapNumber::kValueOffset), xmm0); __ movdbl(FieldOperand(edx, HeapNumber::kValueOffset), xmm0);
} else { // SSE2 not available, use FPU. } else { // SSE2 not available, use FPU.
FloatingPointHelper::LoadFloatSmis(masm, ecx, arg_location); FloatingPointHelper::LoadFloatSmis(masm, ecx);
switch (op_) { switch (op_) {
case Token::ADD: __ faddp(1); break; case Token::ADD: __ faddp(1); break;
case Token::SUB: __ fsubp(1); break; case Token::SUB: __ fsubp(1); break;
@ -7287,12 +7290,10 @@ void GenericBinaryOpStub::GenerateSmiCode(MacroAssembler* masm, Label* slow) {
__ mov(eax, edx); __ mov(eax, edx);
GenerateReturn(masm); GenerateReturn(masm);
if (arg_location == FloatingPointHelper::ARGS_IN_REGISTERS) { __ bind(&after_alloc_failure);
__ bind(&after_alloc_failure); __ mov(edx, eax);
__ mov(edx, eax); __ mov(eax, ebx);
__ mov(eax, ebx); __ jmp(slow);
__ jmp(slow);
}
break; break;
} }
@ -7438,7 +7439,7 @@ void GenericBinaryOpStub::Generate(MacroAssembler* masm) {
__ bind(&non_smi_result); __ bind(&non_smi_result);
// Allocate a heap number if needed. // Allocate a heap number if needed.
__ mov(ebx, Operand(eax)); // ebx: result __ mov(ebx, Operand(eax)); // ebx: result
Label skip_allocation; Label skip_allocation;
switch (mode_) { switch (mode_) {
case OVERWRITE_LEFT: case OVERWRITE_LEFT:
case OVERWRITE_RIGHT: case OVERWRITE_RIGHT:
@ -7881,22 +7882,12 @@ void FloatingPointHelper::LoadSse2Operands(MacroAssembler* masm,
void FloatingPointHelper::LoadSse2Smis(MacroAssembler* masm, void FloatingPointHelper::LoadSse2Smis(MacroAssembler* masm,
Register scratch, Register scratch) {
ArgLocation arg_location) { __ mov(scratch, eax);
if (arg_location == ARGS_IN_REGISTERS) {
__ mov(scratch, eax);
} else {
__ mov(scratch, Operand(esp, 2 * kPointerSize));
}
__ SmiUntag(scratch); // Untag smi before converting to float. __ SmiUntag(scratch); // Untag smi before converting to float.
__ cvtsi2sd(xmm0, Operand(scratch)); __ cvtsi2sd(xmm0, Operand(scratch));
__ mov(scratch, ebx);
if (arg_location == ARGS_IN_REGISTERS) {
__ mov(scratch, ebx);
} else {
__ mov(scratch, Operand(esp, 1 * kPointerSize));
}
__ SmiUntag(scratch); // Untag smi before converting to float. __ SmiUntag(scratch); // Untag smi before converting to float.
__ cvtsi2sd(xmm1, Operand(scratch)); __ cvtsi2sd(xmm1, Operand(scratch));
} }
@ -7944,23 +7935,14 @@ void FloatingPointHelper::LoadFloatOperands(MacroAssembler* masm,
void FloatingPointHelper::LoadFloatSmis(MacroAssembler* masm, void FloatingPointHelper::LoadFloatSmis(MacroAssembler* masm,
Register scratch, Register scratch) {
ArgLocation arg_location) { __ mov(scratch, eax);
if (arg_location == ARGS_IN_REGISTERS) {
__ mov(scratch, eax);
} else {
__ mov(scratch, Operand(esp, 2 * kPointerSize));
}
__ SmiUntag(scratch); __ SmiUntag(scratch);
__ push(scratch); __ push(scratch);
__ fild_s(Operand(esp, 0)); __ fild_s(Operand(esp, 0));
__ pop(scratch); __ pop(scratch);
if (arg_location == ARGS_IN_REGISTERS) { __ mov(scratch, ebx);
__ mov(scratch, ebx);
} else {
__ mov(scratch, Operand(esp, 1 * kPointerSize));
}
__ SmiUntag(scratch); __ SmiUntag(scratch);
__ push(scratch); __ push(scratch);
__ fild_s(Operand(esp, 0)); __ fild_s(Operand(esp, 0));

View File

@ -652,6 +652,11 @@ class GenericBinaryOpStub: public CodeStub {
void GenerateCall(MacroAssembler* masm, Register left, Smi* right); void GenerateCall(MacroAssembler* masm, Register left, Smi* right);
void GenerateCall(MacroAssembler* masm, Smi* left, Register right); void GenerateCall(MacroAssembler* masm, Smi* left, Register right);
Result GenerateCall(MacroAssembler* masm,
VirtualFrame* frame,
Result* left,
Result* right);
private: private:
Token::Value op_; Token::Value op_;
OverwriteMode mode_; OverwriteMode mode_;
@ -700,15 +705,9 @@ class GenericBinaryOpStub: public CodeStub {
void GenerateReturn(MacroAssembler* masm); void GenerateReturn(MacroAssembler* masm);
void GenerateHeapResultAllocation(MacroAssembler* masm, Label* alloc_failure); void GenerateHeapResultAllocation(MacroAssembler* masm, Label* alloc_failure);
// Args in registers are always OK for ADD and SUB. Floating-point MUL and DIV
// are also OK. Though MUL and DIV on SMIs modify the original registers so
// we need to push args on stack anyway.
bool ArgsInRegistersSupported() { bool ArgsInRegistersSupported() {
if (op_ == Token::ADD || op_ == Token::SUB) return true; return op_ == Token::ADD || op_ == Token::SUB
if (op_ == Token::MUL || op_ == Token::DIV) { || op_ == Token::MUL || op_ == Token::DIV;
return flags_ == NO_SMI_CODE_IN_STUB;
}
return false;
} }
bool IsOperationCommutative() { bool IsOperationCommutative() {
return (op_ == Token::ADD) || (op_ == Token::MUL); return (op_ == Token::ADD) || (op_ == Token::MUL);

View File

@ -1560,12 +1560,10 @@ void FullCodeGenerator::VisitCountOperation(CountOperation* expr) {
} }
} }
// Call stub for +1/-1. // Call stub for +1/-1.
__ push(eax);
__ push(Immediate(Smi::FromInt(1)));
GenericBinaryOpStub stub(expr->binary_op(), GenericBinaryOpStub stub(expr->binary_op(),
NO_OVERWRITE, NO_OVERWRITE,
NO_GENERIC_BINARY_FLAGS); NO_GENERIC_BINARY_FLAGS);
__ CallStub(&stub); stub.GenerateCall(masm(), eax, Smi::FromInt(1));
__ bind(&done); __ bind(&done);
// Store the value returned in eax. // Store the value returned in eax.