/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- * vim: set ts=8 sts=2 et sw=2 tw=80: * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #ifndef frontend_StencilXdr_h #define frontend_StencilXdr_h #include "mozilla/RefPtr.h" // RefPtr #include "frontend/ParserAtom.h" // ParserAtom, ParserAtomSpan #include "frontend/Stencil.h" // BitIntStencil, ScopeStencil, BaseParserScopeData #include "vm/Xdr.h" // XDRMode, XDRResult, XDRState namespace JS { class DecodeOptions; } // namespace JS namespace js { class LifoAlloc; class ObjLiteralStencil; class ScriptSource; class SharedImmutableScriptData; class XDRStencilDecoder; class XDRStencilEncoder; namespace frontend { struct CompilationStencil; struct ExtensibleCompilationStencil; struct SharedDataContainer; // Check that we can copy data to disk and restore it in another instance of // the program in a different address space. template struct CanCopyDataToDisk { // Check that the object is fully packed, to save disk space. #ifdef __cpp_lib_has_unique_object_representations static constexpr bool unique_repr = std::has_unique_object_representations(); #else static constexpr bool unique_repr = true; #endif // Approximation which assumes that 32bits variant of the class would not // have pointers if the 64bits variant does not have pointer. static constexpr bool no_pointer = alignof(DataT) < alignof(void*) || sizeof(void*) == sizeof(uint32_t); static constexpr bool value = unique_repr && no_pointer; }; // This is just a namespace class that can be used in friend declarations, // so that the statically declared XDR methods within have access to the // relevant struct internals. class StencilXDR { private: template [[nodiscard]] static XDRResult codeSourceUnretrievableUncompressed( XDRState* xdr, ScriptSource* ss, uint8_t sourceCharSize, uint32_t uncompressedLength); template class Data, XDRMode mode> static void codeSourceRetrievable(ScriptSource* ss); template [[nodiscard]] static XDRResult codeSourceUncompressedData( XDRState* const xdr, ScriptSource* const ss); template [[nodiscard]] static XDRResult codeSourceCompressedData( XDRState* const xdr, ScriptSource* const ss); template static void codeSourceRetrievableData(ScriptSource* ss); template [[nodiscard]] static XDRResult codeSourceData(XDRState* const xdr, ScriptSource* const ss); public: template static XDRResult codeSource(XDRState* xdr, const JS::DecodeOptions* maybeOptions, RefPtr& source); template static XDRResult codeBigInt(XDRState* xdr, LifoAlloc& alloc, BigIntStencil& stencil); template static XDRResult codeObjLiteral(XDRState* xdr, LifoAlloc& alloc, ObjLiteralStencil& stencil); template static XDRResult codeScopeData(XDRState* xdr, LifoAlloc& alloc, ScopeStencil& stencil, BaseParserScopeData*& baseScopeData); template static XDRResult codeSharedData(XDRState* xdr, RefPtr& sisd); template static XDRResult codeSharedDataContainer(XDRState* xdr, SharedDataContainer& sharedData); template static XDRResult codeParserAtom(XDRState* xdr, LifoAlloc& alloc, ParserAtom** atomp); template static XDRResult codeParserAtomSpan(XDRState* xdr, LifoAlloc& alloc, ParserAtomSpan& parserAtomData); template static XDRResult codeModuleRequest(XDRState* xdr, StencilModuleRequest& stencil); template static XDRResult codeModuleRequestVector( XDRState* xdr, StencilModuleMetadata::RequestVector& vector); template static XDRResult codeModuleEntry(XDRState* xdr, StencilModuleEntry& stencil); template static XDRResult codeModuleEntryVector( XDRState* xdr, StencilModuleMetadata::EntryVector& vector); template static XDRResult codeModuleMetadata(XDRState* xdr, StencilModuleMetadata& stencil); static XDRResult checkCompilationStencil(XDRStencilEncoder* encoder, const CompilationStencil& stencil); static XDRResult checkCompilationStencil( const ExtensibleCompilationStencil& stencil); template static XDRResult codeCompilationStencil(XDRState* xdr, CompilationStencil& stencil); }; } /* namespace frontend */ /* * The structure of the Stencil XDR buffer is: * * 1. Header * a. Version * b. ScriptSource * d. Alignment padding * 2. Stencil * a. CompilationStencil */ /* * The stencil decoder accepts `range` as input. * * The decoded stencils are outputted to the default-initialized * `stencil` parameter of `codeStencil` method. * * The decoded stencils borrow the input `buffer`/`range`, and the consumer * has to keep the buffer alive while the decoded stencils are alive. */ class XDRStencilDecoder : public XDRState { using Base = XDRState; public: XDRStencilDecoder(JSContext* cx, FrontendContext* fc, const JS::TranscodeRange& range) : Base(cx, fc, range) { MOZ_ASSERT(JS::IsTranscodingBytecodeAligned(range.begin().get())); } XDRResult codeStencil(const JS::DecodeOptions& options, frontend::CompilationStencil& stencil); const JS::DecodeOptions& options() { MOZ_ASSERT(options_); return *options_; } private: const JS::DecodeOptions* options_ = nullptr; }; class XDRStencilEncoder : public XDRState { using Base = XDRState; public: XDRStencilEncoder(JSContext* cx, FrontendContext* fc, JS::TranscodeBuffer& buffer) : Base(cx, fc, buffer, buffer.length()) { // NOTE: If buffer is empty, buffer.begin() doesn't point valid buffer. MOZ_ASSERT_IF(!buffer.empty(), JS::IsTranscodingBytecodeAligned(buffer.begin())); MOZ_ASSERT(JS::IsTranscodingBytecodeOffsetAligned(buffer.length())); } XDRResult codeStencil(const RefPtr& source, const frontend::CompilationStencil& stencil); XDRResult codeStencil(const frontend::CompilationStencil& stencil); }; } /* namespace js */ #endif /* frontend_StencilXdr_h */