| Index: minidump/minidump_writable.cc
|
| diff --git a/minidump/minidump_writable.cc b/minidump/minidump_writable.cc
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..f1113a4328dd26b1a1bf7ad0562dac9e5c60e45f
|
| --- /dev/null
|
| +++ b/minidump/minidump_writable.cc
|
| @@ -0,0 +1,264 @@
|
| +// 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.
|
| +
|
| +#include "minidump/minidump_writable.h"
|
| +
|
| +#include "base/logging.h"
|
| +#include "util/numeric/safe_assignment.h"
|
| +
|
| +namespace {
|
| +
|
| +const size_t kMaximumAlignment = 16;
|
| +
|
| +} // namespace
|
| +
|
| +namespace crashpad {
|
| +namespace internal {
|
| +
|
| +bool MinidumpWritable::WriteEverything(FileWriterInterface* file_writer) {
|
| + DCHECK_EQ(state_, kStateMutable);
|
| +
|
| + if (!Freeze()) {
|
| + return false;
|
| + }
|
| +
|
| + DCHECK_EQ(state_, kStateFrozen);
|
| +
|
| + off_t offset = 0;
|
| + std::vector<MinidumpWritable*> write_sequence;
|
| + size_t size = WillWriteAtOffset(kPhaseEarly, &offset, &write_sequence);
|
| + if (size == kInvalidSize) {
|
| + return false;
|
| + }
|
| +
|
| + offset += size;
|
| + if (WillWriteAtOffset(kPhaseLate, &offset, &write_sequence) == kInvalidSize) {
|
| + return false;
|
| + }
|
| +
|
| + DCHECK_EQ(state_, kStateWritable);
|
| + DCHECK_EQ(write_sequence.front(), this);
|
| +
|
| + for (MinidumpWritable* writable : write_sequence) {
|
| + if (!writable->WritePaddingAndObject(file_writer)) {
|
| + return false;
|
| + }
|
| + }
|
| +
|
| + DCHECK_EQ(state_, kStateWritten);
|
| +
|
| + return true;
|
| +}
|
| +
|
| +void MinidumpWritable::RegisterRVA(RVA* rva) {
|
| + DCHECK_LE(state_, kStateFrozen);
|
| +
|
| + registered_rvas_.push_back(rva);
|
| +}
|
| +
|
| +void MinidumpWritable::RegisterLocationDescriptor(
|
| + MINIDUMP_LOCATION_DESCRIPTOR* location_descriptor) {
|
| + DCHECK_LE(state_, kStateFrozen);
|
| +
|
| + registered_location_descriptors_.push_back(location_descriptor);
|
| +}
|
| +
|
| +const size_t MinidumpWritable::kInvalidSize =
|
| + std::numeric_limits<size_t>::max();
|
| +
|
| +MinidumpWritable::MinidumpWritable()
|
| + : registered_rvas_(),
|
| + registered_location_descriptors_(),
|
| + leading_pad_bytes_(0),
|
| + state_(kStateMutable) {
|
| +}
|
| +
|
| +MinidumpWritable::~MinidumpWritable() {
|
| +}
|
| +
|
| +bool MinidumpWritable::Freeze() {
|
| + DCHECK_EQ(state_, kStateMutable);
|
| + state_ = kStateFrozen;
|
| +
|
| + std::vector<MinidumpWritable*> children = Children();
|
| + for (MinidumpWritable* child : children) {
|
| + if (!child->Freeze()) {
|
| + return false;
|
| + }
|
| + }
|
| +
|
| + return true;
|
| +}
|
| +
|
| +size_t MinidumpWritable::Alignment() {
|
| + DCHECK_GE(state_, kStateFrozen);
|
| +
|
| + return 4;
|
| +}
|
| +
|
| +std::vector<MinidumpWritable*> MinidumpWritable::Children() {
|
| + DCHECK_GE(state_, kStateFrozen);
|
| +
|
| + return std::vector<MinidumpWritable*>();
|
| +}
|
| +
|
| +MinidumpWritable::Phase MinidumpWritable::WritePhase() {
|
| + return kPhaseEarly;
|
| +}
|
| +
|
| +size_t MinidumpWritable::WillWriteAtOffset(
|
| + Phase phase,
|
| + off_t* offset,
|
| + std::vector<MinidumpWritable*>* write_sequence) {
|
| + off_t local_offset = *offset;
|
| + CHECK_GE(local_offset, 0);
|
| +
|
| + size_t leading_pad_bytes_this_phase;
|
| + size_t size;
|
| + if (phase == WritePhase()) {
|
| + DCHECK_EQ(state_, kStateFrozen);
|
| +
|
| + // Add this object to the sequence of MinidumpWritable objects to be
|
| + // written.
|
| + write_sequence->push_back(this);
|
| +
|
| + size = SizeOfObject();
|
| +
|
| + if (size > 0) {
|
| + // Honor this object’s request to be aligned to a specific byte boundary.
|
| + // Once the alignment is corrected, this object knows exactly what file
|
| + // offset it will be written at.
|
| + size_t alignment = Alignment();
|
| + CHECK_LE(alignment, kMaximumAlignment);
|
| +
|
| + leading_pad_bytes_this_phase =
|
| + (alignment - (local_offset % alignment)) % alignment;
|
| + local_offset += leading_pad_bytes_this_phase;
|
| + *offset = local_offset;
|
| + } else {
|
| + // If the object is size 0, alignment is of no concern.
|
| + leading_pad_bytes_this_phase = 0;
|
| + }
|
| + leading_pad_bytes_ = leading_pad_bytes_this_phase;
|
| +
|
| + // Now that the file offset that this object will be written at is known,
|
| + // let the subclass implementation know in case it’s interested.
|
| + if (!WillWriteAtOffsetImpl(local_offset)) {
|
| + return kInvalidSize;
|
| + }
|
| +
|
| + // Populate the RVA fields in other objects that have registered to point to
|
| + // this one. Typically, a parent object will have registered to point to its
|
| + // children, but this can also occur where no parent-child relationship
|
| + // exists.
|
| + if (!registered_rvas_.empty() ||
|
| + !registered_location_descriptors_.empty()) {
|
| + RVA local_rva;
|
| + if (!AssignIfInRange(&local_rva, local_offset)) {
|
| + LOG(ERROR) << "offset " << local_offset << " out of range";
|
| + return kInvalidSize;
|
| + }
|
| +
|
| + for (RVA* rva : registered_rvas_) {
|
| + *rva = local_rva;
|
| + }
|
| +
|
| + if (!registered_location_descriptors_.empty()) {
|
| + typeof(registered_location_descriptors_[0]->DataSize) local_size;
|
| + if (!AssignIfInRange(&local_size, size)) {
|
| + LOG(ERROR) << "size " << size << " out of range";
|
| + return kInvalidSize;
|
| + }
|
| +
|
| + for (MINIDUMP_LOCATION_DESCRIPTOR* location_descriptor :
|
| + registered_location_descriptors_) {
|
| + location_descriptor->DataSize = local_size;
|
| + location_descriptor->Rva = local_rva;
|
| + }
|
| + }
|
| + }
|
| +
|
| + // This object is now considered writable. However, if it contains RVA or
|
| + // MINIDUMP_LOCATION_DESCRIPTOR fields, they may not be fully updated yet,
|
| + // because it’s the repsonsibility of these fields’ pointees to update them.
|
| + // Once WillWriteAtOffset has completed running for both phases on an entire
|
| + // tree, and the entire tree has moved into kStateFrozen, all RVA and
|
| + // MINIDUMP_LOCATION_DESCRIPTOR fields within that tree will be populated.
|
| + state_ = kStateWritable;
|
| + } else {
|
| + if (phase == kPhaseEarly) {
|
| + DCHECK_EQ(state_, kStateFrozen);
|
| + } else {
|
| + DCHECK_EQ(state_, kStateWritable);
|
| + }
|
| +
|
| + size = 0;
|
| + leading_pad_bytes_this_phase = 0;
|
| + }
|
| +
|
| + // Loop over children regardless of whether this object itself will write
|
| + // during this phase. An object’s children are not required to be written
|
| + // during the same phase as their parent.
|
| + std::vector<MinidumpWritable*> children = Children();
|
| + for (MinidumpWritable* child : children) {
|
| + // Use “auto” here because it’s impossible to know whether size_t (size) or
|
| + // off_t (local_offset) is the wider type, and thus what type the result of
|
| + // adding these two variables will have.
|
| + auto unaligned_child_offset = local_offset + size;
|
| + off_t child_offset;
|
| + if (!AssignIfInRange(&child_offset, unaligned_child_offset)) {
|
| + LOG(ERROR) << "offset " << unaligned_child_offset << " out of range";
|
| + return kInvalidSize;
|
| + }
|
| +
|
| + size_t child_size =
|
| + child->WillWriteAtOffset(phase, &child_offset, write_sequence);
|
| + if (child_size == kInvalidSize) {
|
| + return kInvalidSize;
|
| + }
|
| +
|
| + size += child_size;
|
| + }
|
| +
|
| + return leading_pad_bytes_this_phase + size;
|
| +}
|
| +
|
| +bool MinidumpWritable::WillWriteAtOffsetImpl(off_t offset) {
|
| + return true;
|
| +}
|
| +
|
| +bool MinidumpWritable::WritePaddingAndObject(FileWriterInterface* file_writer) {
|
| + DCHECK_EQ(state_, kStateWritable);
|
| +
|
| + // The number of elements in kZeroes must be at least one less than the
|
| + // maximum Alignment() ever encountered.
|
| + const uint8_t kZeroes[kMaximumAlignment - 1] = {};
|
| + DCHECK_LE(leading_pad_bytes_, arraysize(kZeroes));
|
| +
|
| + if (leading_pad_bytes_) {
|
| + if (!file_writer->Write(&kZeroes, leading_pad_bytes_)) {
|
| + return false;
|
| + }
|
| + }
|
| +
|
| + if (!WriteObject(file_writer)) {
|
| + return false;
|
| + }
|
| +
|
| + state_ = kStateWritten;
|
| + return true;
|
| +}
|
| +
|
| +} // namespace internal
|
| +} // namespace crashpad
|
|
|