chromium/third_party/crashpad/crashpad/snapshot/ios/memory_snapshot_ios_intermediate_dump.cc

// Copyright 2020 The Crashpad Authors
//
// 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 "snapshot/ios/memory_snapshot_ios_intermediate_dump.h"

#include "base/check_op.h"

namespace crashpad {
namespace internal {

void MemorySnapshotIOSIntermediateDump::Initialize(vm_address_t address,
                                                   vm_address_t data,
                                                   vm_size_t size) {
  INITIALIZATION_STATE_SET_INITIALIZING(initialized_);
  address_ = address;
  data_ = data;
  size_ = base::checked_cast<size_t>(size);
  INITIALIZATION_STATE_SET_VALID(initialized_);
}

uint64_t MemorySnapshotIOSIntermediateDump::Address() const {
  INITIALIZATION_STATE_DCHECK_VALID(initialized_);
  return address_;
}

size_t MemorySnapshotIOSIntermediateDump::Size() const {
  INITIALIZATION_STATE_DCHECK_VALID(initialized_);
  return size_;
}

bool MemorySnapshotIOSIntermediateDump::Read(Delegate* delegate) const {
  INITIALIZATION_STATE_DCHECK_VALID(initialized_);

  if (size_ == 0) {
    return delegate->MemorySnapshotDelegateRead(nullptr, size_);
  }

  return delegate->MemorySnapshotDelegateRead(reinterpret_cast<void*>(data_),
                                              size_);
}

const MemorySnapshot* MemorySnapshotIOSIntermediateDump::MergeWithOtherSnapshot(
    const MemorySnapshot* other) const {
  INITIALIZATION_STATE_DCHECK_VALID(initialized_);
  auto other_snapshot =
      reinterpret_cast<const MemorySnapshotIOSIntermediateDump*>(other);

  INITIALIZATION_STATE_DCHECK_VALID(other_snapshot->initialized_);
  if (other_snapshot->address_ < address_) {
    return other_snapshot->MergeWithOtherSnapshot(this);
  }

  CheckedRange<uint64_t, size_t> merged(0, 0);
  if (!LoggingDetermineMergedRange(this, other, &merged))
    return nullptr;

  auto result = std::make_unique<MemorySnapshotIOSIntermediateDump>();
  result->Initialize(merged.base(), data_, merged.size());
  if (size_ == merged.size()) {
    return result.release();
  }

  const uint8_t* data = reinterpret_cast<const uint8_t*>(data_);
  const uint8_t* other_data =
      reinterpret_cast<const uint8_t*>(other_snapshot->data_);
  vm_size_t overlap = merged.size() - other_snapshot->size_;
  result->merged_data_.reserve(merged.size());
  result->merged_data_.insert(result->merged_data_.end(), data, data + overlap);
  result->merged_data_.insert(result->merged_data_.end(),
                              other_data,
                              other_data + other_snapshot->size_);
  result->data_ =
      reinterpret_cast<const vm_address_t>(result->merged_data_.data());
  DCHECK_EQ(result->merged_data_.size(), merged.size());
  return result.release();
}

}  // namespace internal
}  // namespace crashpad