Revert "Reland "[deoptimizer] Change deopt entries into builtins""

This reverts commit fbfa9bf4ec.

Reason for revert: Seems to break arm64 sim CFI build (please see DeoptExitSizeIfFixed) - https://ci.chromium.org/p/v8/builders/ci/V8%20Linux%20-%20arm64%20-%20sim%20-%20CFI/2808

Original change's description:
> Reland "[deoptimizer] Change deopt entries into builtins"
>
> This is a reland of 7f58ced72e
>
> It fixes the different exit size emitted on x64/Atom CPUs due to
> performance tuning in TurboAssembler::Call. Additionally, add
> cctests to verify the fixed size exits.
>
> Original change's description:
> > [deoptimizer] Change deopt entries into builtins
> >
> > While the overall goal of this commit is to change deoptimization
> > entries into builtins, there are multiple related things happening:
> >
> > - Deoptimization entries, formerly stubs (i.e. Code objects generated
> >   at runtime, guaranteed to be immovable), have been converted into
> >   builtins. The major restriction is that we now need to preserve the
> >   kRootRegister, which was formerly used on most architectures to pass
> >   the deoptimization id. The solution differs based on platform.
> > - Renamed DEOPT_ENTRIES_OR_FOR_TESTING code kind to FOR_TESTING.
> > - Removed heap/ support for immovable Code generation.
> > - Removed the DeserializerData class (no longer needed).
> > - arm64: to preserve 4-byte deopt exits, introduced a new optimization
> >   in which the final jump to the deoptimization entry is generated
> >   once per Code object, and deopt exits can continue to emit a
> >   near-call.
> > - arm,ia32,x64: change to fixed-size deopt exits. This reduces exit
> >   sizes by 4/8, 5, and 5 bytes, respectively.
> >
> > On arm the deopt exit size is reduced from 12 (or 16) bytes to 8 bytes
> > by using the same strategy as on arm64 (recalc deopt id from return
> > address). Before:
> >
> >  e300a002       movw r10, <id>
> >  e59fc024       ldr ip, [pc, <entry offset>]
> >  e12fff3c       blx ip
> >
> > After:
> >
> >  e59acb35       ldr ip, [r10, <entry offset>]
> >  e12fff3c       blx ip
> >
> > On arm64 the deopt exit size remains 4 bytes (or 8 bytes in same cases
> > with CFI). Additionally, up to 4 builtin jumps are emitted per Code
> > object (max 32 bytes added overhead per Code object). Before:
> >
> >  9401cdae       bl <entry offset>
> >
> > After:
> >
> >  # eager deoptimization entry jump.
> >  f95b1f50       ldr x16, [x26, <eager entry offset>]
> >  d61f0200       br x16
> >  # lazy deoptimization entry jump.
> >  f95b2b50       ldr x16, [x26, <lazy entry offset>]
> >  d61f0200       br x16
> >  # the deopt exit.
> >  97fffffc       bl <eager deoptimization entry jump offset>
> >
> > On ia32 the deopt exit size is reduced from 10 to 5 bytes. Before:
> >
> >  bb00000000     mov ebx,<id>
> >  e825f5372b     call <entry>
> >
> > After:
> >
> >  e8ea2256ba     call <entry>
> >
> > On x64 the deopt exit size is reduced from 12 to 7 bytes. Before:
> >
> >  49c7c511000000 REX.W movq r13,<id>
> >  e8ea2f0700     call <entry>
> >
> > After:
> >
> >  41ff9560360000 call [r13+<entry offset>]
> >
> > Bug: v8:8661,v8:8768
> > Change-Id: I13e30aedc360474dc818fecc528ce87c3bfeed42
> > Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2465834
> > Commit-Queue: Jakob Gruber <jgruber@chromium.org>
> > Reviewed-by: Ross McIlroy <rmcilroy@chromium.org>
> > Reviewed-by: Tobias Tebbi <tebbi@chromium.org>
> > Reviewed-by: Ulan Degenbaev <ulan@chromium.org>
> > Cr-Commit-Position: refs/heads/master@{#70597}
>
> Tbr: ulan@chromium.org, tebbi@chromium.org, rmcilroy@chromium.org
> Bug: v8:8661,v8:8768,chromium:1140165
> Change-Id: Ibcd5c39c58a70bf2b2ac221aa375fc68d495e144
> Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2485506
> Reviewed-by: Jakob Gruber <jgruber@chromium.org>
> Reviewed-by: Tobias Tebbi <tebbi@chromium.org>
> Commit-Queue: Jakob Gruber <jgruber@chromium.org>
> Cr-Commit-Position: refs/heads/master@{#70655}

TBR=ulan@chromium.org,rmcilroy@chromium.org,jgruber@chromium.org,tebbi@chromium.org

Change-Id: I4739a3475bfd8ee0cfbe4b9a20382f91a6ef1bf0
No-Presubmit: true
No-Tree-Checks: true
No-Try: true
Bug: v8:8661
Bug: v8:8768
Bug: chromium:1140165
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2485223
Reviewed-by: Maya Lekova <mslekova@chromium.org>
Commit-Queue: Maya Lekova <mslekova@chromium.org>
Cr-Commit-Position: refs/heads/master@{#70658}
This commit is contained in:
Maya Lekova 2020-10-20 14:13:57 +00:00 committed by Commit Bot
parent 0cf24e0a22
commit 7c7aa4fa94
77 changed files with 2362 additions and 1919 deletions

View File

@ -3166,250 +3166,6 @@ void Builtins::Generate_MemCopyUint8Uint8(MacroAssembler* masm) {
__ Ret(); __ Ret();
} }
namespace {
// This code tries to be close to ia32 code so that any changes can be
// easily ported.
void Generate_DeoptimizationEntry(MacroAssembler* masm,
DeoptimizeKind deopt_kind) {
Isolate* isolate = masm->isolate();
static constexpr int kDoubleRegsSize =
kDoubleSize * DwVfpRegister::kNumRegisters;
// Save all allocatable VFP registers before messing with them.
{
// We use a run-time check for VFP32DREGS.
CpuFeatureScope scope(masm, VFP32DREGS,
CpuFeatureScope::kDontCheckSupported);
UseScratchRegisterScope temps(masm);
Register scratch = temps.Acquire();
// Check CPU flags for number of registers, setting the Z condition flag.
__ CheckFor32DRegs(scratch);
// Push registers d0-d15, and possibly d16-d31, on the stack.
// If d16-d31 are not pushed, decrease the stack pointer instead.
__ vstm(db_w, sp, d16, d31, ne);
// Okay to not call AllocateStackSpace here because the size is a known
// small number and we need to use condition codes.
__ sub(sp, sp, Operand(16 * kDoubleSize), LeaveCC, eq);
__ vstm(db_w, sp, d0, d15);
}
// Save all general purpose registers before messing with them.
static constexpr int kNumberOfRegisters = Register::kNumRegisters;
STATIC_ASSERT(kNumberOfRegisters == 16);
// Everything but pc, lr and ip which will be saved but not restored.
RegList restored_regs = kJSCallerSaved | kCalleeSaved | ip.bit();
// Push all 16 registers (needed to populate FrameDescription::registers_).
// TODO(v8:1588): Note that using pc with stm is deprecated, so we should
// perhaps handle this a bit differently.
__ stm(db_w, sp, restored_regs | sp.bit() | lr.bit() | pc.bit());
{
UseScratchRegisterScope temps(masm);
Register scratch = temps.Acquire();
__ Move(scratch, ExternalReference::Create(
IsolateAddressId::kCEntryFPAddress, isolate));
__ str(fp, MemOperand(scratch));
}
static constexpr int kSavedRegistersAreaSize =
(kNumberOfRegisters * kPointerSize) + kDoubleRegsSize;
__ mov(r2, Operand(Deoptimizer::kFixedExitSizeMarker));
// Get the address of the location in the code object (r3) (return
// address for lazy deoptimization) and compute the fp-to-sp delta in
// register r4.
__ mov(r3, lr);
__ add(r4, sp, Operand(kSavedRegistersAreaSize));
__ sub(r4, fp, r4);
// Allocate a new deoptimizer object.
// Pass four arguments in r0 to r3 and fifth argument on stack.
__ PrepareCallCFunction(6);
__ mov(r0, Operand(0));
Label context_check;
__ ldr(r1, MemOperand(fp, CommonFrameConstants::kContextOrFrameTypeOffset));
__ JumpIfSmi(r1, &context_check);
__ ldr(r0, MemOperand(fp, StandardFrameConstants::kFunctionOffset));
__ bind(&context_check);
__ mov(r1, Operand(static_cast<int>(deopt_kind)));
// r2: bailout id already loaded.
// r3: code address or 0 already loaded.
__ str(r4, MemOperand(sp, 0 * kPointerSize)); // Fp-to-sp delta.
__ Move(r5, ExternalReference::isolate_address(isolate));
__ str(r5, MemOperand(sp, 1 * kPointerSize)); // Isolate.
// Call Deoptimizer::New().
{
AllowExternalCallThatCantCauseGC scope(masm);
__ CallCFunction(ExternalReference::new_deoptimizer_function(), 6);
}
// Preserve "deoptimizer" object in register r0 and get the input
// frame descriptor pointer to r1 (deoptimizer->input_);
__ ldr(r1, MemOperand(r0, Deoptimizer::input_offset()));
// Copy core registers into FrameDescription::registers_[kNumRegisters].
DCHECK_EQ(Register::kNumRegisters, kNumberOfRegisters);
for (int i = 0; i < kNumberOfRegisters; i++) {
int offset = (i * kPointerSize) + FrameDescription::registers_offset();
__ ldr(r2, MemOperand(sp, i * kPointerSize));
__ str(r2, MemOperand(r1, offset));
}
// Copy VFP registers to
// double_registers_[DoubleRegister::kNumAllocatableRegisters]
int double_regs_offset = FrameDescription::double_registers_offset();
const RegisterConfiguration* config = RegisterConfiguration::Default();
for (int i = 0; i < config->num_allocatable_double_registers(); ++i) {
int code = config->GetAllocatableDoubleCode(i);
int dst_offset = code * kDoubleSize + double_regs_offset;
int src_offset = code * kDoubleSize + kNumberOfRegisters * kPointerSize;
__ vldr(d0, sp, src_offset);
__ vstr(d0, r1, dst_offset);
}
// Mark the stack as not iterable for the CPU profiler which won't be able to
// walk the stack without the return address.
{
UseScratchRegisterScope temps(masm);
Register is_iterable = temps.Acquire();
Register zero = r4;
__ Move(is_iterable, ExternalReference::stack_is_iterable_address(isolate));
__ mov(zero, Operand(0));
__ strb(zero, MemOperand(is_iterable));
}
// Remove the saved registers from the stack.
__ add(sp, sp, Operand(kSavedRegistersAreaSize));
// Compute a pointer to the unwinding limit in register r2; that is
// the first stack slot not part of the input frame.
__ ldr(r2, MemOperand(r1, FrameDescription::frame_size_offset()));
__ add(r2, r2, sp);
// Unwind the stack down to - but not including - the unwinding
// limit and copy the contents of the activation frame to the input
// frame description.
__ add(r3, r1, Operand(FrameDescription::frame_content_offset()));
Label pop_loop;
Label pop_loop_header;
__ b(&pop_loop_header);
__ bind(&pop_loop);
__ pop(r4);
__ str(r4, MemOperand(r3, 0));
__ add(r3, r3, Operand(sizeof(uint32_t)));
__ bind(&pop_loop_header);
__ cmp(r2, sp);
__ b(ne, &pop_loop);
// Compute the output frame in the deoptimizer.
__ push(r0); // Preserve deoptimizer object across call.
// r0: deoptimizer object; r1: scratch.
__ PrepareCallCFunction(1);
// Call Deoptimizer::ComputeOutputFrames().
{
AllowExternalCallThatCantCauseGC scope(masm);
__ CallCFunction(ExternalReference::compute_output_frames_function(), 1);
}
__ pop(r0); // Restore deoptimizer object (class Deoptimizer).
__ ldr(sp, MemOperand(r0, Deoptimizer::caller_frame_top_offset()));
// Replace the current (input) frame with the output frames.
Label outer_push_loop, inner_push_loop, outer_loop_header, inner_loop_header;
// Outer loop state: r4 = current "FrameDescription** output_",
// r1 = one past the last FrameDescription**.
__ ldr(r1, MemOperand(r0, Deoptimizer::output_count_offset()));
__ ldr(r4, MemOperand(r0, Deoptimizer::output_offset())); // r4 is output_.
__ add(r1, r4, Operand(r1, LSL, 2));
__ jmp(&outer_loop_header);
__ bind(&outer_push_loop);
// Inner loop state: r2 = current FrameDescription*, r3 = loop index.
__ ldr(r2, MemOperand(r4, 0)); // output_[ix]
__ ldr(r3, MemOperand(r2, FrameDescription::frame_size_offset()));
__ jmp(&inner_loop_header);
__ bind(&inner_push_loop);
__ sub(r3, r3, Operand(sizeof(uint32_t)));
__ add(r6, r2, Operand(r3));
__ ldr(r6, MemOperand(r6, FrameDescription::frame_content_offset()));
__ push(r6);
__ bind(&inner_loop_header);
__ cmp(r3, Operand::Zero());
__ b(ne, &inner_push_loop); // test for gt?
__ add(r4, r4, Operand(kPointerSize));
__ bind(&outer_loop_header);
__ cmp(r4, r1);
__ b(lt, &outer_push_loop);
__ ldr(r1, MemOperand(r0, Deoptimizer::input_offset()));
for (int i = 0; i < config->num_allocatable_double_registers(); ++i) {
int code = config->GetAllocatableDoubleCode(i);
DwVfpRegister reg = DwVfpRegister::from_code(code);
int src_offset = code * kDoubleSize + double_regs_offset;
__ vldr(reg, r1, src_offset);
}
// Push pc and continuation from the last output frame.
__ ldr(r6, MemOperand(r2, FrameDescription::pc_offset()));
__ push(r6);
__ ldr(r6, MemOperand(r2, FrameDescription::continuation_offset()));
__ push(r6);
// Push the registers from the last output frame.
for (int i = kNumberOfRegisters - 1; i >= 0; i--) {
int offset = (i * kPointerSize) + FrameDescription::registers_offset();
__ ldr(r6, MemOperand(r2, offset));
__ push(r6);
}
// Restore the registers from the stack.
__ ldm(ia_w, sp, restored_regs); // all but pc registers.
{
UseScratchRegisterScope temps(masm);
Register is_iterable = temps.Acquire();
Register one = r4;
__ Move(is_iterable, ExternalReference::stack_is_iterable_address(isolate));
__ mov(one, Operand(1));
__ strb(one, MemOperand(is_iterable));
}
// Remove sp, lr and pc.
__ Drop(3);
{
UseScratchRegisterScope temps(masm);
Register scratch = temps.Acquire();
__ pop(scratch); // get continuation, leave pc on stack
__ pop(lr);
__ Jump(scratch);
}
__ stop();
}
} // namespace
void Builtins::Generate_DeoptimizationEntry_Eager(MacroAssembler* masm) {
Generate_DeoptimizationEntry(masm, DeoptimizeKind::kEager);
}
void Builtins::Generate_DeoptimizationEntry_Soft(MacroAssembler* masm) {
Generate_DeoptimizationEntry(masm, DeoptimizeKind::kSoft);
}
void Builtins::Generate_DeoptimizationEntry_Bailout(MacroAssembler* masm) {
Generate_DeoptimizationEntry(masm, DeoptimizeKind::kBailout);
}
void Builtins::Generate_DeoptimizationEntry_Lazy(MacroAssembler* masm) {
Generate_DeoptimizationEntry(masm, DeoptimizeKind::kLazy);
}
#undef __ #undef __
} // namespace internal } // namespace internal

View File

@ -3619,297 +3619,6 @@ void Builtins::Generate_DirectCEntry(MacroAssembler* masm) {
__ Ret(); __ Ret();
} }
namespace {
void CopyRegListToFrame(MacroAssembler* masm, const Register& dst,
int dst_offset, const CPURegList& reg_list,
const Register& temp0, const Register& temp1,
int src_offset = 0) {
DCHECK_EQ(reg_list.Count() % 2, 0);
UseScratchRegisterScope temps(masm);
CPURegList copy_to_input = reg_list;
int reg_size = reg_list.RegisterSizeInBytes();
DCHECK_EQ(temp0.SizeInBytes(), reg_size);
DCHECK_EQ(temp1.SizeInBytes(), reg_size);
// Compute some temporary addresses to avoid having the macro assembler set
// up a temp with an offset for accesses out of the range of the addressing
// mode.
Register src = temps.AcquireX();
masm->Add(src, sp, src_offset);
masm->Add(dst, dst, dst_offset);
// Write reg_list into the frame pointed to by dst.
for (int i = 0; i < reg_list.Count(); i += 2) {
masm->Ldp(temp0, temp1, MemOperand(src, i * reg_size));
CPURegister reg0 = copy_to_input.PopLowestIndex();
CPURegister reg1 = copy_to_input.PopLowestIndex();
int offset0 = reg0.code() * reg_size;
int offset1 = reg1.code() * reg_size;
// Pair up adjacent stores, otherwise write them separately.
if (offset1 == offset0 + reg_size) {
masm->Stp(temp0, temp1, MemOperand(dst, offset0));
} else {
masm->Str(temp0, MemOperand(dst, offset0));
masm->Str(temp1, MemOperand(dst, offset1));
}
}
masm->Sub(dst, dst, dst_offset);
}
void RestoreRegList(MacroAssembler* masm, const CPURegList& reg_list,
const Register& src_base, int src_offset) {
DCHECK_EQ(reg_list.Count() % 2, 0);
UseScratchRegisterScope temps(masm);
CPURegList restore_list = reg_list;
int reg_size = restore_list.RegisterSizeInBytes();
// Compute a temporary addresses to avoid having the macro assembler set
// up a temp with an offset for accesses out of the range of the addressing
// mode.
Register src = temps.AcquireX();
masm->Add(src, src_base, src_offset);
// No need to restore padreg.
restore_list.Remove(padreg);
// Restore every register in restore_list from src.
while (!restore_list.IsEmpty()) {
CPURegister reg0 = restore_list.PopLowestIndex();
CPURegister reg1 = restore_list.PopLowestIndex();
int offset0 = reg0.code() * reg_size;
if (reg1 == NoCPUReg) {
masm->Ldr(reg0, MemOperand(src, offset0));
break;
}
int offset1 = reg1.code() * reg_size;
// Pair up adjacent loads, otherwise read them separately.
if (offset1 == offset0 + reg_size) {
masm->Ldp(reg0, reg1, MemOperand(src, offset0));
} else {
masm->Ldr(reg0, MemOperand(src, offset0));
masm->Ldr(reg1, MemOperand(src, offset1));
}
}
}
void Generate_DeoptimizationEntry(MacroAssembler* masm,
DeoptimizeKind deopt_kind) {
Isolate* isolate = masm->isolate();
// TODO(all): This code needs to be revisited. We probably only need to save
// caller-saved registers here. Callee-saved registers can be stored directly
// in the input frame.
// Save all allocatable double registers.
CPURegList saved_double_registers(
CPURegister::kVRegister, kDRegSizeInBits,
RegisterConfiguration::Default()->allocatable_double_codes_mask());
DCHECK_EQ(saved_double_registers.Count() % 2, 0);
__ PushCPURegList(saved_double_registers);
// We save all the registers except sp, lr, platform register (x18) and the
// masm scratches.
CPURegList saved_registers(CPURegister::kRegister, kXRegSizeInBits, 0, 28);
saved_registers.Remove(ip0);
saved_registers.Remove(ip1);
saved_registers.Remove(x18);
saved_registers.Combine(fp);
saved_registers.Align();
DCHECK_EQ(saved_registers.Count() % 2, 0);
__ PushCPURegList(saved_registers);
__ Mov(x3, Operand(ExternalReference::Create(
IsolateAddressId::kCEntryFPAddress, isolate)));
__ Str(fp, MemOperand(x3));
const int kSavedRegistersAreaSize =
(saved_registers.Count() * kXRegSize) +
(saved_double_registers.Count() * kDRegSize);
// Floating point registers are saved on the stack above core registers.
const int kDoubleRegistersOffset = saved_registers.Count() * kXRegSize;
Register bailout_id = x2;
Register code_object = x3;
Register fp_to_sp = x4;
__ Mov(bailout_id, Deoptimizer::kFixedExitSizeMarker);
// Get the address of the location in the code object. This is the return
// address for lazy deoptimization.
__ Mov(code_object, lr);
// Compute the fp-to-sp delta.
__ Add(fp_to_sp, sp, kSavedRegistersAreaSize);
__ Sub(fp_to_sp, fp, fp_to_sp);
// Allocate a new deoptimizer object.
__ Ldr(x1, MemOperand(fp, CommonFrameConstants::kContextOrFrameTypeOffset));
// Ensure we can safely load from below fp.
DCHECK_GT(kSavedRegistersAreaSize, -StandardFrameConstants::kFunctionOffset);
__ Ldr(x0, MemOperand(fp, StandardFrameConstants::kFunctionOffset));
// If x1 is a smi, zero x0.
__ Tst(x1, kSmiTagMask);
__ CzeroX(x0, eq);
__ Mov(x1, static_cast<int>(deopt_kind));
// Following arguments are already loaded:
// - x2: bailout id
// - x3: code object address
// - x4: fp-to-sp delta
__ Mov(x5, ExternalReference::isolate_address(isolate));
{
// Call Deoptimizer::New().
AllowExternalCallThatCantCauseGC scope(masm);
__ CallCFunction(ExternalReference::new_deoptimizer_function(), 6);
}
// Preserve "deoptimizer" object in register x0.
Register deoptimizer = x0;
// Get the input frame descriptor pointer.
__ Ldr(x1, MemOperand(deoptimizer, Deoptimizer::input_offset()));
// Copy core registers into the input frame.
CopyRegListToFrame(masm, x1, FrameDescription::registers_offset(),
saved_registers, x2, x3);
// Copy double registers to the input frame.
CopyRegListToFrame(masm, x1, FrameDescription::double_registers_offset(),
saved_double_registers, x2, x3, kDoubleRegistersOffset);
// Mark the stack as not iterable for the CPU profiler which won't be able to
// walk the stack without the return address.
{
UseScratchRegisterScope temps(masm);
Register is_iterable = temps.AcquireX();
__ Mov(is_iterable, ExternalReference::stack_is_iterable_address(isolate));
__ strb(xzr, MemOperand(is_iterable));
}
// Remove the saved registers from the stack.
DCHECK_EQ(kSavedRegistersAreaSize % kXRegSize, 0);
__ Drop(kSavedRegistersAreaSize / kXRegSize);
// Compute a pointer to the unwinding limit in register x2; that is
// the first stack slot not part of the input frame.
Register unwind_limit = x2;
__ Ldr(unwind_limit, MemOperand(x1, FrameDescription::frame_size_offset()));
// Unwind the stack down to - but not including - the unwinding
// limit and copy the contents of the activation frame to the input
// frame description.
__ Add(x3, x1, FrameDescription::frame_content_offset());
__ SlotAddress(x1, 0);
__ Lsr(unwind_limit, unwind_limit, kSystemPointerSizeLog2);
__ Mov(x5, unwind_limit);
__ CopyDoubleWords(x3, x1, x5);
__ Drop(unwind_limit);
// Compute the output frame in the deoptimizer.
__ Push(padreg, x0); // Preserve deoptimizer object across call.
{
// Call Deoptimizer::ComputeOutputFrames().
AllowExternalCallThatCantCauseGC scope(masm);
__ CallCFunction(ExternalReference::compute_output_frames_function(), 1);
}
__ Pop(x4, padreg); // Restore deoptimizer object (class Deoptimizer).
{
UseScratchRegisterScope temps(masm);
Register scratch = temps.AcquireX();
__ Ldr(scratch, MemOperand(x4, Deoptimizer::caller_frame_top_offset()));
__ Mov(sp, scratch);
}
// Replace the current (input) frame with the output frames.
Label outer_push_loop, outer_loop_header;
__ Ldrsw(x1, MemOperand(x4, Deoptimizer::output_count_offset()));
__ Ldr(x0, MemOperand(x4, Deoptimizer::output_offset()));
__ Add(x1, x0, Operand(x1, LSL, kSystemPointerSizeLog2));
__ B(&outer_loop_header);
__ Bind(&outer_push_loop);
Register current_frame = x2;
Register frame_size = x3;
__ Ldr(current_frame, MemOperand(x0, kSystemPointerSize, PostIndex));
__ Ldr(x3, MemOperand(current_frame, FrameDescription::frame_size_offset()));
__ Lsr(frame_size, x3, kSystemPointerSizeLog2);
__ Claim(frame_size);
__ Add(x7, current_frame, FrameDescription::frame_content_offset());
__ SlotAddress(x6, 0);
__ CopyDoubleWords(x6, x7, frame_size);
__ Bind(&outer_loop_header);
__ Cmp(x0, x1);
__ B(lt, &outer_push_loop);
__ Ldr(x1, MemOperand(x4, Deoptimizer::input_offset()));
RestoreRegList(masm, saved_double_registers, x1,
FrameDescription::double_registers_offset());
{
UseScratchRegisterScope temps(masm);
Register is_iterable = temps.AcquireX();
Register one = x4;
__ Mov(is_iterable, ExternalReference::stack_is_iterable_address(isolate));
__ Mov(one, Operand(1));
__ strb(one, MemOperand(is_iterable));
}
// TODO(all): ARM copies a lot (if not all) of the last output frame onto the
// stack, then pops it all into registers. Here, we try to load it directly
// into the relevant registers. Is this correct? If so, we should improve the
// ARM code.
// Restore registers from the last output frame.
// Note that lr is not in the list of saved_registers and will be restored
// later. We can use it to hold the address of last output frame while
// reloading the other registers.
DCHECK(!saved_registers.IncludesAliasOf(lr));
Register last_output_frame = lr;
__ Mov(last_output_frame, current_frame);
RestoreRegList(masm, saved_registers, last_output_frame,
FrameDescription::registers_offset());
UseScratchRegisterScope temps(masm);
temps.Exclude(x17);
Register continuation = x17;
__ Ldr(continuation, MemOperand(last_output_frame,
FrameDescription::continuation_offset()));
__ Ldr(lr, MemOperand(last_output_frame, FrameDescription::pc_offset()));
#ifdef V8_ENABLE_CONTROL_FLOW_INTEGRITY
__ Autibsp();
#endif
__ Br(continuation);
}
} // namespace
void Builtins::Generate_DeoptimizationEntry_Eager(MacroAssembler* masm) {
Generate_DeoptimizationEntry(masm, DeoptimizeKind::kEager);
}
void Builtins::Generate_DeoptimizationEntry_Soft(MacroAssembler* masm) {
Generate_DeoptimizationEntry(masm, DeoptimizeKind::kSoft);
}
void Builtins::Generate_DeoptimizationEntry_Bailout(MacroAssembler* masm) {
Generate_DeoptimizationEntry(masm, DeoptimizeKind::kBailout);
}
void Builtins::Generate_DeoptimizationEntry_Lazy(MacroAssembler* masm) {
Generate_DeoptimizationEntry(masm, DeoptimizeKind::kLazy);
}
#undef __ #undef __
} // namespace internal } // namespace internal

View File

@ -138,10 +138,6 @@ namespace internal {
TFC(CompileLazyDeoptimizedCode, JSTrampoline) \ TFC(CompileLazyDeoptimizedCode, JSTrampoline) \
TFC(InstantiateAsmJs, JSTrampoline) \ TFC(InstantiateAsmJs, JSTrampoline) \
ASM(NotifyDeoptimized, Dummy) \ ASM(NotifyDeoptimized, Dummy) \
ASM(DeoptimizationEntry_Eager, DeoptimizationEntry) \
ASM(DeoptimizationEntry_Soft, DeoptimizationEntry) \
ASM(DeoptimizationEntry_Bailout, DeoptimizationEntry) \
ASM(DeoptimizationEntry_Lazy, DeoptimizationEntry) \
\ \
/* Trampolines called when returning from a deoptimization that expects */ \ /* Trampolines called when returning from a deoptimization that expects */ \
/* to continue in a JavaScript builtin to finish the functionality of a */ \ /* to continue in a JavaScript builtin to finish the functionality of a */ \

View File

@ -3781,205 +3781,6 @@ void Builtins::Generate_MemMove(MacroAssembler* masm) {
MemMoveEmitPopAndReturn(masm); MemMoveEmitPopAndReturn(masm);
} }
namespace {
void Generate_DeoptimizationEntry(MacroAssembler* masm,
DeoptimizeKind deopt_kind) {
Isolate* isolate = masm->isolate();
// Save all general purpose registers before messing with them.
const int kNumberOfRegisters = Register::kNumRegisters;
const int kDoubleRegsSize = kDoubleSize * XMMRegister::kNumRegisters;
__ AllocateStackSpace(kDoubleRegsSize);
const RegisterConfiguration* config = RegisterConfiguration::Default();
for (int i = 0; i < config->num_allocatable_double_registers(); ++i) {
int code = config->GetAllocatableDoubleCode(i);
XMMRegister xmm_reg = XMMRegister::from_code(code);
int offset = code * kDoubleSize;
__ movsd(Operand(esp, offset), xmm_reg);
}
__ pushad();
ExternalReference c_entry_fp_address =
ExternalReference::Create(IsolateAddressId::kCEntryFPAddress, isolate);
__ mov(masm->ExternalReferenceAsOperand(c_entry_fp_address, esi), ebp);
const int kSavedRegistersAreaSize =
kNumberOfRegisters * kSystemPointerSize + kDoubleRegsSize;
// Get the address of the location in the code object
// and compute the fp-to-sp delta in register edx.
__ mov(ecx, Operand(esp, kSavedRegistersAreaSize));
__ lea(edx, Operand(esp, kSavedRegistersAreaSize + 1 * kSystemPointerSize));
__ sub(edx, ebp);
__ neg(edx);
// Allocate a new deoptimizer object.
__ PrepareCallCFunction(6, eax);
__ mov(eax, Immediate(0));
Label context_check;
__ mov(edi, Operand(ebp, CommonFrameConstants::kContextOrFrameTypeOffset));
__ JumpIfSmi(edi, &context_check);
__ mov(eax, Operand(ebp, StandardFrameConstants::kFunctionOffset));
__ bind(&context_check);
__ mov(Operand(esp, 0 * kSystemPointerSize), eax); // Function.
__ mov(Operand(esp, 1 * kSystemPointerSize),
Immediate(static_cast<int>(deopt_kind)));
__ mov(Operand(esp, 2 * kSystemPointerSize),
Immediate(Deoptimizer::kFixedExitSizeMarker)); // Bailout id.
__ mov(Operand(esp, 3 * kSystemPointerSize), ecx); // Code address or 0.
__ mov(Operand(esp, 4 * kSystemPointerSize), edx); // Fp-to-sp delta.
__ Move(Operand(esp, 5 * kSystemPointerSize),
Immediate(ExternalReference::isolate_address(masm->isolate())));
{
AllowExternalCallThatCantCauseGC scope(masm);
__ CallCFunction(ExternalReference::new_deoptimizer_function(), 6);
}
// Preserve deoptimizer object in register eax and get the input
// frame descriptor pointer.
__ mov(esi, Operand(eax, Deoptimizer::input_offset()));
// Fill in the input registers.
for (int i = kNumberOfRegisters - 1; i >= 0; i--) {
int offset =
(i * kSystemPointerSize) + FrameDescription::registers_offset();
__ pop(Operand(esi, offset));
}
int double_regs_offset = FrameDescription::double_registers_offset();
// Fill in the double input registers.
for (int i = 0; i < config->num_allocatable_double_registers(); ++i) {
int code = config->GetAllocatableDoubleCode(i);
int dst_offset = code * kDoubleSize + double_regs_offset;
int src_offset = code * kDoubleSize;
__ movsd(xmm0, Operand(esp, src_offset));
__ movsd(Operand(esi, dst_offset), xmm0);
}
// Clear FPU all exceptions.
// TODO(ulan): Find out why the TOP register is not zero here in some cases,
// and check that the generated code never deoptimizes with unbalanced stack.
__ fnclex();
// Mark the stack as not iterable for the CPU profiler which won't be able to
// walk the stack without the return address.
__ mov_b(__ ExternalReferenceAsOperand(
ExternalReference::stack_is_iterable_address(isolate), edx),
Immediate(0));
// Remove the return address and the double registers.
__ add(esp, Immediate(kDoubleRegsSize + 1 * kSystemPointerSize));
// Compute a pointer to the unwinding limit in register ecx; that is
// the first stack slot not part of the input frame.
__ mov(ecx, Operand(esi, FrameDescription::frame_size_offset()));
__ add(ecx, esp);
// Unwind the stack down to - but not including - the unwinding
// limit and copy the contents of the activation frame to the input
// frame description.
__ lea(edx, Operand(esi, FrameDescription::frame_content_offset()));
Label pop_loop_header;
__ jmp(&pop_loop_header);
Label pop_loop;
__ bind(&pop_loop);
__ pop(Operand(edx, 0));
__ add(edx, Immediate(sizeof(uint32_t)));
__ bind(&pop_loop_header);
__ cmp(ecx, esp);
__ j(not_equal, &pop_loop);
// Compute the output frame in the deoptimizer.
__ push(eax);
__ PrepareCallCFunction(1, esi);
__ mov(Operand(esp, 0 * kSystemPointerSize), eax);
{
AllowExternalCallThatCantCauseGC scope(masm);
__ CallCFunction(ExternalReference::compute_output_frames_function(), 1);
}
__ pop(eax);
__ mov(esp, Operand(eax, Deoptimizer::caller_frame_top_offset()));
// Replace the current (input) frame with the output frames.
Label outer_push_loop, inner_push_loop, outer_loop_header, inner_loop_header;
// Outer loop state: eax = current FrameDescription**, edx = one
// past the last FrameDescription**.
__ mov(edx, Operand(eax, Deoptimizer::output_count_offset()));
__ mov(eax, Operand(eax, Deoptimizer::output_offset()));
__ lea(edx, Operand(eax, edx, times_system_pointer_size, 0));
__ jmp(&outer_loop_header);
__ bind(&outer_push_loop);
// Inner loop state: esi = current FrameDescription*, ecx = loop
// index.
__ mov(esi, Operand(eax, 0));
__ mov(ecx, Operand(esi, FrameDescription::frame_size_offset()));
__ jmp(&inner_loop_header);
__ bind(&inner_push_loop);
__ sub(ecx, Immediate(sizeof(uint32_t)));
__ push(Operand(esi, ecx, times_1, FrameDescription::frame_content_offset()));
__ bind(&inner_loop_header);
__ test(ecx, ecx);
__ j(not_zero, &inner_push_loop);
__ add(eax, Immediate(kSystemPointerSize));
__ bind(&outer_loop_header);
__ cmp(eax, edx);
__ j(below, &outer_push_loop);
// In case of a failed STUB, we have to restore the XMM registers.
for (int i = 0; i < config->num_allocatable_double_registers(); ++i) {
int code = config->GetAllocatableDoubleCode(i);
XMMRegister xmm_reg = XMMRegister::from_code(code);
int src_offset = code * kDoubleSize + double_regs_offset;
__ movsd(xmm_reg, Operand(esi, src_offset));
}
// Push pc and continuation from the last output frame.
__ push(Operand(esi, FrameDescription::pc_offset()));
__ push(Operand(esi, FrameDescription::continuation_offset()));
// Push the registers from the last output frame.
for (int i = 0; i < kNumberOfRegisters; i++) {
int offset =
(i * kSystemPointerSize) + FrameDescription::registers_offset();
__ push(Operand(esi, offset));
}
__ mov_b(__ ExternalReferenceAsOperand(
ExternalReference::stack_is_iterable_address(isolate), edx),
Immediate(1));
// Restore the registers from the stack.
__ popad();
__ InitializeRootRegister();
// Return to the continuation point.
__ ret(0);
}
} // namespace
void Builtins::Generate_DeoptimizationEntry_Eager(MacroAssembler* masm) {
Generate_DeoptimizationEntry(masm, DeoptimizeKind::kEager);
}
void Builtins::Generate_DeoptimizationEntry_Soft(MacroAssembler* masm) {
Generate_DeoptimizationEntry(masm, DeoptimizeKind::kSoft);
}
void Builtins::Generate_DeoptimizationEntry_Bailout(MacroAssembler* masm) {
Generate_DeoptimizationEntry(masm, DeoptimizeKind::kBailout);
}
void Builtins::Generate_DeoptimizationEntry_Lazy(MacroAssembler* masm) {
Generate_DeoptimizationEntry(masm, DeoptimizeKind::kLazy);
}
#undef __ #undef __
} // namespace internal } // namespace internal

View File

@ -4101,223 +4101,6 @@ void Builtins::Generate_DirectCEntry(MacroAssembler* masm) {
__ int3(); // Unused on this architecture. __ int3(); // Unused on this architecture.
} }
namespace {
void Generate_DeoptimizationEntry(MacroAssembler* masm,
DeoptimizeKind deopt_kind) {
Isolate* isolate = masm->isolate();
// Save all double registers, they will later be copied to the deoptimizer's
// FrameDescription.
static constexpr int kDoubleRegsSize =
kDoubleSize * XMMRegister::kNumRegisters;
__ AllocateStackSpace(kDoubleRegsSize);
const RegisterConfiguration* config = RegisterConfiguration::Default();
for (int i = 0; i < config->num_allocatable_double_registers(); ++i) {
int code = config->GetAllocatableDoubleCode(i);
XMMRegister xmm_reg = XMMRegister::from_code(code);
int offset = code * kDoubleSize;
__ Movsd(Operand(rsp, offset), xmm_reg);
}
// Save all general purpose registers, they will later be copied to the
// deoptimizer's FrameDescription.
static constexpr int kNumberOfRegisters = Register::kNumRegisters;
for (int i = 0; i < kNumberOfRegisters; i++) {
__ pushq(Register::from_code(i));
}
static constexpr int kSavedRegistersAreaSize =
kNumberOfRegisters * kSystemPointerSize + kDoubleRegsSize;
static constexpr int kCurrentOffsetToReturnAddress = kSavedRegistersAreaSize;
static constexpr int kCurrentOffsetToParentSP =
kCurrentOffsetToReturnAddress + kPCOnStackSize;
__ Store(
ExternalReference::Create(IsolateAddressId::kCEntryFPAddress, isolate),
rbp);
// We use this to keep the value of the fifth argument temporarily.
// Unfortunately we can't store it directly in r8 (used for passing
// this on linux), since it is another parameter passing register on windows.
Register arg5 = r11;
__ movq(arg_reg_3, Immediate(Deoptimizer::kFixedExitSizeMarker));
// Get the address of the location in the code object
// and compute the fp-to-sp delta in register arg5.
__ movq(arg_reg_4, Operand(rsp, kCurrentOffsetToReturnAddress));
// Load the fp-to-sp-delta.
__ leaq(arg5, Operand(rsp, kCurrentOffsetToParentSP));
__ subq(arg5, rbp);
__ negq(arg5);
// Allocate a new deoptimizer object.
__ PrepareCallCFunction(6);
__ movq(rax, Immediate(0));
Label context_check;
__ movq(rdi, Operand(rbp, CommonFrameConstants::kContextOrFrameTypeOffset));
__ JumpIfSmi(rdi, &context_check);
__ movq(rax, Operand(rbp, StandardFrameConstants::kFunctionOffset));
__ bind(&context_check);
__ movq(arg_reg_1, rax);
__ Set(arg_reg_2, static_cast<int>(deopt_kind));
// Args 3 and 4 are already in the right registers.
// On windows put the arguments on the stack (PrepareCallCFunction
// has created space for this). On linux pass the arguments in r8 and r9.
#ifdef V8_TARGET_OS_WIN
__ movq(Operand(rsp, 4 * kSystemPointerSize), arg5);
__ LoadAddress(arg5, ExternalReference::isolate_address(isolate));
__ movq(Operand(rsp, 5 * kSystemPointerSize), arg5);
#else
__ movq(r8, arg5);
__ LoadAddress(r9, ExternalReference::isolate_address(isolate));
#endif
{
AllowExternalCallThatCantCauseGC scope(masm);
__ CallCFunction(ExternalReference::new_deoptimizer_function(), 6);
}
// Preserve deoptimizer object in register rax and get the input
// frame descriptor pointer.
__ movq(rbx, Operand(rax, Deoptimizer::input_offset()));
// Fill in the input registers.
for (int i = kNumberOfRegisters - 1; i >= 0; i--) {
int offset =
(i * kSystemPointerSize) + FrameDescription::registers_offset();
__ PopQuad(Operand(rbx, offset));
}
// Fill in the double input registers.
int double_regs_offset = FrameDescription::double_registers_offset();
for (int i = 0; i < XMMRegister::kNumRegisters; i++) {
int dst_offset = i * kDoubleSize + double_regs_offset;
__ popq(Operand(rbx, dst_offset));
}
// Mark the stack as not iterable for the CPU profiler which won't be able to
// walk the stack without the return address.
__ movb(__ ExternalReferenceAsOperand(
ExternalReference::stack_is_iterable_address(isolate)),
Immediate(0));
// Remove the return address from the stack.
__ addq(rsp, Immediate(kPCOnStackSize));
// Compute a pointer to the unwinding limit in register rcx; that is
// the first stack slot not part of the input frame.
__ movq(rcx, Operand(rbx, FrameDescription::frame_size_offset()));
__ addq(rcx, rsp);
// Unwind the stack down to - but not including - the unwinding
// limit and copy the contents of the activation frame to the input
// frame description.
__ leaq(rdx, Operand(rbx, FrameDescription::frame_content_offset()));
Label pop_loop_header;
__ jmp(&pop_loop_header);
Label pop_loop;
__ bind(&pop_loop);
__ Pop(Operand(rdx, 0));
__ addq(rdx, Immediate(sizeof(intptr_t)));
__ bind(&pop_loop_header);
__ cmpq(rcx, rsp);
__ j(not_equal, &pop_loop);
// Compute the output frame in the deoptimizer.
__ pushq(rax);
__ PrepareCallCFunction(2);
__ movq(arg_reg_1, rax);
__ LoadAddress(arg_reg_2, ExternalReference::isolate_address(isolate));
{
AllowExternalCallThatCantCauseGC scope(masm);
__ CallCFunction(ExternalReference::compute_output_frames_function(), 2);
}
__ popq(rax);
__ movq(rsp, Operand(rax, Deoptimizer::caller_frame_top_offset()));
// Replace the current (input) frame with the output frames.
Label outer_push_loop, inner_push_loop, outer_loop_header, inner_loop_header;
// Outer loop state: rax = current FrameDescription**, rdx = one past the
// last FrameDescription**.
__ movl(rdx, Operand(rax, Deoptimizer::output_count_offset()));
__ movq(rax, Operand(rax, Deoptimizer::output_offset()));
__ leaq(rdx, Operand(rax, rdx, times_system_pointer_size, 0));
__ jmp(&outer_loop_header);
__ bind(&outer_push_loop);
// Inner loop state: rbx = current FrameDescription*, rcx = loop index.
__ movq(rbx, Operand(rax, 0));
__ movq(rcx, Operand(rbx, FrameDescription::frame_size_offset()));
__ jmp(&inner_loop_header);
__ bind(&inner_push_loop);
__ subq(rcx, Immediate(sizeof(intptr_t)));
__ Push(Operand(rbx, rcx, times_1, FrameDescription::frame_content_offset()));
__ bind(&inner_loop_header);
__ testq(rcx, rcx);
__ j(not_zero, &inner_push_loop);
__ addq(rax, Immediate(kSystemPointerSize));
__ bind(&outer_loop_header);
__ cmpq(rax, rdx);
__ j(below, &outer_push_loop);
for (int i = 0; i < config->num_allocatable_double_registers(); ++i) {
int code = config->GetAllocatableDoubleCode(i);
XMMRegister xmm_reg = XMMRegister::from_code(code);
int src_offset = code * kDoubleSize + double_regs_offset;
__ Movsd(xmm_reg, Operand(rbx, src_offset));
}
// Push pc and continuation from the last output frame.
__ PushQuad(Operand(rbx, FrameDescription::pc_offset()));
__ PushQuad(Operand(rbx, FrameDescription::continuation_offset()));
// Push the registers from the last output frame.
for (int i = 0; i < kNumberOfRegisters; i++) {
int offset =
(i * kSystemPointerSize) + FrameDescription::registers_offset();
__ PushQuad(Operand(rbx, offset));
}
// Restore the registers from the stack.
for (int i = kNumberOfRegisters - 1; i >= 0; i--) {
Register r = Register::from_code(i);
// Do not restore rsp, simply pop the value into the next register
// and overwrite this afterwards.
if (r == rsp) {
DCHECK_GT(i, 0);
r = Register::from_code(i - 1);
}
__ popq(r);
}
__ movb(__ ExternalReferenceAsOperand(
ExternalReference::stack_is_iterable_address(isolate)),
Immediate(1));
// Return to the continuation point.
__ ret(0);
}
} // namespace
void Builtins::Generate_DeoptimizationEntry_Eager(MacroAssembler* masm) {
Generate_DeoptimizationEntry(masm, DeoptimizeKind::kEager);
}
void Builtins::Generate_DeoptimizationEntry_Soft(MacroAssembler* masm) {
Generate_DeoptimizationEntry(masm, DeoptimizeKind::kSoft);
}
void Builtins::Generate_DeoptimizationEntry_Bailout(MacroAssembler* masm) {
Generate_DeoptimizationEntry(masm, DeoptimizeKind::kBailout);
}
void Builtins::Generate_DeoptimizationEntry_Lazy(MacroAssembler* masm) {
Generate_DeoptimizationEntry(masm, DeoptimizeKind::kLazy);
}
#undef __ #undef __
} // namespace internal } // namespace internal

View File

@ -176,8 +176,8 @@ void TurboAssembler::Jump(Handle<Code> code, RelocInfo::Mode rmode,
// size s.t. pc-relative calls may be used. // size s.t. pc-relative calls may be used.
UseScratchRegisterScope temps(this); UseScratchRegisterScope temps(this);
Register scratch = temps.Acquire(); Register scratch = temps.Acquire();
int offset = IsolateData::builtin_entry_slot_offset( int offset = code->builtin_index() * kSystemPointerSize +
static_cast<Builtins::Name>(code->builtin_index())); IsolateData::builtin_entry_table_offset();
ldr(scratch, MemOperand(kRootRegister, offset)); ldr(scratch, MemOperand(kRootRegister, offset));
Jump(scratch, cond); Jump(scratch, cond);
return; return;
@ -266,8 +266,8 @@ void TurboAssembler::Call(Handle<Code> code, RelocInfo::Mode rmode,
// This branch is taken only for specific cctests, where we force isolate // This branch is taken only for specific cctests, where we force isolate
// creation at runtime. At this point, Code space isn't restricted to a // creation at runtime. At this point, Code space isn't restricted to a
// size s.t. pc-relative calls may be used. // size s.t. pc-relative calls may be used.
int offset = IsolateData::builtin_entry_slot_offset( int offset = code->builtin_index() * kSystemPointerSize +
static_cast<Builtins::Name>(code->builtin_index())); IsolateData::builtin_entry_table_offset();
ldr(ip, MemOperand(kRootRegister, offset)); ldr(ip, MemOperand(kRootRegister, offset));
Call(ip, cond); Call(ip, cond);
return; return;
@ -2485,18 +2485,26 @@ void TurboAssembler::ResetSpeculationPoisonRegister() {
mov(kSpeculationPoisonRegister, Operand(-1)); mov(kSpeculationPoisonRegister, Operand(-1));
} }
void TurboAssembler::CallForDeoptimization(Builtins::Name target, int, void TurboAssembler::CallForDeoptimization(Address target, int deopt_id,
Label* exit, DeoptimizeKind kind, Label* exit, DeoptimizeKind kind) {
Label*) {
BlockConstPoolScope block_const_pool(this);
ldr(ip, MemOperand(kRootRegister,
IsolateData::builtin_entry_slot_offset(target)));
Call(ip);
DCHECK_EQ(SizeOfCodeGeneratedSince(exit),
(kind == DeoptimizeKind::kLazy)
? Deoptimizer::kLazyDeoptExitSize
: Deoptimizer::kNonLazyDeoptExitSize);
USE(exit, kind); USE(exit, kind);
NoRootArrayScope no_root_array(this);
// Save the deopt id in r10 (we don't need the roots array from now on).
DCHECK_LE(deopt_id, 0xFFFF);
if (CpuFeatures::IsSupported(ARMv7)) {
// On ARMv7, we can use movw (with a maximum immediate of 0xFFFF)
movw(r10, deopt_id);
} else {
// On ARMv6, we might need two instructions.
mov(r10, Operand(deopt_id & 0xFF)); // Set the low byte.
if (deopt_id >= 0xFF) {
orr(r10, r10, Operand(deopt_id & 0xFF00)); // Set the high byte.
}
}
Call(target, RelocInfo::RUNTIME_ENTRY);
CheckConstPool(false, false);
} }
void TurboAssembler::Trap() { stop(); } void TurboAssembler::Trap() { stop(); }

View File

@ -320,9 +320,10 @@ class V8_EXPORT_PRIVATE TurboAssembler : public TurboAssemblerBase {
// The return address on the stack is used by frame iteration. // The return address on the stack is used by frame iteration.
void StoreReturnAddressAndCall(Register target); void StoreReturnAddressAndCall(Register target);
void CallForDeoptimization(Builtins::Name target, int deopt_id, Label* exit, // This should only be used when assembling a deoptimizer call because of
DeoptimizeKind kind, // the CheckConstPool invocation, which is only needed for deoptimization.
Label* jump_deoptimization_entry_label); void CallForDeoptimization(Address target, int deopt_id, Label* exit,
DeoptimizeKind kind);
// Emit code to discard a non-negative number of pointer-sized elements // Emit code to discard a non-negative number of pointer-sized elements
// from the stack, clobbering only the sp register. // from the stack, clobbering only the sp register.

View File

@ -1878,13 +1878,6 @@ void TurboAssembler::LoadEntryFromBuiltinIndex(Register builtin_index) {
} }
} }
void TurboAssembler::LoadEntryFromBuiltinIndex(Builtins::Name builtin_index,
Register destination) {
Ldr(destination,
MemOperand(kRootRegister,
IsolateData::builtin_entry_slot_offset(builtin_index)));
}
void TurboAssembler::CallBuiltinByIndex(Register builtin_index) { void TurboAssembler::CallBuiltinByIndex(Register builtin_index) {
LoadEntryFromBuiltinIndex(builtin_index); LoadEntryFromBuiltinIndex(builtin_index);
Call(builtin_index); Call(builtin_index);
@ -2010,11 +2003,15 @@ bool TurboAssembler::IsNearCallOffset(int64_t offset) {
return is_int26(offset); return is_int26(offset);
} }
void TurboAssembler::CallForDeoptimization( void TurboAssembler::CallForDeoptimization(Address target, int deopt_id,
Builtins::Name target, int deopt_id, Label* exit, DeoptimizeKind kind, Label* exit, DeoptimizeKind kind) {
Label* jump_deoptimization_entry_label) {
BlockPoolsScope scope(this); BlockPoolsScope scope(this);
bl(jump_deoptimization_entry_label); int64_t offset = static_cast<int64_t>(target) -
static_cast<int64_t>(options().code_range_start);
DCHECK_EQ(offset % kInstrSize, 0);
offset = offset / static_cast<int>(kInstrSize);
DCHECK(IsNearCallOffset(offset));
near_call(static_cast<int>(offset), RelocInfo::RUNTIME_ENTRY);
DCHECK_EQ(SizeOfCodeGeneratedSince(exit), DCHECK_EQ(SizeOfCodeGeneratedSince(exit),
(kind == DeoptimizeKind::kLazy) (kind == DeoptimizeKind::kLazy)
? Deoptimizer::kLazyDeoptExitSize ? Deoptimizer::kLazyDeoptExitSize

View File

@ -968,8 +968,6 @@ class V8_EXPORT_PRIVATE TurboAssembler : public TurboAssemblerBase {
// Load the builtin given by the Smi in |builtin_index| into the same // Load the builtin given by the Smi in |builtin_index| into the same
// register. // register.
void LoadEntryFromBuiltinIndex(Register builtin_index); void LoadEntryFromBuiltinIndex(Register builtin_index);
void LoadEntryFromBuiltinIndex(Builtins::Name builtin_index,
Register destination);
void CallBuiltinByIndex(Register builtin_index) override; void CallBuiltinByIndex(Register builtin_index) override;
void CallBuiltin(int builtin_index); void CallBuiltin(int builtin_index);
@ -982,9 +980,8 @@ class V8_EXPORT_PRIVATE TurboAssembler : public TurboAssemblerBase {
// The return address on the stack is used by frame iteration. // The return address on the stack is used by frame iteration.
void StoreReturnAddressAndCall(Register target); void StoreReturnAddressAndCall(Register target);
void CallForDeoptimization(Builtins::Name target, int deopt_id, Label* exit, void CallForDeoptimization(Address target, int deopt_id, Label* exit,
DeoptimizeKind kind, DeoptimizeKind kind);
Label* jump_deoptimization_entry_label);
// Calls a C function. // Calls a C function.
// The called function is not allowed to trigger a // The called function is not allowed to trigger a

View File

@ -2074,15 +2074,13 @@ void TurboAssembler::ComputeCodeStartAddress(Register dst) {
} }
} }
void TurboAssembler::CallForDeoptimization(Builtins::Name target, int, void TurboAssembler::CallForDeoptimization(Address target, int deopt_id,
Label* exit, DeoptimizeKind kind, Label* exit, DeoptimizeKind kind) {
Label*) {
CallBuiltin(target);
DCHECK_EQ(SizeOfCodeGeneratedSince(exit),
(kind == DeoptimizeKind::kLazy)
? Deoptimizer::kLazyDeoptExitSize
: Deoptimizer::kNonLazyDeoptExitSize);
USE(exit, kind); USE(exit, kind);
NoRootArrayScope no_root_array(this);
// Save the deopt id in ebx (we don't need the roots array from now on).
mov(ebx, deopt_id);
call(target, RelocInfo::RUNTIME_ENTRY);
} }
void TurboAssembler::Trap() { int3(); } void TurboAssembler::Trap() { int3(); }

View File

@ -130,9 +130,8 @@ class V8_EXPORT_PRIVATE TurboAssembler : public TurboAssemblerBase {
void Trap() override; void Trap() override;
void DebugBreak() override; void DebugBreak() override;
void CallForDeoptimization(Builtins::Name target, int deopt_id, Label* exit, void CallForDeoptimization(Address target, int deopt_id, Label* exit,
DeoptimizeKind kind, DeoptimizeKind kind);
Label* jump_deoptimization_entry_label);
// Jump the register contains a smi. // Jump the register contains a smi.
inline void JumpIfSmi(Register value, Label* smi_label, inline void JumpIfSmi(Register value, Label* smi_label,

View File

@ -593,12 +593,6 @@ using DummyDescriptor = VoidDescriptor;
// Dummy descriptor that marks builtins with C calling convention. // Dummy descriptor that marks builtins with C calling convention.
using CCallDescriptor = VoidDescriptor; using CCallDescriptor = VoidDescriptor;
// Marks deoptimization entry builtins. Precise calling conventions currently
// differ based on the platform.
// TODO(jgruber): Once this is unified, we could create a better description
// here.
using DeoptimizationEntryDescriptor = VoidDescriptor;
class AllocateDescriptor : public CallInterfaceDescriptor { class AllocateDescriptor : public CallInterfaceDescriptor {
public: public:
DEFINE_PARAMETERS_NO_CONTEXT(kRequestedSize) DEFINE_PARAMETERS_NO_CONTEXT(kRequestedSize)

View File

@ -99,7 +99,7 @@ void OptimizedCompilationInfo::ConfigureFlags() {
if (FLAG_turbo_splitting) set_splitting(); if (FLAG_turbo_splitting) set_splitting();
break; break;
case CodeKind::BUILTIN: case CodeKind::BUILTIN:
case CodeKind::FOR_TESTING: case CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING:
if (FLAG_turbo_splitting) set_splitting(); if (FLAG_turbo_splitting) set_splitting();
#if ENABLE_GDB_JIT_INTERFACE && DEBUG #if ENABLE_GDB_JIT_INTERFACE && DEBUG
set_source_positions(); set_source_positions();
@ -161,7 +161,7 @@ std::unique_ptr<char[]> OptimizedCompilationInfo::GetDebugName() const {
StackFrame::Type OptimizedCompilationInfo::GetOutputStackFrameType() const { StackFrame::Type OptimizedCompilationInfo::GetOutputStackFrameType() const {
switch (code_kind()) { switch (code_kind()) {
case CodeKind::FOR_TESTING: case CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING:
case CodeKind::BYTECODE_HANDLER: case CodeKind::BYTECODE_HANDLER:
case CodeKind::BUILTIN: case CodeKind::BUILTIN:
return StackFrame::STUB; return StackFrame::STUB;

View File

@ -150,6 +150,9 @@ class V8_EXPORT_PRIVATE OptimizedCompilationInfo final {
return code_kind() == CodeKind::NATIVE_CONTEXT_INDEPENDENT; return code_kind() == CodeKind::NATIVE_CONTEXT_INDEPENDENT;
} }
bool IsTurboprop() const { return code_kind() == CodeKind::TURBOPROP; } bool IsTurboprop() const { return code_kind() == CodeKind::TURBOPROP; }
bool IsStub() const {
return code_kind() == CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING;
}
bool IsWasm() const { return code_kind() == CodeKind::WASM_FUNCTION; } bool IsWasm() const { return code_kind() == CodeKind::WASM_FUNCTION; }
void SetOptimizingForOsr(BailoutId osr_offset, JavaScriptFrame* osr_frame) { void SetOptimizingForOsr(BailoutId osr_offset, JavaScriptFrame* osr_frame) {

View File

@ -476,7 +476,7 @@ void RelocInfo::Print(Isolate* isolate, std::ostream& os) { // NOLINT
os << " " << Builtins::name(code.builtin_index()); os << " " << Builtins::name(code.builtin_index());
} }
os << ") (" << reinterpret_cast<const void*>(target_address()) << ")"; os << ") (" << reinterpret_cast<const void*>(target_address()) << ")";
} else if (IsRuntimeEntry(rmode_)) { } else if (IsRuntimeEntry(rmode_) && isolate->deoptimizer_data() != nullptr) {
// Deoptimization bailouts are stored as runtime entries. // Deoptimization bailouts are stored as runtime entries.
DeoptimizeKind type; DeoptimizeKind type;
if (Deoptimizer::IsDeoptimizationEntry(isolate, target_address(), &type)) { if (Deoptimizer::IsDeoptimizationEntry(isolate, target_address(), &type)) {

View File

@ -1589,13 +1589,6 @@ void TurboAssembler::Call(Handle<Code> code_object, RelocInfo::Mode rmode) {
call(code_object, rmode); call(code_object, rmode);
} }
Operand TurboAssembler::EntryFromBuiltinIndexAsOperand(
Builtins::Name builtin_index) {
DCHECK(root_array_available());
return Operand(kRootRegister,
IsolateData::builtin_entry_slot_offset(builtin_index));
}
Operand TurboAssembler::EntryFromBuiltinIndexAsOperand(Register builtin_index) { Operand TurboAssembler::EntryFromBuiltinIndexAsOperand(Register builtin_index) {
if (SmiValuesAre32Bits()) { if (SmiValuesAre32Bits()) {
// The builtin_index register contains the builtin index as a Smi. // The builtin_index register contains the builtin index as a Smi.
@ -2832,18 +2825,13 @@ void TurboAssembler::ResetSpeculationPoisonRegister() {
Set(kSpeculationPoisonRegister, -1); Set(kSpeculationPoisonRegister, -1);
} }
void TurboAssembler::CallForDeoptimization(Builtins::Name target, int, void TurboAssembler::CallForDeoptimization(Address target, int deopt_id,
Label* exit, DeoptimizeKind kind, Label* exit, DeoptimizeKind kind) {
Label*) {
// Note: Assembler::call is used here on purpose to guarantee fixed-size
// exits even on Atom CPUs; see TurboAssembler::Call for Atom-specific
// performance tuning which emits a different instruction sequence.
call(EntryFromBuiltinIndexAsOperand(target));
DCHECK_EQ(SizeOfCodeGeneratedSince(exit),
(kind == DeoptimizeKind::kLazy)
? Deoptimizer::kLazyDeoptExitSize
: Deoptimizer::kNonLazyDeoptExitSize);
USE(exit, kind); USE(exit, kind);
NoRootArrayScope no_root_array(this);
// Save the deopt id in r13 (we don't need the roots array from now on).
movq(r13, Immediate(deopt_id));
call(target, RelocInfo::RUNTIME_ENTRY);
} }
void TurboAssembler::Trap() { int3(); } void TurboAssembler::Trap() { int3(); }

View File

@ -492,7 +492,6 @@ class V8_EXPORT_PRIVATE TurboAssembler : public TurboAssemblerBase {
void Call(ExternalReference ext); void Call(ExternalReference ext);
void Call(Label* target) { call(target); } void Call(Label* target) { call(target); }
Operand EntryFromBuiltinIndexAsOperand(Builtins::Name builtin_index);
Operand EntryFromBuiltinIndexAsOperand(Register builtin_index); Operand EntryFromBuiltinIndexAsOperand(Register builtin_index);
void CallBuiltinByIndex(Register builtin_index) override; void CallBuiltinByIndex(Register builtin_index) override;
void CallBuiltin(int builtin_index); void CallBuiltin(int builtin_index);
@ -512,9 +511,8 @@ class V8_EXPORT_PRIVATE TurboAssembler : public TurboAssemblerBase {
void RetpolineJump(Register reg); void RetpolineJump(Register reg);
void CallForDeoptimization(Builtins::Name target, int deopt_id, Label* exit, void CallForDeoptimization(Address target, int deopt_id, Label* exit,
DeoptimizeKind kind, DeoptimizeKind kind);
Label* jump_deoptimization_entry_label);
void Trap() override; void Trap() override;
void DebugBreak() override; void DebugBreak() override;

View File

@ -471,11 +471,8 @@ enum class DeoptimizeKind : uint8_t {
kSoft, kSoft,
kBailout, kBailout,
kLazy, kLazy,
kLastDeoptimizeKind = kLazy
}; };
constexpr DeoptimizeKind kFirstDeoptimizeKind = DeoptimizeKind::kEager;
constexpr DeoptimizeKind kLastDeoptimizeKind = DeoptimizeKind::kLazy;
STATIC_ASSERT(static_cast<int>(kFirstDeoptimizeKind) == 0);
constexpr int kDeoptimizeKindCount = static_cast<int>(kLastDeoptimizeKind) + 1;
inline size_t hash_value(DeoptimizeKind kind) { inline size_t hash_value(DeoptimizeKind kind) {
return static_cast<size_t>(kind); return static_cast<size_t>(kind);
} }
@ -490,6 +487,7 @@ inline std::ostream& operator<<(std::ostream& os, DeoptimizeKind kind) {
case DeoptimizeKind::kBailout: case DeoptimizeKind::kBailout:
return os << "Bailout"; return os << "Bailout";
} }
UNREACHABLE();
} }
// Indicates whether the lookup is related to sloppy-mode block-scoped // Indicates whether the lookup is related to sloppy-mode block-scoped

View File

@ -962,7 +962,9 @@ CodeGenerator::CodeGenResult CodeGenerator::AssembleArchInstruction(
case kArchDeoptimize: { case kArchDeoptimize: {
DeoptimizationExit* exit = DeoptimizationExit* exit =
BuildTranslation(instr, -1, 0, OutputFrameStateCombine::Ignore()); BuildTranslation(instr, -1, 0, OutputFrameStateCombine::Ignore());
__ b(exit->label()); CodeGenResult result = AssembleDeoptimizerCall(exit);
if (result != kSuccess) return result;
unwinding_info_writer_.MarkBlockWillExit();
break; break;
} }
case kArchRet: case kArchRet:
@ -3815,10 +3817,7 @@ void CodeGenerator::AssembleReturn(InstructionOperand* pop) {
void CodeGenerator::FinishCode() { __ CheckConstPool(true, false); } void CodeGenerator::FinishCode() { __ CheckConstPool(true, false); }
void CodeGenerator::PrepareForDeoptimizationExits( void CodeGenerator::PrepareForDeoptimizationExits(int deopt_count) {}
ZoneDeque<DeoptimizationExit*>* exits) {
__ CheckConstPool(true, false);
}
void CodeGenerator::AssembleMove(InstructionOperand* source, void CodeGenerator::AssembleMove(InstructionOperand* source,
InstructionOperand* destination) { InstructionOperand* destination) {

View File

@ -3135,35 +3135,13 @@ void CodeGenerator::AssembleReturn(InstructionOperand* pop) {
void CodeGenerator::FinishCode() { __ ForceConstantPoolEmissionWithoutJump(); } void CodeGenerator::FinishCode() { __ ForceConstantPoolEmissionWithoutJump(); }
void CodeGenerator::PrepareForDeoptimizationExits( void CodeGenerator::PrepareForDeoptimizationExits(int deopt_count) {
ZoneDeque<DeoptimizationExit*>* exits) {
__ ForceConstantPoolEmissionWithoutJump(); __ ForceConstantPoolEmissionWithoutJump();
// We are conservative here, assuming all deopts are lazy deopts. // We are conservative here, assuming all deopts are lazy deopts.
DCHECK_GE(Deoptimizer::kLazyDeoptExitSize, DCHECK_GE(Deoptimizer::kLazyDeoptExitSize,
Deoptimizer::kNonLazyDeoptExitSize); Deoptimizer::kNonLazyDeoptExitSize);
__ CheckVeneerPool( __ CheckVeneerPool(false, false,
false, false, deopt_count * Deoptimizer::kLazyDeoptExitSize);
static_cast<int>(exits->size()) * Deoptimizer::kLazyDeoptExitSize);
// Check which deopt kinds exist in this Code object, to avoid emitting jumps
// to unused entries.
bool saw_deopt_kind[kDeoptimizeKindCount] = {false};
for (auto exit : *exits) {
saw_deopt_kind[static_cast<int>(exit->kind())] = true;
}
// Emit the jumps to deoptimization entries.
UseScratchRegisterScope scope(tasm());
Register scratch = scope.AcquireX();
STATIC_ASSERT(static_cast<int>(kFirstDeoptimizeKind) == 0);
for (int i = 0; i < kDeoptimizeKindCount; i++) {
if (!saw_deopt_kind[i]) continue;
__ bind(&jump_deoptimization_entry_labels_[i]);
__ LoadEntryFromBuiltinIndex(Deoptimizer::GetDeoptimizationEntry(
isolate(), static_cast<DeoptimizeKind>(i)),
scratch);
__ Jump(scratch);
}
} }
void CodeGenerator::AssembleMove(InstructionOperand* source, void CodeGenerator::AssembleMove(InstructionOperand* source,

View File

@ -162,10 +162,8 @@ CodeGenerator::CodeGenResult CodeGenerator::AssembleDeoptimizerCall(
DeoptimizeKind deopt_kind = exit->kind(); DeoptimizeKind deopt_kind = exit->kind();
DeoptimizeReason deoptimization_reason = exit->reason(); DeoptimizeReason deoptimization_reason = exit->reason();
Builtins::Name deopt_entry = Address deopt_entry =
Deoptimizer::GetDeoptimizationEntry(tasm()->isolate(), deopt_kind); Deoptimizer::GetDeoptimizationEntry(tasm()->isolate(), deopt_kind);
Label* jump_deoptimization_entry_label =
&jump_deoptimization_entry_labels_[static_cast<int>(deopt_kind)];
if (info()->source_positions()) { if (info()->source_positions()) {
tasm()->RecordDeoptReason(deoptimization_reason, exit->pos(), tasm()->RecordDeoptReason(deoptimization_reason, exit->pos(),
deoptimization_id); deoptimization_id);
@ -179,7 +177,7 @@ CodeGenerator::CodeGenResult CodeGenerator::AssembleDeoptimizerCall(
} }
tasm()->CallForDeoptimization(deopt_entry, deoptimization_id, exit->label(), tasm()->CallForDeoptimization(deopt_entry, deoptimization_id, exit->label(),
deopt_kind, jump_deoptimization_entry_label); deopt_kind);
exit->set_emitted(); exit->set_emitted();
return kSuccess; return kSuccess;
} }
@ -326,7 +324,7 @@ void CodeGenerator::AssembleCode() {
// For some targets, we must make sure that constant and veneer pools are // For some targets, we must make sure that constant and veneer pools are
// emitted before emitting the deoptimization exits. // emitted before emitting the deoptimization exits.
PrepareForDeoptimizationExits(&deoptimization_exits_); PrepareForDeoptimizationExits(static_cast<int>(deoptimization_exits_.size()));
if (Deoptimizer::kSupportsFixedDeoptExitSizes) { if (Deoptimizer::kSupportsFixedDeoptExitSizes) {
deopt_exit_start_offset_ = tasm()->pc_offset(); deopt_exit_start_offset_ = tasm()->pc_offset();
@ -340,7 +338,7 @@ void CodeGenerator::AssembleCode() {
// Deoptimizer::kSupportsFixedDeoptExitSizes is true, lazy deopts // Deoptimizer::kSupportsFixedDeoptExitSizes is true, lazy deopts
// might need additional instructions. // might need additional instructions.
auto cmp = [](const DeoptimizationExit* a, const DeoptimizationExit* b) { auto cmp = [](const DeoptimizationExit* a, const DeoptimizationExit* b) {
static_assert(DeoptimizeKind::kLazy == kLastDeoptimizeKind, static_assert(DeoptimizeKind::kLazy == DeoptimizeKind::kLastDeoptimizeKind,
"lazy deopts are expected to be emitted last"); "lazy deopts are expected to be emitted last");
if (a->kind() != b->kind()) { if (a->kind() != b->kind()) {
return a->kind() < b->kind(); return a->kind() < b->kind();

View File

@ -406,7 +406,7 @@ class V8_EXPORT_PRIVATE CodeGenerator final : public GapResolver::Assembler {
InstructionOperand* op, MachineType type); InstructionOperand* op, MachineType type);
void MarkLazyDeoptSite(); void MarkLazyDeoptSite();
void PrepareForDeoptimizationExits(ZoneDeque<DeoptimizationExit*>* exits); void PrepareForDeoptimizationExits(int deopt_count);
DeoptimizationExit* AddDeoptimizationExit(Instruction* instr, DeoptimizationExit* AddDeoptimizationExit(Instruction* instr,
size_t frame_state_offset); size_t frame_state_offset);
@ -446,14 +446,6 @@ class V8_EXPORT_PRIVATE CodeGenerator final : public GapResolver::Assembler {
int handler_table_offset_ = 0; int handler_table_offset_ = 0;
int last_lazy_deopt_pc_ = 0; int last_lazy_deopt_pc_ = 0;
// Deoptimization exits must be as small as possible, since their count grows
// with function size. {jump_deoptimization_entry_labels_} is an optimization
// to that effect, which extracts the (potentially large) instruction
// sequence for the final jump to the deoptimization entry into a single spot
// per Code object. All deopt exits can then near-call to this label. Note:
// not used on all architectures.
Label jump_deoptimization_entry_labels_[kDeoptimizeKindCount];
// The maximal combined height of all frames produced upon deoptimization, and // The maximal combined height of all frames produced upon deoptimization, and
// the maximal number of pushed arguments for function calls. Applied as an // the maximal number of pushed arguments for function calls. Applied as an
// offset to the first stack check of an optimized function. // offset to the first stack check of an optimized function.

View File

@ -927,7 +927,8 @@ CodeGenerator::CodeGenResult CodeGenerator::AssembleArchInstruction(
case kArchDeoptimize: { case kArchDeoptimize: {
DeoptimizationExit* exit = DeoptimizationExit* exit =
BuildTranslation(instr, -1, 0, OutputFrameStateCombine::Ignore()); BuildTranslation(instr, -1, 0, OutputFrameStateCombine::Ignore());
__ jmp(exit->label()); CodeGenResult result = AssembleDeoptimizerCall(exit);
if (result != kSuccess) return result;
break; break;
} }
case kArchRet: case kArchRet:
@ -4843,8 +4844,7 @@ void CodeGenerator::AssembleReturn(InstructionOperand* pop) {
void CodeGenerator::FinishCode() {} void CodeGenerator::FinishCode() {}
void CodeGenerator::PrepareForDeoptimizationExits( void CodeGenerator::PrepareForDeoptimizationExits(int deopt_count) {}
ZoneDeque<DeoptimizationExit*>* exits) {}
void CodeGenerator::AssembleMove(InstructionOperand* source, void CodeGenerator::AssembleMove(InstructionOperand* source,
InstructionOperand* destination) { InstructionOperand* destination) {

View File

@ -1099,7 +1099,9 @@ CodeGenerator::CodeGenResult CodeGenerator::AssembleArchInstruction(
case kArchDeoptimize: { case kArchDeoptimize: {
DeoptimizationExit* exit = DeoptimizationExit* exit =
BuildTranslation(instr, -1, 0, OutputFrameStateCombine::Ignore()); BuildTranslation(instr, -1, 0, OutputFrameStateCombine::Ignore());
__ jmp(exit->label()); CodeGenResult result = AssembleDeoptimizerCall(exit);
if (result != kSuccess) return result;
unwinding_info_writer_.MarkBlockWillExit();
break; break;
} }
case kArchRet: case kArchRet:
@ -4577,8 +4579,7 @@ void CodeGenerator::AssembleReturn(InstructionOperand* additional_pop_count) {
void CodeGenerator::FinishCode() { tasm()->PatchConstPool(); } void CodeGenerator::FinishCode() { tasm()->PatchConstPool(); }
void CodeGenerator::PrepareForDeoptimizationExits( void CodeGenerator::PrepareForDeoptimizationExits(int deopt_count) {}
ZoneDeque<DeoptimizationExit*>* exits) {}
void CodeGenerator::IncrementStackAccessCounter( void CodeGenerator::IncrementStackAccessCounter(
InstructionOperand* source, InstructionOperand* destination) { InstructionOperand* source, InstructionOperand* destination) {

View File

@ -1164,6 +1164,11 @@ PipelineCompilationJob::Status PipelineCompilationJob::PrepareJobImpl(
if (compilation_info()->is_osr()) data_.InitializeOsrHelper(); if (compilation_info()->is_osr()) data_.InitializeOsrHelper();
// Make sure that we have generated the deopt entries code. This is in order
// to avoid triggering the generation of deopt entries later during code
// assembly.
Deoptimizer::EnsureCodeForDeoptimizationEntries(isolate);
pipeline_.Serialize(); pipeline_.Serialize();
if (!data_.broker()->is_concurrent_inlining()) { if (!data_.broker()->is_concurrent_inlining()) {
@ -3006,6 +3011,7 @@ MaybeHandle<Code> Pipeline::GenerateCodeForTesting(
PipelineImpl pipeline(&data); PipelineImpl pipeline(&data);
Linkage linkage(Linkage::ComputeIncoming(data.instruction_zone(), info)); Linkage linkage(Linkage::ComputeIncoming(data.instruction_zone(), info));
Deoptimizer::EnsureCodeForDeoptimizationEntries(isolate);
{ {
CompilationHandleScope compilation_scope(isolate, info); CompilationHandleScope compilation_scope(isolate, info);
@ -3230,7 +3236,7 @@ bool Pipeline::AllocateRegistersForTesting(const RegisterConfiguration* config,
bool use_mid_tier_register_allocator, bool use_mid_tier_register_allocator,
bool run_verifier) { bool run_verifier) {
OptimizedCompilationInfo info(ArrayVector("testing"), sequence->zone(), OptimizedCompilationInfo info(ArrayVector("testing"), sequence->zone(),
CodeKind::FOR_TESTING); CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING);
ZoneStats zone_stats(sequence->isolate()->allocator()); ZoneStats zone_stats(sequence->isolate()->allocator());
PipelineData data(&zone_stats, &info, sequence->isolate(), sequence); PipelineData data(&zone_stats, &info, sequence->isolate(), sequence);
data.InitializeFrameData(nullptr); data.InitializeFrameData(nullptr);

View File

@ -2,14 +2,246 @@
// Use of this source code is governed by a BSD-style license that can be // Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file. // found in the LICENSE file.
#include "src/codegen/assembler-inl.h"
#include "src/codegen/macro-assembler.h"
#include "src/codegen/register-configuration.h"
#include "src/codegen/safepoint-table.h"
#include "src/deoptimizer/deoptimizer.h" #include "src/deoptimizer/deoptimizer.h"
#include "src/objects/objects-inl.h"
namespace v8 { namespace v8 {
namespace internal { namespace internal {
const bool Deoptimizer::kSupportsFixedDeoptExitSizes = true; const bool Deoptimizer::kSupportsFixedDeoptExitSizes = false;
const int Deoptimizer::kNonLazyDeoptExitSize = 2 * kInstrSize; const int Deoptimizer::kNonLazyDeoptExitSize = 0;
const int Deoptimizer::kLazyDeoptExitSize = 2 * kInstrSize; const int Deoptimizer::kLazyDeoptExitSize = 0;
#define __ masm->
// This code tries to be close to ia32 code so that any changes can be
// easily ported.
void Deoptimizer::GenerateDeoptimizationEntries(MacroAssembler* masm,
Isolate* isolate,
DeoptimizeKind deopt_kind) {
NoRootArrayScope no_root_array(masm);
// Save all general purpose registers before messing with them.
const int kNumberOfRegisters = Register::kNumRegisters;
// Everything but pc, lr and ip which will be saved but not restored.
RegList restored_regs = kJSCallerSaved | kCalleeSaved | ip.bit();
const int kDoubleRegsSize = kDoubleSize * DwVfpRegister::kNumRegisters;
// Save all allocatable VFP registers before messing with them.
{
// We use a run-time check for VFP32DREGS.
CpuFeatureScope scope(masm, VFP32DREGS,
CpuFeatureScope::kDontCheckSupported);
UseScratchRegisterScope temps(masm);
Register scratch = temps.Acquire();
// Check CPU flags for number of registers, setting the Z condition flag.
__ CheckFor32DRegs(scratch);
// Push registers d0-d15, and possibly d16-d31, on the stack.
// If d16-d31 are not pushed, decrease the stack pointer instead.
__ vstm(db_w, sp, d16, d31, ne);
// Okay to not call AllocateStackSpace here because the size is a known
// small number and we need to use condition codes.
__ sub(sp, sp, Operand(16 * kDoubleSize), LeaveCC, eq);
__ vstm(db_w, sp, d0, d15);
}
// Push all 16 registers (needed to populate FrameDescription::registers_).
// TODO(1588) Note that using pc with stm is deprecated, so we should perhaps
// handle this a bit differently.
__ stm(db_w, sp, restored_regs | sp.bit() | lr.bit() | pc.bit());
{
UseScratchRegisterScope temps(masm);
Register scratch = temps.Acquire();
__ mov(scratch, Operand(ExternalReference::Create(
IsolateAddressId::kCEntryFPAddress, isolate)));
__ str(fp, MemOperand(scratch));
}
const int kSavedRegistersAreaSize =
(kNumberOfRegisters * kPointerSize) + kDoubleRegsSize;
// Get the bailout id is passed as r10 by the caller.
__ mov(r2, r10);
// Get the address of the location in the code object (r3) (return
// address for lazy deoptimization) and compute the fp-to-sp delta in
// register r4.
__ mov(r3, lr);
__ add(r4, sp, Operand(kSavedRegistersAreaSize));
__ sub(r4, fp, r4);
// Allocate a new deoptimizer object.
// Pass four arguments in r0 to r3 and fifth argument on stack.
__ PrepareCallCFunction(6);
__ mov(r0, Operand(0));
Label context_check;
__ ldr(r1, MemOperand(fp, CommonFrameConstants::kContextOrFrameTypeOffset));
__ JumpIfSmi(r1, &context_check);
__ ldr(r0, MemOperand(fp, StandardFrameConstants::kFunctionOffset));
__ bind(&context_check);
__ mov(r1, Operand(static_cast<int>(deopt_kind)));
// r2: bailout id already loaded.
// r3: code address or 0 already loaded.
__ str(r4, MemOperand(sp, 0 * kPointerSize)); // Fp-to-sp delta.
__ mov(r5, Operand(ExternalReference::isolate_address(isolate)));
__ str(r5, MemOperand(sp, 1 * kPointerSize)); // Isolate.
// Call Deoptimizer::New().
{
AllowExternalCallThatCantCauseGC scope(masm);
__ CallCFunction(ExternalReference::new_deoptimizer_function(), 6);
}
// Preserve "deoptimizer" object in register r0 and get the input
// frame descriptor pointer to r1 (deoptimizer->input_);
__ ldr(r1, MemOperand(r0, Deoptimizer::input_offset()));
// Copy core registers into FrameDescription::registers_[kNumRegisters].
DCHECK_EQ(Register::kNumRegisters, kNumberOfRegisters);
for (int i = 0; i < kNumberOfRegisters; i++) {
int offset = (i * kPointerSize) + FrameDescription::registers_offset();
__ ldr(r2, MemOperand(sp, i * kPointerSize));
__ str(r2, MemOperand(r1, offset));
}
// Copy VFP registers to
// double_registers_[DoubleRegister::kNumAllocatableRegisters]
int double_regs_offset = FrameDescription::double_registers_offset();
const RegisterConfiguration* config = RegisterConfiguration::Default();
for (int i = 0; i < config->num_allocatable_double_registers(); ++i) {
int code = config->GetAllocatableDoubleCode(i);
int dst_offset = code * kDoubleSize + double_regs_offset;
int src_offset = code * kDoubleSize + kNumberOfRegisters * kPointerSize;
__ vldr(d0, sp, src_offset);
__ vstr(d0, r1, dst_offset);
}
// Mark the stack as not iterable for the CPU profiler which won't be able to
// walk the stack without the return address.
{
UseScratchRegisterScope temps(masm);
Register is_iterable = temps.Acquire();
Register zero = r4;
__ Move(is_iterable, ExternalReference::stack_is_iterable_address(isolate));
__ mov(zero, Operand(0));
__ strb(zero, MemOperand(is_iterable));
}
// Remove the saved registers from the stack.
__ add(sp, sp, Operand(kSavedRegistersAreaSize));
// Compute a pointer to the unwinding limit in register r2; that is
// the first stack slot not part of the input frame.
__ ldr(r2, MemOperand(r1, FrameDescription::frame_size_offset()));
__ add(r2, r2, sp);
// Unwind the stack down to - but not including - the unwinding
// limit and copy the contents of the activation frame to the input
// frame description.
__ add(r3, r1, Operand(FrameDescription::frame_content_offset()));
Label pop_loop;
Label pop_loop_header;
__ b(&pop_loop_header);
__ bind(&pop_loop);
__ pop(r4);
__ str(r4, MemOperand(r3, 0));
__ add(r3, r3, Operand(sizeof(uint32_t)));
__ bind(&pop_loop_header);
__ cmp(r2, sp);
__ b(ne, &pop_loop);
// Compute the output frame in the deoptimizer.
__ push(r0); // Preserve deoptimizer object across call.
// r0: deoptimizer object; r1: scratch.
__ PrepareCallCFunction(1);
// Call Deoptimizer::ComputeOutputFrames().
{
AllowExternalCallThatCantCauseGC scope(masm);
__ CallCFunction(ExternalReference::compute_output_frames_function(), 1);
}
__ pop(r0); // Restore deoptimizer object (class Deoptimizer).
__ ldr(sp, MemOperand(r0, Deoptimizer::caller_frame_top_offset()));
// Replace the current (input) frame with the output frames.
Label outer_push_loop, inner_push_loop, outer_loop_header, inner_loop_header;
// Outer loop state: r4 = current "FrameDescription** output_",
// r1 = one past the last FrameDescription**.
__ ldr(r1, MemOperand(r0, Deoptimizer::output_count_offset()));
__ ldr(r4, MemOperand(r0, Deoptimizer::output_offset())); // r4 is output_.
__ add(r1, r4, Operand(r1, LSL, 2));
__ jmp(&outer_loop_header);
__ bind(&outer_push_loop);
// Inner loop state: r2 = current FrameDescription*, r3 = loop index.
__ ldr(r2, MemOperand(r4, 0)); // output_[ix]
__ ldr(r3, MemOperand(r2, FrameDescription::frame_size_offset()));
__ jmp(&inner_loop_header);
__ bind(&inner_push_loop);
__ sub(r3, r3, Operand(sizeof(uint32_t)));
__ add(r6, r2, Operand(r3));
__ ldr(r6, MemOperand(r6, FrameDescription::frame_content_offset()));
__ push(r6);
__ bind(&inner_loop_header);
__ cmp(r3, Operand::Zero());
__ b(ne, &inner_push_loop); // test for gt?
__ add(r4, r4, Operand(kPointerSize));
__ bind(&outer_loop_header);
__ cmp(r4, r1);
__ b(lt, &outer_push_loop);
__ ldr(r1, MemOperand(r0, Deoptimizer::input_offset()));
for (int i = 0; i < config->num_allocatable_double_registers(); ++i) {
int code = config->GetAllocatableDoubleCode(i);
DwVfpRegister reg = DwVfpRegister::from_code(code);
int src_offset = code * kDoubleSize + double_regs_offset;
__ vldr(reg, r1, src_offset);
}
// Push pc and continuation from the last output frame.
__ ldr(r6, MemOperand(r2, FrameDescription::pc_offset()));
__ push(r6);
__ ldr(r6, MemOperand(r2, FrameDescription::continuation_offset()));
__ push(r6);
// Push the registers from the last output frame.
for (int i = kNumberOfRegisters - 1; i >= 0; i--) {
int offset = (i * kPointerSize) + FrameDescription::registers_offset();
__ ldr(r6, MemOperand(r2, offset));
__ push(r6);
}
// Restore the registers from the stack.
__ ldm(ia_w, sp, restored_regs); // all but pc registers.
{
UseScratchRegisterScope temps(masm);
Register is_iterable = temps.Acquire();
Register one = r4;
__ Move(is_iterable, ExternalReference::stack_is_iterable_address(isolate));
__ mov(one, Operand(1));
__ strb(one, MemOperand(is_iterable));
}
// Remove sp, lr and pc.
__ Drop(3);
{
UseScratchRegisterScope temps(masm);
Register scratch = temps.Acquire();
__ pop(scratch); // get continuation, leave pc on stack
__ pop(lr);
__ Jump(scratch);
}
__ stop();
}
Float32 RegisterValues::GetFloatRegister(unsigned n) const { Float32 RegisterValues::GetFloatRegister(unsigned n) const {
const int kShift = n % 2 == 0 ? 0 : 32; const int kShift = n % 2 == 0 ? 0 : 32;
@ -33,5 +265,7 @@ void FrameDescription::SetCallerConstantPool(unsigned offset, intptr_t value) {
void FrameDescription::SetPc(intptr_t pc) { pc_ = pc; } void FrameDescription::SetPc(intptr_t pc) { pc_ = pc; }
#undef __
} // namespace internal } // namespace internal
} // namespace v8 } // namespace v8

View File

@ -3,7 +3,12 @@
// found in the LICENSE file. // found in the LICENSE file.
#include "src/api/api.h" #include "src/api/api.h"
#include "src/codegen/arm64/assembler-arm64-inl.h"
#include "src/codegen/arm64/macro-assembler-arm64-inl.h"
#include "src/codegen/register-configuration.h"
#include "src/codegen/safepoint-table.h"
#include "src/deoptimizer/deoptimizer.h" #include "src/deoptimizer/deoptimizer.h"
#include "src/execution/frame-constants.h"
#include "src/execution/pointer-authentication.h" #include "src/execution/pointer-authentication.h"
namespace v8 { namespace v8 {
@ -17,6 +22,286 @@ const int Deoptimizer::kLazyDeoptExitSize = 2 * kInstrSize;
const int Deoptimizer::kLazyDeoptExitSize = 1 * kInstrSize; const int Deoptimizer::kLazyDeoptExitSize = 1 * kInstrSize;
#endif #endif
#define __ masm->
namespace {
void CopyRegListToFrame(MacroAssembler* masm, const Register& dst,
int dst_offset, const CPURegList& reg_list,
const Register& temp0, const Register& temp1,
int src_offset = 0) {
DCHECK_EQ(reg_list.Count() % 2, 0);
UseScratchRegisterScope temps(masm);
CPURegList copy_to_input = reg_list;
int reg_size = reg_list.RegisterSizeInBytes();
DCHECK_EQ(temp0.SizeInBytes(), reg_size);
DCHECK_EQ(temp1.SizeInBytes(), reg_size);
// Compute some temporary addresses to avoid having the macro assembler set
// up a temp with an offset for accesses out of the range of the addressing
// mode.
Register src = temps.AcquireX();
masm->Add(src, sp, src_offset);
masm->Add(dst, dst, dst_offset);
// Write reg_list into the frame pointed to by dst.
for (int i = 0; i < reg_list.Count(); i += 2) {
masm->Ldp(temp0, temp1, MemOperand(src, i * reg_size));
CPURegister reg0 = copy_to_input.PopLowestIndex();
CPURegister reg1 = copy_to_input.PopLowestIndex();
int offset0 = reg0.code() * reg_size;
int offset1 = reg1.code() * reg_size;
// Pair up adjacent stores, otherwise write them separately.
if (offset1 == offset0 + reg_size) {
masm->Stp(temp0, temp1, MemOperand(dst, offset0));
} else {
masm->Str(temp0, MemOperand(dst, offset0));
masm->Str(temp1, MemOperand(dst, offset1));
}
}
masm->Sub(dst, dst, dst_offset);
}
void RestoreRegList(MacroAssembler* masm, const CPURegList& reg_list,
const Register& src_base, int src_offset) {
DCHECK_EQ(reg_list.Count() % 2, 0);
UseScratchRegisterScope temps(masm);
CPURegList restore_list = reg_list;
int reg_size = restore_list.RegisterSizeInBytes();
// Compute a temporary addresses to avoid having the macro assembler set
// up a temp with an offset for accesses out of the range of the addressing
// mode.
Register src = temps.AcquireX();
masm->Add(src, src_base, src_offset);
// No need to restore padreg.
restore_list.Remove(padreg);
// Restore every register in restore_list from src.
while (!restore_list.IsEmpty()) {
CPURegister reg0 = restore_list.PopLowestIndex();
CPURegister reg1 = restore_list.PopLowestIndex();
int offset0 = reg0.code() * reg_size;
if (reg1 == NoCPUReg) {
masm->Ldr(reg0, MemOperand(src, offset0));
break;
}
int offset1 = reg1.code() * reg_size;
// Pair up adjacent loads, otherwise read them separately.
if (offset1 == offset0 + reg_size) {
masm->Ldp(reg0, reg1, MemOperand(src, offset0));
} else {
masm->Ldr(reg0, MemOperand(src, offset0));
masm->Ldr(reg1, MemOperand(src, offset1));
}
}
}
} // namespace
void Deoptimizer::GenerateDeoptimizationEntries(MacroAssembler* masm,
Isolate* isolate,
DeoptimizeKind deopt_kind) {
NoRootArrayScope no_root_array(masm);
// TODO(all): This code needs to be revisited. We probably only need to save
// caller-saved registers here. Callee-saved registers can be stored directly
// in the input frame.
// Save all allocatable double registers.
CPURegList saved_double_registers(
CPURegister::kVRegister, kDRegSizeInBits,
RegisterConfiguration::Default()->allocatable_double_codes_mask());
DCHECK_EQ(saved_double_registers.Count() % 2, 0);
__ PushCPURegList(saved_double_registers);
// We save all the registers except sp, lr, platform register (x18) and the
// masm scratches.
CPURegList saved_registers(CPURegister::kRegister, kXRegSizeInBits, 0, 28);
saved_registers.Remove(ip0);
saved_registers.Remove(ip1);
saved_registers.Remove(x18);
saved_registers.Combine(fp);
saved_registers.Align();
DCHECK_EQ(saved_registers.Count() % 2, 0);
__ PushCPURegList(saved_registers);
__ Mov(x3, Operand(ExternalReference::Create(
IsolateAddressId::kCEntryFPAddress, isolate)));
__ Str(fp, MemOperand(x3));
const int kSavedRegistersAreaSize =
(saved_registers.Count() * kXRegSize) +
(saved_double_registers.Count() * kDRegSize);
// Floating point registers are saved on the stack above core registers.
const int kDoubleRegistersOffset = saved_registers.Count() * kXRegSize;
// We don't use a bailout id for arm64, because we can compute the id from the
// address. Pass kMaxUInt32 instead to signify this.
Register bailout_id = x2;
__ Mov(bailout_id, kMaxUInt32);
Register code_object = x3;
Register fp_to_sp = x4;
// Get the address of the location in the code object. This is the return
// address for lazy deoptimization.
__ Mov(code_object, lr);
// Compute the fp-to-sp delta.
__ Add(fp_to_sp, sp, kSavedRegistersAreaSize);
__ Sub(fp_to_sp, fp, fp_to_sp);
// Allocate a new deoptimizer object.
__ Ldr(x1, MemOperand(fp, CommonFrameConstants::kContextOrFrameTypeOffset));
// Ensure we can safely load from below fp.
DCHECK_GT(kSavedRegistersAreaSize, -StandardFrameConstants::kFunctionOffset);
__ Ldr(x0, MemOperand(fp, StandardFrameConstants::kFunctionOffset));
// If x1 is a smi, zero x0.
__ Tst(x1, kSmiTagMask);
__ CzeroX(x0, eq);
__ Mov(x1, static_cast<int>(deopt_kind));
// Following arguments are already loaded:
// - x2: bailout id
// - x3: code object address
// - x4: fp-to-sp delta
__ Mov(x5, ExternalReference::isolate_address(isolate));
{
// Call Deoptimizer::New().
AllowExternalCallThatCantCauseGC scope(masm);
__ CallCFunction(ExternalReference::new_deoptimizer_function(), 6);
}
// Preserve "deoptimizer" object in register x0.
Register deoptimizer = x0;
// Get the input frame descriptor pointer.
__ Ldr(x1, MemOperand(deoptimizer, Deoptimizer::input_offset()));
// Copy core registers into the input frame.
CopyRegListToFrame(masm, x1, FrameDescription::registers_offset(),
saved_registers, x2, x3);
// Copy double registers to the input frame.
CopyRegListToFrame(masm, x1, FrameDescription::double_registers_offset(),
saved_double_registers, x2, x3, kDoubleRegistersOffset);
// Mark the stack as not iterable for the CPU profiler which won't be able to
// walk the stack without the return address.
{
UseScratchRegisterScope temps(masm);
Register is_iterable = temps.AcquireX();
__ Mov(is_iterable, ExternalReference::stack_is_iterable_address(isolate));
__ strb(xzr, MemOperand(is_iterable));
}
// Remove the saved registers from the stack.
DCHECK_EQ(kSavedRegistersAreaSize % kXRegSize, 0);
__ Drop(kSavedRegistersAreaSize / kXRegSize);
// Compute a pointer to the unwinding limit in register x2; that is
// the first stack slot not part of the input frame.
Register unwind_limit = x2;
__ Ldr(unwind_limit, MemOperand(x1, FrameDescription::frame_size_offset()));
// Unwind the stack down to - but not including - the unwinding
// limit and copy the contents of the activation frame to the input
// frame description.
__ Add(x3, x1, FrameDescription::frame_content_offset());
__ SlotAddress(x1, 0);
__ Lsr(unwind_limit, unwind_limit, kSystemPointerSizeLog2);
__ Mov(x5, unwind_limit);
__ CopyDoubleWords(x3, x1, x5);
__ Drop(unwind_limit);
// Compute the output frame in the deoptimizer.
__ Push(padreg, x0); // Preserve deoptimizer object across call.
{
// Call Deoptimizer::ComputeOutputFrames().
AllowExternalCallThatCantCauseGC scope(masm);
__ CallCFunction(ExternalReference::compute_output_frames_function(), 1);
}
__ Pop(x4, padreg); // Restore deoptimizer object (class Deoptimizer).
{
UseScratchRegisterScope temps(masm);
Register scratch = temps.AcquireX();
__ Ldr(scratch, MemOperand(x4, Deoptimizer::caller_frame_top_offset()));
__ Mov(sp, scratch);
}
// Replace the current (input) frame with the output frames.
Label outer_push_loop, outer_loop_header;
__ Ldrsw(x1, MemOperand(x4, Deoptimizer::output_count_offset()));
__ Ldr(x0, MemOperand(x4, Deoptimizer::output_offset()));
__ Add(x1, x0, Operand(x1, LSL, kSystemPointerSizeLog2));
__ B(&outer_loop_header);
__ Bind(&outer_push_loop);
Register current_frame = x2;
Register frame_size = x3;
__ Ldr(current_frame, MemOperand(x0, kSystemPointerSize, PostIndex));
__ Ldr(x3, MemOperand(current_frame, FrameDescription::frame_size_offset()));
__ Lsr(frame_size, x3, kSystemPointerSizeLog2);
__ Claim(frame_size);
__ Add(x7, current_frame, FrameDescription::frame_content_offset());
__ SlotAddress(x6, 0);
__ CopyDoubleWords(x6, x7, frame_size);
__ Bind(&outer_loop_header);
__ Cmp(x0, x1);
__ B(lt, &outer_push_loop);
__ Ldr(x1, MemOperand(x4, Deoptimizer::input_offset()));
RestoreRegList(masm, saved_double_registers, x1,
FrameDescription::double_registers_offset());
{
UseScratchRegisterScope temps(masm);
Register is_iterable = temps.AcquireX();
Register one = x4;
__ Mov(is_iterable, ExternalReference::stack_is_iterable_address(isolate));
__ Mov(one, Operand(1));
__ strb(one, MemOperand(is_iterable));
}
// TODO(all): ARM copies a lot (if not all) of the last output frame onto the
// stack, then pops it all into registers. Here, we try to load it directly
// into the relevant registers. Is this correct? If so, we should improve the
// ARM code.
// Restore registers from the last output frame.
// Note that lr is not in the list of saved_registers and will be restored
// later. We can use it to hold the address of last output frame while
// reloading the other registers.
DCHECK(!saved_registers.IncludesAliasOf(lr));
Register last_output_frame = lr;
__ Mov(last_output_frame, current_frame);
RestoreRegList(masm, saved_registers, last_output_frame,
FrameDescription::registers_offset());
UseScratchRegisterScope temps(masm);
temps.Exclude(x17);
Register continuation = x17;
__ Ldr(continuation, MemOperand(last_output_frame,
FrameDescription::continuation_offset()));
__ Ldr(lr, MemOperand(last_output_frame, FrameDescription::pc_offset()));
#ifdef V8_ENABLE_CONTROL_FLOW_INTEGRITY
__ Autibsp();
#endif
__ Br(continuation);
}
Float32 RegisterValues::GetFloatRegister(unsigned n) const { Float32 RegisterValues::GetFloatRegister(unsigned n) const {
return Float32::FromBits( return Float32::FromBits(
static_cast<uint32_t>(double_registers_[n].get_bits())); static_cast<uint32_t>(double_registers_[n].get_bits()));
@ -46,5 +331,7 @@ void FrameDescription::SetPc(intptr_t pc) {
pc_ = pc; pc_ = pc;
} }
#undef __
} // namespace internal } // namespace internal
} // namespace v8 } // namespace v8

View File

@ -26,7 +26,6 @@
#include "src/objects/debug-objects-inl.h" #include "src/objects/debug-objects-inl.h"
#include "src/objects/heap-number-inl.h" #include "src/objects/heap-number-inl.h"
#include "src/objects/smi.h" #include "src/objects/smi.h"
#include "src/snapshot/embedded/embedded-data.h"
#include "src/tracing/trace-event.h" #include "src/tracing/trace-event.h"
#include "torque-generated/exported-class-definitions.h" #include "torque-generated/exported-class-definitions.h"
@ -174,6 +173,25 @@ class FrameWriter {
unsigned top_offset_; unsigned top_offset_;
}; };
DeoptimizerData::DeoptimizerData(Heap* heap) : heap_(heap), current_(nullptr) {
Code* start = &deopt_entry_code_[0];
Code* end = &deopt_entry_code_[DeoptimizerData::kLastDeoptimizeKind + 1];
strong_roots_entry_ =
heap_->RegisterStrongRoots(FullObjectSlot(start), FullObjectSlot(end));
}
DeoptimizerData::~DeoptimizerData() {
heap_->UnregisterStrongRoots(strong_roots_entry_);
}
Code DeoptimizerData::deopt_entry_code(DeoptimizeKind kind) {
return deopt_entry_code_[static_cast<int>(kind)];
}
void DeoptimizerData::set_deopt_entry_code(DeoptimizeKind kind, Code code) {
deopt_entry_code_[static_cast<int>(kind)] = code;
}
Code Deoptimizer::FindDeoptimizingCode(Address addr) { Code Deoptimizer::FindDeoptimizingCode(Address addr) {
if (function_.IsHeapObject()) { if (function_.IsHeapObject()) {
// Search all deoptimizing code in the native context of the function. // Search all deoptimizing code in the native context of the function.
@ -190,7 +208,7 @@ Code Deoptimizer::FindDeoptimizingCode(Address addr) {
return Code(); return Code();
} }
// We rely on this function not causing a GC. It is called from generated code // We rely on this function not causing a GC. It is called from generated code
// without having a real stack frame in place. // without having a real stack frame in place.
Deoptimizer* Deoptimizer::New(Address raw_function, DeoptimizeKind kind, Deoptimizer* Deoptimizer::New(Address raw_function, DeoptimizeKind kind,
unsigned bailout_id, Address from, unsigned bailout_id, Address from,
@ -198,13 +216,16 @@ Deoptimizer* Deoptimizer::New(Address raw_function, DeoptimizeKind kind,
JSFunction function = JSFunction::cast(Object(raw_function)); JSFunction function = JSFunction::cast(Object(raw_function));
Deoptimizer* deoptimizer = new Deoptimizer(isolate, function, kind, Deoptimizer* deoptimizer = new Deoptimizer(isolate, function, kind,
bailout_id, from, fp_to_sp_delta); bailout_id, from, fp_to_sp_delta);
isolate->set_current_deoptimizer(deoptimizer); CHECK_NULL(isolate->deoptimizer_data()->current_);
isolate->deoptimizer_data()->current_ = deoptimizer;
return deoptimizer; return deoptimizer;
} }
Deoptimizer* Deoptimizer::Grab(Isolate* isolate) { Deoptimizer* Deoptimizer::Grab(Isolate* isolate) {
Deoptimizer* result = isolate->GetAndClearCurrentDeoptimizer(); Deoptimizer* result = isolate->deoptimizer_data()->current_;
CHECK_NOT_NULL(result);
result->DeleteFrameDescriptions(); result->DeleteFrameDescriptions();
isolate->deoptimizer_data()->current_ = nullptr;
return result; return result;
} }
@ -471,6 +492,8 @@ const char* Deoptimizer::MessageFor(DeoptimizeKind kind, bool reuse_code) {
case DeoptimizeKind::kBailout: case DeoptimizeKind::kBailout:
return "bailout"; return "bailout";
} }
FATAL("Unsupported deopt kind");
return nullptr;
} }
namespace { namespace {
@ -513,9 +536,6 @@ Deoptimizer::Deoptimizer(Isolate* isolate, JSFunction function,
deoptimizing_throw_ = true; deoptimizing_throw_ = true;
} }
DCHECK(bailout_id_ == kFixedExitSizeMarker ||
bailout_id_ < kMaxNumberOfEntries);
DCHECK_NE(from, kNullAddress); DCHECK_NE(from, kNullAddress);
compiled_code_ = FindOptimizedCode(); compiled_code_ = FindOptimizedCode();
DCHECK(!compiled_code_.is_null()); DCHECK(!compiled_code_.is_null());
@ -544,7 +564,7 @@ Deoptimizer::Deoptimizer(Isolate* isolate, JSFunction function,
input_ = new (size) FrameDescription(size, parameter_count); input_ = new (size) FrameDescription(size, parameter_count);
if (kSupportsFixedDeoptExitSizes) { if (kSupportsFixedDeoptExitSizes) {
DCHECK_EQ(bailout_id_, kFixedExitSizeMarker); DCHECK_EQ(bailout_id_, kMaxUInt32);
// Calculate bailout id from return address. // Calculate bailout id from return address.
DCHECK_GT(kNonLazyDeoptExitSize, 0); DCHECK_GT(kNonLazyDeoptExitSize, 0);
DCHECK_GT(kLazyDeoptExitSize, 0); DCHECK_GT(kLazyDeoptExitSize, 0);
@ -556,7 +576,7 @@ Deoptimizer::Deoptimizer(Isolate* isolate, JSFunction function,
Address lazy_deopt_start = Address lazy_deopt_start =
deopt_start + non_lazy_deopt_count * kNonLazyDeoptExitSize; deopt_start + non_lazy_deopt_count * kNonLazyDeoptExitSize;
// The deoptimization exits are sorted so that lazy deopt exits appear last. // The deoptimization exits are sorted so that lazy deopt exits appear last.
static_assert(DeoptimizeKind::kLazy == kLastDeoptimizeKind, static_assert(DeoptimizeKind::kLazy == DeoptimizeKind::kLastDeoptimizeKind,
"lazy deopts are expected to be emitted last"); "lazy deopts are expected to be emitted last");
// from_ is the value of the link register after the call to the // from_ is the value of the link register after the call to the
// deoptimizer, so for the last lazy deopt, from_ points to the first // deoptimizer, so for the last lazy deopt, from_ points to the first
@ -615,44 +635,42 @@ void Deoptimizer::DeleteFrameDescriptions() {
#endif // DEBUG #endif // DEBUG
} }
Builtins::Name Deoptimizer::GetDeoptimizationEntry(Isolate* isolate, Address Deoptimizer::GetDeoptimizationEntry(Isolate* isolate,
DeoptimizeKind kind) { DeoptimizeKind kind) {
switch (kind) { DeoptimizerData* data = isolate->deoptimizer_data();
case DeoptimizeKind::kEager: CHECK_LE(kind, DeoptimizerData::kLastDeoptimizeKind);
return Builtins::kDeoptimizationEntry_Eager; CHECK(!data->deopt_entry_code(kind).is_null());
case DeoptimizeKind::kSoft: return data->deopt_entry_code(kind).raw_instruction_start();
return Builtins::kDeoptimizationEntry_Soft;
case DeoptimizeKind::kBailout:
return Builtins::kDeoptimizationEntry_Bailout;
case DeoptimizeKind::kLazy:
return Builtins::kDeoptimizationEntry_Lazy;
}
} }
bool Deoptimizer::IsDeoptimizationEntry(Isolate* isolate, Address addr, bool Deoptimizer::IsDeoptimizationEntry(Isolate* isolate, Address addr,
DeoptimizeKind* type_out) { DeoptimizeKind type) {
Code maybe_code = InstructionStream::TryLookupCode(isolate, addr); DeoptimizerData* data = isolate->deoptimizer_data();
if (maybe_code.is_null()) return false; CHECK_LE(type, DeoptimizerData::kLastDeoptimizeKind);
Code code = data->deopt_entry_code(type);
if (code.is_null()) return false;
return addr == code.raw_instruction_start();
}
Code code = maybe_code; bool Deoptimizer::IsDeoptimizationEntry(Isolate* isolate, Address addr,
switch (code.builtin_index()) { DeoptimizeKind* type) {
case Builtins::kDeoptimizationEntry_Eager: if (IsDeoptimizationEntry(isolate, addr, DeoptimizeKind::kEager)) {
*type_out = DeoptimizeKind::kEager; *type = DeoptimizeKind::kEager;
return true; return true;
case Builtins::kDeoptimizationEntry_Soft:
*type_out = DeoptimizeKind::kSoft;
return true;
case Builtins::kDeoptimizationEntry_Bailout:
*type_out = DeoptimizeKind::kBailout;
return true;
case Builtins::kDeoptimizationEntry_Lazy:
*type_out = DeoptimizeKind::kLazy;
return true;
default:
return false;
} }
if (IsDeoptimizationEntry(isolate, addr, DeoptimizeKind::kSoft)) {
UNREACHABLE(); *type = DeoptimizeKind::kSoft;
return true;
}
if (IsDeoptimizationEntry(isolate, addr, DeoptimizeKind::kLazy)) {
*type = DeoptimizeKind::kLazy;
return true;
}
if (IsDeoptimizationEntry(isolate, addr, DeoptimizeKind::kBailout)) {
*type = DeoptimizeKind::kBailout;
return true;
}
return false;
} }
int Deoptimizer::GetDeoptimizedCodeCount(Isolate* isolate) { int Deoptimizer::GetDeoptimizedCodeCount(Isolate* isolate) {
@ -800,8 +818,8 @@ void Deoptimizer::TraceDeoptMarked(Isolate* isolate) {
// without having a real stack frame in place. // without having a real stack frame in place.
void Deoptimizer::DoComputeOutputFrames() { void Deoptimizer::DoComputeOutputFrames() {
// When we call this function, the return address of the previous frame has // When we call this function, the return address of the previous frame has
// been removed from the stack by the DeoptimizationEntry builtin, so the // been removed from the stack by GenerateDeoptimizationEntries() so the stack
// stack is not iterable by the SafeStackFrameIterator. // is not iterable by the SafeStackFrameIterator.
#if V8_TARGET_ARCH_STORES_RETURN_ADDRESS_ON_STACK #if V8_TARGET_ARCH_STORES_RETURN_ADDRESS_ON_STACK
DCHECK_EQ(0, isolate()->isolate_data()->stack_is_iterable()); DCHECK_EQ(0, isolate()->isolate_data()->stack_is_iterable());
#endif #endif
@ -863,7 +881,7 @@ void Deoptimizer::DoComputeOutputFrames() {
// Do the input frame to output frame(s) translation. // Do the input frame to output frame(s) translation.
size_t count = translated_state_.frames().size(); size_t count = translated_state_.frames().size();
// If we are supposed to go to the catch handler, find the catching frame // If we are supposed to go to the catch handler, find the catching frame
// for the catch and make sure we only deoptimize up to that frame. // for the catch and make sure we only deoptimize upto that frame.
if (deoptimizing_throw_) { if (deoptimizing_throw_) {
size_t catch_handler_frame_index = count; size_t catch_handler_frame_index = count;
for (size_t i = count; i-- > 0;) { for (size_t i = count; i-- > 0;) {
@ -1194,7 +1212,7 @@ void Deoptimizer::DoComputeInterpretedFrame(TranslatedFrame* translated_frame,
: builtins->builtin(Builtins::kInterpreterEnterBytecodeDispatch); : builtins->builtin(Builtins::kInterpreterEnterBytecodeDispatch);
if (is_topmost) { if (is_topmost) {
// Only the pc of the topmost frame needs to be signed since it is // Only the pc of the topmost frame needs to be signed since it is
// authenticated at the end of the DeoptimizationEntry builtin. // authenticated at the end of GenerateDeoptimizationEntries.
const intptr_t top_most_pc = PointerAuthentication::SignAndCheckPC( const intptr_t top_most_pc = PointerAuthentication::SignAndCheckPC(
static_cast<intptr_t>(dispatch_builtin.InstructionStart()), static_cast<intptr_t>(dispatch_builtin.InstructionStart()),
frame_writer.frame()->GetTop()); frame_writer.frame()->GetTop());
@ -1521,7 +1539,7 @@ void Deoptimizer::DoComputeConstructStubFrame(TranslatedFrame* translated_frame,
intptr_t pc_value = static_cast<intptr_t>(start + pc_offset); intptr_t pc_value = static_cast<intptr_t>(start + pc_offset);
if (is_topmost) { if (is_topmost) {
// Only the pc of the topmost frame needs to be signed since it is // Only the pc of the topmost frame needs to be signed since it is
// authenticated at the end of the DeoptimizationEntry builtin. // authenticated at the end of GenerateDeoptimizationEntries.
output_frame->SetPc(PointerAuthentication::SignAndCheckPC( output_frame->SetPc(PointerAuthentication::SignAndCheckPC(
pc_value, frame_writer.frame()->GetTop())); pc_value, frame_writer.frame()->GetTop()));
} else { } else {
@ -1922,7 +1940,7 @@ void Deoptimizer::DoComputeBuiltinContinuation(
mode, frame_info.frame_has_result_stack_slot())); mode, frame_info.frame_has_result_stack_slot()));
if (is_topmost) { if (is_topmost) {
// Only the pc of the topmost frame needs to be signed since it is // Only the pc of the topmost frame needs to be signed since it is
// authenticated at the end of the DeoptimizationEntry builtin. // authenticated at the end of GenerateDeoptimizationEntries.
const intptr_t top_most_pc = PointerAuthentication::SignAndCheckPC( const intptr_t top_most_pc = PointerAuthentication::SignAndCheckPC(
static_cast<intptr_t>(continue_to_builtin.InstructionStart()), static_cast<intptr_t>(continue_to_builtin.InstructionStart()),
frame_writer.frame()->GetTop()); frame_writer.frame()->GetTop());
@ -2019,6 +2037,40 @@ unsigned Deoptimizer::ComputeIncomingArgumentSize(SharedFunctionInfo shared) {
return parameter_slots * kSystemPointerSize; return parameter_slots * kSystemPointerSize;
} }
void Deoptimizer::EnsureCodeForDeoptimizationEntry(Isolate* isolate,
DeoptimizeKind kind) {
CHECK(kind == DeoptimizeKind::kEager || kind == DeoptimizeKind::kSoft ||
kind == DeoptimizeKind::kLazy || kind == DeoptimizeKind::kBailout);
DeoptimizerData* data = isolate->deoptimizer_data();
if (!data->deopt_entry_code(kind).is_null()) return;
MacroAssembler masm(isolate, CodeObjectRequired::kYes,
NewAssemblerBuffer(16 * KB));
masm.set_emit_debug_code(false);
GenerateDeoptimizationEntries(&masm, masm.isolate(), kind);
CodeDesc desc;
masm.GetCode(isolate, &desc);
DCHECK(!RelocInfo::RequiresRelocationAfterCodegen(desc));
// Allocate the code as immovable since the entry addresses will be used
// directly and there is no support for relocating them.
Handle<Code> code = Factory::CodeBuilder(
isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.set_immovable()
.Build();
CHECK(isolate->heap()->IsImmovable(*code));
CHECK(data->deopt_entry_code(kind).is_null());
data->set_deopt_entry_code(kind, *code);
}
void Deoptimizer::EnsureCodeForDeoptimizationEntries(Isolate* isolate) {
EnsureCodeForDeoptimizationEntry(isolate, DeoptimizeKind::kEager);
EnsureCodeForDeoptimizationEntry(isolate, DeoptimizeKind::kLazy);
EnsureCodeForDeoptimizationEntry(isolate, DeoptimizeKind::kSoft);
EnsureCodeForDeoptimizationEntry(isolate, DeoptimizeKind::kBailout);
}
FrameDescription::FrameDescription(uint32_t frame_size, int parameter_count) FrameDescription::FrameDescription(uint32_t frame_size, int parameter_count)
: frame_size_(frame_size), : frame_size_(frame_size),
parameter_count_(parameter_count), parameter_count_(parameter_count),

View File

@ -500,13 +500,12 @@ class Deoptimizer : public Malloced {
static void ComputeOutputFrames(Deoptimizer* deoptimizer); static void ComputeOutputFrames(Deoptimizer* deoptimizer);
V8_EXPORT_PRIVATE static Builtins::Name GetDeoptimizationEntry( static Address GetDeoptimizationEntry(Isolate* isolate, DeoptimizeKind kind);
Isolate* isolate, DeoptimizeKind kind);
// Returns true if {addr} is a deoptimization entry and stores its type in // Returns true if {addr} is a deoptimization entry and stores its type in
// {type_out}. Returns false if {addr} is not a deoptimization entry. // {type}. Returns false if {addr} is not a deoptimization entry.
static bool IsDeoptimizationEntry(Isolate* isolate, Address addr, static bool IsDeoptimizationEntry(Isolate* isolate, Address addr,
DeoptimizeKind* type_out); DeoptimizeKind* type);
// Code generation support. // Code generation support.
static int input_offset() { return offsetof(Deoptimizer, input_); } static int input_offset() { return offsetof(Deoptimizer, input_); }
@ -521,26 +520,25 @@ class Deoptimizer : public Malloced {
V8_EXPORT_PRIVATE static int GetDeoptimizedCodeCount(Isolate* isolate); V8_EXPORT_PRIVATE static int GetDeoptimizedCodeCount(Isolate* isolate);
static const int kNotDeoptimizationEntry = -1;
static void EnsureCodeForDeoptimizationEntry(Isolate* isolate,
DeoptimizeKind kind);
static void EnsureCodeForDeoptimizationEntries(Isolate* isolate);
Isolate* isolate() const { return isolate_; } Isolate* isolate() const { return isolate_; }
static constexpr int kMaxNumberOfEntries = 16384; static const int kMaxNumberOfEntries = 16384;
// This marker is passed to Deoptimizer::New as {bailout_id} on platforms
// that have fixed deopt sizes (see also kSupportsFixedDeoptExitSizes). The
// actual deoptimization id is then calculated from the return address.
static constexpr unsigned kFixedExitSizeMarker = kMaxUInt32;
// Set to true when the architecture supports deoptimization exit sequences // Set to true when the architecture supports deoptimization exit sequences
// of a fixed size, that can be sorted so that the deoptimization index is // of a fixed size, that can be sorted so that the deoptimization index is
// deduced from the address of the deoptimization exit. // deduced from the address of the deoptimization exit.
// TODO(jgruber): Remove this, and support for variable deopt exit sizes, static const bool kSupportsFixedDeoptExitSizes;
// once all architectures use fixed exit sizes.
V8_EXPORT_PRIVATE static const bool kSupportsFixedDeoptExitSizes;
// Size of deoptimization exit sequence. This is only meaningful when // Size of deoptimization exit sequence. This is only meaningful when
// kSupportsFixedDeoptExitSizes is true. // kSupportsFixedDeoptExitSizes is true.
V8_EXPORT_PRIVATE static const int kNonLazyDeoptExitSize; static const int kNonLazyDeoptExitSize;
V8_EXPORT_PRIVATE static const int kLazyDeoptExitSize; static const int kLazyDeoptExitSize;
// Tracing. // Tracing.
static void TraceMarkForDeoptimization(Code code, const char* reason); static void TraceMarkForDeoptimization(Code code, const char* reason);
@ -557,6 +555,9 @@ class Deoptimizer : public Malloced {
Code FindOptimizedCode(); Code FindOptimizedCode();
void DeleteFrameDescriptions(); void DeleteFrameDescriptions();
static bool IsDeoptimizationEntry(Isolate* isolate, Address addr,
DeoptimizeKind type);
void DoComputeOutputFrames(); void DoComputeOutputFrames();
void DoComputeInterpretedFrame(TranslatedFrame* translated_frame, void DoComputeInterpretedFrame(TranslatedFrame* translated_frame,
int frame_index, bool goto_catch_handler); int frame_index, bool goto_catch_handler);
@ -578,6 +579,10 @@ class Deoptimizer : public Malloced {
static unsigned ComputeIncomingArgumentSize(SharedFunctionInfo shared); static unsigned ComputeIncomingArgumentSize(SharedFunctionInfo shared);
static unsigned ComputeOutgoingArgumentSize(Code code, unsigned bailout_id); static unsigned ComputeOutgoingArgumentSize(Code code, unsigned bailout_id);
static void GenerateDeoptimizationEntries(MacroAssembler* masm,
Isolate* isolate,
DeoptimizeKind kind);
static void MarkAllCodeForContext(NativeContext native_context); static void MarkAllCodeForContext(NativeContext native_context);
static void DeoptimizeMarkedCodeForContext(NativeContext native_context); static void DeoptimizeMarkedCodeForContext(NativeContext native_context);
// Searches the list of known deoptimizing code for a Code object // Searches the list of known deoptimizing code for a Code object
@ -822,6 +827,36 @@ class FrameDescription {
} }
}; };
class DeoptimizerData {
public:
explicit DeoptimizerData(Heap* heap);
~DeoptimizerData();
#ifdef DEBUG
bool IsDeoptEntryCode(Code code) const {
for (int i = 0; i < kLastDeoptimizeKind + 1; i++) {
if (code == deopt_entry_code_[i]) return true;
}
return false;
}
#endif // DEBUG
private:
Heap* heap_;
static const int kLastDeoptimizeKind =
static_cast<int>(DeoptimizeKind::kLastDeoptimizeKind);
Code deopt_entry_code_[kLastDeoptimizeKind + 1];
Code deopt_entry_code(DeoptimizeKind kind);
void set_deopt_entry_code(DeoptimizeKind kind, Code code);
Deoptimizer* current_;
StrongRootsEntry* strong_roots_entry_;
friend class Deoptimizer;
DISALLOW_COPY_AND_ASSIGN(DeoptimizerData);
};
class TranslationBuffer { class TranslationBuffer {
public: public:
explicit TranslationBuffer(Zone* zone) : contents_(zone) {} explicit TranslationBuffer(Zone* zone) : contents_(zone) {}

View File

@ -4,14 +4,201 @@
#if V8_TARGET_ARCH_IA32 #if V8_TARGET_ARCH_IA32
#include "src/codegen/macro-assembler.h"
#include "src/codegen/register-configuration.h"
#include "src/codegen/safepoint-table.h"
#include "src/deoptimizer/deoptimizer.h" #include "src/deoptimizer/deoptimizer.h"
#include "src/execution/frame-constants.h"
namespace v8 { namespace v8 {
namespace internal { namespace internal {
const bool Deoptimizer::kSupportsFixedDeoptExitSizes = true; const bool Deoptimizer::kSupportsFixedDeoptExitSizes = false;
const int Deoptimizer::kNonLazyDeoptExitSize = 5; const int Deoptimizer::kNonLazyDeoptExitSize = 0;
const int Deoptimizer::kLazyDeoptExitSize = 5; const int Deoptimizer::kLazyDeoptExitSize = 0;
#define __ masm->
void Deoptimizer::GenerateDeoptimizationEntries(MacroAssembler* masm,
Isolate* isolate,
DeoptimizeKind deopt_kind) {
NoRootArrayScope no_root_array(masm);
// Save all general purpose registers before messing with them.
const int kNumberOfRegisters = Register::kNumRegisters;
const int kDoubleRegsSize = kDoubleSize * XMMRegister::kNumRegisters;
__ AllocateStackSpace(kDoubleRegsSize);
const RegisterConfiguration* config = RegisterConfiguration::Default();
for (int i = 0; i < config->num_allocatable_double_registers(); ++i) {
int code = config->GetAllocatableDoubleCode(i);
XMMRegister xmm_reg = XMMRegister::from_code(code);
int offset = code * kDoubleSize;
__ movsd(Operand(esp, offset), xmm_reg);
}
__ pushad();
ExternalReference c_entry_fp_address =
ExternalReference::Create(IsolateAddressId::kCEntryFPAddress, isolate);
__ mov(masm->ExternalReferenceAsOperand(c_entry_fp_address, esi), ebp);
const int kSavedRegistersAreaSize =
kNumberOfRegisters * kSystemPointerSize + kDoubleRegsSize;
// The bailout id is passed in ebx by the caller.
// Get the address of the location in the code object
// and compute the fp-to-sp delta in register edx.
__ mov(ecx, Operand(esp, kSavedRegistersAreaSize));
__ lea(edx, Operand(esp, kSavedRegistersAreaSize + 1 * kSystemPointerSize));
__ sub(edx, ebp);
__ neg(edx);
// Allocate a new deoptimizer object.
__ PrepareCallCFunction(6, eax);
__ mov(eax, Immediate(0));
Label context_check;
__ mov(edi, Operand(ebp, CommonFrameConstants::kContextOrFrameTypeOffset));
__ JumpIfSmi(edi, &context_check);
__ mov(eax, Operand(ebp, StandardFrameConstants::kFunctionOffset));
__ bind(&context_check);
__ mov(Operand(esp, 0 * kSystemPointerSize), eax); // Function.
__ mov(Operand(esp, 1 * kSystemPointerSize),
Immediate(static_cast<int>(deopt_kind)));
__ mov(Operand(esp, 2 * kSystemPointerSize), ebx); // Bailout id.
__ mov(Operand(esp, 3 * kSystemPointerSize), ecx); // Code address or 0.
__ mov(Operand(esp, 4 * kSystemPointerSize), edx); // Fp-to-sp delta.
__ mov(Operand(esp, 5 * kSystemPointerSize),
Immediate(ExternalReference::isolate_address(isolate)));
{
AllowExternalCallThatCantCauseGC scope(masm);
__ CallCFunction(ExternalReference::new_deoptimizer_function(), 6);
}
// Preserve deoptimizer object in register eax and get the input
// frame descriptor pointer.
__ mov(esi, Operand(eax, Deoptimizer::input_offset()));
// Fill in the input registers.
for (int i = kNumberOfRegisters - 1; i >= 0; i--) {
int offset =
(i * kSystemPointerSize) + FrameDescription::registers_offset();
__ pop(Operand(esi, offset));
}
int double_regs_offset = FrameDescription::double_registers_offset();
// Fill in the double input registers.
for (int i = 0; i < config->num_allocatable_double_registers(); ++i) {
int code = config->GetAllocatableDoubleCode(i);
int dst_offset = code * kDoubleSize + double_regs_offset;
int src_offset = code * kDoubleSize;
__ movsd(xmm0, Operand(esp, src_offset));
__ movsd(Operand(esi, dst_offset), xmm0);
}
// Clear FPU all exceptions.
// TODO(ulan): Find out why the TOP register is not zero here in some cases,
// and check that the generated code never deoptimizes with unbalanced stack.
__ fnclex();
// Mark the stack as not iterable for the CPU profiler which won't be able to
// walk the stack without the return address.
__ mov_b(__ ExternalReferenceAsOperand(
ExternalReference::stack_is_iterable_address(isolate), edx),
Immediate(0));
// Remove the return address and the double registers.
__ add(esp, Immediate(kDoubleRegsSize + 1 * kSystemPointerSize));
// Compute a pointer to the unwinding limit in register ecx; that is
// the first stack slot not part of the input frame.
__ mov(ecx, Operand(esi, FrameDescription::frame_size_offset()));
__ add(ecx, esp);
// Unwind the stack down to - but not including - the unwinding
// limit and copy the contents of the activation frame to the input
// frame description.
__ lea(edx, Operand(esi, FrameDescription::frame_content_offset()));
Label pop_loop_header;
__ jmp(&pop_loop_header);
Label pop_loop;
__ bind(&pop_loop);
__ pop(Operand(edx, 0));
__ add(edx, Immediate(sizeof(uint32_t)));
__ bind(&pop_loop_header);
__ cmp(ecx, esp);
__ j(not_equal, &pop_loop);
// Compute the output frame in the deoptimizer.
__ push(eax);
__ PrepareCallCFunction(1, esi);
__ mov(Operand(esp, 0 * kSystemPointerSize), eax);
{
AllowExternalCallThatCantCauseGC scope(masm);
__ CallCFunction(ExternalReference::compute_output_frames_function(), 1);
}
__ pop(eax);
__ mov(esp, Operand(eax, Deoptimizer::caller_frame_top_offset()));
// Replace the current (input) frame with the output frames.
Label outer_push_loop, inner_push_loop, outer_loop_header, inner_loop_header;
// Outer loop state: eax = current FrameDescription**, edx = one
// past the last FrameDescription**.
__ mov(edx, Operand(eax, Deoptimizer::output_count_offset()));
__ mov(eax, Operand(eax, Deoptimizer::output_offset()));
__ lea(edx, Operand(eax, edx, times_system_pointer_size, 0));
__ jmp(&outer_loop_header);
__ bind(&outer_push_loop);
// Inner loop state: esi = current FrameDescription*, ecx = loop
// index.
__ mov(esi, Operand(eax, 0));
__ mov(ecx, Operand(esi, FrameDescription::frame_size_offset()));
__ jmp(&inner_loop_header);
__ bind(&inner_push_loop);
__ sub(ecx, Immediate(sizeof(uint32_t)));
__ push(Operand(esi, ecx, times_1, FrameDescription::frame_content_offset()));
__ bind(&inner_loop_header);
__ test(ecx, ecx);
__ j(not_zero, &inner_push_loop);
__ add(eax, Immediate(kSystemPointerSize));
__ bind(&outer_loop_header);
__ cmp(eax, edx);
__ j(below, &outer_push_loop);
// In case of a failed STUB, we have to restore the XMM registers.
for (int i = 0; i < config->num_allocatable_double_registers(); ++i) {
int code = config->GetAllocatableDoubleCode(i);
XMMRegister xmm_reg = XMMRegister::from_code(code);
int src_offset = code * kDoubleSize + double_regs_offset;
__ movsd(xmm_reg, Operand(esi, src_offset));
}
// Push pc and continuation from the last output frame.
__ push(Operand(esi, FrameDescription::pc_offset()));
__ push(Operand(esi, FrameDescription::continuation_offset()));
// Push the registers from the last output frame.
for (int i = 0; i < kNumberOfRegisters; i++) {
int offset =
(i * kSystemPointerSize) + FrameDescription::registers_offset();
__ push(Operand(esi, offset));
}
__ mov_b(__ ExternalReferenceAsOperand(
ExternalReference::stack_is_iterable_address(isolate), edx),
Immediate(1));
// Restore the registers from the stack.
__ popad();
__ InitializeRootRegister();
// Return to the continuation point.
__ ret(0);
}
Float32 RegisterValues::GetFloatRegister(unsigned n) const { Float32 RegisterValues::GetFloatRegister(unsigned n) const {
return Float32::FromBits( return Float32::FromBits(
@ -33,6 +220,8 @@ void FrameDescription::SetCallerConstantPool(unsigned offset, intptr_t value) {
void FrameDescription::SetPc(intptr_t pc) { pc_ = pc; } void FrameDescription::SetPc(intptr_t pc) { pc_ = pc; }
#undef __
} // namespace internal } // namespace internal
} // namespace v8 } // namespace v8

View File

@ -4,14 +4,217 @@
#if V8_TARGET_ARCH_X64 #if V8_TARGET_ARCH_X64
#include "src/codegen/macro-assembler.h"
#include "src/codegen/register-configuration.h"
#include "src/codegen/safepoint-table.h"
#include "src/deoptimizer/deoptimizer.h" #include "src/deoptimizer/deoptimizer.h"
#include "src/objects/objects-inl.h"
namespace v8 { namespace v8 {
namespace internal { namespace internal {
const bool Deoptimizer::kSupportsFixedDeoptExitSizes = true; const bool Deoptimizer::kSupportsFixedDeoptExitSizes = false;
const int Deoptimizer::kNonLazyDeoptExitSize = 7; const int Deoptimizer::kNonLazyDeoptExitSize = 0;
const int Deoptimizer::kLazyDeoptExitSize = 7; const int Deoptimizer::kLazyDeoptExitSize = 0;
#define __ masm->
void Deoptimizer::GenerateDeoptimizationEntries(MacroAssembler* masm,
Isolate* isolate,
DeoptimizeKind deopt_kind) {
NoRootArrayScope no_root_array(masm);
// Save all general purpose registers before messing with them.
const int kNumberOfRegisters = Register::kNumRegisters;
const int kDoubleRegsSize = kDoubleSize * XMMRegister::kNumRegisters;
__ AllocateStackSpace(kDoubleRegsSize);
const RegisterConfiguration* config = RegisterConfiguration::Default();
for (int i = 0; i < config->num_allocatable_double_registers(); ++i) {
int code = config->GetAllocatableDoubleCode(i);
XMMRegister xmm_reg = XMMRegister::from_code(code);
int offset = code * kDoubleSize;
__ Movsd(Operand(rsp, offset), xmm_reg);
}
// We push all registers onto the stack, even though we do not need
// to restore all later.
for (int i = 0; i < kNumberOfRegisters; i++) {
Register r = Register::from_code(i);
__ pushq(r);
}
const int kSavedRegistersAreaSize =
kNumberOfRegisters * kSystemPointerSize + kDoubleRegsSize;
__ Store(
ExternalReference::Create(IsolateAddressId::kCEntryFPAddress, isolate),
rbp);
// We use this to keep the value of the fifth argument temporarily.
// Unfortunately we can't store it directly in r8 (used for passing
// this on linux), since it is another parameter passing register on windows.
Register arg5 = r11;
// The bailout id is passed using r13 on the stack.
__ movq(arg_reg_3, r13);
// Get the address of the location in the code object
// and compute the fp-to-sp delta in register arg5.
__ movq(arg_reg_4, Operand(rsp, kSavedRegistersAreaSize));
__ leaq(arg5, Operand(rsp, kSavedRegistersAreaSize + kPCOnStackSize));
__ subq(arg5, rbp);
__ negq(arg5);
// Allocate a new deoptimizer object.
__ PrepareCallCFunction(6);
__ movq(rax, Immediate(0));
Label context_check;
__ movq(rdi, Operand(rbp, CommonFrameConstants::kContextOrFrameTypeOffset));
__ JumpIfSmi(rdi, &context_check);
__ movq(rax, Operand(rbp, StandardFrameConstants::kFunctionOffset));
__ bind(&context_check);
__ movq(arg_reg_1, rax);
__ Set(arg_reg_2, static_cast<int>(deopt_kind));
// Args 3 and 4 are already in the right registers.
// On windows put the arguments on the stack (PrepareCallCFunction
// has created space for this). On linux pass the arguments in r8 and r9.
#ifdef V8_TARGET_OS_WIN
__ movq(Operand(rsp, 4 * kSystemPointerSize), arg5);
__ LoadAddress(arg5, ExternalReference::isolate_address(isolate));
__ movq(Operand(rsp, 5 * kSystemPointerSize), arg5);
#else
__ movq(r8, arg5);
__ LoadAddress(r9, ExternalReference::isolate_address(isolate));
#endif
{
AllowExternalCallThatCantCauseGC scope(masm);
__ CallCFunction(ExternalReference::new_deoptimizer_function(), 6);
}
// Preserve deoptimizer object in register rax and get the input
// frame descriptor pointer.
__ movq(rbx, Operand(rax, Deoptimizer::input_offset()));
// Fill in the input registers.
for (int i = kNumberOfRegisters - 1; i >= 0; i--) {
int offset =
(i * kSystemPointerSize) + FrameDescription::registers_offset();
__ PopQuad(Operand(rbx, offset));
}
// Fill in the double input registers.
int double_regs_offset = FrameDescription::double_registers_offset();
for (int i = 0; i < XMMRegister::kNumRegisters; i++) {
int dst_offset = i * kDoubleSize + double_regs_offset;
__ popq(Operand(rbx, dst_offset));
}
// Mark the stack as not iterable for the CPU profiler which won't be able to
// walk the stack without the return address.
__ movb(__ ExternalReferenceAsOperand(
ExternalReference::stack_is_iterable_address(isolate)),
Immediate(0));
// Remove the return address from the stack.
__ addq(rsp, Immediate(kPCOnStackSize));
// Compute a pointer to the unwinding limit in register rcx; that is
// the first stack slot not part of the input frame.
__ movq(rcx, Operand(rbx, FrameDescription::frame_size_offset()));
__ addq(rcx, rsp);
// Unwind the stack down to - but not including - the unwinding
// limit and copy the contents of the activation frame to the input
// frame description.
__ leaq(rdx, Operand(rbx, FrameDescription::frame_content_offset()));
Label pop_loop_header;
__ jmp(&pop_loop_header);
Label pop_loop;
__ bind(&pop_loop);
__ Pop(Operand(rdx, 0));
__ addq(rdx, Immediate(sizeof(intptr_t)));
__ bind(&pop_loop_header);
__ cmpq(rcx, rsp);
__ j(not_equal, &pop_loop);
// Compute the output frame in the deoptimizer.
__ pushq(rax);
__ PrepareCallCFunction(2);
__ movq(arg_reg_1, rax);
__ LoadAddress(arg_reg_2, ExternalReference::isolate_address(isolate));
{
AllowExternalCallThatCantCauseGC scope(masm);
__ CallCFunction(ExternalReference::compute_output_frames_function(), 2);
}
__ popq(rax);
__ movq(rsp, Operand(rax, Deoptimizer::caller_frame_top_offset()));
// Replace the current (input) frame with the output frames.
Label outer_push_loop, inner_push_loop, outer_loop_header, inner_loop_header;
// Outer loop state: rax = current FrameDescription**, rdx = one past the
// last FrameDescription**.
__ movl(rdx, Operand(rax, Deoptimizer::output_count_offset()));
__ movq(rax, Operand(rax, Deoptimizer::output_offset()));
__ leaq(rdx, Operand(rax, rdx, times_system_pointer_size, 0));
__ jmp(&outer_loop_header);
__ bind(&outer_push_loop);
// Inner loop state: rbx = current FrameDescription*, rcx = loop index.
__ movq(rbx, Operand(rax, 0));
__ movq(rcx, Operand(rbx, FrameDescription::frame_size_offset()));
__ jmp(&inner_loop_header);
__ bind(&inner_push_loop);
__ subq(rcx, Immediate(sizeof(intptr_t)));
__ Push(Operand(rbx, rcx, times_1, FrameDescription::frame_content_offset()));
__ bind(&inner_loop_header);
__ testq(rcx, rcx);
__ j(not_zero, &inner_push_loop);
__ addq(rax, Immediate(kSystemPointerSize));
__ bind(&outer_loop_header);
__ cmpq(rax, rdx);
__ j(below, &outer_push_loop);
for (int i = 0; i < config->num_allocatable_double_registers(); ++i) {
int code = config->GetAllocatableDoubleCode(i);
XMMRegister xmm_reg = XMMRegister::from_code(code);
int src_offset = code * kDoubleSize + double_regs_offset;
__ Movsd(xmm_reg, Operand(rbx, src_offset));
}
// Push pc and continuation from the last output frame.
__ PushQuad(Operand(rbx, FrameDescription::pc_offset()));
__ PushQuad(Operand(rbx, FrameDescription::continuation_offset()));
// Push the registers from the last output frame.
for (int i = 0; i < kNumberOfRegisters; i++) {
int offset =
(i * kSystemPointerSize) + FrameDescription::registers_offset();
__ PushQuad(Operand(rbx, offset));
}
// Restore the registers from the stack.
for (int i = kNumberOfRegisters - 1; i >= 0; i--) {
Register r = Register::from_code(i);
// Do not restore rsp, simply pop the value into the next register
// and overwrite this afterwards.
if (r == rsp) {
DCHECK_GT(i, 0);
r = Register::from_code(i - 1);
}
__ popq(r);
}
__ movb(__ ExternalReferenceAsOperand(
ExternalReference::stack_is_iterable_address(isolate)),
Immediate(1));
// Return to the continuation point.
__ ret(0);
}
Float32 RegisterValues::GetFloatRegister(unsigned n) const { Float32 RegisterValues::GetFloatRegister(unsigned n) const {
return Float32::FromBits( return Float32::FromBits(
@ -33,6 +236,8 @@ void FrameDescription::SetCallerConstantPool(unsigned offset, intptr_t value) {
void FrameDescription::SetPc(intptr_t pc) { pc_ = pc; } void FrameDescription::SetPc(intptr_t pc) { pc_ = pc; }
#undef __
} // namespace internal } // namespace internal
} // namespace v8 } // namespace v8

View File

@ -253,7 +253,8 @@ static void PrintRelocInfo(StringBuilder* out, Isolate* isolate,
host.as_wasm_code()->native_module()->GetRuntimeStubId( host.as_wasm_code()->native_module()->GetRuntimeStubId(
relocinfo->wasm_stub_call_address())); relocinfo->wasm_stub_call_address()));
out->AddFormatted(" ;; wasm stub: %s", runtime_stub_name); out->AddFormatted(" ;; wasm stub: %s", runtime_stub_name);
} else if (RelocInfo::IsRuntimeEntry(rmode) && isolate != nullptr) { } else if (RelocInfo::IsRuntimeEntry(rmode) && isolate &&
isolate->deoptimizer_data() != nullptr) {
// A runtime entry relocinfo might be a deoptimization bailout. // A runtime entry relocinfo might be a deoptimization bailout.
Address addr = relocinfo->target_address(); Address addr = relocinfo->target_address();
DeoptimizeKind type; DeoptimizeKind type;

View File

@ -1296,8 +1296,9 @@ void JSFunction::JSFunctionPrint(std::ostream& os) { // NOLINT
os << "\n - kind: " << shared().kind(); os << "\n - kind: " << shared().kind();
os << "\n - context: " << Brief(context()); os << "\n - context: " << Brief(context());
os << "\n - code: " << Brief(code()); os << "\n - code: " << Brief(code());
if (code().kind() == CodeKind::FOR_TESTING) { if (code().kind() == CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING) {
os << "\n - FOR_TESTING"; os << "\n - FunctionTester function";
} else if (ActiveTierIsIgnition()) { } else if (ActiveTierIsIgnition()) {
os << "\n - interpreted"; os << "\n - interpreted";
if (shared().HasBytecodeArray()) { if (shared().HasBytecodeArray()) {

View File

@ -57,10 +57,6 @@ class IsolateData final {
static constexpr int builtin_entry_table_offset() { static constexpr int builtin_entry_table_offset() {
return kBuiltinEntryTableOffset - kIsolateRootBias; return kBuiltinEntryTableOffset - kIsolateRootBias;
} }
static constexpr int builtin_entry_slot_offset(Builtins::Name builtin_index) {
CONSTEXPR_DCHECK(Builtins::IsBuiltinId(builtin_index));
return builtin_entry_table_offset() + builtin_index * kSystemPointerSize;
}
// Root-register-relative offset of the builtins table. // Root-register-relative offset of the builtins table.
static constexpr int builtins_table_offset() { static constexpr int builtins_table_offset() {

View File

@ -3076,6 +3076,8 @@ void Isolate::Deinit() {
ReleaseSharedPtrs(); ReleaseSharedPtrs();
delete deoptimizer_data_;
deoptimizer_data_ = nullptr;
string_table_.reset(); string_table_.reset();
builtins_.TearDown(); builtins_.TearDown();
bootstrapper_->TearDown(); bootstrapper_->TearDown();
@ -3535,6 +3537,8 @@ bool Isolate::Init(SnapshotData* startup_snapshot_data,
} }
DCHECK_NOT_NULL(wasm_engine_); DCHECK_NOT_NULL(wasm_engine_);
deoptimizer_data_ = new DeoptimizerData(heap());
if (setup_delegate_ == nullptr) { if (setup_delegate_ == nullptr) {
setup_delegate_ = new SetupIsolateDelegate(create_heap_objects); setup_delegate_ = new SetupIsolateDelegate(create_heap_objects);
} }

View File

@ -77,7 +77,7 @@ class CompilationStatistics;
class CompilerDispatcher; class CompilerDispatcher;
class Counters; class Counters;
class Debug; class Debug;
class Deoptimizer; class DeoptimizerData;
class DescriptorLookupCache; class DescriptorLookupCache;
class EmbeddedFileWriterInterface; class EmbeddedFileWriterInterface;
class EternalHandles; class EternalHandles;
@ -1025,17 +1025,7 @@ class V8_EXPORT_PRIVATE Isolate final : private HiddenFactory {
StubCache* load_stub_cache() { return load_stub_cache_; } StubCache* load_stub_cache() { return load_stub_cache_; }
StubCache* store_stub_cache() { return store_stub_cache_; } StubCache* store_stub_cache() { return store_stub_cache_; }
Deoptimizer* GetAndClearCurrentDeoptimizer() { DeoptimizerData* deoptimizer_data() { return deoptimizer_data_; }
Deoptimizer* result = current_deoptimizer_;
CHECK_NOT_NULL(result);
current_deoptimizer_ = nullptr;
return result;
}
void set_current_deoptimizer(Deoptimizer* deoptimizer) {
DCHECK_NULL(current_deoptimizer_);
DCHECK_NOT_NULL(deoptimizer);
current_deoptimizer_ = deoptimizer;
}
bool deoptimizer_lazy_throw() const { return deoptimizer_lazy_throw_; } bool deoptimizer_lazy_throw() const { return deoptimizer_lazy_throw_; }
void set_deoptimizer_lazy_throw(bool value) { void set_deoptimizer_lazy_throw(bool value) {
deoptimizer_lazy_throw_ = value; deoptimizer_lazy_throw_ = value;
@ -1741,7 +1731,7 @@ class V8_EXPORT_PRIVATE Isolate final : private HiddenFactory {
Logger* logger_ = nullptr; Logger* logger_ = nullptr;
StubCache* load_stub_cache_ = nullptr; StubCache* load_stub_cache_ = nullptr;
StubCache* store_stub_cache_ = nullptr; StubCache* store_stub_cache_ = nullptr;
Deoptimizer* current_deoptimizer_ = nullptr; DeoptimizerData* deoptimizer_data_ = nullptr;
bool deoptimizer_lazy_throw_ = false; bool deoptimizer_lazy_throw_ = false;
MaterializedObjectStore* materialized_object_store_ = nullptr; MaterializedObjectStore* materialized_object_store_ = nullptr;
bool capture_stack_trace_for_uncaught_exceptions_ = false; bool capture_stack_trace_for_uncaught_exceptions_ = false;

View File

@ -150,6 +150,10 @@ MaybeHandle<Code> Factory::CodeBuilder::BuildInternal(
if (result.is_null()) return MaybeHandle<Code>(); if (result.is_null()) return MaybeHandle<Code>();
} }
if (!is_movable_) {
result = heap->EnsureImmovableCode(result, object_size);
}
// The code object has not been fully initialized yet. We rely on the // The code object has not been fully initialized yet. We rely on the
// fact that no allocation will happen from this point on. // fact that no allocation will happen from this point on.
DisallowHeapAllocation no_gc; DisallowHeapAllocation no_gc;

View File

@ -837,6 +837,11 @@ class V8_EXPORT_PRIVATE Factory : public FactoryBase<Factory> {
return *this; return *this;
} }
CodeBuilder& set_immovable() {
is_movable_ = false;
return *this;
}
CodeBuilder& set_is_turbofanned() { CodeBuilder& set_is_turbofanned() {
is_turbofanned_ = true; is_turbofanned_ = true;
return *this; return *this;
@ -883,6 +888,7 @@ class V8_EXPORT_PRIVATE Factory : public FactoryBase<Factory> {
BasicBlockProfilerData* profiler_data_ = nullptr; BasicBlockProfilerData* profiler_data_ = nullptr;
bool is_executable_ = true; bool is_executable_ = true;
bool read_only_data_container_ = false; bool read_only_data_container_ = false;
bool is_movable_ = true;
bool is_turbofanned_ = false; bool is_turbofanned_ = false;
int stack_slots_ = 0; int stack_slots_ = 0;
}; };

View File

@ -4984,6 +4984,33 @@ void Heap::DisableInlineAllocation() {
} }
} }
HeapObject Heap::EnsureImmovableCode(HeapObject heap_object, int object_size) {
// Code objects which should stay at a fixed address are allocated either
// in the first page of code space, in large object space, or (during
// snapshot creation) the containing page is marked as immovable.
DCHECK(!heap_object.is_null());
#ifndef V8_ENABLE_THIRD_PARTY_HEAP
DCHECK(code_space_->Contains(heap_object));
#endif
DCHECK_GE(object_size, 0);
if (!Heap::IsImmovable(heap_object)) {
if (isolate()->serializer_enabled() ||
code_space_->first_page()->Contains(heap_object.address())) {
BasicMemoryChunk::FromHeapObject(heap_object)->MarkNeverEvacuate();
} else {
// Discard the first code allocation, which was on a page where it could
// be moved.
CreateFillerObjectAt(heap_object.address(), object_size,
ClearRecordedSlots::kNo);
heap_object = AllocateRawCodeInLargeObjectSpace(object_size);
UnprotectAndRegisterMemoryChunk(heap_object);
ZapCodeObject(heap_object.address(), object_size);
OnAllocationEvent(heap_object, object_size);
}
}
return heap_object;
}
HeapObject Heap::AllocateRawWithLightRetrySlowPath( HeapObject Heap::AllocateRawWithLightRetrySlowPath(
int size, AllocationType allocation, AllocationOrigin origin, int size, AllocationType allocation, AllocationOrigin origin,
AllocationAlignment alignment) { AllocationAlignment alignment) {
@ -5036,6 +5063,40 @@ HeapObject Heap::AllocateRawWithRetryOrFailSlowPath(
return HeapObject(); return HeapObject();
} }
// TODO(jkummerow): Refactor this. AllocateRaw should take an "immovability"
// parameter and just do what's necessary.
HeapObject Heap::AllocateRawCodeInLargeObjectSpace(int size) {
AllocationResult alloc = code_lo_space()->AllocateRaw(size);
HeapObject result;
if (alloc.To(&result)) {
DCHECK(result != ReadOnlyRoots(this).exception());
return result;
}
// Two GCs before panicking.
for (int i = 0; i < 2; i++) {
CollectGarbage(alloc.RetrySpace(),
GarbageCollectionReason::kAllocationFailure);
alloc = code_lo_space()->AllocateRaw(size);
if (alloc.To(&result)) {
DCHECK(result != ReadOnlyRoots(this).exception());
return result;
}
}
isolate()->counters()->gc_last_resort_from_handles()->Increment();
CollectAllAvailableGarbage(GarbageCollectionReason::kLastResort);
{
AlwaysAllocateScope scope(this);
alloc = code_lo_space()->AllocateRaw(size);
}
if (alloc.To(&result)) {
DCHECK(result != ReadOnlyRoots(this).exception());
return result;
}
// TODO(1181417): Fix this.
FatalProcessOutOfMemory("CALL_AND_RETRY_LAST");
return HeapObject();
}
void Heap::SetUp() { void Heap::SetUp() {
#ifdef V8_ENABLE_ALLOCATION_TIMEOUT #ifdef V8_ENABLE_ALLOCATION_TIMEOUT
allocation_timeout_ = NextAllocationTimeout(); allocation_timeout_ = NextAllocationTimeout();

View File

@ -1977,10 +1977,17 @@ class Heap {
int size, AllocationType allocation, AllocationOrigin origin, int size, AllocationType allocation, AllocationOrigin origin,
AllocationAlignment alignment = kWordAligned); AllocationAlignment alignment = kWordAligned);
V8_WARN_UNUSED_RESULT HeapObject AllocateRawCodeInLargeObjectSpace(int size);
// Allocates a heap object based on the map. // Allocates a heap object based on the map.
V8_WARN_UNUSED_RESULT AllocationResult Allocate(Map map, V8_WARN_UNUSED_RESULT AllocationResult Allocate(Map map,
AllocationType allocation); AllocationType allocation);
// Takes a code object and checks if it is on memory which is not subject to
// compaction. This method will return a new code object on an immovable
// memory location if the original code object was movable.
HeapObject EnsureImmovableCode(HeapObject heap_object, int object_size);
// Allocates a partial map for bootstrapping. // Allocates a partial map for bootstrapping.
V8_WARN_UNUSED_RESULT AllocationResult V8_WARN_UNUSED_RESULT AllocationResult
AllocatePartialMap(InstanceType instance_type, int instance_size); AllocatePartialMap(InstanceType instance_type, int instance_size);

View File

@ -2175,7 +2175,7 @@ void ExistingCodeLogger::LogCodeObject(Object object) {
return; // We log this later using LogCompiledFunctions. return; // We log this later using LogCompiledFunctions.
case CodeKind::BYTECODE_HANDLER: case CodeKind::BYTECODE_HANDLER:
return; // We log it later by walking the dispatch table. return; // We log it later by walking the dispatch table.
case CodeKind::FOR_TESTING: case CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING:
description = "STUB code"; description = "STUB code";
tag = CodeEventListener::STUB_TAG; tag = CodeEventListener::STUB_TAG;
break; break;

View File

@ -11,20 +11,22 @@
namespace v8 { namespace v8 {
namespace internal { namespace internal {
#define CODE_KIND_LIST(V) \ // TODO(jgruber): Convert deopt entries to builtins and rename
V(TURBOFAN) \ // DEOPT_ENTRIES_OR_FOR_TESTING to FOR_TESTING.
V(BYTECODE_HANDLER) \ #define CODE_KIND_LIST(V) \
V(FOR_TESTING) \ V(TURBOFAN) \
V(BUILTIN) \ V(BYTECODE_HANDLER) \
V(REGEXP) \ V(DEOPT_ENTRIES_OR_FOR_TESTING) \
V(WASM_FUNCTION) \ V(BUILTIN) \
V(WASM_TO_CAPI_FUNCTION) \ V(REGEXP) \
V(WASM_TO_JS_FUNCTION) \ V(WASM_FUNCTION) \
V(JS_TO_WASM_FUNCTION) \ V(WASM_TO_CAPI_FUNCTION) \
V(JS_TO_JS_FUNCTION) \ V(WASM_TO_JS_FUNCTION) \
V(C_WASM_ENTRY) \ V(JS_TO_WASM_FUNCTION) \
V(INTERPRETED_FUNCTION) \ V(JS_TO_JS_FUNCTION) \
V(NATIVE_CONTEXT_INDEPENDENT) \ V(C_WASM_ENTRY) \
V(INTERPRETED_FUNCTION) \
V(NATIVE_CONTEXT_INDEPENDENT) \
V(TURBOPROP) V(TURBOPROP)
enum class CodeKind { enum class CodeKind {

View File

@ -89,6 +89,13 @@ bool IsUnexpectedCodeObject(Isolate* isolate, HeapObject obj) {
if (!obj.IsCode()) return false; if (!obj.IsCode()) return false;
Code code = Code::cast(obj); Code code = Code::cast(obj);
// TODO(v8:8768): Deopt entry code should not be serialized.
if (code.kind() == CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING &&
isolate->deoptimizer_data() != nullptr) {
if (isolate->deoptimizer_data()->IsDeoptEntryCode(code)) return false;
}
if (code.kind() == CodeKind::REGEXP) return false; if (code.kind() == CodeKind::REGEXP) return false;
if (!code.is_builtin()) return true; if (!code.is_builtin()) return true;
if (code.is_off_heap_trampoline()) return false; if (code.is_off_heap_trampoline()) return false;

View File

@ -21,8 +21,9 @@ Handle<Code> AssembleCodeImpl(std::function<void(MacroAssembler&)> assemble) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
if (FLAG_print_code) { if (FLAG_print_code) {
code->Print(); code->Print();
} }

View File

@ -23,7 +23,8 @@ class CodeAssemblerTester {
const char* name = "test") const char* name = "test")
: zone_(isolate->allocator(), ZONE_NAME, kCompressGraphZone), : zone_(isolate->allocator(), ZONE_NAME, kCompressGraphZone),
scope_(isolate), scope_(isolate),
state_(isolate, &zone_, descriptor, CodeKind::FOR_TESTING, name, state_(isolate, &zone_, descriptor,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING, name,
PoisoningMitigationLevel::kDontPoison, Builtins::kNoBuiltinId) {} PoisoningMitigationLevel::kDontPoison, Builtins::kNoBuiltinId) {}
// Test generating code for a stub. Assumes VoidDescriptor call interface. // Test generating code for a stub. Assumes VoidDescriptor call interface.
@ -47,7 +48,8 @@ class CodeAssemblerTester {
const char* name = "test") const char* name = "test")
: zone_(isolate->allocator(), ZONE_NAME, kCompressGraphZone), : zone_(isolate->allocator(), ZONE_NAME, kCompressGraphZone),
scope_(isolate), scope_(isolate),
state_(isolate, &zone_, call_descriptor, CodeKind::FOR_TESTING, name, state_(isolate, &zone_, call_descriptor,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING, name,
PoisoningMitigationLevel::kDontPoison, Builtins::kNoBuiltinId) {} PoisoningMitigationLevel::kDontPoison, Builtins::kNoBuiltinId) {}
CodeAssemblerState* state() { return &state_; } CodeAssemblerState* state() { return &state_; }

View File

@ -91,7 +91,7 @@ class RawMachineAssemblerTester : public HandleAndZoneScope,
} }
private: private:
CodeKind kind_ = CodeKind::FOR_TESTING; CodeKind kind_ = CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING;
MaybeHandle<Code> code_; MaybeHandle<Code> code_;
}; };

View File

@ -965,10 +965,10 @@ class CodeGeneratorTester {
int extra_stack_space = 0) int extra_stack_space = 0)
: zone_(environment->main_zone()), : zone_(environment->main_zone()),
info_(ArrayVector("test"), environment->main_zone(), info_(ArrayVector("test"), environment->main_zone(),
CodeKind::FOR_TESTING), CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING),
linkage_(environment->test_descriptor()), linkage_(environment->test_descriptor()),
frame_(environment->test_descriptor()->CalculateFixedFrameSize( frame_(environment->test_descriptor()->CalculateFixedFrameSize(
CodeKind::FOR_TESTING)) { CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)) {
// Pick half of the stack parameters at random and move them into spill // Pick half of the stack parameters at random and move them into spill
// slots, separated by `extra_stack_space` bytes. // slots, separated by `extra_stack_space` bytes.
// When testing a move with stack slots using CheckAssembleMove or // When testing a move with stack slots using CheckAssembleMove or

View File

@ -109,7 +109,7 @@ TEST(TestLinkageStubCall) {
Zone zone(isolate->allocator(), ZONE_NAME); Zone zone(isolate->allocator(), ZONE_NAME);
Callable callable = Builtins::CallableFor(isolate, Builtins::kToNumber); Callable callable = Builtins::CallableFor(isolate, Builtins::kToNumber);
OptimizedCompilationInfo info(ArrayVector("test"), &zone, OptimizedCompilationInfo info(ArrayVector("test"), &zone,
CodeKind::FOR_TESTING); CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING);
auto call_descriptor = Linkage::GetStubCallDescriptor( auto call_descriptor = Linkage::GetStubCallDescriptor(
&zone, callable.descriptor(), 0, CallDescriptor::kNoFlags, &zone, callable.descriptor(), 0, CallDescriptor::kNoFlags,
Operator::kNoProperties); Operator::kNoProperties);
@ -130,7 +130,7 @@ TEST(TestFPLinkageStubCall) {
Callable callable = Callable callable =
Builtins::CallableFor(isolate, Builtins::kWasmFloat64ToNumber); Builtins::CallableFor(isolate, Builtins::kWasmFloat64ToNumber);
OptimizedCompilationInfo info(ArrayVector("test"), &zone, OptimizedCompilationInfo info(ArrayVector("test"), &zone,
CodeKind::FOR_TESTING); CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING);
auto call_descriptor = Linkage::GetStubCallDescriptor( auto call_descriptor = Linkage::GetStubCallDescriptor(
&zone, callable.descriptor(), 0, CallDescriptor::kNoFlags, &zone, callable.descriptor(), 0, CallDescriptor::kNoFlags,
Operator::kNoProperties); Operator::kNoProperties);

View File

@ -246,7 +246,7 @@ Handle<Code> CompileGraph(const char* name, CallDescriptor* call_descriptor,
Graph* graph, Schedule* schedule = nullptr) { Graph* graph, Schedule* schedule = nullptr) {
Isolate* isolate = CcTest::InitIsolateOnce(); Isolate* isolate = CcTest::InitIsolateOnce();
OptimizedCompilationInfo info(ArrayVector("testing"), graph->zone(), OptimizedCompilationInfo info(ArrayVector("testing"), graph->zone(),
CodeKind::FOR_TESTING); CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING);
Handle<Code> code = Pipeline::GenerateCodeForTesting( Handle<Code> code = Pipeline::GenerateCodeForTesting(
&info, isolate, call_descriptor, graph, &info, isolate, call_descriptor, graph,
AssemblerOptions::Default(isolate), schedule) AssemblerOptions::Default(isolate), schedule)

View File

@ -48,6 +48,7 @@
V(StressHandles) \ V(StressHandles) \
V(TestMemoryReducerSampleJsCalls) \ V(TestMemoryReducerSampleJsCalls) \
V(TestSizeOfObjects) \ V(TestSizeOfObjects) \
V(Regress5831) \
V(Regress10560) \ V(Regress10560) \
V(Regress538257) \ V(Regress538257) \
V(Regress587004) \ V(Regress587004) \

View File

@ -395,7 +395,9 @@ UNINITIALIZED_TEST(ConcurrentRecordRelocSlot) {
CodeDesc desc; CodeDesc desc;
masm.GetCode(i_isolate, &desc); masm.GetCode(i_isolate, &desc);
Handle<Code> code_handle = Handle<Code> code_handle =
Factory::CodeBuilder(i_isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(i_isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
heap::AbandonCurrentlyFreeMemory(heap->old_space()); heap::AbandonCurrentlyFreeMemory(heap->old_space());
Handle<HeapNumber> value_handle( Handle<HeapNumber> value_handle(
i_isolate->factory()->NewHeapNumber<AllocationType::kOld>(1.1)); i_isolate->factory()->NewHeapNumber<AllocationType::kOld>(1.1));

View File

@ -206,8 +206,9 @@ HEAP_TEST(TestNewSpaceRefsInCopiedCode) {
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
Handle<Code> copy; Handle<Code> copy;
{ {
@ -230,8 +231,9 @@ static void CheckFindCodeObject(Isolate* isolate) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
CHECK(code->IsCode()); CHECK(code->IsCode());
HeapObject obj = HeapObject::cast(*code); HeapObject obj = HeapObject::cast(*code);
@ -242,8 +244,9 @@ static void CheckFindCodeObject(Isolate* isolate) {
CHECK_EQ(*code, found); CHECK_EQ(*code, found);
} }
Handle<Code> copy = Handle<Code> copy = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
HeapObject obj_copy = HeapObject::cast(*copy); HeapObject obj_copy = HeapObject::cast(*copy);
Object not_right = Object not_right =
isolate->FindCodeObject(obj_copy.address() + obj_copy.Size() / 2); isolate->FindCodeObject(obj_copy.address() + obj_copy.Size() / 2);
@ -6517,6 +6520,68 @@ HEAP_TEST(Regress670675) {
DCHECK(marking->IsStopped()); DCHECK(marking->IsStopped());
} }
namespace {
Handle<Code> GenerateDummyImmovableCode(Isolate* isolate) {
Assembler assm(AssemblerOptions{});
const int kNumberOfNops = 1 << 10;
for (int i = 0; i < kNumberOfNops; i++) {
assm.nop(); // supported on all architectures
}
CodeDesc desc;
assm.GetCode(isolate, &desc);
Handle<Code> code = Factory::CodeBuilder(
isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.set_immovable()
.Build();
CHECK(code->IsCode());
return code;
}
} // namespace
HEAP_TEST(Regress5831) {
CcTest::InitializeVM();
Heap* heap = CcTest::heap();
Isolate* isolate = CcTest::i_isolate();
HandleScope handle_scope(isolate);
// Used to ensure that the generated code is not collected.
const int kInitialSize = 32;
Handle<FixedArray> array = isolate->factory()->NewFixedArray(kInitialSize);
// Ensure that all immovable code space pages are full and we overflow into
// LO_SPACE.
const int kMaxIterations = 1 << 16;
bool overflowed_into_lospace = false;
for (int i = 0; i < kMaxIterations; i++) {
Handle<Code> code = GenerateDummyImmovableCode(isolate);
array = FixedArray::SetAndGrow(isolate, array, i, code);
CHECK(heap->code_space()->Contains(*code) ||
heap->code_lo_space()->Contains(*code));
if (heap->code_lo_space()->Contains(*code)) {
overflowed_into_lospace = true;
break;
}
}
CHECK(overflowed_into_lospace);
// Fake a serializer run.
isolate->serializer_enabled_ = true;
// Generate the code.
Handle<Code> code = GenerateDummyImmovableCode(isolate);
CHECK_GE(MemoryChunkLayout::MaxRegularCodeObjectSize(), code->Size());
CHECK(!heap->code_space()->first_page()->Contains(code->address()));
// Ensure it's not in large object space.
MemoryChunk* chunk = MemoryChunk::FromHeapObject(*code);
CHECK(chunk->owner_identity() != LO_SPACE);
CHECK(chunk->NeverEvacuate());
}
HEAP_TEST(RegressMissingWriteBarrierInAllocate) { HEAP_TEST(RegressMissingWriteBarrierInAllocate) {
if (!FLAG_incremental_marking) return; if (!FLAG_incremental_marking) return;
ManualGCScope manual_gc_scope; ManualGCScope manual_gc_scope;
@ -7258,8 +7323,9 @@ TEST(Regress10900) {
masm.Push(ReadOnlyRoots(heap).undefined_value_handle()); masm.Push(ReadOnlyRoots(heap).undefined_value_handle());
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
{ {
// Generate multiple code pages. // Generate multiple code pages.
CodeSpaceMemoryModificationScope modification_scope(isolate->heap()); CodeSpaceMemoryModificationScope modification_scope(isolate->heap());

View File

@ -51,7 +51,9 @@ TEST(WeakReferencesBasic) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
CHECK(code->IsCode()); CHECK(code->IsCode());
lh->set_data1(HeapObjectReference::Weak(*code)); lh->set_data1(HeapObjectReference::Weak(*code));

View File

@ -203,7 +203,8 @@ TEST(TryProbeStubCache) {
// Generate some number of handlers. // Generate some number of handlers.
for (int i = 0; i < 30; i++) { for (int i = 0; i < 30; i++) {
handlers.push_back(CreateCodeOfKind(CodeKind::FOR_TESTING)); handlers.push_back(
CreateCodeOfKind(CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING));
} }
// Ensure that GC does happen because from now on we are going to fill our // Ensure that GC does happen because from now on we are going to fill our

View File

@ -60,8 +60,9 @@ TEST(0) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -96,8 +97,9 @@ TEST(1) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -141,8 +143,9 @@ TEST(2) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -187,8 +190,9 @@ TEST(3) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -318,7 +322,9 @@ TEST(4) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -380,7 +386,9 @@ TEST(5) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -410,8 +418,9 @@ TEST(6) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -477,8 +486,9 @@ static void TestRoundingMode(VCVTTypes types,
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -659,8 +669,9 @@ TEST(8) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -768,8 +779,9 @@ TEST(9) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -873,8 +885,9 @@ TEST(10) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -967,8 +980,9 @@ TEST(11) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -1093,7 +1107,9 @@ TEST(13) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -1164,8 +1180,9 @@ TEST(14) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -2045,7 +2062,9 @@ TEST(15) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -2320,8 +2339,9 @@ TEST(16) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -2399,7 +2419,9 @@ TEST(sdiv) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -2459,7 +2481,9 @@ TEST(udiv) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -2487,8 +2511,9 @@ TEST(smmla) {
__ bx(lr); __ bx(lr);
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -2512,8 +2537,9 @@ TEST(smmul) {
__ bx(lr); __ bx(lr);
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -2537,8 +2563,9 @@ TEST(sxtb) {
__ bx(lr); __ bx(lr);
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -2562,8 +2589,9 @@ TEST(sxtab) {
__ bx(lr); __ bx(lr);
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -2587,8 +2615,9 @@ TEST(sxth) {
__ bx(lr); __ bx(lr);
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -2612,8 +2641,9 @@ TEST(sxtah) {
__ bx(lr); __ bx(lr);
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -2637,8 +2667,9 @@ TEST(uxtb) {
__ bx(lr); __ bx(lr);
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -2662,8 +2693,9 @@ TEST(uxtab) {
__ bx(lr); __ bx(lr);
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -2687,8 +2719,9 @@ TEST(uxth) {
__ bx(lr); __ bx(lr);
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -2712,8 +2745,9 @@ TEST(uxtah) {
__ bx(lr); __ bx(lr);
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -2754,7 +2788,9 @@ TEST(rbit) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
@ -2834,7 +2870,8 @@ TEST(code_relative_offset) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING) Handle<Code> code = Factory::CodeBuilder(
isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.set_self_reference(code_object) .set_self_reference(code_object)
.Build(); .Build();
auto f = GeneratedCode<F_iiiii>::FromCode(*code); auto f = GeneratedCode<F_iiiii>::FromCode(*code);
@ -2874,8 +2911,9 @@ TEST(msr_mrs) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -2972,7 +3010,9 @@ TEST(ARMv8_float32_vrintX) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -3073,7 +3113,9 @@ TEST(ARMv8_vrintX) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -3210,7 +3252,9 @@ TEST(ARMv8_vsel) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -3301,7 +3345,9 @@ TEST(ARMv8_vminmax_f64) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -3381,7 +3427,9 @@ TEST(ARMv8_vminmax_f32) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -3514,7 +3562,9 @@ static GeneratedCode<F_ppiii> GenerateMacroFloatMinMax(
CodeDesc desc; CodeDesc desc;
assm.GetCode(assm.isolate(), &desc); assm.GetCode(assm.isolate(), &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(assm.isolate(), desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(assm.isolate(), desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -3676,8 +3726,9 @@ TEST(unaligned_loads) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -3719,8 +3770,9 @@ TEST(unaligned_stores) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -3819,8 +3871,9 @@ TEST(vswp) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -4032,7 +4085,9 @@ TEST(split_add_immediate) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -4052,7 +4107,9 @@ TEST(split_add_immediate) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -4075,7 +4132,9 @@ TEST(split_add_immediate) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);

View File

@ -158,16 +158,18 @@ static void InitializeVM() {
#define RUN() simulator.RunFrom(reinterpret_cast<Instruction*>(code->entry())) #define RUN() simulator.RunFrom(reinterpret_cast<Instruction*>(code->entry()))
#define END() \ #define END() \
__ Debug("End test.", __LINE__, TRACE_DISABLE | LOG_ALL); \ __ Debug("End test.", __LINE__, TRACE_DISABLE | LOG_ALL); \
core.Dump(&masm); \ core.Dump(&masm); \
__ PopCalleeSavedRegisters(); \ __ PopCalleeSavedRegisters(); \
__ Ret(); \ __ Ret(); \
{ \ { \
CodeDesc desc; \ CodeDesc desc; \
__ GetCode(masm.isolate(), &desc); \ __ GetCode(masm.isolate(), &desc); \
code = Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); \ code = Factory::CodeBuilder(isolate, desc, \
if (FLAG_print_code) code->Print(); \ CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING) \
.Build(); \
if (FLAG_print_code) code->Print(); \
} }
#else // ifdef USE_SIMULATOR. #else // ifdef USE_SIMULATOR.
@ -204,15 +206,17 @@ static void InitializeVM() {
f.Call(); \ f.Call(); \
} }
#define END() \ #define END() \
core.Dump(&masm); \ core.Dump(&masm); \
__ PopCalleeSavedRegisters(); \ __ PopCalleeSavedRegisters(); \
__ Ret(); \ __ Ret(); \
{ \ { \
CodeDesc desc; \ CodeDesc desc; \
__ GetCode(masm.isolate(), &desc); \ __ GetCode(masm.isolate(), &desc); \
code = Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); \ code = Factory::CodeBuilder(isolate, desc, \
if (FLAG_print_code) code->Print(); \ CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING) \
.Build(); \
if (FLAG_print_code) code->Print(); \
} }
#endif // ifdef USE_SIMULATOR. #endif // ifdef USE_SIMULATOR.
@ -14883,7 +14887,8 @@ TEST(pool_size) {
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
code = Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING) code = Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.set_self_reference(masm.CodeObject()) .set_self_reference(masm.CodeObject())
.Build(); .Build();

View File

@ -27,14 +27,14 @@
#include <stdlib.h> #include <stdlib.h>
#include "src/init/v8.h"
#include "src/base/platform/platform.h" #include "src/base/platform/platform.h"
#include "src/base/utils/random-number-generator.h" #include "src/base/utils/random-number-generator.h"
#include "src/codegen/assembler-inl.h" #include "src/codegen/assembler-inl.h"
#include "src/codegen/macro-assembler.h" #include "src/codegen/macro-assembler.h"
#include "src/deoptimizer/deoptimizer.h"
#include "src/diagnostics/disassembler.h" #include "src/diagnostics/disassembler.h"
#include "src/heap/factory.h" #include "src/heap/factory.h"
#include "src/init/v8.h"
#include "src/utils/ostreams.h" #include "src/utils/ostreams.h"
#include "test/cctest/cctest.h" #include "test/cctest/cctest.h"
@ -62,8 +62,9 @@ TEST(AssemblerIa320) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -100,8 +101,9 @@ TEST(AssemblerIa321) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -142,8 +144,9 @@ TEST(AssemblerIa322) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -171,8 +174,9 @@ TEST(AssemblerIa323) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -200,8 +204,9 @@ TEST(AssemblerIa324) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -228,8 +233,9 @@ TEST(AssemblerIa325) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
F0 f = FUNCTION_CAST<F0>(code->entry()); F0 f = FUNCTION_CAST<F0>(code->entry());
int res = f(); int res = f();
CHECK_EQ(42, res); CHECK_EQ(42, res);
@ -261,8 +267,9 @@ TEST(AssemblerIa326) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -293,8 +300,9 @@ TEST(AssemblerIa328) {
__ ret(0); __ ret(0);
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -376,8 +384,9 @@ TEST(AssemblerMultiByteNop) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
CHECK(code->IsCode()); CHECK(code->IsCode());
F0 f = FUNCTION_CAST<F0>(code->entry()); F0 f = FUNCTION_CAST<F0>(code->entry());
@ -427,8 +436,9 @@ void DoSSE2(const v8::FunctionCallbackInfo<v8::Value>& args) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
F0 f = FUNCTION_CAST<F0>(code->entry()); F0 f = FUNCTION_CAST<F0>(code->entry());
int res = f(); int res = f();
@ -492,8 +502,9 @@ TEST(AssemblerIa32Extractps) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -531,8 +542,9 @@ TEST(AssemblerIa32SSE) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -564,8 +576,9 @@ TEST(AssemblerIa32SSE3) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -792,8 +805,9 @@ TEST(AssemblerX64FMA_sd) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -1020,8 +1034,9 @@ TEST(AssemblerX64FMA_ss) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -1128,8 +1143,9 @@ TEST(AssemblerIa32BMI1) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -1176,8 +1192,9 @@ TEST(AssemblerIa32LZCNT) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -1224,8 +1241,9 @@ TEST(AssemblerIa32POPCNT) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -1370,8 +1388,9 @@ TEST(AssemblerIa32BMI2) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -1414,8 +1433,9 @@ TEST(AssemblerIa32JumpTables1) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -1462,8 +1482,9 @@ TEST(AssemblerIa32JumpTables2) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -1505,8 +1526,9 @@ TEST(Regress621926) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
@ -1517,29 +1539,6 @@ TEST(Regress621926) {
CHECK_EQ(1, f()); CHECK_EQ(1, f());
} }
TEST(DeoptExitSizeIsFixed) {
CHECK(Deoptimizer::kSupportsFixedDeoptExitSizes);
Isolate* isolate = CcTest::i_isolate();
HandleScope handles(isolate);
v8::internal::byte buffer[256];
MacroAssembler masm(isolate, v8::internal::CodeObjectRequired::kYes,
ExternalAssemblerBuffer(buffer, sizeof(buffer)));
STATIC_ASSERT(static_cast<int>(kFirstDeoptimizeKind) == 0);
for (int i = 0; i < kDeoptimizeKindCount; i++) {
DeoptimizeKind kind = static_cast<DeoptimizeKind>(i);
Builtins::Name target = Deoptimizer::GetDeoptimizationEntry(isolate, kind);
Label before_exit;
masm.bind(&before_exit);
masm.CallForDeoptimization(target, 42, &before_exit, kind, nullptr);
CHECK_EQ(masm.SizeOfCodeGeneratedSince(&before_exit),
kind == DeoptimizeKind::kLazy
? Deoptimizer::kLazyDeoptExitSize
: Deoptimizer::kNonLazyDeoptExitSize);
}
}
#undef __ #undef __
} // namespace internal } // namespace internal

View File

@ -64,8 +64,9 @@ TEST(MIPS0) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F2>::FromCode(*code); auto f = GeneratedCode<F2>::FromCode(*code);
int res = reinterpret_cast<int>(f.Call(0xAB0, 0xC, 0, 0, 0)); int res = reinterpret_cast<int>(f.Call(0xAB0, 0xC, 0, 0, 0));
CHECK_EQ(static_cast<int32_t>(0xABC), res); CHECK_EQ(static_cast<int32_t>(0xABC), res);
@ -99,8 +100,9 @@ TEST(MIPS1) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F1>::FromCode(*code); auto f = GeneratedCode<F1>::FromCode(*code);
int res = reinterpret_cast<int>(f.Call(50, 0, 0, 0, 0)); int res = reinterpret_cast<int>(f.Call(50, 0, 0, 0, 0));
CHECK_EQ(1275, res); CHECK_EQ(1275, res);
@ -236,8 +238,9 @@ TEST(MIPS2) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F2>::FromCode(*code); auto f = GeneratedCode<F2>::FromCode(*code);
int res = reinterpret_cast<int>(f.Call(0xAB0, 0xC, 0, 0, 0)); int res = reinterpret_cast<int>(f.Call(0xAB0, 0xC, 0, 0, 0));
CHECK_EQ(static_cast<int32_t>(0x31415926), res); CHECK_EQ(static_cast<int32_t>(0x31415926), res);
@ -336,8 +339,9 @@ TEST(MIPS3) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
// Double test values. // Double test values.
t.a = 1.5e14; t.a = 1.5e14;
@ -438,8 +442,9 @@ TEST(MIPS4) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
t.a = 1.5e22; t.a = 1.5e22;
t.b = 2.75e11; t.b = 2.75e11;
@ -499,8 +504,9 @@ TEST(MIPS5) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
t.a = 1.5e4; t.a = 1.5e4;
t.b = 2.75e8; t.b = 2.75e8;
@ -567,8 +573,9 @@ TEST(MIPS6) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
t.ui = 0x11223344; t.ui = 0x11223344;
t.si = 0x99AABBCC; t.si = 0x99AABBCC;
@ -659,8 +666,9 @@ TEST(MIPS7) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
t.a = 1.5e14; t.a = 1.5e14;
t.b = 2.75e11; t.b = 2.75e11;
@ -756,7 +764,9 @@ TEST(MIPS8) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
t.input = 0x12345678; t.input = 0x12345678;
f.Call(&t, 0x0, 0, 0, 0); f.Call(&t, 0x0, 0, 0, 0);
@ -800,8 +810,9 @@ TEST(MIPS9) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
USE(code); USE(code);
} }
@ -851,8 +862,9 @@ TEST(MIPS10) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
t.a = 2.147483646e+09; // 0x7FFFFFFE -> 0xFF80000041DFFFFF as double. t.a = 2.147483646e+09; // 0x7FFFFFFE -> 0xFF80000041DFFFFF as double.
t.b_word = 0x0FF00FF0; // 0x0FF00FF0 -> 0x as double. t.b_word = 0x0FF00FF0; // 0x0FF00FF0 -> 0x as double.
@ -978,8 +990,9 @@ TEST(MIPS11) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
t.reg_init = 0xAABBCCDD; t.reg_init = 0xAABBCCDD;
t.mem_init = 0x11223344; t.mem_init = 0x11223344;
@ -1103,8 +1116,9 @@ TEST(MIPS12) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
t.x = 1; t.x = 1;
t.y = 2; t.y = 2;
@ -1156,8 +1170,9 @@ TEST(MIPS13) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
t.cvt_big_in = 0xFFFFFFFF; t.cvt_big_in = 0xFFFFFFFF;
@ -1276,8 +1291,9 @@ TEST(MIPS14) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
t.round_up_in = 123.51; t.round_up_in = 123.51;
@ -1381,7 +1397,9 @@ TEST(seleqz_selnez) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
(f.Call(&test, 0, 0, 0, 0)); (f.Call(&test, 0, 0, 0, 0));
@ -1495,7 +1513,9 @@ TEST(min_max) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (int i = 0; i < kTableLength; i++) { for (int i = 0; i < kTableLength; i++) {
test.a = inputsa[i]; test.a = inputsa[i];
@ -1605,7 +1625,9 @@ TEST(rint_d) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (int j = 0; j < 4; j++) { for (int j = 0; j < 4; j++) {
@ -1652,7 +1674,9 @@ TEST(sel) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
const int test_size = 3; const int test_size = 3;
@ -1784,7 +1808,9 @@ TEST(rint_s) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (int j = 0; j < 4; j++) { for (int j = 0; j < 4; j++) {
@ -1827,8 +1853,9 @@ TEST(Cvt_d_uw) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (int i = 0; i < kTableLength; i++) { for (int i = 0; i < kTableLength; i++) {
test.input = inputs[i]; test.input = inputs[i];
@ -1909,7 +1936,9 @@ TEST(mina_maxa) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (int i = 0; i < kTableLength; i++) { for (int i = 0; i < kTableLength; i++) {
test.a = inputsa[i]; test.a = inputsa[i];
@ -1989,7 +2018,9 @@ TEST(trunc_l) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (int i = 0; i < kTableLength; i++) { for (int i = 0; i < kTableLength; i++) {
test.a = inputs_D[i]; test.a = inputs_D[i];
@ -2069,7 +2100,9 @@ TEST(movz_movn) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (int i = 0; i < kTableLength; i++) { for (int i = 0; i < kTableLength; i++) {
test.a = inputs_D[i]; test.a = inputs_D[i];
@ -2170,7 +2203,9 @@ TEST(movt_movd) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
(f.Call(&test, 0, 0, 0, 0)); (f.Call(&test, 0, 0, 0, 0));
@ -2254,8 +2289,9 @@ TEST(cvt_w_d) {
Test test; Test test;
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (int j = 0; j < 4; j++) { for (int j = 0; j < 4; j++) {
test.fcsr = fcsr_inputs[j]; test.fcsr = fcsr_inputs[j];
@ -2321,8 +2357,9 @@ TEST(trunc_w) {
Test test; Test test;
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (int i = 0; i < kTableLength; i++) { for (int i = 0; i < kTableLength; i++) {
test.a = inputs_D[i]; test.a = inputs_D[i];
@ -2390,8 +2427,9 @@ TEST(round_w) {
Test test; Test test;
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (int i = 0; i < kTableLength; i++) { for (int i = 0; i < kTableLength; i++) {
test.a = inputs_D[i]; test.a = inputs_D[i];
@ -2462,7 +2500,9 @@ TEST(round_l) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (int i = 0; i < kTableLength; i++) { for (int i = 0; i < kTableLength; i++) {
test.a = inputs_D[i]; test.a = inputs_D[i];
@ -2534,8 +2574,9 @@ TEST(sub) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (int i = 0; i < kTableLength; i++) { for (int i = 0; i < kTableLength; i++) {
test.a = inputfs_S[i]; test.a = inputfs_S[i];
@ -2613,8 +2654,9 @@ TEST(sqrt_rsqrt_recip) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (int i = 0; i < kTableLength; i++) { for (int i = 0; i < kTableLength; i++) {
@ -2693,8 +2735,9 @@ TEST(neg) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (int i = 0; i < kTableLength; i++) { for (int i = 0; i < kTableLength; i++) {
test.a = inputs_S[i]; test.a = inputs_S[i];
@ -2750,8 +2793,9 @@ TEST(mul) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (int i = 0; i < kTableLength; i++) { for (int i = 0; i < kTableLength; i++) {
test.a = inputfs_S[i]; test.a = inputfs_S[i];
@ -2806,8 +2850,9 @@ TEST(mov) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (int i = 0; i < kTableLength; i++) { for (int i = 0; i < kTableLength; i++) {
test.a = inputs_D[i]; test.a = inputs_D[i];
@ -2873,8 +2918,9 @@ TEST(floor_w) {
Test test; Test test;
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (int i = 0; i < kTableLength; i++) { for (int i = 0; i < kTableLength; i++) {
test.a = inputs_D[i]; test.a = inputs_D[i];
@ -2945,7 +2991,9 @@ TEST(floor_l) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (int i = 0; i < kTableLength; i++) { for (int i = 0; i < kTableLength; i++) {
test.a = inputs_D[i]; test.a = inputs_D[i];
@ -3016,8 +3064,9 @@ TEST(ceil_w) {
Test test; Test test;
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (int i = 0; i < kTableLength; i++) { for (int i = 0; i < kTableLength; i++) {
test.a = inputs_D[i]; test.a = inputs_D[i];
@ -3088,7 +3137,9 @@ TEST(ceil_l) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (int i = 0; i < kTableLength; i++) { for (int i = 0; i < kTableLength; i++) {
test.a = inputs_D[i]; test.a = inputs_D[i];
@ -3155,8 +3206,9 @@ TEST(jump_tables1) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -3220,8 +3272,9 @@ TEST(jump_tables2) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -3292,8 +3345,9 @@ TEST(jump_tables3) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -3344,7 +3398,9 @@ TEST(BITSWAP) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
t.r1 = 0x781A15C3; t.r1 = 0x781A15C3;
t.r2 = 0x8B71FCDE; t.r2 = 0x8B71FCDE;
@ -3478,7 +3534,9 @@ TEST(class_fmt) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
t.dSignalingNan = std::numeric_limits<double>::signaling_NaN(); t.dSignalingNan = std::numeric_limits<double>::signaling_NaN();
@ -3568,8 +3626,9 @@ TEST(ABS) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
test.a = -2.0; test.a = -2.0;
test.b = -2.0; test.b = -2.0;
@ -3661,8 +3720,9 @@ TEST(ADD_FMT) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
test.a = 2.0; test.a = 2.0;
test.b = 3.0; test.b = 3.0;
@ -3816,7 +3876,9 @@ TEST(C_COND_FMT) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
test.dOp1 = 2.0; test.dOp1 = 2.0;
test.dOp2 = 3.0; test.dOp2 = 3.0;
@ -4016,7 +4078,9 @@ TEST(CMP_COND_FMT) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
uint64_t dTrue = 0xFFFFFFFFFFFFFFFF; uint64_t dTrue = 0xFFFFFFFFFFFFFFFF;
uint64_t dFalse = 0x0000000000000000; uint64_t dFalse = 0x0000000000000000;
@ -4201,8 +4265,9 @@ TEST(CVT) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
test.cvt_d_s_in = -0.51; test.cvt_d_s_in = -0.51;
@ -4413,8 +4478,9 @@ TEST(DIV_FMT) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
@ -4505,8 +4571,9 @@ uint32_t run_align(uint32_t rs_value, uint32_t rt_value, uint8_t bp) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F2>::FromCode(*code); auto f = GeneratedCode<F2>::FromCode(*code);
@ -4560,8 +4627,9 @@ uint32_t run_aluipc(int16_t offset) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F2>::FromCode(*code); auto f = GeneratedCode<F2>::FromCode(*code);
PC = (uint32_t)code->entry(); // Set the program counter. PC = (uint32_t)code->entry(); // Set the program counter.
@ -4613,8 +4681,9 @@ uint32_t run_auipc(int16_t offset) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F2>::FromCode(*code); auto f = GeneratedCode<F2>::FromCode(*code);
PC = (uint32_t)code->entry(); // Set the program counter. PC = (uint32_t)code->entry(); // Set the program counter.
@ -4688,8 +4757,9 @@ uint32_t run_lwpc(int offset) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F2>::FromCode(*code); auto f = GeneratedCode<F2>::FromCode(*code);
@ -4768,8 +4838,9 @@ uint32_t run_jic(int16_t offset) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F2>::FromCode(*code); auto f = GeneratedCode<F2>::FromCode(*code);
@ -4839,8 +4910,9 @@ uint64_t run_beqzc(int32_t value, int32_t offset) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F2>::FromCode(*code); auto f = GeneratedCode<F2>::FromCode(*code);
@ -4945,8 +5017,9 @@ void run_bz_bnz(TestCaseMsaBranch* input, Branch GenerateBranch,
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -5115,8 +5188,9 @@ uint32_t run_jialc(int16_t offset) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F2>::FromCode(*code); auto f = GeneratedCode<F2>::FromCode(*code);
@ -5163,8 +5237,9 @@ static uint32_t run_addiupc(int32_t imm19) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F2>::FromCode(*code); auto f = GeneratedCode<F2>::FromCode(*code);
PC = (uint32_t)code->entry(); // Set the program counter. PC = (uint32_t)code->entry(); // Set the program counter.
@ -5245,8 +5320,9 @@ int32_t run_bc(int32_t offset) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F2>::FromCode(*code); auto f = GeneratedCode<F2>::FromCode(*code);
@ -5326,8 +5402,9 @@ int32_t run_balc(int32_t offset) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F2>::FromCode(*code); auto f = GeneratedCode<F2>::FromCode(*code);
@ -5350,8 +5427,9 @@ uint32_t run_aui(uint32_t rs, uint16_t offset) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F2>::FromCode(*code); auto f = GeneratedCode<F2>::FromCode(*code);
@ -5439,8 +5517,9 @@ uint32_t run_bal(int16_t offset) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F2>::FromCode(*code); auto f = GeneratedCode<F2>::FromCode(*code);
@ -5491,8 +5570,9 @@ TEST(Trampoline) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F2>::FromCode(*code); auto f = GeneratedCode<F2>::FromCode(*code);
int32_t res = reinterpret_cast<int32_t>(f.Call(42, 42, 0, 0, 0)); int32_t res = reinterpret_cast<int32_t>(f.Call(42, 42, 0, 0, 0));
@ -5618,8 +5698,9 @@ void helper_madd_msub_maddf_msubf(F func) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
const size_t kTableLength = sizeof(test_cases) / sizeof(TestCaseMaddMsub<T>); const size_t kTableLength = sizeof(test_cases) / sizeof(TestCaseMaddMsub<T>);
@ -5703,8 +5784,9 @@ uint32_t run_Subu(uint32_t imm, int32_t num_instr) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F2>::FromCode(*code); auto f = GeneratedCode<F2>::FromCode(*code);
uint32_t res = reinterpret_cast<uint32_t>(f.Call(0, 0, 0, 0, 0)); uint32_t res = reinterpret_cast<uint32_t>(f.Call(0, 0, 0, 0, 0));
@ -5807,8 +5889,9 @@ TEST(MSA_fill_copy) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -5875,8 +5958,9 @@ TEST(MSA_fill_copy_2) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -5932,8 +6016,9 @@ TEST(MSA_fill_copy_3) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -5977,8 +6062,9 @@ void run_msa_insert(int32_t rs_value, int n, msa_reg_t* w) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -6078,7 +6164,9 @@ TEST(MSA_move_v) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -6123,7 +6211,9 @@ void run_msa_sldi(OperFunc GenerateOperation,
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -6207,8 +6297,9 @@ void run_msa_ctc_cfc(uint32_t value) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -6317,8 +6408,9 @@ void run_msa_i8(SecondaryField opcode, uint64_t ws_lo, uint64_t ws_hi,
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -6495,8 +6587,9 @@ uint32_t run_Ins(uint32_t imm, uint32_t source, uint16_t pos, uint16_t size) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F2>::FromCode(*code); auto f = GeneratedCode<F2>::FromCode(*code);
uint32_t res = reinterpret_cast<uint32_t>(f.Call(0, 0, 0, 0, 0)); uint32_t res = reinterpret_cast<uint32_t>(f.Call(0, 0, 0, 0, 0));
@ -6545,8 +6638,9 @@ uint32_t run_Ext(uint32_t source, uint16_t pos, uint16_t size) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F2>::FromCode(*code); auto f = GeneratedCode<F2>::FromCode(*code);
uint32_t res = reinterpret_cast<uint32_t>(f.Call(0, 0, 0, 0, 0)); uint32_t res = reinterpret_cast<uint32_t>(f.Call(0, 0, 0, 0, 0));
@ -6607,8 +6701,9 @@ void run_msa_i5(struct TestCaseMsaI5* input, bool i5_sign_ext,
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -7027,8 +7122,9 @@ void run_msa_2r(const struct TestCaseMsa2R* input,
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -8077,8 +8173,9 @@ void run_msa_vector(struct TestCaseMsaVector* input,
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -8165,8 +8262,9 @@ void run_msa_bit(struct TestCaseMsaBit* input, InstFunc GenerateInstructionFunc,
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -8638,8 +8736,9 @@ void run_msa_i10(int32_t input, InstFunc GenerateVectorInstructionFunc,
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -8716,8 +8815,9 @@ void run_msa_mi10(InstFunc GenerateVectorInstructionFunc) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -8795,8 +8895,9 @@ void run_msa_3r(struct TestCaseMsa3R* input, InstFunc GenerateI5InstructionFunc,
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -9801,8 +9902,9 @@ void run_msa_3rf(const struct TestCaseMsa3RF* input,
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif

File diff suppressed because it is too large Load Diff

View File

@ -60,8 +60,9 @@ TEST(0) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif
@ -96,8 +97,9 @@ TEST(1) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif
@ -144,8 +146,9 @@ TEST(2) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif
@ -213,8 +216,9 @@ TEST(3) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif
@ -327,7 +331,7 @@ TEST(4) {
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Object code = isolate->heap()->CreateCode( Object code = isolate->heap()->CreateCode(
desc, desc,
CodeKind::FOR_TESTING, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING,
Handle<Code>())->ToObjectChecked(); Handle<Code>())->ToObjectChecked();
CHECK(code->IsCode()); CHECK(code->IsCode());
#ifdef DEBUG #ifdef DEBUG
@ -387,7 +391,7 @@ TEST(5) {
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Object code = isolate->heap()->CreateCode( Object code = isolate->heap()->CreateCode(
desc, desc,
CodeKind::FOR_TESTING, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING,
Handle<Code>())->ToObjectChecked(); Handle<Code>())->ToObjectChecked();
CHECK(code->IsCode()); CHECK(code->IsCode());
#ifdef DEBUG #ifdef DEBUG
@ -422,7 +426,7 @@ TEST(6) {
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Object code = isolate->heap()->CreateCode( Object code = isolate->heap()->CreateCode(
desc, desc,
CodeKind::FOR_TESTING, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING,
Handle<Code>())->ToObjectChecked(); Handle<Code>())->ToObjectChecked();
CHECK(code->IsCode()); CHECK(code->IsCode());
#ifdef DEBUG #ifdef DEBUG
@ -497,7 +501,7 @@ static void TestRoundingMode(VCVTTypes types,
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Object code = isolate->heap()->CreateCode( Object code = isolate->heap()->CreateCode(
desc, desc,
CodeKind::FOR_TESTING, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING,
Handle<Code>())->ToObjectChecked(); Handle<Code>())->ToObjectChecked();
CHECK(code->IsCode()); CHECK(code->IsCode());
#ifdef DEBUG #ifdef DEBUG
@ -684,7 +688,7 @@ TEST(8) {
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Object code = isolate->heap()->CreateCode( Object code = isolate->heap()->CreateCode(
desc, desc,
CodeKind::FOR_TESTING, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING,
Handle<Code>())->ToObjectChecked(); Handle<Code>())->ToObjectChecked();
CHECK(code->IsCode()); CHECK(code->IsCode());
#ifdef DEBUG #ifdef DEBUG
@ -799,7 +803,7 @@ TEST(9) {
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Object code = isolate->heap()->CreateCode( Object code = isolate->heap()->CreateCode(
desc, desc,
CodeKind::FOR_TESTING, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING,
Handle<Code>())->ToObjectChecked(); Handle<Code>())->ToObjectChecked();
CHECK(code->IsCode()); CHECK(code->IsCode());
#ifdef DEBUG #ifdef DEBUG
@ -910,7 +914,7 @@ TEST(10) {
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Object code = isolate->heap()->CreateCode( Object code = isolate->heap()->CreateCode(
desc, desc,
CodeKind::FOR_TESTING, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING,
Handle<Code>())->ToObjectChecked(); Handle<Code>())->ToObjectChecked();
CHECK(code->IsCode()); CHECK(code->IsCode());
#ifdef DEBUG #ifdef DEBUG
@ -1007,7 +1011,7 @@ TEST(11) {
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Object code = isolate->heap()->CreateCode( Object code = isolate->heap()->CreateCode(
desc, desc,
CodeKind::FOR_TESTING, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING,
Handle<Code>())->ToObjectChecked(); Handle<Code>())->ToObjectChecked();
CHECK(code->IsCode()); CHECK(code->IsCode());
#ifdef DEBUG #ifdef DEBUG

View File

@ -63,8 +63,9 @@ TEST(0) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif
@ -102,8 +103,9 @@ TEST(1) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif
@ -153,8 +155,9 @@ TEST(2) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif
@ -208,8 +211,9 @@ TEST(3) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif
@ -250,7 +254,7 @@ TEST(4) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = isolate->factory()->NewCode( Handle<Code> code = isolate->factory()->NewCode(
desc, CodeKind::FOR_TESTING, Handle<Code>()); desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING, Handle<Code>());
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif
@ -278,7 +282,7 @@ TEST(5) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = isolate->factory()->NewCode( Handle<Code> code = isolate->factory()->NewCode(
desc, CodeKind::FOR_TESTING, Handle<Code>()); desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING, Handle<Code>());
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif
@ -312,7 +316,7 @@ TEST(6) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = isolate->factory()->NewCode( Handle<Code> code = isolate->factory()->NewCode(
desc, CodeKind::FOR_TESTING, Handle<Code>()); desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING, Handle<Code>());
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif
@ -344,7 +348,7 @@ TEST(7) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = isolate->factory()->NewCode( Handle<Code> code = isolate->factory()->NewCode(
desc, CodeKind::FOR_TESTING, Handle<Code>()); desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING, Handle<Code>());
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif
@ -375,7 +379,7 @@ TEST(8) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = isolate->factory()->NewCode( Handle<Code> code = isolate->factory()->NewCode(
desc, CodeKind::FOR_TESTING, Handle<Code>()); desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING, Handle<Code>());
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif
@ -402,7 +406,7 @@ TEST(9) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = isolate->factory()->NewCode( Handle<Code> code = isolate->factory()->NewCode(
desc, CodeKind::FOR_TESTING, Handle<Code>()); desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING, Handle<Code>());
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif
@ -486,8 +490,9 @@ TEST(10) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif
@ -540,8 +545,9 @@ TEST(11) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif
@ -594,8 +600,9 @@ TEST(12) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif
@ -658,8 +665,9 @@ TEST(13) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif
@ -749,8 +757,9 @@ TEST(14) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif
@ -839,8 +848,9 @@ TEST(15) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif
@ -886,8 +896,9 @@ TEST(16) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif
@ -960,8 +971,9 @@ TEST(17) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif
@ -1052,8 +1064,9 @@ TEST(18) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
code->Print(); code->Print();
#endif #endif

View File

@ -743,8 +743,9 @@ TEST(AssemblerMultiByteNop) {
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F0>::FromCode(*code); auto f = GeneratedCode<F0>::FromCode(*code);
int res = f.Call(); int res = f.Call();
@ -800,8 +801,9 @@ void DoSSE2(const v8::FunctionCallbackInfo<v8::Value>& args) {
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F0>::FromCode(*code); auto f = GeneratedCode<F0>::FromCode(*code);
int res = f.Call(); int res = f.Call();
@ -865,8 +867,9 @@ TEST(AssemblerX64Extractps) {
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -902,8 +905,9 @@ TEST(AssemblerX64SSE) {
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -933,8 +937,9 @@ TEST(AssemblerX64SSE3) {
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -1158,8 +1163,9 @@ TEST(AssemblerX64FMA_sd) {
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -1384,8 +1390,9 @@ TEST(AssemblerX64FMA_ss) {
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -1460,8 +1467,9 @@ TEST(AssemblerX64SSE_ss) {
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -1546,8 +1554,9 @@ TEST(AssemblerX64AVX_ss) {
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -1786,8 +1795,9 @@ TEST(AssemblerX64AVX_sd) {
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -1978,8 +1988,9 @@ TEST(AssemblerX64BMI1) {
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -2038,8 +2049,9 @@ TEST(AssemblerX64LZCNT) {
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -2098,8 +2110,9 @@ TEST(AssemblerX64POPCNT) {
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -2361,8 +2374,9 @@ TEST(AssemblerX64BMI2) {
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -2405,8 +2419,9 @@ TEST(AssemblerX64JumpTables1) {
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -2453,8 +2468,9 @@ TEST(AssemblerX64JumpTables2) {
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -2510,8 +2526,9 @@ TEST(AssemblerX64vmovups) {
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);

View File

@ -39,9 +39,10 @@ TEST(CodeLayoutWithoutUnwindingInfo) {
code_desc.unwinding_info_size = 0; code_desc.unwinding_info_size = 0;
code_desc.origin = nullptr; code_desc.origin = nullptr;
Handle<Code> code = Factory::CodeBuilder(CcTest::i_isolate(), code_desc, Handle<Code> code =
CodeKind::FOR_TESTING) Factory::CodeBuilder(CcTest::i_isolate(), code_desc,
.Build(); CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
CHECK(!code->has_unwinding_info()); CHECK(!code->has_unwinding_info());
CHECK_EQ(code->raw_instruction_size(), buffer_size); CHECK_EQ(code->raw_instruction_size(), buffer_size);
@ -86,9 +87,10 @@ TEST(CodeLayoutWithUnwindingInfo) {
code_desc.unwinding_info_size = unwinding_info_size; code_desc.unwinding_info_size = unwinding_info_size;
code_desc.origin = nullptr; code_desc.origin = nullptr;
Handle<Code> code = Factory::CodeBuilder(CcTest::i_isolate(), code_desc, Handle<Code> code =
CodeKind::FOR_TESTING) Factory::CodeBuilder(CcTest::i_isolate(), code_desc,
.Build(); CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
CHECK(code->has_unwinding_info()); CHECK(code->has_unwinding_info());
CHECK_EQ(code->raw_instruction_size(), buffer_size + unwinding_info_size); CHECK_EQ(code->raw_instruction_size(), buffer_size + unwinding_info_size);

View File

@ -131,7 +131,7 @@ Handle<JSFunction> CreateCsaDescriptorArrayLookup(Isolate* isolate) {
compiler::CodeAssemblerTester asm_tester( compiler::CodeAssemblerTester asm_tester(
isolate, kNumParams + 1, // +1 to include receiver. isolate, kNumParams + 1, // +1 to include receiver.
CodeKind::FOR_TESTING); CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING);
{ {
CodeStubAssembler m(asm_tester.state()); CodeStubAssembler m(asm_tester.state());
@ -176,7 +176,7 @@ Handle<JSFunction> CreateCsaTransitionArrayLookup(Isolate* isolate) {
const int kNumParams = 2; const int kNumParams = 2;
compiler::CodeAssemblerTester asm_tester( compiler::CodeAssemblerTester asm_tester(
isolate, kNumParams + 1, // +1 to include receiver. isolate, kNumParams + 1, // +1 to include receiver.
CodeKind::FOR_TESTING); CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING);
{ {
CodeStubAssembler m(asm_tester.state()); CodeStubAssembler m(asm_tester.state());

View File

@ -988,8 +988,9 @@ TEST(DisasmIa320) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
USE(code); USE(code);
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;

View File

@ -993,8 +993,9 @@ TEST(DisasmX64) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
USE(code); USE(code);
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;

View File

@ -4068,7 +4068,8 @@ TEST(WeakReference) {
i::CodeDesc desc; i::CodeDesc desc;
assm.GetCode(i_isolate, &desc); assm.GetCode(i_isolate, &desc);
i::Handle<i::Code> code = i::Handle<i::Code> code =
i::Factory::CodeBuilder(i_isolate, desc, i::CodeKind::FOR_TESTING) i::Factory::CodeBuilder(i_isolate, desc,
i::CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build(); .Build();
CHECK(code->IsCode()); CHECK(code->IsCode());

View File

@ -29,7 +29,6 @@
#include "src/codegen/assembler-inl.h" #include "src/codegen/assembler-inl.h"
#include "src/codegen/macro-assembler.h" #include "src/codegen/macro-assembler.h"
#include "src/deoptimizer/deoptimizer.h"
#include "src/execution/simulator.h" #include "src/execution/simulator.h"
#include "src/init/v8.h" #include "src/init/v8.h"
#include "src/objects/objects-inl.h" #include "src/objects/objects-inl.h"
@ -147,8 +146,9 @@ TEST(ExtractLane) {
CodeDesc desc; CodeDesc desc;
masm->GetCode(isolate, &desc); masm->GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -278,8 +278,9 @@ TEST(ReplaceLane) {
CodeDesc desc; CodeDesc desc;
masm->GetCode(isolate, &desc); masm->GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -310,29 +311,6 @@ TEST(ReplaceLane) {
} }
} }
TEST(DeoptExitSizeIsFixed) {
CHECK(Deoptimizer::kSupportsFixedDeoptExitSizes);
Isolate* isolate = CcTest::i_isolate();
HandleScope handles(isolate);
auto buffer = AllocateAssemblerBuffer();
MacroAssembler masm(isolate, v8::internal::CodeObjectRequired::kYes,
buffer->CreateView());
STATIC_ASSERT(static_cast<int>(kFirstDeoptimizeKind) == 0);
for (int i = 0; i < kDeoptimizeKindCount; i++) {
DeoptimizeKind kind = static_cast<DeoptimizeKind>(i);
Builtins::Name target = Deoptimizer::GetDeoptimizationEntry(isolate, kind);
Label before_exit;
masm.bind(&before_exit);
masm.CallForDeoptimization(target, 42, &before_exit, kind, &before_exit);
CHECK_EQ(masm.SizeOfCodeGeneratedSince(&before_exit),
kind == DeoptimizeKind::kLazy
? Deoptimizer::kLazyDeoptExitSize
: Deoptimizer::kNonLazyDeoptExitSize);
}
}
#undef __ #undef __
} // namespace test_macro_assembler_arm } // namespace test_macro_assembler_arm

View File

@ -27,13 +27,13 @@
#include <stdlib.h> #include <stdlib.h>
#include "src/init/v8.h"
#include "src/base/platform/platform.h" #include "src/base/platform/platform.h"
#include "src/codegen/arm64/assembler-arm64-inl.h" #include "src/codegen/arm64/assembler-arm64-inl.h"
#include "src/codegen/macro-assembler.h" #include "src/codegen/macro-assembler.h"
#include "src/deoptimizer/deoptimizer.h"
#include "src/execution/simulator.h" #include "src/execution/simulator.h"
#include "src/heap/factory.h" #include "src/heap/factory.h"
#include "src/init/v8.h"
#include "src/objects/objects-inl.h" #include "src/objects/objects-inl.h"
#include "src/objects/smi.h" #include "src/objects/smi.h"
#include "src/utils/ostreams.h" #include "src/utils/ostreams.h"
@ -65,8 +65,9 @@ TEST(EmbeddedObj) {
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -91,29 +92,6 @@ TEST(EmbeddedObj) {
#endif // V8_COMPRESS_POINTERS #endif // V8_COMPRESS_POINTERS
} }
TEST(DeoptExitSizeIsFixed) {
CHECK(Deoptimizer::kSupportsFixedDeoptExitSizes);
Isolate* isolate = CcTest::i_isolate();
HandleScope handles(isolate);
auto buffer = AllocateAssemblerBuffer();
MacroAssembler masm(isolate, v8::internal::CodeObjectRequired::kYes,
buffer->CreateView());
STATIC_ASSERT(static_cast<int>(kFirstDeoptimizeKind) == 0);
for (int i = 0; i < kDeoptimizeKindCount; i++) {
DeoptimizeKind kind = static_cast<DeoptimizeKind>(i);
Builtins::Name target = Deoptimizer::GetDeoptimizationEntry(isolate, kind);
Label before_exit;
masm.bind(&before_exit);
masm.CallForDeoptimization(target, 42, &before_exit, kind, &before_exit);
CHECK_EQ(masm.SizeOfCodeGeneratedSince(&before_exit),
kind == DeoptimizeKind::kLazy
? Deoptimizer::kLazyDeoptExitSize
: Deoptimizer::kNonLazyDeoptExitSize);
}
}
#undef __ #undef __
} // namespace test_macro_assembler_arm64 } // namespace test_macro_assembler_arm64

View File

@ -88,8 +88,9 @@ TEST(BYTESWAP) {
CodeDesc desc; CodeDesc desc;
masm->GetCode(isolate, &desc); masm->GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (size_t i = 0; i < arraysize(test_values); i++) { for (size_t i = 0; i < arraysize(test_values); i++) {
@ -198,8 +199,9 @@ TEST(jump_tables4) {
CodeDesc desc; CodeDesc desc;
masm->GetCode(isolate, &desc); masm->GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -261,8 +263,9 @@ TEST(jump_tables5) {
CodeDesc desc; CodeDesc desc;
masm->GetCode(isolate, &desc); masm->GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -349,8 +352,9 @@ TEST(jump_tables6) {
CodeDesc desc; CodeDesc desc;
masm->GetCode(isolate, &desc); masm->GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -374,8 +378,9 @@ static uint32_t run_lsa(uint32_t rt, uint32_t rs, int8_t sa) {
CodeDesc desc; CodeDesc desc;
assembler.GetCode(isolate, &desc); assembler.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F1>::FromCode(*code); auto f = GeneratedCode<F1>::FromCode(*code);
@ -502,8 +507,9 @@ RET_TYPE run_Cvt(IN_TYPE x, Func GenerateConvertInstructionFunc) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F_CVT>::FromCode(*code); auto f = GeneratedCode<F_CVT>::FromCode(*code);
@ -615,7 +621,9 @@ TEST(OverflowInstructions) {
CodeDesc desc; CodeDesc desc;
masm->GetCode(isolate, &desc); masm->GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
t.lhs = ii; t.lhs = ii;
t.rhs = jj; t.rhs = jj;
@ -737,8 +745,9 @@ TEST(min_max_nan) {
CodeDesc desc; CodeDesc desc;
masm->GetCode(isolate, &desc); masm->GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (int i = 0; i < kTableLength; i++) { for (int i = 0; i < kTableLength; i++) {
test.a = inputsa[i]; test.a = inputsa[i];
@ -772,8 +781,9 @@ bool run_Unaligned(char* memory_buffer, int32_t in_offset, int32_t out_offset,
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F_CVT>::FromCode(*code); auto f = GeneratedCode<F_CVT>::FromCode(*code);
@ -1019,8 +1029,9 @@ bool run_Sltu(uint32_t rs, uint32_t rd, Func GenerateSltuInstructionFunc) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F_CVT>::FromCode(*code); auto f = GeneratedCode<F_CVT>::FromCode(*code);
int32_t res = reinterpret_cast<int32_t>(f.Call(rs, rd, 0, 0, 0)); int32_t res = reinterpret_cast<int32_t>(f.Call(rs, rd, 0, 0, 0));
@ -1114,7 +1125,8 @@ static GeneratedCode<F4> GenerateMacroFloat32MinMax(MacroAssembler* masm) {
CodeDesc desc; CodeDesc desc;
masm->GetCode(masm->isolate(), &desc); masm->GetCode(masm->isolate(), &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(masm->isolate(), desc, CodeKind::FOR_TESTING) Factory::CodeBuilder(masm->isolate(), desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build(); .Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
@ -1256,7 +1268,8 @@ static GeneratedCode<F4> GenerateMacroFloat64MinMax(MacroAssembler* masm) {
CodeDesc desc; CodeDesc desc;
masm->GetCode(masm->isolate(), &desc); masm->GetCode(masm->isolate(), &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(masm->isolate(), desc, CodeKind::FOR_TESTING) Factory::CodeBuilder(masm->isolate(), desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build(); .Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;

View File

@ -108,8 +108,9 @@ TEST(BYTESWAP) {
CodeDesc desc; CodeDesc desc;
masm->GetCode(isolate, &desc); masm->GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (size_t i = 0; i < arraysize(test_values); i++) { for (size_t i = 0; i < arraysize(test_values); i++) {
@ -163,8 +164,9 @@ TEST(LoadConstants) {
CodeDesc desc; CodeDesc desc;
masm->GetCode(isolate, &desc); masm->GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<FV>::FromCode(*code); auto f = GeneratedCode<FV>::FromCode(*code);
(void)f.Call(reinterpret_cast<int64_t>(result), 0, 0, 0, 0); (void)f.Call(reinterpret_cast<int64_t>(result), 0, 0, 0, 0);
@ -206,8 +208,9 @@ TEST(LoadAddress) {
CodeDesc desc; CodeDesc desc;
masm->GetCode(isolate, &desc); masm->GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<FV>::FromCode(*code); auto f = GeneratedCode<FV>::FromCode(*code);
(void)f.Call(0, 0, 0, 0, 0); (void)f.Call(0, 0, 0, 0, 0);
@ -263,8 +266,9 @@ TEST(jump_tables4) {
CodeDesc desc; CodeDesc desc;
masm->GetCode(isolate, &desc); masm->GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -333,8 +337,9 @@ TEST(jump_tables5) {
CodeDesc desc; CodeDesc desc;
masm->GetCode(isolate, &desc); masm->GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -422,8 +427,9 @@ TEST(jump_tables6) {
CodeDesc desc; CodeDesc desc;
masm->GetCode(isolate, &desc); masm->GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
code->Print(std::cout); code->Print(std::cout);
#endif #endif
@ -447,8 +453,9 @@ static uint64_t run_lsa(uint32_t rt, uint32_t rs, int8_t sa) {
CodeDesc desc; CodeDesc desc;
assembler.GetCode(isolate, &desc); assembler.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F1>::FromCode(*code); auto f = GeneratedCode<F1>::FromCode(*code);
@ -527,8 +534,9 @@ static uint64_t run_dlsa(uint64_t rt, uint64_t rs, int8_t sa) {
CodeDesc desc; CodeDesc desc;
assembler.GetCode(isolate, &desc); assembler.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<FV>::FromCode(*code); auto f = GeneratedCode<FV>::FromCode(*code);
@ -677,8 +685,9 @@ RET_TYPE run_Cvt(IN_TYPE x, Func GenerateConvertInstructionFunc) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F_CVT>::FromCode(*code); auto f = GeneratedCode<F_CVT>::FromCode(*code);
@ -853,7 +862,9 @@ TEST(OverflowInstructions) {
CodeDesc desc; CodeDesc desc;
masm->GetCode(isolate, &desc); masm->GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate, desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
t.lhs = ii; t.lhs = ii;
t.rhs = jj; t.rhs = jj;
@ -975,8 +986,9 @@ TEST(min_max_nan) {
CodeDesc desc; CodeDesc desc;
masm->GetCode(isolate, &desc); masm->GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F3>::FromCode(*code); auto f = GeneratedCode<F3>::FromCode(*code);
for (int i = 0; i < kTableLength; i++) { for (int i = 0; i < kTableLength; i++) {
test.a = inputsa[i]; test.a = inputsa[i];
@ -1010,8 +1022,9 @@ bool run_Unaligned(char* memory_buffer, int32_t in_offset, int32_t out_offset,
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F_CVT>::FromCode(*code); auto f = GeneratedCode<F_CVT>::FromCode(*code);
@ -1374,8 +1387,9 @@ bool run_Sltu(uint64_t rs, uint64_t rd, Func GenerateSltuInstructionFunc) {
CodeDesc desc; CodeDesc desc;
assm.GetCode(isolate, &desc); assm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
auto f = GeneratedCode<F_CVT>::FromCode(*code); auto f = GeneratedCode<F_CVT>::FromCode(*code);
int64_t res = reinterpret_cast<int64_t>(f.Call(rs, rd, 0, 0, 0)); int64_t res = reinterpret_cast<int64_t>(f.Call(rs, rd, 0, 0, 0));
@ -1469,7 +1483,8 @@ static GeneratedCode<F4> GenerateMacroFloat32MinMax(MacroAssembler* masm) {
CodeDesc desc; CodeDesc desc;
masm->GetCode(masm->isolate(), &desc); masm->GetCode(masm->isolate(), &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(masm->isolate(), desc, CodeKind::FOR_TESTING) Factory::CodeBuilder(masm->isolate(), desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build(); .Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;
@ -1611,7 +1626,8 @@ static GeneratedCode<F4> GenerateMacroFloat64MinMax(MacroAssembler* masm) {
CodeDesc desc; CodeDesc desc;
masm->GetCode(masm->isolate(), &desc); masm->GetCode(masm->isolate(), &desc);
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(masm->isolate(), desc, CodeKind::FOR_TESTING) Factory::CodeBuilder(masm->isolate(), desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build(); .Build();
#ifdef DEBUG #ifdef DEBUG
StdoutStream os; StdoutStream os;

View File

@ -27,13 +27,13 @@
#include <stdlib.h> #include <stdlib.h>
#include "src/init/v8.h"
#include "src/base/platform/platform.h" #include "src/base/platform/platform.h"
#include "src/codegen/macro-assembler.h" #include "src/codegen/macro-assembler.h"
#include "src/codegen/x64/assembler-x64-inl.h" #include "src/codegen/x64/assembler-x64-inl.h"
#include "src/deoptimizer/deoptimizer.h"
#include "src/execution/simulator.h" #include "src/execution/simulator.h"
#include "src/heap/factory.h" #include "src/heap/factory.h"
#include "src/init/v8.h"
#include "src/objects/objects-inl.h" #include "src/objects/objects-inl.h"
#include "src/objects/smi.h" #include "src/objects/smi.h"
#include "src/utils/ostreams.h" #include "src/utils/ostreams.h"
@ -449,8 +449,9 @@ TEST(EmbeddedObj) {
CodeDesc desc; CodeDesc desc;
masm->GetCode(isolate, &desc); masm->GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
#ifdef OBJECT_PRINT #ifdef OBJECT_PRINT
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
@ -1033,29 +1034,6 @@ TEST(AreAliased) {
DCHECK(AreAliased(rax, no_reg, rbx, no_reg, rcx, no_reg, rdx, rax, no_reg)); DCHECK(AreAliased(rax, no_reg, rbx, no_reg, rcx, no_reg, rdx, rax, no_reg));
} }
TEST(DeoptExitSizeIsFixed) {
CHECK(Deoptimizer::kSupportsFixedDeoptExitSizes);
Isolate* isolate = CcTest::i_isolate();
HandleScope handles(isolate);
auto buffer = AllocateAssemblerBuffer();
MacroAssembler masm(isolate, v8::internal::CodeObjectRequired::kYes,
buffer->CreateView());
STATIC_ASSERT(static_cast<int>(kFirstDeoptimizeKind) == 0);
for (int i = 0; i < kDeoptimizeKindCount; i++) {
DeoptimizeKind kind = static_cast<DeoptimizeKind>(i);
Builtins::Name target = Deoptimizer::GetDeoptimizationEntry(isolate, kind);
Label before_exit;
masm.bind(&before_exit);
masm.CallForDeoptimization(target, 42, &before_exit, kind, nullptr);
CHECK_EQ(masm.SizeOfCodeGeneratedSince(&before_exit),
kind == DeoptimizeKind::kLazy
? Deoptimizer::kLazyDeoptExitSize
: Deoptimizer::kNonLazyDeoptExitSize);
}
}
#undef __ #undef __
} // namespace test_macro_assembler_x64 } // namespace test_macro_assembler_x64

View File

@ -204,8 +204,9 @@ void TestInvalidateExclusiveAccess(TestData initial_data, MemoryAccess access1,
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
TestData t = initial_data; TestData t = initial_data;
Simulator::current(isolate)->Call<void>(code->entry(), &t); Simulator::current(isolate)->Call<void>(code->entry(), &t);
@ -276,8 +277,9 @@ int ExecuteMemoryAccess(Isolate* isolate, TestData* test_data,
CodeDesc desc; CodeDesc desc;
masm.GetCode(isolate, &desc); masm.GetCode(isolate, &desc);
Handle<Code> code = Handle<Code> code = Factory::CodeBuilder(
Factory::CodeBuilder(isolate, desc, CodeKind::FOR_TESTING).Build(); isolate, desc, CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
Simulator::current(isolate)->Call<void>(code->entry(), test_data); Simulator::current(isolate)->Call<void>(code->entry(), test_data);
return Simulator::current(isolate)->wreg(0); return Simulator::current(isolate)->wreg(0);
} }

View File

@ -240,7 +240,7 @@ extern "C" int LLVMFuzzerTestOneInput(const uint8_t* data, size_t size) {
callee.Return(static_cast<int>(desc->ReturnCount()), returns.get()); callee.Return(static_cast<int>(desc->ReturnCount()), returns.get());
OptimizedCompilationInfo info(ArrayVector("testing"), &zone, OptimizedCompilationInfo info(ArrayVector("testing"), &zone,
CodeKind::FOR_TESTING); CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING);
Handle<Code> code = Handle<Code> code =
Pipeline::GenerateCodeForTesting(&info, i_isolate, desc, callee.graph(), Pipeline::GenerateCodeForTesting(&info, i_isolate, desc, callee.graph(),
AssemblerOptions::Default(i_isolate), AssemblerOptions::Default(i_isolate),
@ -286,7 +286,7 @@ extern "C" int LLVMFuzzerTestOneInput(const uint8_t* data, size_t size) {
// Call the wrapper. // Call the wrapper.
OptimizedCompilationInfo wrapper_info(ArrayVector("wrapper"), &zone, OptimizedCompilationInfo wrapper_info(ArrayVector("wrapper"), &zone,
CodeKind::FOR_TESTING); CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING);
Handle<Code> wrapper_code = Handle<Code> wrapper_code =
Pipeline::GenerateCodeForTesting( Pipeline::GenerateCodeForTesting(
&wrapper_info, i_isolate, wrapper_desc, caller.graph(), &wrapper_info, i_isolate, wrapper_desc, caller.graph(),

View File

@ -155,7 +155,9 @@ TEST_P(TurboAssemblerTestMoveObjectAndSlot, MoveObjectAndSlot) {
tasm.GetCode(nullptr, &desc); tasm.GetCode(nullptr, &desc);
if (FLAG_print_code) { if (FLAG_print_code) {
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate(), desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate(), desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
} }

View File

@ -161,7 +161,9 @@ TEST_P(TurboAssemblerTestMoveObjectAndSlot, MoveObjectAndSlot) {
tasm.GetCode(nullptr, &desc); tasm.GetCode(nullptr, &desc);
if (FLAG_print_code) { if (FLAG_print_code) {
Handle<Code> code = Handle<Code> code =
Factory::CodeBuilder(isolate(), desc, CodeKind::FOR_TESTING).Build(); Factory::CodeBuilder(isolate(), desc,
CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING)
.Build();
StdoutStream os; StdoutStream os;
code->Print(os); code->Print(os);
} }

View File

@ -24,7 +24,7 @@ CodeStubAssemblerTestState::CodeStubAssemblerTestState(
CodeStubAssemblerTest* test) CodeStubAssemblerTest* test)
: compiler::CodeAssemblerState( : compiler::CodeAssemblerState(
test->isolate(), test->zone(), VoidDescriptor{}, test->isolate(), test->zone(), VoidDescriptor{},
CodeKind::FOR_TESTING, "test", CodeKind::DEOPT_ENTRIES_OR_FOR_TESTING, "test",
PoisoningMitigationLevel::kPoisonCriticalOnly) {} PoisoningMitigationLevel::kPoisonCriticalOnly) {}
TARGET_TEST_F(CodeStubAssemblerTest, SmiTag) { TARGET_TEST_F(CodeStubAssemblerTest, SmiTag) {