chromium/third_party/crashpad/crashpad/snapshot/fuchsia/memory_map_fuchsia.cc

// Copyright 2018 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/fuchsia/memory_map_fuchsia.h"

#include "base/fuchsia/fuchsia_logging.h"
#include "util/numeric/checked_range.h"

namespace crashpad {

MemoryMapFuchsia::MemoryMapFuchsia() = default;

MemoryMapFuchsia::~MemoryMapFuchsia() = default;

bool MemoryMapFuchsia::Initialize(const zx::process& process) {
  INITIALIZATION_STATE_SET_INITIALIZING(initialized_);

  // There's no way to know what an appropriate buffer size is before starting.
  // Start at a size that should be more than enough for any reasonable process.
  map_entries_.resize(4096);

  // Retrieving the maps is racy with new mappings being created, so retry this
  // loop up to |tries| times until the number of actual mappings retrieved
  // matches those available.
  int tries = 5;
  for (;;) {
    size_t actual;
    size_t available;
    zx_status_t status =
        process.get_info(ZX_INFO_PROCESS_MAPS,
                         &map_entries_[0],
                         map_entries_.size() * sizeof(map_entries_[0]),
                         &actual,
                         &available);
    if (status != ZX_OK) {
      ZX_LOG(ERROR, status) << "zx_object_get_info ZX_INFO_PROCESS_MAPS";
      map_entries_.clear();
      return false;
    }
    if (actual < available && tries-- > 0) {
      // Make the buffer slightly larger than |available| to attempt to account
      // for the race between here and the next retrieval.
      map_entries_.resize(available + 20);
      continue;
    }

    map_entries_.resize(actual);

    INITIALIZATION_STATE_SET_VALID(initialized_);
    return true;
  }
}

bool MemoryMapFuchsia::FindMappingForAddress(zx_vaddr_t address,
                                             zx_info_maps_t* map) const {
  bool found = false;
  zx_info_maps_t result = {};
  for (const auto& m : map_entries_) {
    CheckedRange<zx_vaddr_t, size_t> range(m.base, m.size);
    if (range.ContainsValue(address)) {
      if (!found || m.depth > result.depth) {
        result = m;
        found = true;
      }
    }
  }

  if (found) {
    *map = result;
  }
  return found;
}

}  // namespace crashpad