1 // Copyright 2017 the V8 project authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "src/wasm/baseline/liftoff-compiler.h"
6 
7 #include "src/assembler-inl.h"
8 #include "src/base/optional.h"
9 // TODO(clemensh): Remove dependences on compiler stuff.
10 #include "src/compiler/linkage.h"
11 #include "src/compiler/wasm-compiler.h"
12 #include "src/counters.h"
13 #include "src/macro-assembler-inl.h"
14 #include "src/tracing/trace-event.h"
15 #include "src/wasm/baseline/liftoff-assembler.h"
16 #include "src/wasm/function-body-decoder-impl.h"
17 #include "src/wasm/function-compiler.h"
18 #include "src/wasm/memory-tracing.h"
19 #include "src/wasm/wasm-engine.h"
20 #include "src/wasm/wasm-linkage.h"
21 #include "src/wasm/wasm-objects.h"
22 #include "src/wasm/wasm-opcodes.h"
23 
24 namespace v8 {
25 namespace internal {
26 namespace wasm {
27 
28 constexpr auto kRegister = LiftoffAssembler::VarState::kRegister;
29 constexpr auto KIntConst = LiftoffAssembler::VarState::KIntConst;
30 constexpr auto kStack = LiftoffAssembler::VarState::kStack;
31 
32 namespace {
33 
34 #define __ asm_.
35 
36 #define TRACE(...)                                            \
37   do {                                                        \
38     if (FLAG_trace_liftoff) PrintF("[liftoff] " __VA_ARGS__); \
39   } while (false)
40 
41 #define WASM_INSTANCE_OBJECT_OFFSET(name) \
42   (WasmInstanceObject::k##name##Offset - kHeapObjectTag)
43 
44 #define LOAD_INSTANCE_FIELD(dst, name, type)                       \
45   __ LoadFromInstance(dst.gp(), WASM_INSTANCE_OBJECT_OFFSET(name), \
46                       LoadType(type).size());
47 
48 #ifdef DEBUG
49 #define DEBUG_CODE_COMMENT(str) \
50   do {                          \
51     __ RecordComment(str);      \
52   } while (false)
53 #else
54 #define DEBUG_CODE_COMMENT(str) ((void)0)
55 #endif
56 
57 constexpr LoadType::LoadTypeValue kPointerLoadType =
58     kPointerSize == 8 ? LoadType::kI64Load : LoadType::kI32Load;
59 
60 #if V8_TARGET_ARCH_ARM64
61 // On ARM64, the Assembler keeps track of pointers to Labels to resolve
62 // branches to distant targets. Moving labels would confuse the Assembler,
63 // thus store the label on the heap and keep a unique_ptr.
64 class MovableLabel {
65  public:
66   MOVE_ONLY_NO_DEFAULT_CONSTRUCTOR(MovableLabel);
MovableLabel()67   MovableLabel() : label_(new Label()) {}
68 
get()69   Label* get() { return label_.get(); }
70 
71  private:
72   std::unique_ptr<Label> label_;
73 };
74 #else
75 // On all other platforms, just store the Label directly.
76 class MovableLabel {
77  public:
78   MOVE_ONLY_WITH_DEFAULT_CONSTRUCTORS(MovableLabel);
79 
get()80   Label* get() { return &label_; }
81 
82  private:
83   Label label_;
84 };
85 #endif
86 
GetLoweredCallDescriptor(Zone * zone,compiler::CallDescriptor * call_desc)87 compiler::CallDescriptor* GetLoweredCallDescriptor(
88     Zone* zone, compiler::CallDescriptor* call_desc) {
89   return kPointerSize == 4 ? compiler::GetI32WasmCallDescriptor(zone, call_desc)
90                            : call_desc;
91 }
92 
93 constexpr ValueType kTypesArr_ilfd[] = {kWasmI32, kWasmI64, kWasmF32, kWasmF64};
94 constexpr Vector<const ValueType> kTypes_ilfd = ArrayVector(kTypesArr_ilfd);
95 
96 class LiftoffCompiler {
97  public:
98   MOVE_ONLY_NO_DEFAULT_CONSTRUCTOR(LiftoffCompiler);
99 
100   // TODO(clemensh): Make this a template parameter.
101   static constexpr Decoder::ValidateFlag validate = Decoder::kValidate;
102 
103   using Value = ValueBase;
104 
105   struct ElseState {
106     MovableLabel label;
107     LiftoffAssembler::CacheState state;
108   };
109 
110   struct Control : public ControlWithNamedConstructors<Control, Value> {
111     MOVE_ONLY_WITH_DEFAULT_CONSTRUCTORS(Control);
112 
113     std::unique_ptr<ElseState> else_state;
114     LiftoffAssembler::CacheState label_state;
115     MovableLabel label;
116   };
117 
118   using FullDecoder = WasmFullDecoder<validate, LiftoffCompiler>;
119 
120   struct OutOfLineCode {
121     MovableLabel label;
122     MovableLabel continuation;
123     WasmCode::RuntimeStubId stub;
124     WasmCodePosition position;
125     LiftoffRegList regs_to_save;
126     uint32_t pc;  // for trap handler.
127 
128     // Named constructors:
Trapv8::internal::wasm::__anon2863bf240111::LiftoffCompiler::OutOfLineCode129     static OutOfLineCode Trap(WasmCode::RuntimeStubId s, WasmCodePosition pos,
130                               uint32_t pc) {
131       DCHECK_LT(0, pos);
132       return {{}, {}, s, pos, {}, pc};
133     }
StackCheckv8::internal::wasm::__anon2863bf240111::LiftoffCompiler::OutOfLineCode134     static OutOfLineCode StackCheck(WasmCodePosition pos, LiftoffRegList regs) {
135       return {{}, {}, WasmCode::kWasmStackGuard, pos, regs, 0};
136     }
137   };
138 
LiftoffCompiler(compiler::CallDescriptor * call_descriptor,ModuleEnv * env,Zone * compilation_zone)139   LiftoffCompiler(compiler::CallDescriptor* call_descriptor, ModuleEnv* env,
140                   Zone* compilation_zone)
141       : descriptor_(
142             GetLoweredCallDescriptor(compilation_zone, call_descriptor)),
143         env_(env),
144         compilation_zone_(compilation_zone),
145         safepoint_table_builder_(compilation_zone_) {}
146 
~LiftoffCompiler()147   ~LiftoffCompiler() { BindUnboundLabels(nullptr); }
148 
ok() const149   bool ok() const { return ok_; }
150 
GetCode(CodeDesc * desc)151   void GetCode(CodeDesc* desc) { asm_.GetCode(nullptr, desc); }
152 
GetSourcePositionTable()153   OwnedVector<uint8_t> GetSourcePositionTable() {
154     return source_position_table_builder_.ToSourcePositionTableVector();
155   }
156 
GetProtectedInstructions() const157   OwnedVector<trap_handler::ProtectedInstructionData> GetProtectedInstructions()
158       const {
159     return OwnedVector<trap_handler::ProtectedInstructionData>::Of(
160         protected_instructions_);
161   }
162 
GetTotalFrameSlotCount() const163   uint32_t GetTotalFrameSlotCount() const {
164     return __ GetTotalFrameSlotCount();
165   }
166 
unsupported(FullDecoder * decoder,const char * reason)167   void unsupported(FullDecoder* decoder, const char* reason) {
168     ok_ = false;
169     TRACE("unsupported: %s\n", reason);
170     decoder->errorf(decoder->pc(), "unsupported liftoff operation: %s", reason);
171     BindUnboundLabels(decoder);
172   }
173 
DidAssemblerBailout(FullDecoder * decoder)174   bool DidAssemblerBailout(FullDecoder* decoder) {
175     if (decoder->failed() || !__ did_bailout()) return false;
176     unsupported(decoder, __ bailout_reason());
177     return true;
178   }
179 
CheckSupportedType(FullDecoder * decoder,Vector<const ValueType> supported_types,ValueType type,const char * context)180   bool CheckSupportedType(FullDecoder* decoder,
181                           Vector<const ValueType> supported_types,
182                           ValueType type, const char* context) {
183     char buffer[128];
184     // Check supported types.
185     for (ValueType supported : supported_types) {
186       if (type == supported) return true;
187     }
188     SNPrintF(ArrayVector(buffer), "%s %s", ValueTypes::TypeName(type), context);
189     unsupported(decoder, buffer);
190     return false;
191   }
192 
GetSafepointTableOffset() const193   int GetSafepointTableOffset() const {
194     return safepoint_table_builder_.GetCodeOffset();
195   }
196 
BindUnboundLabels(FullDecoder * decoder)197   void BindUnboundLabels(FullDecoder* decoder) {
198 #ifdef DEBUG
199     // Bind all labels now, otherwise their destructor will fire a DCHECK error
200     // if they where referenced before.
201     uint32_t control_depth = decoder ? decoder->control_depth() : 0;
202     for (uint32_t i = 0; i < control_depth; ++i) {
203       Control* c = decoder->control_at(i);
204       Label* label = c->label.get();
205       if (!label->is_bound()) __ bind(label);
206       if (c->else_state) {
207         Label* else_label = c->else_state->label.get();
208         if (!else_label->is_bound()) __ bind(else_label);
209       }
210     }
211     for (auto& ool : out_of_line_code_) {
212       if (!ool.label.get()->is_bound()) __ bind(ool.label.get());
213     }
214 #endif
215   }
216 
StartFunction(FullDecoder * decoder)217   void StartFunction(FullDecoder* decoder) {
218     int num_locals = decoder->NumLocals();
219     __ set_num_locals(num_locals);
220     for (int i = 0; i < num_locals; ++i) {
221       __ set_local_type(i, decoder->GetLocalType(i));
222     }
223   }
224 
CollectReservedRegsForParameters(uint32_t input_idx_start,uint32_t num_params,LiftoffRegList & param_regs)225   void CollectReservedRegsForParameters(uint32_t input_idx_start,
226                                         uint32_t num_params,
227                                         LiftoffRegList& param_regs) {
228     uint32_t input_idx = input_idx_start;
229     for (uint32_t param_idx = 0; param_idx < num_params; ++param_idx) {
230       ValueType type = __ local_type(param_idx);
231       const int num_lowered_params = 1 + needs_reg_pair(type);
232       RegClass rc = num_lowered_params == 1 ? reg_class_for(type) : kGpReg;
233 
234       for (int pair_idx = 0; pair_idx < num_lowered_params; ++pair_idx) {
235         compiler::LinkageLocation param_loc =
236             descriptor_->GetInputLocation(input_idx + pair_idx);
237         if (param_loc.IsRegister()) {
238           DCHECK(!param_loc.IsAnyRegister());
239           int reg_code = param_loc.AsRegister();
240           RegList cache_regs = rc == kGpReg ? kLiftoffAssemblerGpCacheRegs
241                                             : kLiftoffAssemblerFpCacheRegs;
242           if (cache_regs & (1 << reg_code)) {
243             LiftoffRegister in_reg = LiftoffRegister::from_code(rc, reg_code);
244             param_regs.set(in_reg);
245           }
246         }
247       }
248       input_idx += num_lowered_params;
249     }
250   }
251 
252   // Returns the number of inputs processed (1 or 2).
ProcessParameter(ValueType type,uint32_t input_idx)253   uint32_t ProcessParameter(ValueType type, uint32_t input_idx) {
254     const int num_lowered_params = 1 + needs_reg_pair(type);
255     // Initialize to anything, will be set in the loop and used afterwards.
256     LiftoffRegister reg = LiftoffRegister::from_code(kGpReg, 0);
257     RegClass rc = num_lowered_params == 1 ? reg_class_for(type) : kGpReg;
258     LiftoffRegList pinned;
259     for (int pair_idx = 0; pair_idx < num_lowered_params; ++pair_idx) {
260       compiler::LinkageLocation param_loc =
261           descriptor_->GetInputLocation(input_idx + pair_idx);
262       // Initialize to anything, will be set in both arms of the if.
263       LiftoffRegister in_reg = LiftoffRegister::from_code(kGpReg, 0);
264       if (param_loc.IsRegister()) {
265         DCHECK(!param_loc.IsAnyRegister());
266         int reg_code = param_loc.AsRegister();
267         RegList cache_regs = rc == kGpReg ? kLiftoffAssemblerGpCacheRegs
268                                           : kLiftoffAssemblerFpCacheRegs;
269         if (cache_regs & (1 << reg_code)) {
270           // This is a cache register, just use it.
271           in_reg = LiftoffRegister::from_code(rc, reg_code);
272         } else {
273           // Move to a cache register (spill one if necessary).
274           // Note that we cannot create a {LiftoffRegister} for reg_code, since
275           // {LiftoffRegister} can only store cache regs.
276           LiftoffRegister in_reg = __ GetUnusedRegister(rc, pinned);
277           if (rc == kGpReg) {
278             __ Move(in_reg.gp(), Register::from_code(reg_code), type);
279           } else {
280             __ Move(in_reg.fp(), DoubleRegister::from_code(reg_code), type);
281           }
282         }
283       } else if (param_loc.IsCallerFrameSlot()) {
284         in_reg = __ GetUnusedRegister(rc, pinned);
285         ValueType lowered_type = num_lowered_params == 1 ? type : kWasmI32;
286         __ LoadCallerFrameSlot(in_reg, -param_loc.AsCallerFrameSlot(),
287                                lowered_type);
288       }
289       reg = pair_idx == 0 ? in_reg
290                           : LiftoffRegister::ForPair(reg.gp(), in_reg.gp());
291       pinned.set(reg);
292     }
293     __ PushRegister(type, reg);
294     return num_lowered_params;
295   }
296 
StackCheck(WasmCodePosition position)297   void StackCheck(WasmCodePosition position) {
298     if (FLAG_wasm_no_stack_checks || !env_->runtime_exception_support) return;
299     out_of_line_code_.push_back(
300         OutOfLineCode::StackCheck(position, __ cache_state()->used_registers));
301     OutOfLineCode& ool = out_of_line_code_.back();
302     LiftoffRegister limit_address = __ GetUnusedRegister(kGpReg);
303     LOAD_INSTANCE_FIELD(limit_address, StackLimitAddress, kPointerLoadType);
304     __ StackCheck(ool.label.get(), limit_address.gp());
305     __ bind(ool.continuation.get());
306   }
307 
StartFunctionBody(FullDecoder * decoder,Control * block)308   void StartFunctionBody(FullDecoder* decoder, Control* block) {
309     for (uint32_t i = 0; i < __ num_locals(); ++i) {
310       if (!CheckSupportedType(decoder, kTypes_ilfd, __ local_type(i), "param"))
311         return;
312     }
313 
314     // Input 0 is the call target, the instance is at 1.
315     constexpr int kInstanceParameterIndex = 1;
316     // Store the instance parameter to a special stack slot.
317     compiler::LinkageLocation instance_loc =
318         descriptor_->GetInputLocation(kInstanceParameterIndex);
319     DCHECK(instance_loc.IsRegister());
320     DCHECK(!instance_loc.IsAnyRegister());
321     Register instance_reg = Register::from_code(instance_loc.AsRegister());
322     DCHECK_EQ(kWasmInstanceRegister, instance_reg);
323 
324     // Parameter 0 is the instance parameter.
325     uint32_t num_params =
326         static_cast<uint32_t>(decoder->sig_->parameter_count());
327 
328     __ EnterFrame(StackFrame::WASM_COMPILED);
329     __ set_has_frame(true);
330     pc_offset_stack_frame_construction_ = __ PrepareStackFrame();
331     // {PrepareStackFrame} is the first platform-specific assembler method.
332     // If this failed, we can bail out immediately, avoiding runtime overhead
333     // and potential failures because of other unimplemented methods.
334     // A platform implementing {PrepareStackFrame} must ensure that we can
335     // finish compilation without errors even if we hit unimplemented
336     // LiftoffAssembler methods.
337     if (DidAssemblerBailout(decoder)) return;
338 
339     __ SpillInstance(instance_reg);
340     // Input 0 is the code target, 1 is the instance. First parameter at 2.
341     uint32_t input_idx = kInstanceParameterIndex + 1;
342     for (uint32_t param_idx = 0; param_idx < num_params; ++param_idx) {
343       input_idx += ProcessParameter(__ local_type(param_idx), input_idx);
344     }
345     DCHECK_EQ(input_idx, descriptor_->InputCount());
346     // Set to a gp register, to mark this uninitialized.
347     LiftoffRegister zero_double_reg(Register::from_code<0>());
348     DCHECK(zero_double_reg.is_gp());
349     for (uint32_t param_idx = num_params; param_idx < __ num_locals();
350          ++param_idx) {
351       ValueType type = decoder->GetLocalType(param_idx);
352       switch (type) {
353         case kWasmI32:
354           __ cache_state()->stack_state.emplace_back(kWasmI32, uint32_t{0});
355           break;
356         case kWasmI64:
357           __ cache_state()->stack_state.emplace_back(kWasmI64, uint32_t{0});
358           break;
359         case kWasmF32:
360         case kWasmF64:
361           if (zero_double_reg.is_gp()) {
362             // Note: This might spill one of the registers used to hold
363             // parameters.
364             zero_double_reg = __ GetUnusedRegister(kFpReg);
365             // Zero is represented by the bit pattern 0 for both f32 and f64.
366             __ LoadConstant(zero_double_reg, WasmValue(0.));
367           }
368           __ PushRegister(type, zero_double_reg);
369           break;
370         default:
371           UNIMPLEMENTED();
372       }
373     }
374     block->label_state.stack_base = __ num_locals();
375 
376     // The function-prologue stack check is associated with position 0, which
377     // is never a position of any instruction in the function.
378     StackCheck(0);
379 
380     DCHECK_EQ(__ num_locals(), __ cache_state()->stack_height());
381   }
382 
GenerateOutOfLineCode(OutOfLineCode & ool)383   void GenerateOutOfLineCode(OutOfLineCode& ool) {
384     __ bind(ool.label.get());
385     const bool is_stack_check = ool.stub == WasmCode::kWasmStackGuard;
386     const bool is_mem_out_of_bounds =
387         ool.stub == WasmCode::kThrowWasmTrapMemOutOfBounds;
388 
389     if (is_mem_out_of_bounds && env_->use_trap_handler) {
390       uint32_t pc = static_cast<uint32_t>(__ pc_offset());
391       DCHECK_EQ(pc, __ pc_offset());
392       protected_instructions_.emplace_back(
393           trap_handler::ProtectedInstructionData{ool.pc, pc});
394     }
395 
396     if (!env_->runtime_exception_support) {
397       // We cannot test calls to the runtime in cctest/test-run-wasm.
398       // Therefore we emit a call to C here instead of a call to the runtime.
399       // In this mode, we never generate stack checks.
400       DCHECK(!is_stack_check);
401       __ CallTrapCallbackForTesting();
402       __ LeaveFrame(StackFrame::WASM_COMPILED);
403       __ DropStackSlotsAndRet(
404           static_cast<uint32_t>(descriptor_->StackParameterCount()));
405       return;
406     }
407 
408     if (!ool.regs_to_save.is_empty()) __ PushRegisters(ool.regs_to_save);
409 
410     source_position_table_builder_.AddPosition(
411         __ pc_offset(), SourcePosition(ool.position), false);
412     __ CallRuntimeStub(ool.stub);
413     safepoint_table_builder_.DefineSafepoint(&asm_, Safepoint::kSimple, 0,
414                                              Safepoint::kNoLazyDeopt);
415     DCHECK_EQ(ool.continuation.get()->is_bound(), is_stack_check);
416     if (!ool.regs_to_save.is_empty()) __ PopRegisters(ool.regs_to_save);
417     if (is_stack_check) {
418       __ emit_jump(ool.continuation.get());
419     } else {
420       __ AssertUnreachable(AbortReason::kUnexpectedReturnFromWasmTrap);
421     }
422   }
423 
FinishFunction(FullDecoder * decoder)424   void FinishFunction(FullDecoder* decoder) {
425     if (DidAssemblerBailout(decoder)) return;
426     for (OutOfLineCode& ool : out_of_line_code_) {
427       GenerateOutOfLineCode(ool);
428     }
429     __ PatchPrepareStackFrame(pc_offset_stack_frame_construction_,
430                               __ GetTotalFrameSlotCount());
431     __ FinishCode();
432     safepoint_table_builder_.Emit(&asm_, __ GetTotalFrameSlotCount());
433     // The previous calls may have also generated a bailout.
434     DidAssemblerBailout(decoder);
435   }
436 
OnFirstError(FullDecoder * decoder)437   void OnFirstError(FullDecoder* decoder) {
438     ok_ = false;
439     BindUnboundLabels(decoder);
440     asm_.AbortCompilation();
441   }
442 
NextInstruction(FullDecoder * decoder,WasmOpcode opcode)443   void NextInstruction(FullDecoder* decoder, WasmOpcode opcode) {
444     TraceCacheState(decoder);
445     SLOW_DCHECK(__ ValidateCacheState());
446     DEBUG_CODE_COMMENT(WasmOpcodes::OpcodeName(opcode));
447   }
448 
Block(FullDecoder * decoder,Control * block)449   void Block(FullDecoder* decoder, Control* block) {
450     block->label_state.stack_base = __ cache_state()->stack_height();
451   }
452 
Loop(FullDecoder * decoder,Control * loop)453   void Loop(FullDecoder* decoder, Control* loop) {
454     loop->label_state.stack_base = __ cache_state()->stack_height();
455 
456     // Before entering a loop, spill all locals to the stack, in order to free
457     // the cache registers, and to avoid unnecessarily reloading stack values
458     // into registers at branches.
459     // TODO(clemensh): Come up with a better strategy here, involving
460     // pre-analysis of the function.
461     __ SpillLocals();
462 
463     // Loop labels bind at the beginning of the block.
464     __ bind(loop->label.get());
465 
466     // Save the current cache state for the merge when jumping to this loop.
467     loop->label_state.Split(*__ cache_state());
468 
469     // Execute a stack check in the loop header.
470     StackCheck(decoder->position());
471   }
472 
Try(FullDecoder * decoder,Control * block)473   void Try(FullDecoder* decoder, Control* block) {
474     unsupported(decoder, "try");
475   }
476 
If(FullDecoder * decoder,const Value & cond,Control * if_block)477   void If(FullDecoder* decoder, const Value& cond, Control* if_block) {
478     DCHECK_EQ(if_block, decoder->control_at(0));
479     DCHECK(if_block->is_if());
480 
481     if (if_block->start_merge.arity > 0 || if_block->end_merge.arity > 1)
482       return unsupported(decoder, "multi-value if");
483 
484     // Allocate the else state.
485     if_block->else_state = base::make_unique<ElseState>();
486 
487     // Test the condition, jump to else if zero.
488     Register value = __ PopToRegister().gp();
489     __ emit_cond_jump(kEqual, if_block->else_state->label.get(), kWasmI32,
490                       value);
491 
492     if_block->label_state.stack_base = __ cache_state()->stack_height();
493     // Store the state (after popping the value) for executing the else branch.
494     if_block->else_state->state.Split(*__ cache_state());
495   }
496 
FallThruTo(FullDecoder * decoder,Control * c)497   void FallThruTo(FullDecoder* decoder, Control* c) {
498     if (c->end_merge.reached) {
499       __ MergeFullStackWith(c->label_state);
500     } else if (c->is_onearmed_if()) {
501       c->label_state.InitMerge(*__ cache_state(), __ num_locals(),
502                                c->br_merge()->arity);
503       __ MergeFullStackWith(c->label_state);
504     } else {
505       c->label_state.Split(*__ cache_state());
506     }
507     TraceCacheState(decoder);
508   }
509 
PopControl(FullDecoder * decoder,Control * c)510   void PopControl(FullDecoder* decoder, Control* c) {
511     if (!c->is_loop() && c->end_merge.reached) {
512       __ cache_state()->Steal(c->label_state);
513     }
514     if (!c->label.get()->is_bound()) {
515       __ bind(c->label.get());
516     }
517   }
518 
EndControl(FullDecoder * decoder,Control * c)519   void EndControl(FullDecoder* decoder, Control* c) {}
520 
521   enum CCallReturn : bool { kHasReturn = true, kNoReturn = false };
522 
GenerateCCall(const LiftoffRegister * result_regs,FunctionSig * sig,ValueType out_argument_type,const LiftoffRegister * arg_regs,ExternalReference ext_ref)523   void GenerateCCall(const LiftoffRegister* result_regs, FunctionSig* sig,
524                      ValueType out_argument_type,
525                      const LiftoffRegister* arg_regs,
526                      ExternalReference ext_ref) {
527     // Before making a call, spill all cache registers.
528     __ SpillAllRegisters();
529 
530     // Store arguments on our stack, then align the stack for calling to C.
531     int param_bytes = 0;
532     for (ValueType param_type : sig->parameters()) {
533       param_bytes += ValueTypes::MemSize(param_type);
534     }
535     int out_arg_bytes = out_argument_type == kWasmStmt
536                             ? 0
537                             : ValueTypes::MemSize(out_argument_type);
538     int stack_bytes = std::max(param_bytes, out_arg_bytes);
539     __ CallC(sig, arg_regs, result_regs, out_argument_type, stack_bytes,
540              ext_ref);
541   }
542 
543   template <ValueType src_type, ValueType result_type, class EmitFn>
EmitUnOp(EmitFn fn)544   void EmitUnOp(EmitFn fn) {
545     static RegClass src_rc = reg_class_for(src_type);
546     static RegClass result_rc = reg_class_for(result_type);
547     LiftoffRegister src = __ PopToRegister();
548     LiftoffRegister dst = src_rc == result_rc
549                               ? __ GetUnusedRegister(result_rc, {src})
550                               : __ GetUnusedRegister(result_rc);
551     fn(dst, src);
552     __ PushRegister(result_type, dst);
553   }
554 
EmitI32UnOpWithCFallback(bool (LiftoffAssembler::* emit_fn)(Register,Register),ExternalReference (* fallback_fn)())555   void EmitI32UnOpWithCFallback(bool (LiftoffAssembler::*emit_fn)(Register,
556                                                                   Register),
557                                 ExternalReference (*fallback_fn)()) {
558     auto emit_with_c_fallback = [=](LiftoffRegister dst, LiftoffRegister src) {
559       if (emit_fn && (asm_.*emit_fn)(dst.gp(), src.gp())) return;
560       ExternalReference ext_ref = fallback_fn();
561       ValueType sig_i_i_reps[] = {kWasmI32, kWasmI32};
562       FunctionSig sig_i_i(1, 1, sig_i_i_reps);
563       GenerateCCall(&dst, &sig_i_i, kWasmStmt, &src, ext_ref);
564     };
565     EmitUnOp<kWasmI32, kWasmI32>(emit_with_c_fallback);
566   }
567 
568   template <ValueType type>
EmitFloatUnOpWithCFallback(bool (LiftoffAssembler::* emit_fn)(DoubleRegister,DoubleRegister),ExternalReference (* fallback_fn)())569   void EmitFloatUnOpWithCFallback(
570       bool (LiftoffAssembler::*emit_fn)(DoubleRegister, DoubleRegister),
571       ExternalReference (*fallback_fn)()) {
572     auto emit_with_c_fallback = [=](LiftoffRegister dst, LiftoffRegister src) {
573       if ((asm_.*emit_fn)(dst.fp(), src.fp())) return;
574       ExternalReference ext_ref = fallback_fn();
575       ValueType sig_reps[] = {type};
576       FunctionSig sig(0, 1, sig_reps);
577       GenerateCCall(&dst, &sig, type, &src, ext_ref);
578     };
579     EmitUnOp<type, type>(emit_with_c_fallback);
580   }
581 
582   enum TypeConversionTrapping : bool { kCanTrap = true, kNoTrap = false };
583   template <ValueType dst_type, ValueType src_type,
584             TypeConversionTrapping can_trap>
EmitTypeConversion(WasmOpcode opcode,ExternalReference (* fallback_fn)(),WasmCodePosition trap_position)585   void EmitTypeConversion(WasmOpcode opcode, ExternalReference (*fallback_fn)(),
586                           WasmCodePosition trap_position) {
587     static constexpr RegClass src_rc = reg_class_for(src_type);
588     static constexpr RegClass dst_rc = reg_class_for(dst_type);
589     LiftoffRegister src = __ PopToRegister();
590     LiftoffRegister dst = src_rc == dst_rc ? __ GetUnusedRegister(dst_rc, {src})
591                                            : __ GetUnusedRegister(dst_rc);
592     DCHECK_EQ(!!can_trap, trap_position > 0);
593     Label* trap = can_trap ? AddOutOfLineTrap(
594                                  trap_position,
595                                  WasmCode::kThrowWasmTrapFloatUnrepresentable)
596                            : nullptr;
597     if (!__ emit_type_conversion(opcode, dst, src, trap)) {
598       DCHECK_NOT_NULL(fallback_fn);
599       ExternalReference ext_ref = fallback_fn();
600       if (can_trap) {
601         // External references for potentially trapping conversions return int.
602         ValueType sig_reps[] = {kWasmI32, src_type};
603         FunctionSig sig(1, 1, sig_reps);
604         LiftoffRegister ret_reg =
605             __ GetUnusedRegister(kGpReg, LiftoffRegList::ForRegs(dst));
606         LiftoffRegister dst_regs[] = {ret_reg, dst};
607         GenerateCCall(dst_regs, &sig, dst_type, &src, ext_ref);
608         __ emit_cond_jump(kEqual, trap, kWasmI32, ret_reg.gp());
609       } else {
610         ValueType sig_reps[] = {src_type};
611         FunctionSig sig(0, 1, sig_reps);
612         GenerateCCall(&dst, &sig, dst_type, &src, ext_ref);
613       }
614     }
615     __ PushRegister(dst_type, dst);
616   }
617 
UnOp(FullDecoder * decoder,WasmOpcode opcode,FunctionSig *,const Value & value,Value * result)618   void UnOp(FullDecoder* decoder, WasmOpcode opcode, FunctionSig*,
619             const Value& value, Value* result) {
620 #define CASE_I32_UNOP(opcode, fn)                       \
621   case WasmOpcode::kExpr##opcode:                       \
622     EmitUnOp<kWasmI32, kWasmI32>(                       \
623         [=](LiftoffRegister dst, LiftoffRegister src) { \
624           __ emit_##fn(dst.gp(), src.gp());             \
625         });                                             \
626     break;
627 #define CASE_FLOAT_UNOP(opcode, type, fn)               \
628   case WasmOpcode::kExpr##opcode:                       \
629     EmitUnOp<kWasm##type, kWasm##type>(                 \
630         [=](LiftoffRegister dst, LiftoffRegister src) { \
631           __ emit_##fn(dst.fp(), src.fp());             \
632         });                                             \
633     break;
634 #define CASE_FLOAT_UNOP_WITH_CFALLBACK(opcode, type, fn)                    \
635   case WasmOpcode::kExpr##opcode:                                           \
636     EmitFloatUnOpWithCFallback<kWasm##type>(&LiftoffAssembler::emit_##fn,   \
637                                             &ExternalReference::wasm_##fn); \
638     break;
639 #define CASE_TYPE_CONVERSION(opcode, dst_type, src_type, ext_ref, can_trap) \
640   case WasmOpcode::kExpr##opcode:                                           \
641     EmitTypeConversion<kWasm##dst_type, kWasm##src_type, can_trap>(         \
642         kExpr##opcode, ext_ref, can_trap ? decoder->position() : 0);        \
643     break;
644     switch (opcode) {
645       CASE_I32_UNOP(I32Eqz, i32_eqz)
646       CASE_I32_UNOP(I32Clz, i32_clz)
647       CASE_I32_UNOP(I32Ctz, i32_ctz)
648       CASE_FLOAT_UNOP(F32Abs, F32, f32_abs)
649       CASE_FLOAT_UNOP(F32Neg, F32, f32_neg)
650       CASE_FLOAT_UNOP(F32Ceil, F32, f32_ceil)
651       CASE_FLOAT_UNOP(F32Floor, F32, f32_floor)
652       CASE_FLOAT_UNOP(F32Trunc, F32, f32_trunc)
653       CASE_FLOAT_UNOP(F32NearestInt, F32, f32_nearest_int)
654       CASE_FLOAT_UNOP(F32Sqrt, F32, f32_sqrt)
655       CASE_FLOAT_UNOP(F64Abs, F64, f64_abs)
656       CASE_FLOAT_UNOP(F64Neg, F64, f64_neg)
657       CASE_FLOAT_UNOP_WITH_CFALLBACK(F64Ceil, F64, f64_ceil)
658       CASE_FLOAT_UNOP_WITH_CFALLBACK(F64Floor, F64, f64_floor)
659       CASE_FLOAT_UNOP_WITH_CFALLBACK(F64Trunc, F64, f64_trunc)
660       CASE_FLOAT_UNOP_WITH_CFALLBACK(F64NearestInt, F64, f64_nearest_int)
661       CASE_FLOAT_UNOP(F64Sqrt, F64, f64_sqrt)
662       CASE_TYPE_CONVERSION(I32ConvertI64, I32, I64, nullptr, kNoTrap)
663       CASE_TYPE_CONVERSION(I32SConvertF32, I32, F32, nullptr, kCanTrap)
664       CASE_TYPE_CONVERSION(I32UConvertF32, I32, F32, nullptr, kCanTrap)
665       CASE_TYPE_CONVERSION(I32SConvertF64, I32, F64, nullptr, kCanTrap)
666       CASE_TYPE_CONVERSION(I32UConvertF64, I32, F64, nullptr, kCanTrap)
667       CASE_TYPE_CONVERSION(I32ReinterpretF32, I32, F32, nullptr, kNoTrap)
668       CASE_TYPE_CONVERSION(I64SConvertI32, I64, I32, nullptr, kNoTrap)
669       CASE_TYPE_CONVERSION(I64UConvertI32, I64, I32, nullptr, kNoTrap)
670       CASE_TYPE_CONVERSION(I64SConvertF32, I64, F32,
671                            &ExternalReference::wasm_float32_to_int64, kCanTrap)
672       CASE_TYPE_CONVERSION(I64UConvertF32, I64, F32,
673                            &ExternalReference::wasm_float32_to_uint64, kCanTrap)
674       CASE_TYPE_CONVERSION(I64SConvertF64, I64, F64,
675                            &ExternalReference::wasm_float64_to_int64, kCanTrap)
676       CASE_TYPE_CONVERSION(I64UConvertF64, I64, F64,
677                            &ExternalReference::wasm_float64_to_uint64, kCanTrap)
678       CASE_TYPE_CONVERSION(I64ReinterpretF64, I64, F64, nullptr, kNoTrap)
679       CASE_TYPE_CONVERSION(F32SConvertI32, F32, I32, nullptr, kNoTrap)
680       CASE_TYPE_CONVERSION(F32UConvertI32, F32, I32, nullptr, kNoTrap)
681       CASE_TYPE_CONVERSION(F32SConvertI64, F32, I64,
682                            &ExternalReference::wasm_int64_to_float32, kNoTrap)
683       CASE_TYPE_CONVERSION(F32UConvertI64, F32, I64,
684                            &ExternalReference::wasm_uint64_to_float32, kNoTrap)
685       CASE_TYPE_CONVERSION(F32ConvertF64, F32, F64, nullptr, kNoTrap)
686       CASE_TYPE_CONVERSION(F32ReinterpretI32, F32, I32, nullptr, kNoTrap)
687       CASE_TYPE_CONVERSION(F64SConvertI32, F64, I32, nullptr, kNoTrap)
688       CASE_TYPE_CONVERSION(F64UConvertI32, F64, I32, nullptr, kNoTrap)
689       CASE_TYPE_CONVERSION(F64SConvertI64, F64, I64,
690                            &ExternalReference::wasm_int64_to_float64, kNoTrap)
691       CASE_TYPE_CONVERSION(F64UConvertI64, F64, I64,
692                            &ExternalReference::wasm_uint64_to_float64, kNoTrap)
693       CASE_TYPE_CONVERSION(F64ConvertF32, F64, F32, nullptr, kNoTrap)
694       CASE_TYPE_CONVERSION(F64ReinterpretI64, F64, I64, nullptr, kNoTrap)
695       case kExprI32Popcnt:
696         EmitI32UnOpWithCFallback(&LiftoffAssembler::emit_i32_popcnt,
697                                  &ExternalReference::wasm_word32_popcnt);
698         break;
699       case WasmOpcode::kExprI64Eqz:
700         EmitUnOp<kWasmI64, kWasmI32>(
701             [=](LiftoffRegister dst, LiftoffRegister src) {
702               __ emit_i64_eqz(dst.gp(), src);
703             });
704         break;
705       default:
706         return unsupported(decoder, WasmOpcodes::OpcodeName(opcode));
707     }
708 #undef CASE_I32_UNOP
709 #undef CASE_FLOAT_UNOP
710 #undef CASE_FLOAT_UNOP_WITH_CFALLBACK
711 #undef CASE_TYPE_CONVERSION
712   }
713 
714   template <ValueType src_type, ValueType result_type, typename EmitFn>
EmitBinOp(EmitFn fn)715   void EmitBinOp(EmitFn fn) {
716     static constexpr RegClass src_rc = reg_class_for(src_type);
717     static constexpr RegClass result_rc = reg_class_for(result_type);
718     LiftoffRegister rhs = __ PopToRegister();
719     LiftoffRegister lhs = __ PopToRegister(LiftoffRegList::ForRegs(rhs));
720     LiftoffRegister dst = src_rc == result_rc
721                               ? __ GetUnusedRegister(result_rc, {lhs, rhs})
722                               : __ GetUnusedRegister(result_rc);
723     fn(dst, lhs, rhs);
724     __ PushRegister(result_type, dst);
725   }
726 
EmitDivOrRem64CCall(LiftoffRegister dst,LiftoffRegister lhs,LiftoffRegister rhs,ExternalReference ext_ref,Label * trap_by_zero,Label * trap_unrepresentable=nullptr)727   void EmitDivOrRem64CCall(LiftoffRegister dst, LiftoffRegister lhs,
728                            LiftoffRegister rhs, ExternalReference ext_ref,
729                            Label* trap_by_zero,
730                            Label* trap_unrepresentable = nullptr) {
731     // Cannot emit native instructions, build C call.
732     LiftoffRegister ret =
733         __ GetUnusedRegister(kGpReg, LiftoffRegList::ForRegs(dst));
734     LiftoffRegister tmp =
735         __ GetUnusedRegister(kGpReg, LiftoffRegList::ForRegs(dst, ret));
736     LiftoffRegister arg_regs[] = {lhs, rhs};
737     LiftoffRegister result_regs[] = {ret, dst};
738     ValueType sig_types[] = {kWasmI32, kWasmI64, kWasmI64};
739     // <i64, i64> -> i32 (with i64 output argument)
740     FunctionSig sig(1, 2, sig_types);
741     GenerateCCall(result_regs, &sig, kWasmI64, arg_regs, ext_ref);
742     __ LoadConstant(tmp, WasmValue(int32_t{0}));
743     __ emit_cond_jump(kEqual, trap_by_zero, kWasmI32, ret.gp(), tmp.gp());
744     if (trap_unrepresentable) {
745       __ LoadConstant(tmp, WasmValue(int32_t{-1}));
746       __ emit_cond_jump(kEqual, trap_unrepresentable, kWasmI32, ret.gp(),
747                         tmp.gp());
748     }
749   }
750 
BinOp(FullDecoder * decoder,WasmOpcode opcode,FunctionSig *,const Value & lhs,const Value & rhs,Value * result)751   void BinOp(FullDecoder* decoder, WasmOpcode opcode, FunctionSig*,
752              const Value& lhs, const Value& rhs, Value* result) {
753 #define CASE_I32_BINOP(opcode, fn)                                           \
754   case WasmOpcode::kExpr##opcode:                                            \
755     return EmitBinOp<kWasmI32, kWasmI32>(                                    \
756         [=](LiftoffRegister dst, LiftoffRegister lhs, LiftoffRegister rhs) { \
757           __ emit_##fn(dst.gp(), lhs.gp(), rhs.gp());                        \
758         });
759 #define CASE_I64_BINOP(opcode, fn)                                           \
760   case WasmOpcode::kExpr##opcode:                                            \
761     return EmitBinOp<kWasmI64, kWasmI64>(                                    \
762         [=](LiftoffRegister dst, LiftoffRegister lhs, LiftoffRegister rhs) { \
763           __ emit_##fn(dst, lhs, rhs);                                       \
764         });
765 #define CASE_FLOAT_BINOP(opcode, type, fn)                                   \
766   case WasmOpcode::kExpr##opcode:                                            \
767     return EmitBinOp<kWasm##type, kWasm##type>(                              \
768         [=](LiftoffRegister dst, LiftoffRegister lhs, LiftoffRegister rhs) { \
769           __ emit_##fn(dst.fp(), lhs.fp(), rhs.fp());                        \
770         });
771 #define CASE_I32_CMPOP(opcode, cond)                                         \
772   case WasmOpcode::kExpr##opcode:                                            \
773     return EmitBinOp<kWasmI32, kWasmI32>(                                    \
774         [=](LiftoffRegister dst, LiftoffRegister lhs, LiftoffRegister rhs) { \
775           __ emit_i32_set_cond(cond, dst.gp(), lhs.gp(), rhs.gp());          \
776         });
777 #define CASE_I64_CMPOP(opcode, cond)                                         \
778   case WasmOpcode::kExpr##opcode:                                            \
779     return EmitBinOp<kWasmI64, kWasmI32>(                                    \
780         [=](LiftoffRegister dst, LiftoffRegister lhs, LiftoffRegister rhs) { \
781           __ emit_i64_set_cond(cond, dst.gp(), lhs, rhs);                    \
782         });
783 #define CASE_F32_CMPOP(opcode, cond)                                         \
784   case WasmOpcode::kExpr##opcode:                                            \
785     return EmitBinOp<kWasmF32, kWasmI32>(                                    \
786         [=](LiftoffRegister dst, LiftoffRegister lhs, LiftoffRegister rhs) { \
787           __ emit_f32_set_cond(cond, dst.gp(), lhs.fp(), rhs.fp());          \
788         });
789 #define CASE_F64_CMPOP(opcode, cond)                                         \
790   case WasmOpcode::kExpr##opcode:                                            \
791     return EmitBinOp<kWasmF64, kWasmI32>(                                    \
792         [=](LiftoffRegister dst, LiftoffRegister lhs, LiftoffRegister rhs) { \
793           __ emit_f64_set_cond(cond, dst.gp(), lhs.fp(), rhs.fp());          \
794         });
795 #define CASE_I32_SHIFTOP(opcode, fn)                                         \
796   case WasmOpcode::kExpr##opcode:                                            \
797     return EmitBinOp<kWasmI32, kWasmI32>(                                    \
798         [=](LiftoffRegister dst, LiftoffRegister lhs, LiftoffRegister rhs) { \
799           __ emit_##fn(dst.gp(), lhs.gp(), rhs.gp(), {});                    \
800         });
801 #define CASE_I64_SHIFTOP(opcode, fn)                                           \
802   case WasmOpcode::kExpr##opcode:                                              \
803     return EmitBinOp<kWasmI64, kWasmI64>([=](LiftoffRegister dst,              \
804                                              LiftoffRegister src,              \
805                                              LiftoffRegister amount) {         \
806       __ emit_##fn(dst, src, amount.is_pair() ? amount.low_gp() : amount.gp(), \
807                    {});                                                        \
808     });
809 #define CASE_CCALL_BINOP(opcode, type, ext_ref_fn)                           \
810   case WasmOpcode::kExpr##opcode:                                            \
811     return EmitBinOp<kWasmI32, kWasmI32>(                                    \
812         [=](LiftoffRegister dst, LiftoffRegister lhs, LiftoffRegister rhs) { \
813           LiftoffRegister args[] = {lhs, rhs};                               \
814           auto ext_ref = ExternalReference::ext_ref_fn();                    \
815           ValueType sig_i_ii_reps[] = {kWasmI32, kWasmI32, kWasmI32};        \
816           FunctionSig sig_i_ii(1, 2, sig_i_ii_reps);                         \
817           GenerateCCall(&dst, &sig_i_ii, kWasmStmt, args, ext_ref);          \
818         });
819     switch (opcode) {
820       CASE_I32_BINOP(I32Add, i32_add)
821       CASE_I32_BINOP(I32Sub, i32_sub)
822       CASE_I32_BINOP(I32Mul, i32_mul)
823       CASE_I32_BINOP(I32And, i32_and)
824       CASE_I32_BINOP(I32Ior, i32_or)
825       CASE_I32_BINOP(I32Xor, i32_xor)
826       CASE_I64_BINOP(I64And, i64_and)
827       CASE_I64_BINOP(I64Ior, i64_or)
828       CASE_I64_BINOP(I64Xor, i64_xor)
829       CASE_I32_CMPOP(I32Eq, kEqual)
830       CASE_I32_CMPOP(I32Ne, kUnequal)
831       CASE_I32_CMPOP(I32LtS, kSignedLessThan)
832       CASE_I32_CMPOP(I32LtU, kUnsignedLessThan)
833       CASE_I32_CMPOP(I32GtS, kSignedGreaterThan)
834       CASE_I32_CMPOP(I32GtU, kUnsignedGreaterThan)
835       CASE_I32_CMPOP(I32LeS, kSignedLessEqual)
836       CASE_I32_CMPOP(I32LeU, kUnsignedLessEqual)
837       CASE_I32_CMPOP(I32GeS, kSignedGreaterEqual)
838       CASE_I32_CMPOP(I32GeU, kUnsignedGreaterEqual)
839       CASE_I64_BINOP(I64Add, i64_add)
840       CASE_I64_BINOP(I64Sub, i64_sub)
841       CASE_I64_BINOP(I64Mul, i64_mul)
842       CASE_I64_CMPOP(I64Eq, kEqual)
843       CASE_I64_CMPOP(I64Ne, kUnequal)
844       CASE_I64_CMPOP(I64LtS, kSignedLessThan)
845       CASE_I64_CMPOP(I64LtU, kUnsignedLessThan)
846       CASE_I64_CMPOP(I64GtS, kSignedGreaterThan)
847       CASE_I64_CMPOP(I64GtU, kUnsignedGreaterThan)
848       CASE_I64_CMPOP(I64LeS, kSignedLessEqual)
849       CASE_I64_CMPOP(I64LeU, kUnsignedLessEqual)
850       CASE_I64_CMPOP(I64GeS, kSignedGreaterEqual)
851       CASE_I64_CMPOP(I64GeU, kUnsignedGreaterEqual)
852       CASE_F32_CMPOP(F32Eq, kEqual)
853       CASE_F32_CMPOP(F32Ne, kUnequal)
854       CASE_F32_CMPOP(F32Lt, kUnsignedLessThan)
855       CASE_F32_CMPOP(F32Gt, kUnsignedGreaterThan)
856       CASE_F32_CMPOP(F32Le, kUnsignedLessEqual)
857       CASE_F32_CMPOP(F32Ge, kUnsignedGreaterEqual)
858       CASE_F64_CMPOP(F64Eq, kEqual)
859       CASE_F64_CMPOP(F64Ne, kUnequal)
860       CASE_F64_CMPOP(F64Lt, kUnsignedLessThan)
861       CASE_F64_CMPOP(F64Gt, kUnsignedGreaterThan)
862       CASE_F64_CMPOP(F64Le, kUnsignedLessEqual)
863       CASE_F64_CMPOP(F64Ge, kUnsignedGreaterEqual)
864       CASE_I32_SHIFTOP(I32Shl, i32_shl)
865       CASE_I32_SHIFTOP(I32ShrS, i32_sar)
866       CASE_I32_SHIFTOP(I32ShrU, i32_shr)
867       CASE_I64_SHIFTOP(I64Shl, i64_shl)
868       CASE_I64_SHIFTOP(I64ShrS, i64_sar)
869       CASE_I64_SHIFTOP(I64ShrU, i64_shr)
870       CASE_CCALL_BINOP(I32Rol, I32, wasm_word32_rol)
871       CASE_CCALL_BINOP(I32Ror, I32, wasm_word32_ror)
872       CASE_FLOAT_BINOP(F32Add, F32, f32_add)
873       CASE_FLOAT_BINOP(F32Sub, F32, f32_sub)
874       CASE_FLOAT_BINOP(F32Mul, F32, f32_mul)
875       CASE_FLOAT_BINOP(F32Div, F32, f32_div)
876       CASE_FLOAT_BINOP(F32Min, F32, f32_min)
877       CASE_FLOAT_BINOP(F32Max, F32, f32_max)
878       CASE_FLOAT_BINOP(F64Add, F64, f64_add)
879       CASE_FLOAT_BINOP(F64Sub, F64, f64_sub)
880       CASE_FLOAT_BINOP(F64Mul, F64, f64_mul)
881       CASE_FLOAT_BINOP(F64Div, F64, f64_div)
882       CASE_FLOAT_BINOP(F64Min, F64, f64_min)
883       CASE_FLOAT_BINOP(F64Max, F64, f64_max)
884       case WasmOpcode::kExprI32DivS:
885         EmitBinOp<kWasmI32, kWasmI32>([this, decoder](LiftoffRegister dst,
886                                                       LiftoffRegister lhs,
887                                                       LiftoffRegister rhs) {
888           WasmCodePosition position = decoder->position();
889           AddOutOfLineTrap(position, WasmCode::kThrowWasmTrapDivByZero);
890           // Adding the second trap might invalidate the pointer returned for
891           // the first one, thus get both pointers afterwards.
892           AddOutOfLineTrap(position,
893                            WasmCode::kThrowWasmTrapDivUnrepresentable);
894           Label* div_by_zero = out_of_line_code_.end()[-2].label.get();
895           Label* div_unrepresentable = out_of_line_code_.end()[-1].label.get();
896           __ emit_i32_divs(dst.gp(), lhs.gp(), rhs.gp(), div_by_zero,
897                            div_unrepresentable);
898         });
899         break;
900       case WasmOpcode::kExprI32DivU:
901         EmitBinOp<kWasmI32, kWasmI32>([this, decoder](LiftoffRegister dst,
902                                                       LiftoffRegister lhs,
903                                                       LiftoffRegister rhs) {
904           Label* div_by_zero = AddOutOfLineTrap(
905               decoder->position(), WasmCode::kThrowWasmTrapDivByZero);
906           __ emit_i32_divu(dst.gp(), lhs.gp(), rhs.gp(), div_by_zero);
907         });
908         break;
909       case WasmOpcode::kExprI32RemS:
910         EmitBinOp<kWasmI32, kWasmI32>([this, decoder](LiftoffRegister dst,
911                                                       LiftoffRegister lhs,
912                                                       LiftoffRegister rhs) {
913           Label* rem_by_zero = AddOutOfLineTrap(
914               decoder->position(), WasmCode::kThrowWasmTrapRemByZero);
915           __ emit_i32_rems(dst.gp(), lhs.gp(), rhs.gp(), rem_by_zero);
916         });
917         break;
918       case WasmOpcode::kExprI32RemU:
919         EmitBinOp<kWasmI32, kWasmI32>([this, decoder](LiftoffRegister dst,
920                                                       LiftoffRegister lhs,
921                                                       LiftoffRegister rhs) {
922           Label* rem_by_zero = AddOutOfLineTrap(
923               decoder->position(), WasmCode::kThrowWasmTrapRemByZero);
924           __ emit_i32_remu(dst.gp(), lhs.gp(), rhs.gp(), rem_by_zero);
925         });
926         break;
927       case WasmOpcode::kExprI64DivS:
928         EmitBinOp<kWasmI64, kWasmI64>([this, decoder](LiftoffRegister dst,
929                                                       LiftoffRegister lhs,
930                                                       LiftoffRegister rhs) {
931           WasmCodePosition position = decoder->position();
932           AddOutOfLineTrap(position, WasmCode::kThrowWasmTrapDivByZero);
933           // Adding the second trap might invalidate the pointer returned for
934           // the first one, thus get both pointers afterwards.
935           AddOutOfLineTrap(position,
936                            WasmCode::kThrowWasmTrapDivUnrepresentable);
937           Label* div_by_zero = out_of_line_code_.end()[-2].label.get();
938           Label* div_unrepresentable = out_of_line_code_.end()[-1].label.get();
939           if (!__ emit_i64_divs(dst, lhs, rhs, div_by_zero,
940                                 div_unrepresentable)) {
941             ExternalReference ext_ref = ExternalReference::wasm_int64_div();
942             EmitDivOrRem64CCall(dst, lhs, rhs, ext_ref, div_by_zero,
943                                 div_unrepresentable);
944           }
945         });
946         break;
947       case WasmOpcode::kExprI64DivU:
948         EmitBinOp<kWasmI64, kWasmI64>([this, decoder](LiftoffRegister dst,
949                                                       LiftoffRegister lhs,
950                                                       LiftoffRegister rhs) {
951           Label* div_by_zero = AddOutOfLineTrap(
952               decoder->position(), WasmCode::kThrowWasmTrapDivByZero);
953           if (!__ emit_i64_divu(dst, lhs, rhs, div_by_zero)) {
954             ExternalReference ext_ref = ExternalReference::wasm_uint64_div();
955             EmitDivOrRem64CCall(dst, lhs, rhs, ext_ref, div_by_zero);
956           }
957         });
958         break;
959       case WasmOpcode::kExprI64RemS:
960         EmitBinOp<kWasmI64, kWasmI64>([this, decoder](LiftoffRegister dst,
961                                                       LiftoffRegister lhs,
962                                                       LiftoffRegister rhs) {
963           Label* rem_by_zero = AddOutOfLineTrap(
964               decoder->position(), WasmCode::kThrowWasmTrapRemByZero);
965           if (!__ emit_i64_rems(dst, lhs, rhs, rem_by_zero)) {
966             ExternalReference ext_ref = ExternalReference::wasm_int64_mod();
967             EmitDivOrRem64CCall(dst, lhs, rhs, ext_ref, rem_by_zero);
968           }
969         });
970         break;
971       case WasmOpcode::kExprI64RemU:
972         EmitBinOp<kWasmI64, kWasmI64>([this, decoder](LiftoffRegister dst,
973                                                       LiftoffRegister lhs,
974                                                       LiftoffRegister rhs) {
975           Label* rem_by_zero = AddOutOfLineTrap(
976               decoder->position(), WasmCode::kThrowWasmTrapRemByZero);
977           if (!__ emit_i64_remu(dst, lhs, rhs, rem_by_zero)) {
978             ExternalReference ext_ref = ExternalReference::wasm_uint64_mod();
979             EmitDivOrRem64CCall(dst, lhs, rhs, ext_ref, rem_by_zero);
980           }
981         });
982         break;
983       default:
984         return unsupported(decoder, WasmOpcodes::OpcodeName(opcode));
985     }
986 #undef CASE_I32_BINOP
987 #undef CASE_I64_BINOP
988 #undef CASE_FLOAT_BINOP
989 #undef CASE_I32_CMPOP
990 #undef CASE_I64_CMPOP
991 #undef CASE_F32_CMPOP
992 #undef CASE_F64_CMPOP
993 #undef CASE_I32_SHIFTOP
994 #undef CASE_I64_SHIFTOP
995 #undef CASE_CCALL_BINOP
996   }
997 
I32Const(FullDecoder * decoder,Value * result,int32_t value)998   void I32Const(FullDecoder* decoder, Value* result, int32_t value) {
999     __ cache_state()->stack_state.emplace_back(kWasmI32, value);
1000   }
1001 
I64Const(FullDecoder * decoder,Value * result,int64_t value)1002   void I64Const(FullDecoder* decoder, Value* result, int64_t value) {
1003     // The {VarState} stores constant values as int32_t, thus we only store
1004     // 64-bit constants in this field if it fits in an int32_t. Larger values
1005     // cannot be used as immediate value anyway, so we can also just put them in
1006     // a register immediately.
1007     int32_t value_i32 = static_cast<int32_t>(value);
1008     if (value_i32 == value) {
1009       __ cache_state()->stack_state.emplace_back(kWasmI64, value_i32);
1010     } else {
1011       LiftoffRegister reg = __ GetUnusedRegister(reg_class_for(kWasmI64));
1012       __ LoadConstant(reg, WasmValue(value));
1013       __ PushRegister(kWasmI64, reg);
1014     }
1015   }
1016 
F32Const(FullDecoder * decoder,Value * result,float value)1017   void F32Const(FullDecoder* decoder, Value* result, float value) {
1018     LiftoffRegister reg = __ GetUnusedRegister(kFpReg);
1019     __ LoadConstant(reg, WasmValue(value));
1020     __ PushRegister(kWasmF32, reg);
1021   }
1022 
F64Const(FullDecoder * decoder,Value * result,double value)1023   void F64Const(FullDecoder* decoder, Value* result, double value) {
1024     LiftoffRegister reg = __ GetUnusedRegister(kFpReg);
1025     __ LoadConstant(reg, WasmValue(value));
1026     __ PushRegister(kWasmF64, reg);
1027   }
1028 
RefNull(FullDecoder * decoder,Value * result)1029   void RefNull(FullDecoder* decoder, Value* result) {
1030     unsupported(decoder, "ref_null");
1031   }
1032 
Drop(FullDecoder * decoder,const Value & value)1033   void Drop(FullDecoder* decoder, const Value& value) {
1034     auto& slot = __ cache_state()->stack_state.back();
1035     // If the dropped slot contains a register, decrement it's use count.
1036     if (slot.is_reg()) __ cache_state()->dec_used(slot.reg());
1037     __ cache_state()->stack_state.pop_back();
1038   }
1039 
DoReturn(FullDecoder * decoder,Vector<Value> values,bool implicit)1040   void DoReturn(FullDecoder* decoder, Vector<Value> values, bool implicit) {
1041     if (implicit) {
1042       DCHECK_EQ(1, decoder->control_depth());
1043       Control* func_block = decoder->control_at(0);
1044       __ bind(func_block->label.get());
1045       __ cache_state()->Steal(func_block->label_state);
1046     }
1047     if (!values.is_empty()) {
1048       if (values.size() > 1) return unsupported(decoder, "multi-return");
1049       LiftoffRegister reg = __ PopToRegister();
1050       LiftoffRegister return_reg =
1051           kNeedI64RegPair && values[0].type == kWasmI64
1052               ? LiftoffRegister::ForPair(kGpReturnRegisters[0],
1053                                          kGpReturnRegisters[1])
1054               : reg_class_for(values[0].type) == kGpReg
1055                     ? LiftoffRegister(kGpReturnRegisters[0])
1056                     : LiftoffRegister(kFpReturnRegisters[0]);
1057       if (reg != return_reg) __ Move(return_reg, reg, values[0].type);
1058     }
1059     __ LeaveFrame(StackFrame::WASM_COMPILED);
1060     __ DropStackSlotsAndRet(
1061         static_cast<uint32_t>(descriptor_->StackParameterCount()));
1062   }
1063 
GetLocal(FullDecoder * decoder,Value * result,const LocalIndexImmediate<validate> & imm)1064   void GetLocal(FullDecoder* decoder, Value* result,
1065                 const LocalIndexImmediate<validate>& imm) {
1066     auto& slot = __ cache_state()->stack_state[imm.index];
1067     DCHECK_EQ(slot.type(), imm.type);
1068     switch (slot.loc()) {
1069       case kRegister:
1070         __ PushRegister(slot.type(), slot.reg());
1071         break;
1072       case KIntConst:
1073         __ cache_state()->stack_state.emplace_back(imm.type, slot.i32_const());
1074         break;
1075       case kStack: {
1076         auto rc = reg_class_for(imm.type);
1077         LiftoffRegister reg = __ GetUnusedRegister(rc);
1078         __ Fill(reg, imm.index, imm.type);
1079         __ PushRegister(slot.type(), reg);
1080         break;
1081       }
1082     }
1083   }
1084 
SetLocalFromStackSlot(LiftoffAssembler::VarState & dst_slot,uint32_t local_index)1085   void SetLocalFromStackSlot(LiftoffAssembler::VarState& dst_slot,
1086                              uint32_t local_index) {
1087     auto& state = *__ cache_state();
1088     ValueType type = dst_slot.type();
1089     if (dst_slot.is_reg()) {
1090       LiftoffRegister slot_reg = dst_slot.reg();
1091       if (state.get_use_count(slot_reg) == 1) {
1092         __ Fill(dst_slot.reg(), state.stack_height() - 1, type);
1093         return;
1094       }
1095       state.dec_used(slot_reg);
1096       dst_slot.MakeStack();
1097     }
1098     DCHECK_EQ(type, __ local_type(local_index));
1099     RegClass rc = reg_class_for(type);
1100     LiftoffRegister dst_reg = __ GetUnusedRegister(rc);
1101     __ Fill(dst_reg, __ cache_state()->stack_height() - 1, type);
1102     dst_slot = LiftoffAssembler::VarState(type, dst_reg);
1103     __ cache_state()->inc_used(dst_reg);
1104   }
1105 
SetLocal(uint32_t local_index,bool is_tee)1106   void SetLocal(uint32_t local_index, bool is_tee) {
1107     auto& state = *__ cache_state();
1108     auto& source_slot = state.stack_state.back();
1109     auto& target_slot = state.stack_state[local_index];
1110     switch (source_slot.loc()) {
1111       case kRegister:
1112         if (target_slot.is_reg()) state.dec_used(target_slot.reg());
1113         target_slot = source_slot;
1114         if (is_tee) state.inc_used(target_slot.reg());
1115         break;
1116       case KIntConst:
1117         if (target_slot.is_reg()) state.dec_used(target_slot.reg());
1118         target_slot = source_slot;
1119         break;
1120       case kStack:
1121         SetLocalFromStackSlot(target_slot, local_index);
1122         break;
1123     }
1124     if (!is_tee) __ cache_state()->stack_state.pop_back();
1125   }
1126 
SetLocal(FullDecoder * decoder,const Value & value,const LocalIndexImmediate<validate> & imm)1127   void SetLocal(FullDecoder* decoder, const Value& value,
1128                 const LocalIndexImmediate<validate>& imm) {
1129     SetLocal(imm.index, false);
1130   }
1131 
TeeLocal(FullDecoder * decoder,const Value & value,Value * result,const LocalIndexImmediate<validate> & imm)1132   void TeeLocal(FullDecoder* decoder, const Value& value, Value* result,
1133                 const LocalIndexImmediate<validate>& imm) {
1134     SetLocal(imm.index, true);
1135   }
1136 
GetGlobalBaseAndOffset(const WasmGlobal * global,LiftoffRegList & pinned,uint32_t * offset)1137   LiftoffRegister GetGlobalBaseAndOffset(const WasmGlobal* global,
1138                                          LiftoffRegList& pinned,
1139                                          uint32_t* offset) {
1140     LiftoffRegister addr = pinned.set(__ GetUnusedRegister(kGpReg));
1141     if (global->mutability && global->imported) {
1142       LOAD_INSTANCE_FIELD(addr, ImportedMutableGlobals, kPointerLoadType);
1143       __ Load(addr, addr.gp(), no_reg, global->index * sizeof(Address),
1144               kPointerLoadType, pinned);
1145       *offset = 0;
1146     } else {
1147       LOAD_INSTANCE_FIELD(addr, GlobalsStart, kPointerLoadType);
1148       *offset = global->offset;
1149     }
1150     return addr;
1151   }
1152 
GetGlobal(FullDecoder * decoder,Value * result,const GlobalIndexImmediate<validate> & imm)1153   void GetGlobal(FullDecoder* decoder, Value* result,
1154                  const GlobalIndexImmediate<validate>& imm) {
1155     const auto* global = &env_->module->globals[imm.index];
1156     if (!CheckSupportedType(decoder, kTypes_ilfd, global->type, "global"))
1157       return;
1158     LiftoffRegList pinned;
1159     uint32_t offset = 0;
1160     LiftoffRegister addr = GetGlobalBaseAndOffset(global, pinned, &offset);
1161     LiftoffRegister value =
1162         pinned.set(__ GetUnusedRegister(reg_class_for(global->type), pinned));
1163     LoadType type = LoadType::ForValueType(global->type);
1164     __ Load(value, addr.gp(), no_reg, offset, type, pinned);
1165     __ PushRegister(global->type, value);
1166   }
1167 
SetGlobal(FullDecoder * decoder,const Value & value,const GlobalIndexImmediate<validate> & imm)1168   void SetGlobal(FullDecoder* decoder, const Value& value,
1169                  const GlobalIndexImmediate<validate>& imm) {
1170     auto* global = &env_->module->globals[imm.index];
1171     if (!CheckSupportedType(decoder, kTypes_ilfd, global->type, "global"))
1172       return;
1173     LiftoffRegList pinned;
1174     uint32_t offset = 0;
1175     LiftoffRegister addr = GetGlobalBaseAndOffset(global, pinned, &offset);
1176     LiftoffRegister reg = pinned.set(__ PopToRegister(pinned));
1177     StoreType type = StoreType::ForValueType(global->type);
1178     __ Store(addr.gp(), no_reg, offset, reg, type, pinned);
1179   }
1180 
Unreachable(FullDecoder * decoder)1181   void Unreachable(FullDecoder* decoder) {
1182     Label* unreachable_label = AddOutOfLineTrap(
1183         decoder->position(), WasmCode::kThrowWasmTrapUnreachable);
1184     __ emit_jump(unreachable_label);
1185     __ AssertUnreachable(AbortReason::kUnexpectedReturnFromWasmTrap);
1186   }
1187 
Select(FullDecoder * decoder,const Value & cond,const Value & fval,const Value & tval,Value * result)1188   void Select(FullDecoder* decoder, const Value& cond, const Value& fval,
1189               const Value& tval, Value* result) {
1190     LiftoffRegList pinned;
1191     Register condition = pinned.set(__ PopToRegister()).gp();
1192     ValueType type = __ cache_state()->stack_state.end()[-1].type();
1193     DCHECK_EQ(type, __ cache_state()->stack_state.end()[-2].type());
1194     LiftoffRegister false_value = pinned.set(__ PopToRegister(pinned));
1195     LiftoffRegister true_value = __ PopToRegister(pinned);
1196     LiftoffRegister dst =
1197         __ GetUnusedRegister(true_value.reg_class(), {true_value, false_value});
1198     __ PushRegister(type, dst);
1199 
1200     // Now emit the actual code to move either {true_value} or {false_value}
1201     // into {dst}.
1202     Label cont;
1203     Label case_false;
1204     __ emit_cond_jump(kEqual, &case_false, kWasmI32, condition);
1205     if (dst != true_value) __ Move(dst, true_value, type);
1206     __ emit_jump(&cont);
1207 
1208     __ bind(&case_false);
1209     if (dst != false_value) __ Move(dst, false_value, type);
1210     __ bind(&cont);
1211   }
1212 
Br(Control * target)1213   void Br(Control* target) {
1214     if (!target->br_merge()->reached) {
1215       target->label_state.InitMerge(*__ cache_state(), __ num_locals(),
1216                                     target->br_merge()->arity);
1217     }
1218     __ MergeStackWith(target->label_state, target->br_merge()->arity);
1219     __ jmp(target->label.get());
1220   }
1221 
Br(FullDecoder * decoder,Control * target)1222   void Br(FullDecoder* decoder, Control* target) { Br(target); }
1223 
BrIf(FullDecoder * decoder,const Value & cond,Control * target)1224   void BrIf(FullDecoder* decoder, const Value& cond, Control* target) {
1225     Label cont_false;
1226     Register value = __ PopToRegister().gp();
1227     __ emit_cond_jump(kEqual, &cont_false, kWasmI32, value);
1228 
1229     Br(target);
1230     __ bind(&cont_false);
1231   }
1232 
1233   // Generate a branch table case, potentially reusing previously generated
1234   // stack transfer code.
GenerateBrCase(FullDecoder * decoder,uint32_t br_depth,std::map<uint32_t,MovableLabel> & br_targets)1235   void GenerateBrCase(FullDecoder* decoder, uint32_t br_depth,
1236                       std::map<uint32_t, MovableLabel>& br_targets) {
1237     MovableLabel& label = br_targets[br_depth];
1238     if (label.get()->is_bound()) {
1239       __ jmp(label.get());
1240     } else {
1241       __ bind(label.get());
1242       Br(decoder->control_at(br_depth));
1243     }
1244   }
1245 
1246   // Generate a branch table for input in [min, max).
1247   // TODO(wasm): Generate a real branch table (like TF TableSwitch).
GenerateBrTable(FullDecoder * decoder,LiftoffRegister tmp,LiftoffRegister value,uint32_t min,uint32_t max,BranchTableIterator<validate> & table_iterator,std::map<uint32_t,MovableLabel> & br_targets)1248   void GenerateBrTable(FullDecoder* decoder, LiftoffRegister tmp,
1249                        LiftoffRegister value, uint32_t min, uint32_t max,
1250                        BranchTableIterator<validate>& table_iterator,
1251                        std::map<uint32_t, MovableLabel>& br_targets) {
1252     DCHECK_LT(min, max);
1253     // Check base case.
1254     if (max == min + 1) {
1255       DCHECK_EQ(min, table_iterator.cur_index());
1256       GenerateBrCase(decoder, table_iterator.next(), br_targets);
1257       return;
1258     }
1259 
1260     uint32_t split = min + (max - min) / 2;
1261     Label upper_half;
1262     __ LoadConstant(tmp, WasmValue(split));
1263     __ emit_cond_jump(kUnsignedGreaterEqual, &upper_half, kWasmI32, value.gp(),
1264                       tmp.gp());
1265     // Emit br table for lower half:
1266     GenerateBrTable(decoder, tmp, value, min, split, table_iterator,
1267                     br_targets);
1268     __ bind(&upper_half);
1269     // Emit br table for upper half:
1270     GenerateBrTable(decoder, tmp, value, split, max, table_iterator,
1271                     br_targets);
1272   }
1273 
BrTable(FullDecoder * decoder,const BranchTableImmediate<validate> & imm,const Value & key)1274   void BrTable(FullDecoder* decoder, const BranchTableImmediate<validate>& imm,
1275                const Value& key) {
1276     LiftoffRegList pinned;
1277     LiftoffRegister value = pinned.set(__ PopToRegister());
1278     BranchTableIterator<validate> table_iterator(decoder, imm);
1279     std::map<uint32_t, MovableLabel> br_targets;
1280 
1281     if (imm.table_count > 0) {
1282       LiftoffRegister tmp = __ GetUnusedRegister(kGpReg, pinned);
1283       __ LoadConstant(tmp, WasmValue(uint32_t{imm.table_count}));
1284       Label case_default;
1285       __ emit_cond_jump(kUnsignedGreaterEqual, &case_default, kWasmI32,
1286                         value.gp(), tmp.gp());
1287 
1288       GenerateBrTable(decoder, tmp, value, 0, imm.table_count, table_iterator,
1289                       br_targets);
1290 
1291       __ bind(&case_default);
1292     }
1293 
1294     // Generate the default case.
1295     GenerateBrCase(decoder, table_iterator.next(), br_targets);
1296     DCHECK(!table_iterator.has_next());
1297   }
1298 
Else(FullDecoder * decoder,Control * if_block)1299   void Else(FullDecoder* decoder, Control* if_block) {
1300     if (if_block->reachable()) __ emit_jump(if_block->label.get());
1301     __ bind(if_block->else_state->label.get());
1302     __ cache_state()->Steal(if_block->else_state->state);
1303   }
1304 
AddOutOfLineTrap(WasmCodePosition position,WasmCode::RuntimeStubId stub,uint32_t pc=0)1305   Label* AddOutOfLineTrap(WasmCodePosition position,
1306                           WasmCode::RuntimeStubId stub, uint32_t pc = 0) {
1307     DCHECK(!FLAG_wasm_no_bounds_checks);
1308     // The pc is needed for memory OOB trap with trap handler enabled. Other
1309     // callers should not even compute it.
1310     DCHECK_EQ(pc != 0, stub == WasmCode::kThrowWasmTrapMemOutOfBounds &&
1311                            env_->use_trap_handler);
1312 
1313     out_of_line_code_.push_back(OutOfLineCode::Trap(stub, position, pc));
1314     return out_of_line_code_.back().label.get();
1315   }
1316 
1317   // Returns true if the memory access is statically known to be out of bounds
1318   // (a jump to the trap was generated then); return false otherwise.
BoundsCheckMem(FullDecoder * decoder,uint32_t access_size,uint32_t offset,Register index,LiftoffRegList pinned)1319   bool BoundsCheckMem(FullDecoder* decoder, uint32_t access_size,
1320                       uint32_t offset, Register index, LiftoffRegList pinned) {
1321     const bool statically_oob = access_size > env_->max_memory_size ||
1322                                 offset > env_->max_memory_size - access_size;
1323 
1324     if (!statically_oob &&
1325         (FLAG_wasm_no_bounds_checks || env_->use_trap_handler)) {
1326       return false;
1327     }
1328 
1329     // TODO(wasm): This adds protected instruction information for the jump
1330     // instruction we are about to generate. It would be better to just not add
1331     // protected instruction info when the pc is 0.
1332     Label* trap_label = AddOutOfLineTrap(
1333         decoder->position(), WasmCode::kThrowWasmTrapMemOutOfBounds,
1334         env_->use_trap_handler ? __ pc_offset() : 0);
1335 
1336     if (statically_oob) {
1337       __ emit_jump(trap_label);
1338       Control* current_block = decoder->control_at(0);
1339       if (current_block->reachable()) {
1340         current_block->reachability = kSpecOnlyReachable;
1341       }
1342       return true;
1343     }
1344 
1345     DCHECK(!env_->use_trap_handler);
1346     DCHECK(!FLAG_wasm_no_bounds_checks);
1347 
1348     uint64_t end_offset = uint64_t{offset} + access_size - 1u;
1349 
1350     // If the end offset is larger than the smallest memory, dynamically check
1351     // the end offset against the actual memory size, which is not known at
1352     // compile time. Otherwise, only one check is required (see below).
1353     LiftoffRegister end_offset_reg =
1354         pinned.set(__ GetUnusedRegister(kGpReg, pinned));
1355     LiftoffRegister mem_size = __ GetUnusedRegister(kGpReg, pinned);
1356     LOAD_INSTANCE_FIELD(mem_size, MemorySize, kPointerLoadType);
1357 
1358     if (kPointerSize == 8) {
1359       __ LoadConstant(end_offset_reg, WasmValue(end_offset));
1360     } else {
1361       __ LoadConstant(end_offset_reg,
1362                       WasmValue(static_cast<uint32_t>(end_offset)));
1363     }
1364 
1365     if (end_offset >= env_->min_memory_size) {
1366       __ emit_cond_jump(kUnsignedGreaterEqual, trap_label,
1367                         LiftoffAssembler::kWasmIntPtr, end_offset_reg.gp(),
1368                         mem_size.gp());
1369     }
1370 
1371     // Just reuse the end_offset register for computing the effective size.
1372     LiftoffRegister effective_size_reg = end_offset_reg;
1373     __ emit_ptrsize_sub(effective_size_reg.gp(), mem_size.gp(),
1374                         end_offset_reg.gp());
1375 
1376     __ emit_i32_to_intptr(index, index);
1377 
1378     __ emit_cond_jump(kUnsignedGreaterEqual, trap_label,
1379                       LiftoffAssembler::kWasmIntPtr, index,
1380                       effective_size_reg.gp());
1381     return false;
1382   }
1383 
TraceMemoryOperation(bool is_store,MachineRepresentation rep,Register index,uint32_t offset,WasmCodePosition position)1384   void TraceMemoryOperation(bool is_store, MachineRepresentation rep,
1385                             Register index, uint32_t offset,
1386                             WasmCodePosition position) {
1387     // Before making the runtime call, spill all cache registers.
1388     __ SpillAllRegisters();
1389 
1390     LiftoffRegList pinned = LiftoffRegList::ForRegs(index);
1391     // Get one register for computing the address (offset + index).
1392     LiftoffRegister address = pinned.set(__ GetUnusedRegister(kGpReg, pinned));
1393     // Compute offset+index in address.
1394     __ LoadConstant(address, WasmValue(offset));
1395     __ emit_i32_add(address.gp(), address.gp(), index);
1396 
1397     // Get a register to hold the stack slot for MemoryTracingInfo.
1398     LiftoffRegister info = pinned.set(__ GetUnusedRegister(kGpReg, pinned));
1399     // Allocate stack slot for MemoryTracingInfo.
1400     __ AllocateStackSlot(info.gp(), sizeof(MemoryTracingInfo));
1401 
1402     // Now store all information into the MemoryTracingInfo struct.
1403     __ Store(info.gp(), no_reg, offsetof(MemoryTracingInfo, address), address,
1404              StoreType::kI32Store, pinned);
1405     __ LoadConstant(address, WasmValue(is_store ? 1 : 0));
1406     __ Store(info.gp(), no_reg, offsetof(MemoryTracingInfo, is_store), address,
1407              StoreType::kI32Store8, pinned);
1408     __ LoadConstant(address, WasmValue(static_cast<int>(rep)));
1409     __ Store(info.gp(), no_reg, offsetof(MemoryTracingInfo, mem_rep), address,
1410              StoreType::kI32Store8, pinned);
1411 
1412     source_position_table_builder_.AddPosition(__ pc_offset(),
1413                                                SourcePosition(position), false);
1414 
1415     Register args[] = {info.gp()};
1416     GenerateRuntimeCall(Runtime::kWasmTraceMemory, arraysize(args), args);
1417     __ DeallocateStackSlot(sizeof(MemoryTracingInfo));
1418   }
1419 
GenerateRuntimeCall(Runtime::FunctionId runtime_function,int num_args,Register * args)1420   void GenerateRuntimeCall(Runtime::FunctionId runtime_function, int num_args,
1421                            Register* args) {
1422     auto call_descriptor = compiler::Linkage::GetRuntimeCallDescriptor(
1423         compilation_zone_, runtime_function, num_args,
1424         compiler::Operator::kNoProperties, compiler::CallDescriptor::kNoFlags);
1425     // Currently, only one argument is supported. More arguments require some
1426     // caution for the parallel register moves (reuse StackTransferRecipe).
1427     DCHECK_EQ(1, num_args);
1428     constexpr size_t kInputShift = 1;  // Input 0 is the call target.
1429     compiler::LinkageLocation param_loc =
1430         call_descriptor->GetInputLocation(kInputShift);
1431     if (param_loc.IsRegister()) {
1432       Register reg = Register::from_code(param_loc.AsRegister());
1433       __ Move(LiftoffRegister(reg), LiftoffRegister(args[0]),
1434               LiftoffAssembler::kWasmIntPtr);
1435     } else {
1436       DCHECK(param_loc.IsCallerFrameSlot());
1437       LiftoffStackSlots stack_slots(&asm_);
1438       stack_slots.Add(LiftoffAssembler::VarState(LiftoffAssembler::kWasmIntPtr,
1439                                                  LiftoffRegister(args[0])));
1440       stack_slots.Construct();
1441     }
1442 
1443     // Set context to zero (Smi::kZero) for the runtime call.
1444     __ TurboAssembler::Move(kContextRegister, Smi::kZero);
1445     LiftoffRegister centry(kJavaScriptCallCodeStartRegister);
1446     LOAD_INSTANCE_FIELD(centry, CEntryStub, kPointerLoadType);
1447     __ CallRuntimeWithCEntry(runtime_function, centry.gp());
1448     safepoint_table_builder_.DefineSafepoint(&asm_, Safepoint::kSimple, 0,
1449                                              Safepoint::kNoLazyDeopt);
1450   }
1451 
AddMemoryMasking(LiftoffRegister index,uint32_t * offset,LiftoffRegList & pinned)1452   LiftoffRegister AddMemoryMasking(LiftoffRegister index, uint32_t* offset,
1453                                    LiftoffRegList& pinned) {
1454     if (!FLAG_untrusted_code_mitigations || env_->use_trap_handler) {
1455       return index;
1456     }
1457     DEBUG_CODE_COMMENT("Mask memory index");
1458     // Make sure that we can overwrite {index}.
1459     if (__ cache_state()->is_used(index)) {
1460       LiftoffRegister old_index = index;
1461       pinned.clear(old_index);
1462       index = pinned.set(__ GetUnusedRegister(kGpReg, pinned));
1463       if (index != old_index) __ Move(index.gp(), old_index.gp(), kWasmI32);
1464     }
1465     LiftoffRegister tmp = __ GetUnusedRegister(kGpReg, pinned);
1466     __ LoadConstant(tmp, WasmValue(*offset));
1467     __ emit_i32_add(index.gp(), index.gp(), tmp.gp());
1468     LOAD_INSTANCE_FIELD(tmp, MemoryMask, LoadType::kI32Load);
1469     __ emit_i32_and(index.gp(), index.gp(), tmp.gp());
1470     *offset = 0;
1471     return index;
1472   }
1473 
LoadMem(FullDecoder * decoder,LoadType type,const MemoryAccessImmediate<validate> & imm,const Value & index_val,Value * result)1474   void LoadMem(FullDecoder* decoder, LoadType type,
1475                const MemoryAccessImmediate<validate>& imm,
1476                const Value& index_val, Value* result) {
1477     ValueType value_type = type.value_type();
1478     if (!CheckSupportedType(decoder, kTypes_ilfd, value_type, "load")) return;
1479     LiftoffRegList pinned;
1480     LiftoffRegister index = pinned.set(__ PopToRegister());
1481     if (BoundsCheckMem(decoder, type.size(), imm.offset, index.gp(), pinned)) {
1482       return;
1483     }
1484     uint32_t offset = imm.offset;
1485     index = AddMemoryMasking(index, &offset, pinned);
1486     DEBUG_CODE_COMMENT("Load from memory");
1487     LiftoffRegister addr = pinned.set(__ GetUnusedRegister(kGpReg, pinned));
1488     LOAD_INSTANCE_FIELD(addr, MemoryStart, kPointerLoadType);
1489     RegClass rc = reg_class_for(value_type);
1490     LiftoffRegister value = pinned.set(__ GetUnusedRegister(rc, pinned));
1491     uint32_t protected_load_pc = 0;
1492     __ Load(value, addr.gp(), index.gp(), offset, type, pinned,
1493             &protected_load_pc, true);
1494     if (env_->use_trap_handler) {
1495       AddOutOfLineTrap(decoder->position(),
1496                        WasmCode::kThrowWasmTrapMemOutOfBounds,
1497                        protected_load_pc);
1498     }
1499     __ PushRegister(value_type, value);
1500 
1501     if (FLAG_wasm_trace_memory) {
1502       TraceMemoryOperation(false, type.mem_type().representation(), index.gp(),
1503                            offset, decoder->position());
1504     }
1505   }
1506 
StoreMem(FullDecoder * decoder,StoreType type,const MemoryAccessImmediate<validate> & imm,const Value & index_val,const Value & value_val)1507   void StoreMem(FullDecoder* decoder, StoreType type,
1508                 const MemoryAccessImmediate<validate>& imm,
1509                 const Value& index_val, const Value& value_val) {
1510     ValueType value_type = type.value_type();
1511     if (!CheckSupportedType(decoder, kTypes_ilfd, value_type, "store")) return;
1512     LiftoffRegList pinned;
1513     LiftoffRegister value = pinned.set(__ PopToRegister());
1514     LiftoffRegister index = pinned.set(__ PopToRegister(pinned));
1515     if (BoundsCheckMem(decoder, type.size(), imm.offset, index.gp(), pinned)) {
1516       return;
1517     }
1518     uint32_t offset = imm.offset;
1519     index = AddMemoryMasking(index, &offset, pinned);
1520     DEBUG_CODE_COMMENT("Store to memory");
1521     LiftoffRegister addr = pinned.set(__ GetUnusedRegister(kGpReg, pinned));
1522     LOAD_INSTANCE_FIELD(addr, MemoryStart, kPointerLoadType);
1523     uint32_t protected_store_pc = 0;
1524     __ Store(addr.gp(), index.gp(), offset, value, type, pinned,
1525              &protected_store_pc, true);
1526     if (env_->use_trap_handler) {
1527       AddOutOfLineTrap(decoder->position(),
1528                        WasmCode::kThrowWasmTrapMemOutOfBounds,
1529                        protected_store_pc);
1530     }
1531     if (FLAG_wasm_trace_memory) {
1532       TraceMemoryOperation(true, type.mem_rep(), index.gp(), offset,
1533                            decoder->position());
1534     }
1535   }
1536 
CurrentMemoryPages(FullDecoder * decoder,Value * result)1537   void CurrentMemoryPages(FullDecoder* decoder, Value* result) {
1538     LiftoffRegList pinned;
1539     LiftoffRegister mem_size = pinned.set(__ GetUnusedRegister(kGpReg));
1540     LiftoffRegister tmp_const =
1541         pinned.set(__ GetUnusedRegister(kGpReg, pinned));
1542     LOAD_INSTANCE_FIELD(mem_size, MemorySize, LoadType::kI32Load);
1543     // TODO(clemensh): Shift by immediate directly.
1544     __ LoadConstant(tmp_const,
1545                     WasmValue(int32_t{WhichPowerOf2(kWasmPageSize)}));
1546     __ emit_i32_shr(mem_size.gp(), mem_size.gp(), tmp_const.gp(), pinned);
1547     __ PushRegister(kWasmI32, mem_size);
1548   }
1549 
GrowMemory(FullDecoder * decoder,const Value & value,Value * result_val)1550   void GrowMemory(FullDecoder* decoder, const Value& value, Value* result_val) {
1551     // Pop the input, then spill all cache registers to make the runtime call.
1552     LiftoffRegList pinned;
1553     LiftoffRegister input = pinned.set(__ PopToRegister());
1554     __ SpillAllRegisters();
1555 
1556     constexpr Register kGpReturnReg = kGpReturnRegisters[0];
1557     static_assert(kLiftoffAssemblerGpCacheRegs & Register::bit<kGpReturnReg>(),
1558                   "first return register is a cache register (needs more "
1559                   "complex code here otherwise)");
1560     LiftoffRegister result = pinned.set(LiftoffRegister(kGpReturnReg));
1561 
1562     WasmGrowMemoryDescriptor descriptor;
1563     DCHECK_EQ(0, descriptor.GetStackParameterCount());
1564     DCHECK_EQ(1, descriptor.GetRegisterParameterCount());
1565     DCHECK_EQ(ValueTypes::MachineTypeFor(kWasmI32),
1566               descriptor.GetParameterType(0));
1567 
1568     Register param_reg = descriptor.GetRegisterParameter(0);
1569     if (input.gp() != param_reg) __ Move(param_reg, input.gp(), kWasmI32);
1570 
1571     __ CallRuntimeStub(WasmCode::kWasmGrowMemory);
1572     safepoint_table_builder_.DefineSafepoint(&asm_, Safepoint::kSimple, 0,
1573                                              Safepoint::kNoLazyDeopt);
1574 
1575     if (kReturnRegister0 != result.gp()) {
1576       __ Move(result.gp(), kReturnRegister0, kWasmI32);
1577     }
1578 
1579     __ PushRegister(kWasmI32, result);
1580   }
1581 
CallDirect(FullDecoder * decoder,const CallFunctionImmediate<validate> & imm,const Value args[],Value returns[])1582   void CallDirect(FullDecoder* decoder,
1583                   const CallFunctionImmediate<validate>& imm,
1584                   const Value args[], Value returns[]) {
1585     if (imm.sig->return_count() > 1)
1586       return unsupported(decoder, "multi-return");
1587     if (imm.sig->return_count() == 1 &&
1588         !CheckSupportedType(decoder, kTypes_ilfd, imm.sig->GetReturn(0),
1589                             "return"))
1590       return;
1591 
1592     auto call_descriptor =
1593         compiler::GetWasmCallDescriptor(compilation_zone_, imm.sig);
1594     call_descriptor =
1595         GetLoweredCallDescriptor(compilation_zone_, call_descriptor);
1596 
1597     if (imm.index < env_->module->num_imported_functions) {
1598       // A direct call to an imported function.
1599       LiftoffRegList pinned;
1600       LiftoffRegister tmp = pinned.set(__ GetUnusedRegister(kGpReg, pinned));
1601       LiftoffRegister target = pinned.set(__ GetUnusedRegister(kGpReg, pinned));
1602 
1603       LiftoffRegister imported_targets = tmp;
1604       LOAD_INSTANCE_FIELD(imported_targets, ImportedFunctionTargets,
1605                           kPointerLoadType);
1606       __ Load(target, imported_targets.gp(), no_reg,
1607               imm.index * sizeof(Address), kPointerLoadType, pinned);
1608 
1609       LiftoffRegister imported_instances = tmp;
1610       LOAD_INSTANCE_FIELD(imported_instances, ImportedFunctionInstances,
1611                           kPointerLoadType);
1612       LiftoffRegister target_instance = tmp;
1613       __ Load(target_instance, imported_instances.gp(), no_reg,
1614               compiler::FixedArrayOffsetMinusTag(imm.index), kPointerLoadType,
1615               pinned);
1616 
1617       LiftoffRegister* explicit_instance = &target_instance;
1618       Register target_reg = target.gp();
1619       __ PrepareCall(imm.sig, call_descriptor, &target_reg, explicit_instance);
1620       source_position_table_builder_.AddPosition(
1621           __ pc_offset(), SourcePosition(decoder->position()), false);
1622 
1623       __ CallIndirect(imm.sig, call_descriptor, target_reg);
1624 
1625       safepoint_table_builder_.DefineSafepoint(&asm_, Safepoint::kSimple, 0,
1626                                                Safepoint::kNoLazyDeopt);
1627 
1628       __ FinishCall(imm.sig, call_descriptor);
1629     } else {
1630       // A direct call within this module just gets the current instance.
1631       __ PrepareCall(imm.sig, call_descriptor);
1632 
1633       source_position_table_builder_.AddPosition(
1634           __ pc_offset(), SourcePosition(decoder->position()), false);
1635 
1636       // Just encode the function index. This will be patched at instantiation.
1637       Address addr = static_cast<Address>(imm.index);
1638       __ CallNativeWasmCode(addr);
1639 
1640       safepoint_table_builder_.DefineSafepoint(&asm_, Safepoint::kSimple, 0,
1641                                                Safepoint::kNoLazyDeopt);
1642 
1643       __ FinishCall(imm.sig, call_descriptor);
1644     }
1645   }
1646 
CallIndirect(FullDecoder * decoder,const Value & index_val,const CallIndirectImmediate<validate> & imm,const Value args[],Value returns[])1647   void CallIndirect(FullDecoder* decoder, const Value& index_val,
1648                     const CallIndirectImmediate<validate>& imm,
1649                     const Value args[], Value returns[]) {
1650     if (imm.sig->return_count() > 1) {
1651       return unsupported(decoder, "multi-return");
1652     }
1653     if (imm.sig->return_count() == 1 &&
1654         !CheckSupportedType(decoder, kTypes_ilfd, imm.sig->GetReturn(0),
1655                             "return")) {
1656       return;
1657     }
1658 
1659     // Pop the index.
1660     LiftoffRegister index = __ PopToRegister();
1661     // If that register is still being used after popping, we move it to another
1662     // register, because we want to modify that register.
1663     if (__ cache_state()->is_used(index)) {
1664       LiftoffRegister new_index =
1665           __ GetUnusedRegister(kGpReg, LiftoffRegList::ForRegs(index));
1666       __ Move(new_index, index, kWasmI32);
1667       index = new_index;
1668     }
1669 
1670     LiftoffRegList pinned = LiftoffRegList::ForRegs(index);
1671     // Get three temporary registers.
1672     LiftoffRegister table = pinned.set(__ GetUnusedRegister(kGpReg, pinned));
1673     LiftoffRegister tmp_const =
1674         pinned.set(__ GetUnusedRegister(kGpReg, pinned));
1675     LiftoffRegister scratch = pinned.set(__ GetUnusedRegister(kGpReg, pinned));
1676 
1677     // Bounds check against the table size.
1678     Label* invalid_func_label = AddOutOfLineTrap(
1679         decoder->position(), WasmCode::kThrowWasmTrapFuncInvalid);
1680 
1681     uint32_t canonical_sig_num = env_->module->signature_ids[imm.sig_index];
1682     DCHECK_GE(canonical_sig_num, 0);
1683     DCHECK_GE(kMaxInt, canonical_sig_num);
1684 
1685     // Compare against table size stored in
1686     // {instance->indirect_function_table_size}.
1687     LOAD_INSTANCE_FIELD(tmp_const, IndirectFunctionTableSize,
1688                         LoadType::kI32Load);
1689     __ emit_cond_jump(kUnsignedGreaterEqual, invalid_func_label, kWasmI32,
1690                       index.gp(), tmp_const.gp());
1691 
1692     // Mask the index to prevent SSCA.
1693     if (FLAG_untrusted_code_mitigations) {
1694       DEBUG_CODE_COMMENT("Mask indirect call index");
1695       // mask = ((index - size) & ~index) >> 31
1696       // Reuse allocated registers; note: size is still stored in {tmp_const}.
1697       LiftoffRegister diff = table;
1698       LiftoffRegister neg_index = tmp_const;
1699       LiftoffRegister mask = scratch;
1700       // 1) diff = index - size
1701       __ emit_i32_sub(diff.gp(), index.gp(), tmp_const.gp());
1702       // 2) neg_index = ~index
1703       __ LoadConstant(neg_index, WasmValue(int32_t{-1}));
1704       __ emit_i32_xor(neg_index.gp(), neg_index.gp(), index.gp());
1705       // 3) mask = diff & neg_index
1706       __ emit_i32_and(mask.gp(), diff.gp(), neg_index.gp());
1707       // 4) mask = mask >> 31
1708       __ LoadConstant(tmp_const, WasmValue(int32_t{31}));
1709       __ emit_i32_sar(mask.gp(), mask.gp(), tmp_const.gp(), pinned);
1710 
1711       // Apply mask.
1712       __ emit_i32_and(index.gp(), index.gp(), mask.gp());
1713     }
1714 
1715     DEBUG_CODE_COMMENT("Check indirect call signature");
1716     // Load the signature from {instance->ift_sig_ids[key]}
1717     LOAD_INSTANCE_FIELD(table, IndirectFunctionTableSigIds, kPointerLoadType);
1718     __ LoadConstant(tmp_const,
1719                     WasmValue(static_cast<uint32_t>(sizeof(uint32_t))));
1720     // TODO(wasm): use a emit_i32_shli() instead of a multiply.
1721     // (currently cannot use shl on ia32/x64 because it clobbers %rcx).
1722     __ emit_i32_mul(index.gp(), index.gp(), tmp_const.gp());
1723     __ Load(scratch, table.gp(), index.gp(), 0, LoadType::kI32Load, pinned);
1724 
1725     // Compare against expected signature.
1726     __ LoadConstant(tmp_const, WasmValue(canonical_sig_num));
1727 
1728     Label* sig_mismatch_label = AddOutOfLineTrap(
1729         decoder->position(), WasmCode::kThrowWasmTrapFuncSigMismatch);
1730     __ emit_cond_jump(kUnequal, sig_mismatch_label,
1731                       LiftoffAssembler::kWasmIntPtr, scratch.gp(),
1732                       tmp_const.gp());
1733 
1734     DEBUG_CODE_COMMENT("Execute indirect call");
1735     if (kPointerSize == 8) {
1736       // {index} has already been multiplied by 4. Multiply by another 2.
1737       __ LoadConstant(tmp_const, WasmValue(2));
1738       __ emit_i32_mul(index.gp(), index.gp(), tmp_const.gp());
1739     }
1740 
1741     // Load the target from {instance->ift_targets[key]}
1742     LOAD_INSTANCE_FIELD(table, IndirectFunctionTableTargets, kPointerLoadType);
1743     __ Load(scratch, table.gp(), index.gp(), 0, kPointerLoadType, pinned);
1744 
1745     // Load the instance from {instance->ift_instances[key]}
1746     LOAD_INSTANCE_FIELD(table, IndirectFunctionTableInstances,
1747                         kPointerLoadType);
1748     __ Load(tmp_const, table.gp(), index.gp(),
1749             (FixedArray::kHeaderSize - kHeapObjectTag), kPointerLoadType,
1750             pinned);
1751     LiftoffRegister* explicit_instance = &tmp_const;
1752 
1753     source_position_table_builder_.AddPosition(
1754         __ pc_offset(), SourcePosition(decoder->position()), false);
1755 
1756     auto call_descriptor =
1757         compiler::GetWasmCallDescriptor(compilation_zone_, imm.sig);
1758     call_descriptor =
1759         GetLoweredCallDescriptor(compilation_zone_, call_descriptor);
1760 
1761     Register target = scratch.gp();
1762     __ PrepareCall(imm.sig, call_descriptor, &target, explicit_instance);
1763     __ CallIndirect(imm.sig, call_descriptor, target);
1764 
1765     safepoint_table_builder_.DefineSafepoint(&asm_, Safepoint::kSimple, 0,
1766                                              Safepoint::kNoLazyDeopt);
1767 
1768     __ FinishCall(imm.sig, call_descriptor);
1769   }
1770 
SimdOp(FullDecoder * decoder,WasmOpcode opcode,Vector<Value> args,Value * result)1771   void SimdOp(FullDecoder* decoder, WasmOpcode opcode, Vector<Value> args,
1772               Value* result) {
1773     unsupported(decoder, "simd");
1774   }
SimdLaneOp(FullDecoder * decoder,WasmOpcode opcode,const SimdLaneImmediate<validate> & imm,const Vector<Value> inputs,Value * result)1775   void SimdLaneOp(FullDecoder* decoder, WasmOpcode opcode,
1776                   const SimdLaneImmediate<validate>& imm,
1777                   const Vector<Value> inputs, Value* result) {
1778     unsupported(decoder, "simd");
1779   }
SimdShiftOp(FullDecoder * decoder,WasmOpcode opcode,const SimdShiftImmediate<validate> & imm,const Value & input,Value * result)1780   void SimdShiftOp(FullDecoder* decoder, WasmOpcode opcode,
1781                    const SimdShiftImmediate<validate>& imm, const Value& input,
1782                    Value* result) {
1783     unsupported(decoder, "simd");
1784   }
Simd8x16ShuffleOp(FullDecoder * decoder,const Simd8x16ShuffleImmediate<validate> & imm,const Value & input0,const Value & input1,Value * result)1785   void Simd8x16ShuffleOp(FullDecoder* decoder,
1786                          const Simd8x16ShuffleImmediate<validate>& imm,
1787                          const Value& input0, const Value& input1,
1788                          Value* result) {
1789     unsupported(decoder, "simd");
1790   }
Throw(FullDecoder * decoder,const ExceptionIndexImmediate<validate> &,Control * block,const Vector<Value> & args)1791   void Throw(FullDecoder* decoder, const ExceptionIndexImmediate<validate>&,
1792              Control* block, const Vector<Value>& args) {
1793     unsupported(decoder, "throw");
1794   }
CatchException(FullDecoder * decoder,const ExceptionIndexImmediate<validate> & imm,Control * block,Vector<Value> caught_values)1795   void CatchException(FullDecoder* decoder,
1796                       const ExceptionIndexImmediate<validate>& imm,
1797                       Control* block, Vector<Value> caught_values) {
1798     unsupported(decoder, "catch");
1799   }
AtomicOp(FullDecoder * decoder,WasmOpcode opcode,Vector<Value> args,const MemoryAccessImmediate<validate> & imm,Value * result)1800   void AtomicOp(FullDecoder* decoder, WasmOpcode opcode, Vector<Value> args,
1801                 const MemoryAccessImmediate<validate>& imm, Value* result) {
1802     unsupported(decoder, "atomicop");
1803   }
1804 
1805  private:
1806   LiftoffAssembler asm_;
1807   compiler::CallDescriptor* const descriptor_;
1808   ModuleEnv* const env_;
1809   bool ok_ = true;
1810   std::vector<OutOfLineCode> out_of_line_code_;
1811   SourcePositionTableBuilder source_position_table_builder_;
1812   std::vector<trap_handler::ProtectedInstructionData> protected_instructions_;
1813   // Zone used to store information during compilation. The result will be
1814   // stored independently, such that this zone can die together with the
1815   // LiftoffCompiler after compilation.
1816   Zone* compilation_zone_;
1817   SafepointTableBuilder safepoint_table_builder_;
1818   // The pc offset of the instructions to reserve the stack frame. Needed to
1819   // patch the actually needed stack size in the end.
1820   uint32_t pc_offset_stack_frame_construction_ = 0;
1821 
TraceCacheState(FullDecoder * decoder) const1822   void TraceCacheState(FullDecoder* decoder) const {
1823 #ifdef DEBUG
1824     if (!FLAG_trace_liftoff || !FLAG_trace_wasm_decoder) return;
1825     StdoutStream os;
1826     for (int control_depth = decoder->control_depth() - 1; control_depth >= -1;
1827          --control_depth) {
1828       auto* cache_state =
1829           control_depth == -1 ? __ cache_state()
1830                               : &decoder->control_at(control_depth)
1831                                      ->label_state;
1832       os << PrintCollection(cache_state->stack_state);
1833       if (control_depth != -1) PrintF("; ");
1834     }
1835     os << "\n";
1836 #endif
1837   }
1838 };
1839 
1840 }  // namespace
1841 
ExecuteCompilation(WasmFeatures * detected)1842 bool LiftoffCompilationUnit::ExecuteCompilation(WasmFeatures* detected) {
1843   TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("v8.wasm"),
1844                "ExecuteLiftoffCompilation");
1845   base::ElapsedTimer compile_timer;
1846   if (FLAG_trace_wasm_decode_time) {
1847     compile_timer.Start();
1848   }
1849 
1850   Zone zone(wasm_unit_->wasm_engine_->allocator(), "LiftoffCompilationZone");
1851   const WasmModule* module =
1852       wasm_unit_->env_ ? wasm_unit_->env_->module : nullptr;
1853   auto call_descriptor =
1854       compiler::GetWasmCallDescriptor(&zone, wasm_unit_->func_body_.sig);
1855   base::Optional<TimedHistogramScope> liftoff_compile_time_scope(
1856       base::in_place, wasm_unit_->counters_->liftoff_compile_time());
1857   WasmFullDecoder<Decoder::kValidate, LiftoffCompiler> decoder(
1858       &zone, module, wasm_unit_->native_module_->enabled_features(), detected,
1859       wasm_unit_->func_body_, call_descriptor, wasm_unit_->env_, &zone);
1860   decoder.Decode();
1861   liftoff_compile_time_scope.reset();
1862   LiftoffCompiler* compiler = &decoder.interface();
1863   if (decoder.failed()) return false;  // validation error
1864   if (!compiler->ok()) {
1865     // Liftoff compilation failed.
1866     wasm_unit_->counters_->liftoff_unsupported_functions()->Increment();
1867     return false;
1868   }
1869 
1870   wasm_unit_->counters_->liftoff_compiled_functions()->Increment();
1871 
1872   if (FLAG_trace_wasm_decode_time) {
1873     double compile_ms = compile_timer.Elapsed().InMillisecondsF();
1874     PrintF(
1875         "wasm-compilation liftoff phase 1 ok: %u bytes, %0.3f ms decode and "
1876         "compile\n",
1877         static_cast<unsigned>(wasm_unit_->func_body_.end -
1878                               wasm_unit_->func_body_.start),
1879         compile_ms);
1880   }
1881 
1882   CodeDesc desc;
1883   compiler->GetCode(&desc);
1884   OwnedVector<byte> source_positions = compiler->GetSourcePositionTable();
1885   OwnedVector<trap_handler::ProtectedInstructionData> protected_instructions =
1886       compiler->GetProtectedInstructions();
1887   uint32_t frame_slot_count = compiler->GetTotalFrameSlotCount();
1888   int safepoint_table_offset = compiler->GetSafepointTableOffset();
1889 
1890   code_ = wasm_unit_->native_module_->AddCode(
1891       wasm_unit_->func_index_, desc, frame_slot_count, safepoint_table_offset,
1892       0, std::move(protected_instructions), std::move(source_positions),
1893       WasmCode::kLiftoff);
1894   wasm_unit_->native_module_->PublishCode(code_);
1895 
1896   return true;
1897 }
1898 
FinishCompilation(ErrorThrower *)1899 WasmCode* LiftoffCompilationUnit::FinishCompilation(ErrorThrower*) {
1900   return code_;
1901 }
1902 
1903 #undef __
1904 #undef TRACE
1905 #undef WASM_INSTANCE_OBJECT_OFFSET
1906 #undef LOAD_INSTANCE_FIELD
1907 #undef DEBUG_CODE_COMMENT
1908 
1909 }  // namespace wasm
1910 }  // namespace internal
1911 }  // namespace v8
1912