// 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 #include #include #include #include #include "base/logging.h" #include "base/macros.h" #include "base/numerics/safe_conversions.h" #include "base/strings/string_piece.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 an annotation’s name, in bytes. static constexpr size_t kNameMaxLength = 64; //! \brief The maximum size of an annotation’s value, 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; // 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(Type::kUserDefinedStart); const UnderlyingType user_type = start + value; DCHECK(user_type > start) << "User-defined Type is 0 or overflows"; return static_cast(user_type); #else return static_cast( static_cast(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& 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 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 class StringAnnotation : public Annotation { public: //! \brief A constructor tag that enables braced initialization in C arrays. //! //! \sa StringAnnotation() enum class Tag { kArray }; //! \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 Constructs a new StringAnnotation with the given \a name. //! //! This constructor takes the ArrayInitializerTag for use when //! initializing a C array of annotations. The main constructor is //! explicit and cannot be brace-initialized. As an example: //! //! \code //! static crashpad::StringAnnotation<32> annotations[] = { //! {"name-1", crashpad::StringAnnotation<32>::Tag::kArray}, //! {"name-2", crashpad::StringAnnotation<32>::Tag::kArray}, //! {"name-3", crashpad::StringAnnotation<32>::Tag::kArray}, //! }; //! \endcode //! //! \param[in] name The Annotation name. //! \param[in] tag A constructor tag. constexpr StringAnnotation(const char name[], Tag tag) : StringAnnotation(name) {} //! \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(strlen(value)))); } //! \brief Sets the Annotation's string value. //! //! \param[in] string The string value. void Set(base::StringPiece string) { Annotation::ValueSizeType size = std::min(MaxSize, base::saturated_cast(string.size())); memcpy(value_, string.data(), size); // Check for no embedded `NUL` characters. DCHECK(!memchr(value_, '\0', size)) << "embedded NUL"; SetSize(size); } const base::StringPiece value() const { return base::StringPiece(value_, size()); } 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_