Source code
Revision control
Copy as Markdown
Other Tools
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*-
* vim: set ts=8 sts=2 et sw=2 tw=80:
*
* Copyright 2021 Mozilla Foundation
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "wasm/WasmCodegenTypes.h"
#include "wasm/WasmExprType.h"
#include "wasm/WasmStubs.h"
#include "wasm/WasmTypeDef.h"
#include "wasm/WasmValidate.h"
#include "wasm/WasmValue.h"
using mozilla::MakeEnumeratedRange;
using mozilla::PodZero;
using namespace js;
using namespace js::wasm;
ArgTypeVector::ArgTypeVector(const FuncType& funcType)
: args_(funcType.args()),
hasStackResults_(ABIResultIter::HasStackResults(
ResultType::Vector(funcType.results()))) {}
bool TrapSiteVectorArray::empty() const {
for (Trap trap : MakeEnumeratedRange(Trap::Limit)) {
if (!(*this)[trap].empty()) {
return false;
}
}
return true;
}
#ifdef DEBUG
const char* js::wasm::NameOfTrap(Trap trap) {
switch (trap) {
case Trap::Unreachable:
return "Unreachable";
case Trap::IntegerOverflow:
return "IntegerOverflow";
case Trap::InvalidConversionToInteger:
return "InvalidConversionToInteger";
case Trap::IntegerDivideByZero:
return "IntegerDivideByZero";
case Trap::OutOfBounds:
return "OutOfBounds";
case Trap::UnalignedAccess:
return "UnalignedAccess";
case Trap::IndirectCallToNull:
return "IndirectCallToNull";
case Trap::IndirectCallBadSig:
return "IndirectCallBadSig";
case Trap::NullPointerDereference:
return "NullPointerDereference";
case Trap::BadCast:
return "BadCast";
case Trap::StackOverflow:
return "StackOverflow";
case Trap::CheckInterrupt:
return "CheckInterrupt";
case Trap::ThrowReported:
return "ThrowReported";
case Trap::Limit:
return "Limit";
default:
return "NameOfTrap:unknown";
}
}
const char* js::wasm::NameOfTrapMachineInsn(TrapMachineInsn tmi) {
switch (tmi) {
case TrapMachineInsn::OfficialUD:
return "OfficialUD";
case TrapMachineInsn::Load8:
return "Load8";
case TrapMachineInsn::Load16:
return "Load16";
case TrapMachineInsn::Load32:
return "Load32";
case TrapMachineInsn::Load64:
return "Load64";
case TrapMachineInsn::Load128:
return "Load128";
case TrapMachineInsn::Store8:
return "Store8";
case TrapMachineInsn::Store16:
return "Store16";
case TrapMachineInsn::Store32:
return "Store32";
case TrapMachineInsn::Store64:
return "Store64";
case TrapMachineInsn::Store128:
return "Store128";
case TrapMachineInsn::Atomic:
return "Atomic";
default:
return "NameOfTrapMachineInsn::unknown";
}
}
#endif // DEBUG
void TrapSiteVectorArray::clear() {
for (Trap trap : MakeEnumeratedRange(Trap::Limit)) {
(*this)[trap].clear();
}
}
void TrapSiteVectorArray::swap(TrapSiteVectorArray& rhs) {
for (Trap trap : MakeEnumeratedRange(Trap::Limit)) {
(*this)[trap].swap(rhs[trap]);
}
}
void TrapSiteVectorArray::shrinkStorageToFit() {
for (Trap trap : MakeEnumeratedRange(Trap::Limit)) {
(*this)[trap].shrinkStorageToFit();
}
}
size_t TrapSiteVectorArray::sumOfLengths() const {
size_t ret = 0;
for (Trap trap : MakeEnumeratedRange(Trap::Limit)) {
ret += (*this)[trap].length();
}
return ret;
}
size_t TrapSiteVectorArray::sizeOfExcludingThis(
mozilla::MallocSizeOf mallocSizeOf) const {
size_t ret = 0;
for (Trap trap : MakeEnumeratedRange(Trap::Limit)) {
ret += (*this)[trap].sizeOfExcludingThis(mallocSizeOf);
}
return ret;
}
CodeRange::CodeRange(Kind kind, Offsets offsets)
: begin_(offsets.begin), ret_(0), end_(offsets.end), kind_(kind) {
MOZ_ASSERT(begin_ <= end_);
PodZero(&u);
#ifdef DEBUG
switch (kind_) {
case FarJumpIsland:
case TrapExit:
case Throw:
break;
default:
MOZ_CRASH("should use more specific constructor");
}
#endif
}
CodeRange::CodeRange(Kind kind, uint32_t funcIndex, Offsets offsets)
: begin_(offsets.begin), ret_(0), end_(offsets.end), kind_(kind) {
u.funcIndex_ = funcIndex;
u.func.beginToUncheckedCallEntry_ = 0;
u.func.beginToTierEntry_ = 0;
u.func.hasUnwindInfo_ = false;
MOZ_ASSERT(isEntry());
MOZ_ASSERT(begin_ <= end_);
}
CodeRange::CodeRange(Kind kind, CallableOffsets offsets)
: begin_(offsets.begin), ret_(offsets.ret), end_(offsets.end), kind_(kind) {
MOZ_ASSERT(begin_ < ret_);
MOZ_ASSERT(ret_ < end_);
PodZero(&u);
#ifdef DEBUG
switch (kind_) {
case DebugStub:
case BuiltinThunk:
case RequestTierUpStub:
break;
default:
MOZ_CRASH("should use more specific constructor");
}
#endif
}
CodeRange::CodeRange(Kind kind, uint32_t funcIndex, CallableOffsets offsets)
: begin_(offsets.begin), ret_(offsets.ret), end_(offsets.end), kind_(kind) {
MOZ_ASSERT(isImportExit() || isJitEntry());
MOZ_ASSERT(begin_ < ret_);
MOZ_ASSERT(ret_ < end_);
u.funcIndex_ = funcIndex;
u.func.beginToUncheckedCallEntry_ = 0;
u.func.beginToTierEntry_ = 0;
u.func.hasUnwindInfo_ = false;
}
CodeRange::CodeRange(Kind kind, uint32_t funcIndex, ImportOffsets offsets)
: begin_(offsets.begin), ret_(offsets.ret), end_(offsets.end), kind_(kind) {
MOZ_ASSERT(isImportJitExit());
MOZ_ASSERT(begin_ < ret_);
MOZ_ASSERT(ret_ < end_);
uint32_t entry = offsets.afterFallbackCheck;
MOZ_ASSERT(begin_ <= entry && entry <= ret_);
u.funcIndex_ = funcIndex;
u.jitExitEntry_ = entry - begin_;
}
CodeRange::CodeRange(uint32_t funcIndex, FuncOffsets offsets,
bool hasUnwindInfo)
: begin_(offsets.begin),
ret_(offsets.ret),
end_(offsets.end),
kind_(Function) {
MOZ_ASSERT(begin_ < ret_);
MOZ_ASSERT(ret_ < end_);
MOZ_ASSERT(offsets.uncheckedCallEntry - begin_ <= UINT16_MAX);
MOZ_ASSERT(offsets.tierEntry - begin_ <= UINT16_MAX);
u.funcIndex_ = funcIndex;
u.func.beginToUncheckedCallEntry_ = offsets.uncheckedCallEntry - begin_;
u.func.beginToTierEntry_ = offsets.tierEntry - begin_;
u.func.hasUnwindInfo_ = hasUnwindInfo;
}
const CodeRange* wasm::LookupInSorted(const CodeRangeVector& codeRanges,
CodeRange::OffsetInCode target) {
size_t lowerBound = 0;
size_t upperBound = codeRanges.length();
size_t match;
if (!BinarySearch(codeRanges, lowerBound, upperBound, target, &match)) {
return nullptr;
}
return &codeRanges[match];
}
CallIndirectId CallIndirectId::forAsmJSFunc() {
return CallIndirectId(CallIndirectIdKind::AsmJS);
}
CallIndirectId CallIndirectId::forFunc(const CodeMetadata& codeMeta,
uint32_t funcIndex) {
// asm.js tables are homogenous and don't require a signature check
if (codeMeta.isAsmJS()) {
return CallIndirectId::forAsmJSFunc();
}
FuncDesc func = codeMeta.funcs[funcIndex];
if (!func.canRefFunc()) {
return CallIndirectId();
}
return CallIndirectId::forFuncType(codeMeta,
codeMeta.funcs[funcIndex].typeIndex);
}
CallIndirectId CallIndirectId::forFuncType(const CodeMetadata& codeMeta,
uint32_t funcTypeIndex) {
// asm.js tables are homogenous and don't require a signature check
if (codeMeta.isAsmJS()) {
return CallIndirectId::forAsmJSFunc();
}
const TypeDef& typeDef = codeMeta.types->type(funcTypeIndex);
const FuncType& funcType = typeDef.funcType();
CallIndirectId callIndirectId;
if (funcType.hasImmediateTypeId()) {
callIndirectId.kind_ = CallIndirectIdKind::Immediate;
callIndirectId.immediate_ = funcType.immediateTypeId();
} else {
callIndirectId.kind_ = CallIndirectIdKind::Global;
callIndirectId.global_.instanceDataOffset_ =
codeMeta.offsetOfTypeDef(funcTypeIndex);
callIndirectId.global_.hasSuperType_ = typeDef.superTypeDef() != nullptr;
}
return callIndirectId;
}
CalleeDesc CalleeDesc::function(uint32_t funcIndex) {
CalleeDesc c;
c.which_ = Func;
c.u.funcIndex_ = funcIndex;
return c;
}
CalleeDesc CalleeDesc::import(uint32_t instanceDataOffset) {
CalleeDesc c;
c.which_ = Import;
c.u.import.instanceDataOffset_ = instanceDataOffset;
return c;
}
CalleeDesc CalleeDesc::wasmTable(const CodeMetadata& codeMeta,
const TableDesc& desc, uint32_t tableIndex,
CallIndirectId callIndirectId) {
CalleeDesc c;
c.which_ = WasmTable;
c.u.table.instanceDataOffset_ =
codeMeta.offsetOfTableInstanceData(tableIndex);
c.u.table.minLength_ = desc.initialLength();
c.u.table.maxLength_ = desc.maximumLength();
c.u.table.callIndirectId_ = callIndirectId;
return c;
}
CalleeDesc CalleeDesc::asmJSTable(const CodeMetadata& codeMeta,
uint32_t tableIndex) {
CalleeDesc c;
c.which_ = AsmJSTable;
c.u.table.instanceDataOffset_ =
codeMeta.offsetOfTableInstanceData(tableIndex);
return c;
}
CalleeDesc CalleeDesc::builtin(SymbolicAddress callee) {
CalleeDesc c;
c.which_ = Builtin;
c.u.builtin_ = callee;
return c;
}
CalleeDesc CalleeDesc::builtinInstanceMethod(SymbolicAddress callee) {
CalleeDesc c;
c.which_ = BuiltinInstanceMethod;
c.u.builtin_ = callee;
return c;
}
CalleeDesc CalleeDesc::wasmFuncRef() {
CalleeDesc c;
c.which_ = FuncRef;
return c;
}