blob: cf9db5ba189e58e1bf40bd305a011f3153dfca08 [file] [log] [blame]
// Copyright 2015 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "mojo/public/cpp/bindings/lib/validation_util.h"
#include <limits>
#include <string>
#include "mojo/public/cpp/bindings/lib/bindings_serialization.h"
#include "mojo/public/cpp/bindings/lib/validation_errors.h"
namespace mojo {
namespace internal {
bool ValidateEncodedPointer(const uint64_t* offset) {
// - Make sure |*offset| is no more than 32-bits.
// - Cast |offset| to uintptr_t so overflow behavior is well defined across
// 32-bit and 64-bit systems.
return *offset <= std::numeric_limits<uint32_t>::max() &&
(reinterpret_cast<uintptr_t>(offset) +
static_cast<uint32_t>(*offset) >=
reinterpret_cast<uintptr_t>(offset));
}
ValidationError ValidateStructHeaderAndClaimMemory(
const void* data,
BoundsChecker* bounds_checker,
std::string* err) {
if (!IsAligned(data)) {
MOJO_INTERNAL_DEBUG_SET_ERROR_MSG(err) << "";
return ValidationError::MISALIGNED_OBJECT;
}
if (!bounds_checker->IsValidRange(data, sizeof(StructHeader))) {
MOJO_INTERNAL_DEBUG_SET_ERROR_MSG(err) << "";
return ValidationError::ILLEGAL_MEMORY_RANGE;
}
const StructHeader* header = static_cast<const StructHeader*>(data);
if (header->num_bytes < sizeof(StructHeader)) {
MOJO_INTERNAL_DEBUG_SET_ERROR_MSG(err) << "";
return ValidationError::UNEXPECTED_STRUCT_HEADER;
}
if (!bounds_checker->ClaimMemory(data, header->num_bytes)) {
MOJO_INTERNAL_DEBUG_SET_ERROR_MSG(err) << "";
return ValidationError::ILLEGAL_MEMORY_RANGE;
}
return ValidationError::NONE;
}
} // namespace internal
} // namespace mojo