2022-09-06 19:14:07 -04:00
|
|
|
// Copyright 2018 The Crashpad Authors
|
2018-02-01 15:54:56 -08:00
|
|
|
//
|
|
|
|
// 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/memory_snapshot.h"
|
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
|
|
|
|
#include "base/format_macros.h"
|
2020-06-18 15:35:28 +02:00
|
|
|
#include "base/logging.h"
|
2018-02-01 15:54:56 -08:00
|
|
|
#include "base/strings/stringprintf.h"
|
|
|
|
#include "util/numeric/checked_range.h"
|
|
|
|
|
|
|
|
namespace crashpad {
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
bool DetermineMergedRangeImpl(bool log,
|
|
|
|
const MemorySnapshot* a,
|
|
|
|
const MemorySnapshot* b,
|
|
|
|
CheckedRange<uint64_t, size_t>* merged) {
|
|
|
|
if (a->Size() == 0) {
|
|
|
|
LOG_IF(ERROR, log) << base::StringPrintf(
|
|
|
|
"invalid empty range at 0x%" PRIx64, a->Address());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (b->Size() == 0) {
|
|
|
|
LOG_IF(ERROR, log) << base::StringPrintf(
|
|
|
|
"invalid empty range at 0x%" PRIx64, b->Address());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
CheckedRange<uint64_t, size_t> range_a(a->Address(), a->Size());
|
|
|
|
if (!range_a.IsValid()) {
|
|
|
|
LOG_IF(ERROR, log) << base::StringPrintf("invalid range at 0x%" PRIx64
|
|
|
|
", size %" PRIuS,
|
|
|
|
range_a.base(),
|
|
|
|
range_a.size());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
CheckedRange<uint64_t, size_t> range_b(b->Address(), b->Size());
|
|
|
|
if (!range_b.IsValid()) {
|
|
|
|
LOG_IF(ERROR, log) << base::StringPrintf("invalid range at 0x%" PRIx64
|
|
|
|
", size %" PRIuS,
|
|
|
|
range_b.base(),
|
|
|
|
range_b.size());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!range_a.OverlapsRange(range_b) && range_a.end() != range_b.base() &&
|
|
|
|
range_b.end() != range_a.base()) {
|
|
|
|
LOG_IF(ERROR, log) << base::StringPrintf(
|
|
|
|
"ranges not overlapping or abutting: (0x%" PRIx64 ", size %" PRIuS
|
|
|
|
") and (0x%" PRIx64 ", size %" PRIuS ")",
|
|
|
|
range_a.base(),
|
|
|
|
range_a.size(),
|
|
|
|
range_b.base(),
|
|
|
|
range_b.size());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (merged) {
|
|
|
|
uint64_t base = std::min(range_a.base(), range_b.base());
|
|
|
|
uint64_t end = std::max(range_a.end(), range_b.end());
|
|
|
|
size_t size = static_cast<size_t>(end - base);
|
|
|
|
merged->SetRange(base, size);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
bool LoggingDetermineMergedRange(const MemorySnapshot* a,
|
|
|
|
const MemorySnapshot* b,
|
|
|
|
CheckedRange<uint64_t, size_t>* merged) {
|
|
|
|
return DetermineMergedRangeImpl(true, a, b, merged);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DetermineMergedRange(const MemorySnapshot* a,
|
|
|
|
const MemorySnapshot* b,
|
|
|
|
CheckedRange<uint64_t, size_t>* merged) {
|
|
|
|
return DetermineMergedRangeImpl(false, a, b, merged);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace crashpad
|