MIPS: Do stack checks while pushing locals.

Port r20176 (a1bfd83)

BUG=
R=plind44@gmail.com

Review URL: https://codereview.chromium.org/208323010

Patch from Balazs Kilvady <kilvadyb@homejinni.com>.

git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@20196 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
This commit is contained in:
plind44@gmail.com 2014-03-24 14:11:26 +00:00
parent f1bacf8fff
commit 2c8229f609

View File

@ -120,6 +120,24 @@ class JumpPatchSite BASE_EMBEDDED {
};
static void EmitStackCheck(MacroAssembler* masm_,
Register stack_limit_scratch,
int pointers = 0,
Register scratch = sp) {
Isolate* isolate = masm_->isolate();
Label ok;
ASSERT(scratch.is(sp) == (pointers == 0));
if (pointers != 0) {
__ Subu(scratch, sp, Operand(pointers * kPointerSize));
}
__ LoadRoot(stack_limit_scratch, Heap::kStackLimitRootIndex);
__ Branch(&ok, hs, scratch, Operand(stack_limit_scratch));
PredictableCodeSizeScope predictable(masm_, 4 * Assembler::kInstrSize);
__ Call(isolate->builtins()->StackCheck(), RelocInfo::CODE_TARGET);
__ bind(&ok);
}
// Generate code for a JS function. On entry to the function the receiver
// and arguments have been pushed on the stack left to right. The actual
// argument count matches the formal parameter count expected by the
@ -187,22 +205,30 @@ void FullCodeGenerator::Generate() {
// Generators allocate locals, if any, in context slots.
ASSERT(!info->function()->is_generator() || locals_count == 0);
if (locals_count > 0) {
// Emit a loop to initialize stack cells for locals when optimizing for
// size. Otherwise, unroll the loop for maximum performance.
if (locals_count >= 128) {
EmitStackCheck(masm_, a2, locals_count, t5);
}
__ LoadRoot(t5, Heap::kUndefinedValueRootIndex);
if ((FLAG_optimize_for_size && locals_count > 4) ||
!is_int16(locals_count)) {
Label loop;
__ Subu(a2, sp, Operand(locals_count * kPointerSize));
__ bind(&loop);
__ Subu(sp, sp, Operand(kPointerSize));
__ Branch(&loop, gt, sp, Operand(a2), USE_DELAY_SLOT);
__ sw(t5, MemOperand(sp, 0)); // Push in the delay slot.
} else {
__ Subu(sp, sp, Operand(locals_count * kPointerSize));
for (int i = 0; i < locals_count; i++) {
int kMaxPushes = FLAG_optimize_for_size ? 4 : 32;
if (locals_count >= kMaxPushes) {
int loop_iterations = locals_count / kMaxPushes;
__ li(a2, Operand(loop_iterations));
Label loop_header;
__ bind(&loop_header);
// Do pushes.
__ Subu(sp, sp, Operand(kMaxPushes * kPointerSize));
for (int i = 0; i < kMaxPushes; i++) {
__ sw(t5, MemOperand(sp, i * kPointerSize));
}
// Continue loop if not done.
__ Subu(a2, a2, Operand(1));
__ Branch(&loop_header, ne, a2, Operand(zero_reg));
}
int remaining = locals_count % kMaxPushes;
// Emit the remaining pushes.
__ Subu(sp, sp, Operand(remaining * kPointerSize));
for (int i = 0; i < remaining; i++) {
__ sw(t5, MemOperand(sp, i * kPointerSize));
}
}
}
@ -313,11 +339,7 @@ void FullCodeGenerator::Generate() {
{ Comment cmnt(masm_, "[ Stack check");
PrepareForBailoutForId(BailoutId::Declarations(), NO_REGISTERS);
Label ok;
__ LoadRoot(t0, Heap::kStackLimitRootIndex);
__ Branch(&ok, hs, sp, Operand(t0));
__ Call(isolate()->builtins()->StackCheck(), RelocInfo::CODE_TARGET);
__ bind(&ok);
EmitStackCheck(masm_, at);
}
{ Comment cmnt(masm_, "[ Body");