mirror of
https://github.com/chromium/crashpad.git
synced 2025-01-15 10:07:56 +08:00
ef262d1ee3
Change-Id: I45c1afe73e8570dfcedde6da01375a4533bb355a Reviewed-on: https://chromium-review.googlesource.com/741891 Reviewed-by: Robert Sesek <rsesek@chromium.org>
224 lines
7.6 KiB
C++
224 lines
7.6 KiB
C++
// Copyright 2017 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.
|
|
|
|
#ifndef CRASHPAD_CLIENT_ANNOTATION_H_
|
|
#define CRASHPAD_CLIENT_ANNOTATION_H_
|
|
|
|
#include <algorithm>
|
|
#include <atomic>
|
|
|
|
#include <stdint.h>
|
|
#include <string.h>
|
|
#include <sys/types.h>
|
|
|
|
#include "base/logging.h"
|
|
#include "base/macros.h"
|
|
#include "base/numerics/safe_conversions.h"
|
|
#include "build/build_config.h"
|
|
|
|
namespace crashpad {
|
|
|
|
class AnnotationList;
|
|
|
|
//! \brief Base class for an annotation, which records a name-value pair of
|
|
//! arbitrary data when set.
|
|
//!
|
|
//! After an annotation is declared, its `value_ptr_` will not be captured in a
|
|
//! crash report until a call to \a SetSize() specifies how much data from the
|
|
//! value should be recorded.
|
|
//!
|
|
//! Annotations should be declared with static storage duration.
|
|
//!
|
|
//! An example declaration and usage:
|
|
//!
|
|
//! \code
|
|
//! // foo.cc:
|
|
//!
|
|
//! namespace {
|
|
//! char g_buffer[1024];
|
|
//! crashpad::Annotation g_buffer_annotation(
|
|
//! crashpad::Annotation::Type::kString, "buffer_head", g_buffer);
|
|
//! } // namespace
|
|
//!
|
|
//! void OnBufferProduced(size_t n) {
|
|
//! // Capture the head of the buffer, in case we crash when parsing it.
|
|
//! g_buffer_annotation.SetSize(std::min(64, n));
|
|
//!
|
|
//! // Start parsing the header.
|
|
//! Frobinate(g_buffer, n);
|
|
//! }
|
|
//! \endcode
|
|
//!
|
|
//! Annotation objects are not inherently thread-safe. To manipulate them
|
|
//! from multiple threads, external synchronization must be used.
|
|
//!
|
|
//! Annotation objects should never be destroyed. Once they are Set(), they
|
|
//! are permanently referenced by a global object.
|
|
class Annotation {
|
|
public:
|
|
//! \brief The maximum length of the #name field in bytes.
|
|
static constexpr size_t kNameMaxLength = 64;
|
|
|
|
//! \brief The maximum size of the #value field in bytes.
|
|
static constexpr size_t kValueMaxSize = 2048;
|
|
|
|
//! \brief The type used for \a SetSize().
|
|
using ValueSizeType = uint32_t;
|
|
|
|
//! \brief The type of data stored in the annotation.
|
|
enum class Type : uint16_t {
|
|
//! \brief An invalid annotation. Reserved for internal use.
|
|
kInvalid = 0,
|
|
|
|
//! \brief A `NUL`-terminated C-string.
|
|
kString = 1,
|
|
|
|
//! \brief Clients may declare their own custom types by using values
|
|
//! greater than this.
|
|
kUserDefinedStart = 0x8000,
|
|
};
|
|
|
|
//! \brief Creates a user-defined Annotation::Type.
|
|
//!
|
|
//! This exists to remove the casting overhead of `enum class`.
|
|
//!
|
|
//! \param[in] value A value used to create a user-defined type.
|
|
//!
|
|
//! \returns The value added to Type::kUserDefinedStart and casted.
|
|
constexpr static Type UserDefinedType(uint16_t value) {
|
|
using UnderlyingType = std::underlying_type<Type>::type;
|
|
// MSVS 2015 doesn't have full C++14 support and complains about local
|
|
// variables defined in a constexpr function, which is valid. Avoid them
|
|
// and the also-problematic DCHECK until all the infrastructure is updated:
|
|
// https://crbug.com/crashpad/201.
|
|
#if !defined(OS_WIN) || (defined(_MSC_VER) && _MSC_VER >= 1910)
|
|
const UnderlyingType start =
|
|
static_cast<UnderlyingType>(Type::kUserDefinedStart);
|
|
const UnderlyingType user_type = start + value;
|
|
DCHECK(user_type > start) << "User-defined Type is 0 or overflows";
|
|
return static_cast<Type>(user_type);
|
|
#else
|
|
return static_cast<Type>(
|
|
static_cast<UnderlyingType>(Type::kUserDefinedStart) + value);
|
|
#endif
|
|
}
|
|
|
|
//! \brief Constructs a new annotation.
|
|
//!
|
|
//! Upon construction, the annotation will not be included in any crash
|
|
//! reports until \sa SetSize() is called with a value greater than `0`.
|
|
//!
|
|
//! \param[in] type The data type of the value of the annotation.
|
|
//! \param[in] name A `NUL`-terminated C-string name for the annotation. Names
|
|
//! do not have to be unique, though not all crash processors may handle
|
|
//! Annotations with the same name. Names should be constexpr data with
|
|
//! static storage duration.
|
|
//! \param[in] value_ptr A pointer to the value for the annotation. The
|
|
//! pointer may not be changed once associated with an annotation, but
|
|
//! the data may be mutated.
|
|
constexpr Annotation(Type type, const char name[], void* const value_ptr)
|
|
: link_node_(nullptr),
|
|
name_(name),
|
|
value_ptr_(value_ptr),
|
|
size_(0),
|
|
type_(type) {}
|
|
|
|
//! \brief Specifies the number of bytes in \a value_ptr_ to include when
|
|
//! generating a crash report.
|
|
//!
|
|
//! A size of `0` indicates that no value should be recorded and is the
|
|
//! equivalent of calling \sa Clear().
|
|
//!
|
|
//! This method does not mutate the data referenced by the annotation, it
|
|
//! merely updates the annotation system's bookkeeping.
|
|
//!
|
|
//! Subclasses of this base class that provide additional Set methods to
|
|
//! mutate the value of the annotation must call always call this method.
|
|
//!
|
|
//! \param[in] size The number of bytes.
|
|
void SetSize(ValueSizeType size);
|
|
|
|
//! \brief Marks the annotation as cleared, indicating the \a value_ptr_
|
|
//! should not be included in a crash report.
|
|
//!
|
|
//! This method does not mutate the data referenced by the annotation, it
|
|
//! merely updates the annotation system's bookkeeping.
|
|
void Clear();
|
|
|
|
//! \brief Tests whether the annotation has been set.
|
|
bool is_set() const { return size_ > 0; }
|
|
|
|
Type type() const { return type_; }
|
|
ValueSizeType size() const { return size_; }
|
|
const char* name() const { return name_; }
|
|
const void* value() const { return value_ptr_; }
|
|
|
|
protected:
|
|
friend class AnnotationList;
|
|
|
|
std::atomic<Annotation*>& link_node() { return link_node_; }
|
|
|
|
private:
|
|
//! \brief Linked list next-node pointer. Accessed only by \sa AnnotationList.
|
|
//!
|
|
//! This will be null until the first call to \sa SetSize(), after which the
|
|
//! presence of the pointer will prevent the node from being added to the
|
|
//! list again.
|
|
std::atomic<Annotation*> link_node_;
|
|
|
|
const char* const name_;
|
|
void* const value_ptr_;
|
|
ValueSizeType size_;
|
|
const Type type_;
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(Annotation);
|
|
};
|
|
|
|
//! \brief An \sa Annotation that stores a `NUL`-terminated C-string value.
|
|
//!
|
|
//! The storage for the value is allocated by the annotation and the template
|
|
//! parameter \a MaxSize controls the maxmium length for the value.
|
|
//!
|
|
//! It is expected that the string value be valid UTF-8, although this is not
|
|
//! validated.
|
|
template <Annotation::ValueSizeType MaxSize>
|
|
class StringAnnotation : public Annotation {
|
|
public:
|
|
//! \brief Constructs a new StringAnnotation with the given \a name.
|
|
//!
|
|
//! \param[in] name The Annotation name.
|
|
constexpr explicit StringAnnotation(const char name[])
|
|
: Annotation(Type::kString, name, value_), value_() {}
|
|
|
|
//! \brief Sets the Annotation's string value.
|
|
//!
|
|
//! \param[in] value The `NUL`-terminated C-string value.
|
|
void Set(const char* value) {
|
|
strncpy(value_, value, MaxSize);
|
|
SetSize(
|
|
std::min(MaxSize, base::saturated_cast<ValueSizeType>(strlen(value))));
|
|
}
|
|
|
|
private:
|
|
// This value is not `NUL`-terminated, since the size is stored by the base
|
|
// annotation.
|
|
char value_[MaxSize];
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(StringAnnotation);
|
|
};
|
|
|
|
} // namespace crashpad
|
|
|
|
#endif // CRASHPAD_CLIENT_ANNOTATION_H_
|