/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2; -*- */ /* 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 mozilla_devtools_DeserializedNode__ #define mozilla_devtools_DeserializedNode__ #include "js/UbiNode.h" #include "js/UniquePtr.h" #include "mozilla/devtools/CoreDump.pb.h" #include "mozilla/Maybe.h" #include "mozilla/Move.h" #include "mozilla/Vector.h" // `Deserialized{Node,Edge}` translate protobuf messages from our core dump // format into structures we can rely upon for implementing `JS::ubi::Node` // specializations on top of. All of the properties of the protobuf messages are // optional for future compatibility, and this is the layer where we validate // that the properties that do actually exist in any given message fulfill our // semantic requirements. // // Both `DeserializedNode` and `DeserializedEdge` are always owned by a // `HeapSnapshot` instance, and their lifetimes must not extend after that of // their owning `HeapSnapshot`. namespace mozilla { namespace devtools { class HeapSnapshot; using NodeId = uint64_t; using StackFrameId = uint64_t; // A `DeserializedEdge` represents an edge in the heap graph pointing to the // node with id equal to `DeserializedEdge::referent` that we deserialized from // a core dump. struct DeserializedEdge { NodeId referent; // A borrowed reference to a string owned by this node's owning HeapSnapshot. const char16_t* name; explicit DeserializedEdge(NodeId referent, const char16_t* edgeName = nullptr) : referent(referent) , name(edgeName) { } DeserializedEdge(DeserializedEdge&& rhs); DeserializedEdge& operator=(DeserializedEdge&& rhs); private: DeserializedEdge(const DeserializedEdge&) = delete; DeserializedEdge& operator=(const DeserializedEdge&) = delete; }; // A `DeserializedNode` is a node in the heap graph that we deserialized from a // core dump. struct DeserializedNode { using EdgeVector = Vector<DeserializedEdge>; using UniqueStringPtr = UniquePtr<char16_t[]>; NodeId id; JS::ubi::CoarseType coarseType; // A borrowed reference to a string owned by this node's owning HeapSnapshot. const char16_t* typeName; uint64_t size; EdgeVector edges; Maybe<StackFrameId> allocationStack; // A borrowed reference to a string owned by this node's owning HeapSnapshot. const char* jsObjectClassName; // A borrowed reference to a string owned by this node's owning HeapSnapshot. const char* scriptFilename; // A weak pointer to this node's owning `HeapSnapshot`. Safe without // AddRef'ing because this node's lifetime is equal to that of its owner. HeapSnapshot* owner; DeserializedNode(NodeId id, JS::ubi::CoarseType coarseType, const char16_t* typeName, uint64_t size, EdgeVector&& edges, Maybe<StackFrameId> allocationStack, const char* className, const char* filename, HeapSnapshot& owner) : id(id) , coarseType(coarseType) , typeName(typeName) , size(size) , edges(Move(edges)) , allocationStack(allocationStack) , jsObjectClassName(className) , scriptFilename(filename) , owner(&owner) { } virtual ~DeserializedNode() { } DeserializedNode(DeserializedNode&& rhs) : id(rhs.id) , coarseType(rhs.coarseType) , typeName(rhs.typeName) , size(rhs.size) , edges(Move(rhs.edges)) , allocationStack(rhs.allocationStack) , jsObjectClassName(rhs.jsObjectClassName) , scriptFilename(rhs.scriptFilename) , owner(rhs.owner) { } DeserializedNode& operator=(DeserializedNode&& rhs) { MOZ_ASSERT(&rhs != this); this->~DeserializedNode(); new(this) DeserializedNode(Move(rhs)); return *this; } // Get a borrowed reference to the given edge's referent. This method is // virtual to provide a hook for gmock and gtest. virtual JS::ubi::Node getEdgeReferent(const DeserializedEdge& edge); struct HashPolicy; protected: // This is only for use with `MockDeserializedNode` in testing. DeserializedNode(NodeId id, const char16_t* typeName, uint64_t size) : id(id) , coarseType(JS::ubi::CoarseType::Other) , typeName(typeName) , size(size) , edges() , allocationStack(Nothing()) , jsObjectClassName(nullptr) , scriptFilename(nullptr) , owner(nullptr) { } private: DeserializedNode(const DeserializedNode&) = delete; DeserializedNode& operator=(const DeserializedNode&) = delete; }; static inline js::HashNumber hashIdDerivedFromPtr(uint64_t id) { // NodeIds and StackFrameIds are always 64 bits, but they are derived from // the original referents' addresses, which could have been either 32 or 64 // bits long. As such, NodeId and StackFrameId have little entropy in their // bottom three bits, and may or may not have entropy in their upper 32 // bits. This hash should manage both cases well. id >>= 3; return js::HashNumber((id >> 32) ^ id); } struct DeserializedNode::HashPolicy { using Lookup = NodeId; static js::HashNumber hash(const Lookup& lookup) { return hashIdDerivedFromPtr(lookup); } static bool match(const DeserializedNode& existing, const Lookup& lookup) { return existing.id == lookup; } }; // A `DeserializedStackFrame` is a stack frame referred to by a thing in the // heap graph that we deserialized from a core dump. struct DeserializedStackFrame { StackFrameId id; Maybe<StackFrameId> parent; uint32_t line; uint32_t column; // Borrowed references to strings owned by this DeserializedStackFrame's // owning HeapSnapshot. const char16_t* source; const char16_t* functionDisplayName; bool isSystem; bool isSelfHosted; // A weak pointer to this frame's owning `HeapSnapshot`. Safe without // AddRef'ing because this frame's lifetime is equal to that of its owner. HeapSnapshot* owner; explicit DeserializedStackFrame(StackFrameId id, const Maybe<StackFrameId>& parent, uint32_t line, uint32_t column, const char16_t* source, const char16_t* functionDisplayName, bool isSystem, bool isSelfHosted, HeapSnapshot& owner) : id(id) , parent(parent) , line(line) , column(column) , source(source) , functionDisplayName(functionDisplayName) , isSystem(isSystem) , isSelfHosted(isSelfHosted) , owner(&owner) { MOZ_ASSERT(source); } JS::ubi::StackFrame getParentStackFrame() const; struct HashPolicy; protected: // This is exposed only for MockDeserializedStackFrame in the gtests. explicit DeserializedStackFrame() : id(0) , parent(Nothing()) , line(0) , column(0) , source(nullptr) , functionDisplayName(nullptr) , isSystem(false) , isSelfHosted(false) , owner(nullptr) { }; }; struct DeserializedStackFrame::HashPolicy { using Lookup = StackFrameId; static js::HashNumber hash(const Lookup& lookup) { return hashIdDerivedFromPtr(lookup); } static bool match(const DeserializedStackFrame& existing, const Lookup& lookup) { return existing.id == lookup; } }; } // namespace devtools } // namespace mozilla namespace JS { namespace ubi { using mozilla::devtools::DeserializedNode; using mozilla::devtools::DeserializedStackFrame; template<> class Concrete<DeserializedNode> : public Base { protected: explicit Concrete(DeserializedNode* ptr) : Base(ptr) { } DeserializedNode& get() const { return *static_cast<DeserializedNode*>(ptr); } public: static void construct(void* storage, DeserializedNode* ptr) { new (storage) Concrete(ptr); } CoarseType coarseType() const final { return get().coarseType; } Id identifier() const override { return get().id; } bool isLive() const override { return false; } const char16_t* typeName() const override; Node::Size size(mozilla::MallocSizeOf mallocSizeof) const override; const char* jsObjectClassName() const override { return get().jsObjectClassName; } const char* scriptFilename() const final { return get().scriptFilename; } bool hasAllocationStack() const override { return get().allocationStack.isSome(); } StackFrame allocationStack() const override; // We ignore the `bool wantNames` parameter because we can't control whether // the core dump was serialized with edge names or not. js::UniquePtr<EdgeRange> edges(JSContext* cx, bool) const override; static const char16_t concreteTypeName[]; }; template<> class ConcreteStackFrame<DeserializedStackFrame> : public BaseStackFrame { protected: explicit ConcreteStackFrame(DeserializedStackFrame* ptr) : BaseStackFrame(ptr) { } DeserializedStackFrame& get() const { return *static_cast<DeserializedStackFrame*>(ptr); } public: static void construct(void* storage, DeserializedStackFrame* ptr) { new (storage) ConcreteStackFrame(ptr); } uint64_t identifier() const override { return get().id; } uint32_t line() const override { return get().line; } uint32_t column() const override { return get().column; } bool isSystem() const override { return get().isSystem; } bool isSelfHosted(JSContext* cx) const override { return get().isSelfHosted; } void trace(JSTracer* trc) override { } AtomOrTwoByteChars source() const override { return AtomOrTwoByteChars(get().source); } AtomOrTwoByteChars functionDisplayName() const override { return AtomOrTwoByteChars(get().functionDisplayName); } StackFrame parent() const override; bool constructSavedFrameStack(JSContext* cx, MutableHandleObject outSavedFrameStack) const override; }; } // namespace ubi } // namespace JS #endif // mozilla_devtools_DeserializedNode__