2022-04-02 01:21:55 +00:00
|
|
|
|
// Copyright 2014 The Crashpad Authors. All rights reserved.
|
|
|
|
|
//
|
|
|
|
|
// 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
|
|
|
|
|
//
|
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
//
|
|
|
|
|
// 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.
|
|
|
|
|
|
|
|
|
|
#ifndef CRASHPAD_MINIDUMP_MINIDUMP_WRITABLE_H_
|
|
|
|
|
#define CRASHPAD_MINIDUMP_MINIDUMP_WRITABLE_H_
|
|
|
|
|
|
|
|
|
|
#include <windows.h>
|
|
|
|
|
#include <dbghelp.h>
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
|
|
|
|
|
#include <limits>
|
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
|
|
#include "util/file/file_io.h"
|
|
|
|
|
|
|
|
|
|
namespace crashpad {
|
|
|
|
|
|
|
|
|
|
class FileWriterInterface;
|
|
|
|
|
|
|
|
|
|
namespace internal {
|
|
|
|
|
|
|
|
|
|
//! \brief The base class for all content that might be written to a minidump
|
|
|
|
|
//! file.
|
|
|
|
|
class MinidumpWritable {
|
|
|
|
|
public:
|
|
|
|
|
MinidumpWritable(const MinidumpWritable&) = delete;
|
|
|
|
|
MinidumpWritable& operator=(const MinidumpWritable&) = delete;
|
|
|
|
|
|
|
|
|
|
virtual ~MinidumpWritable();
|
|
|
|
|
|
|
|
|
|
//! \brief Writes an object and all of its children to a minidump file.
|
|
|
|
|
//!
|
|
|
|
|
//! Use this on the root object of a tree of MinidumpWritable objects,
|
|
|
|
|
//! typically on a MinidumpFileWriter object.
|
|
|
|
|
//!
|
|
|
|
|
//! \param[in] file_writer The file writer to receive the minidump file’s
|
|
|
|
|
//! content.
|
|
|
|
|
//!
|
|
|
|
|
//! \return `true` on success. `false` on failure, with an appropriate message
|
|
|
|
|
//! logged.
|
|
|
|
|
//!
|
|
|
|
|
//! \note Valid in #kStateMutable, and transitions the object and the entire
|
|
|
|
|
//! tree beneath it through all states to #kStateWritten.
|
|
|
|
|
//!
|
|
|
|
|
//! \note This method should rarely be overridden.
|
|
|
|
|
virtual bool WriteEverything(FileWriterInterface* file_writer);
|
|
|
|
|
|
|
|
|
|
//! \brief Registers a file offset pointer as one that should point to the
|
|
|
|
|
//! object on which this method is called.
|
|
|
|
|
//!
|
|
|
|
|
//! Once the file offset at which an object will be written is known (when it
|
|
|
|
|
//! enters #kStateWritable), registered RVA pointers will be updated.
|
|
|
|
|
//!
|
|
|
|
|
//! \param[in] rva A pointer to storage for the file offset that should
|
|
|
|
|
//! contain this object’s writable file offset, once it is known.
|
|
|
|
|
//!
|
|
|
|
|
//! \note Valid in #kStateFrozen or any preceding state.
|
|
|
|
|
//
|
|
|
|
|
// This is public instead of protected because objects of derived classes need
|
|
|
|
|
// to be able to register their own pointers with distinct objects.
|
|
|
|
|
void RegisterRVA(RVA* rva);
|
2022-08-16 00:48:53 +00:00
|
|
|
|
void RegisterRVA(RVA64* rva);
|
2022-04-02 01:21:55 +00:00
|
|
|
|
|
|
|
|
|
//! \brief Registers a location descriptor as one that should point to the
|
|
|
|
|
//! object on which this method is called.
|
|
|
|
|
//!
|
|
|
|
|
//! Once an object’s size and the file offset at it will be written is known
|
|
|
|
|
//! (when it enters #kStateFrozen), the relevant data in registered location
|
|
|
|
|
//! descriptors will be updated.
|
|
|
|
|
//!
|
|
|
|
|
//! \param[in] location_descriptor A pointer to a location descriptor that
|
|
|
|
|
//! should contain this object’s writable size and file offset, once they
|
|
|
|
|
//! are known.
|
|
|
|
|
//!
|
|
|
|
|
//! \note Valid in #kStateFrozen or any preceding state.
|
|
|
|
|
//
|
|
|
|
|
// This is public instead of protected because objects of derived classes need
|
|
|
|
|
// to be able to register their own pointers with distinct objects.
|
|
|
|
|
void RegisterLocationDescriptor(
|
|
|
|
|
MINIDUMP_LOCATION_DESCRIPTOR* location_descriptor);
|
2022-08-16 00:48:53 +00:00
|
|
|
|
void RegisterLocationDescriptor(
|
|
|
|
|
MINIDUMP_LOCATION_DESCRIPTOR64* location_descriptor64);
|
2022-04-02 01:21:55 +00:00
|
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
//! \brief Identifies the state of an object.
|
|
|
|
|
//!
|
|
|
|
|
//! Objects will normally transition through each of these states as they are
|
|
|
|
|
//! created, populated with data, and then written to a minidump file.
|
|
|
|
|
enum State {
|
|
|
|
|
//! \brief The object’s properties can be modified.
|
|
|
|
|
kStateMutable = 0,
|
|
|
|
|
|
|
|
|
|
//! \brief The object is “frozen”.
|
|
|
|
|
//!
|
|
|
|
|
//! Its properties cannot be modified. Pointers to file offsets of other
|
|
|
|
|
//! structures may not yet be valid.
|
|
|
|
|
kStateFrozen,
|
|
|
|
|
|
|
|
|
|
//! \brief The object is writable.
|
|
|
|
|
//!
|
|
|
|
|
//! The file offset at which it will be written is known. Pointers to file
|
|
|
|
|
//! offsets of other structures are valid when all objects in a tree are in
|
|
|
|
|
//! this state.
|
|
|
|
|
kStateWritable,
|
|
|
|
|
|
|
|
|
|
//! \brief The object has been written to a minidump file.
|
|
|
|
|
kStateWritten,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
//! \brief Identifies the phase during which an object will be written to a
|
|
|
|
|
//! minidump file.
|
|
|
|
|
enum Phase {
|
|
|
|
|
//! \brief Objects that are written to a minidump file “early”.
|
|
|
|
|
//!
|
|
|
|
|
//! The normal sequence is for an object to write itself and then write all
|
|
|
|
|
//! of its children.
|
|
|
|
|
kPhaseEarly = 0,
|
|
|
|
|
|
|
|
|
|
//! \brief Objects that are written to a minidump file “late”.
|
|
|
|
|
//!
|
|
|
|
|
//! Some objects, such as those capturing memory region snapshots, are
|
|
|
|
|
//! written to minidump files after all other objects. This “late” phase
|
|
|
|
|
//! identifies such objects. This is useful to improve spatial locality in
|
|
|
|
|
//! minidump files in accordance with expected access patterns: unlike most
|
|
|
|
|
//! other data, memory snapshots are large and do not usually need to be
|
|
|
|
|
//! consulted in their entirety in order to process a minidump file.
|
|
|
|
|
kPhaseLate,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
//! \brief A size value used to signal failure by methods that return
|
|
|
|
|
//! `size_t`.
|
|
|
|
|
static constexpr size_t kInvalidSize = std::numeric_limits<size_t>::max();
|
|
|
|
|
|
|
|
|
|
MinidumpWritable();
|
|
|
|
|
|
|
|
|
|
//! \brief The state of the object.
|
|
|
|
|
State state() const { return state_; }
|
|
|
|
|
|
|
|
|
|
//! \brief Transitions the object from #kStateMutable to #kStateFrozen.
|
|
|
|
|
//!
|
|
|
|
|
//! The default implementation marks the object as frozen and recursively
|
|
|
|
|
//! calls Freeze() on all of its children. Subclasses may override this method
|
|
|
|
|
//! to perform processing that should only be done once callers have finished
|
|
|
|
|
//! populating an object with data. Typically, a subclass implementation would
|
|
|
|
|
//! call RegisterRVA() or RegisterLocationDescriptor() on other objects as
|
|
|
|
|
//! appropriate, because at the time Freeze() runs, the in-memory locations of
|
|
|
|
|
//! RVAs and location descriptors are known and will not change for the
|
|
|
|
|
//! remaining duration of an object’s lifetime.
|
|
|
|
|
//!
|
|
|
|
|
//! \return `true` on success. `false` on failure, with an appropriate message
|
|
|
|
|
//! logged.
|
|
|
|
|
virtual bool Freeze();
|
|
|
|
|
|
|
|
|
|
//! \brief Returns the amount of space that this object will consume when
|
|
|
|
|
//! written to a minidump file, in bytes, not including any leading or
|
|
|
|
|
//! trailing padding necessary to maintain proper alignment.
|
|
|
|
|
//!
|
|
|
|
|
//! \note Valid in #kStateFrozen or any subsequent state.
|
|
|
|
|
virtual size_t SizeOfObject() = 0;
|
|
|
|
|
|
|
|
|
|
//! \brief Returns the object’s desired byte-boundary alignment.
|
|
|
|
|
//!
|
|
|
|
|
//! The default implementation returns `4`. Subclasses may override this as
|
|
|
|
|
//! needed.
|
|
|
|
|
//!
|
|
|
|
|
//! \note Valid in #kStateFrozen or any subsequent state.
|
|
|
|
|
virtual size_t Alignment();
|
|
|
|
|
|
|
|
|
|
//! \brief Returns the object’s children.
|
|
|
|
|
//!
|
|
|
|
|
//! \note Valid in #kStateFrozen or any subsequent state.
|
|
|
|
|
virtual std::vector<MinidumpWritable*> Children();
|
|
|
|
|
|
|
|
|
|
//! \brief Returns the object’s desired write phase.
|
|
|
|
|
//!
|
|
|
|
|
//! The default implementation returns #kPhaseEarly. Subclasses may override
|
|
|
|
|
//! this method to alter their write phase.
|
|
|
|
|
//!
|
|
|
|
|
//! \note Valid in any state.
|
|
|
|
|
virtual Phase WritePhase();
|
|
|
|
|
|
|
|
|
|
//! \brief Prepares the object to be written at a known file offset,
|
|
|
|
|
//! transitioning it from #kStateFrozen to #kStateWritable.
|
|
|
|
|
//!
|
|
|
|
|
//! This method is responsible for determining the final file offset of the
|
|
|
|
|
//! object, which may be increased from \a offset to meet alignment
|
|
|
|
|
//! requirements. It calls WillWriteAtOffsetImpl() for the benefit of
|
|
|
|
|
//! subclasses. It populates all RVAs and location descriptors registered with
|
|
|
|
|
//! it via RegisterRVA() and RegisterLocationDescriptor(). It also recurses
|
|
|
|
|
//! into all known children.
|
|
|
|
|
//!
|
|
|
|
|
//! \param[in] phase The phase during which the object will be written. If
|
|
|
|
|
//! this does not match Phase(), processing is suppressed, although
|
|
|
|
|
//! recursive processing will still occur on all children. This addresses
|
|
|
|
|
//! the case where parents and children do not write in the same phase.
|
|
|
|
|
//! \param[in] offset The file offset at which the object will be written. The
|
|
|
|
|
//! offset may need to be adjusted for alignment.
|
|
|
|
|
//! \param[out] write_sequence This object will append itself to this list,
|
|
|
|
|
//! such that on return from a recursive tree of WillWriteAtOffset()
|
|
|
|
|
//! calls, elements of the vector will be organized in the sequence that
|
|
|
|
|
//! the objects will be written to the minidump file.
|
|
|
|
|
//!
|
|
|
|
|
//! \return The file size consumed by this object and all children, including
|
|
|
|
|
//! any padding inserted to meet alignment requirements. On failure,
|
|
|
|
|
//! #kInvalidSize, with an appropriate message logged.
|
|
|
|
|
//!
|
|
|
|
|
//! \note This method cannot be overridden. Subclasses that need to perform
|
|
|
|
|
//! processing when an object transitions to #kStateWritable should
|
|
|
|
|
//! implement WillWriteAtOffsetImpl(), which is called by this method.
|
|
|
|
|
size_t WillWriteAtOffset(Phase phase,
|
|
|
|
|
FileOffset* offset,
|
|
|
|
|
std::vector<MinidumpWritable*>* write_sequence);
|
|
|
|
|
|
|
|
|
|
//! \brief Called once an object’s writable file offset is determined, as it
|
|
|
|
|
//! transitions into #kStateWritable.
|
|
|
|
|
//!
|
|
|
|
|
//! Subclasses can override this method if they need to provide additional
|
|
|
|
|
//! processing once their writable file offset is known. Typically, this will
|
|
|
|
|
//! be done by subclasses that handle certain RVAs themselves instead of using
|
|
|
|
|
//! the RegisterRVA() interface.
|
|
|
|
|
//!
|
|
|
|
|
//! \param[in] offset The file offset at which the object will be written. The
|
|
|
|
|
//! value passed to this method will already have been adjusted to meet
|
|
|
|
|
//! alignment requirements.
|
|
|
|
|
//!
|
|
|
|
|
//! \return `true` on success. `false` on error, indicating that the minidump
|
|
|
|
|
//! file should not be written.
|
|
|
|
|
//!
|
|
|
|
|
//! \note Valid in #kStateFrozen. The object will transition to
|
|
|
|
|
//! #kStateWritable after this method returns.
|
|
|
|
|
virtual bool WillWriteAtOffsetImpl(FileOffset offset);
|
|
|
|
|
|
|
|
|
|
//! \brief Writes the object, transitioning it from #kStateWritable to
|
|
|
|
|
//! #kStateWritten.
|
|
|
|
|
//!
|
|
|
|
|
//! Writes any padding necessary to meet alignment requirements, and then
|
|
|
|
|
//! calls WriteObject() to write the object’s content.
|
|
|
|
|
//!
|
|
|
|
|
//! \param[in] file_writer The file writer to receive the object’s content.
|
|
|
|
|
//!
|
|
|
|
|
//! \return `true` on success. `false` on error with an appropriate message
|
|
|
|
|
//! logged.
|
|
|
|
|
//!
|
|
|
|
|
//! \note This method cannot be overridden. Subclasses must override
|
|
|
|
|
//! WriteObject().
|
|
|
|
|
bool WritePaddingAndObject(FileWriterInterface* file_writer);
|
|
|
|
|
|
|
|
|
|
//! \brief Writes the object’s content.
|
|
|
|
|
//!
|
|
|
|
|
//! \param[in] file_writer The file writer to receive the object’s content.
|
|
|
|
|
//!
|
|
|
|
|
//! \return `true` on success. `false` on error, indicating that the content
|
|
|
|
|
//! could not be written to the minidump file.
|
|
|
|
|
//!
|
|
|
|
|
//! \note Valid in #kStateWritable. The object will transition to
|
|
|
|
|
//! #kStateWritten after this method returns.
|
|
|
|
|
virtual bool WriteObject(FileWriterInterface* file_writer) = 0;
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
std::vector<RVA*> registered_rvas_; // weak
|
|
|
|
|
|
2022-08-16 00:48:53 +00:00
|
|
|
|
std::vector<RVA64*> registered_rva64s_; // weak
|
|
|
|
|
|
2022-04-02 01:21:55 +00:00
|
|
|
|
// weak
|
|
|
|
|
std::vector<MINIDUMP_LOCATION_DESCRIPTOR*> registered_location_descriptors_;
|
|
|
|
|
|
2022-08-16 00:48:53 +00:00
|
|
|
|
// weak
|
|
|
|
|
std::vector<MINIDUMP_LOCATION_DESCRIPTOR64*>
|
|
|
|
|
registered_location_descriptor64s_;
|
|
|
|
|
|
2022-04-02 01:21:55 +00:00
|
|
|
|
size_t leading_pad_bytes_;
|
|
|
|
|
State state_;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
} // namespace internal
|
|
|
|
|
} // namespace crashpad
|
|
|
|
|
|
|
|
|
|
#endif // CRASHPAD_MINIDUMP_MINIDUMP_WRITABLE_H_
|