| // 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 "base/strings/string_piece.h" |
| #include "build/build_config.h" |
| #include "nb/cpp14oncpp11.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 = 5 * 4096; |
| |
| //! \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 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<ValueSizeType>(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<ValueSizeType>(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_ |