// Copyright 2017 The Crashpad Authors // // 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 "client/annotation.h" #include #include #include "client/annotation_list.h" #include "client/crashpad_info.h" #include "gtest/gtest.h" #include "test/gtest_death.h" #include "util/misc/clock.h" #include "util/synchronization/scoped_spin_guard.h" #include "util/thread/thread.h" namespace crashpad { namespace test { namespace { class SpinGuardAnnotation final : public Annotation { public: SpinGuardAnnotation(Annotation::Type type, const char name[]) : Annotation(type, name, &value_, ConcurrentAccessGuardMode::kScopedSpinGuard) {} bool Set(bool value, uint64_t spin_guard_timeout_ns) { auto guard = TryCreateScopedSpinGuard(spin_guard_timeout_ns); if (!guard) { return false; } value_ = value; SetSize(sizeof(value_)); return true; } private: bool value_; }; class ScopedSpinGuardUnlockThread final : public Thread { public: ScopedSpinGuardUnlockThread(ScopedSpinGuard scoped_spin_guard, uint64_t sleep_time_ns) : scoped_spin_guard_(std::move(scoped_spin_guard)), sleep_time_ns_(sleep_time_ns) {} private: void ThreadMain() override { SleepNanoseconds(sleep_time_ns_); // Move the ScopedSpinGuard member into a local variable which is // destroyed when ThreadMain() returns. ScopedSpinGuard local_scoped_spin_guard(std::move(scoped_spin_guard_)); // After this point, local_scoped_spin_guard will be destroyed and unlocked. } ScopedSpinGuard scoped_spin_guard_; const uint64_t sleep_time_ns_; }; class Annotation : public testing::Test { public: void SetUp() override { CrashpadInfo::GetCrashpadInfo()->set_annotations_list(&annotations_); } void TearDown() override { CrashpadInfo::GetCrashpadInfo()->set_annotations_list(nullptr); } size_t AnnotationsCount() { size_t result = 0; for (auto* annotation : annotations_) { if (annotation->is_set()) ++result; } return result; } protected: crashpad::AnnotationList annotations_; }; TEST_F(Annotation, Basics) { constexpr crashpad::Annotation::Type kType = crashpad::Annotation::UserDefinedType(1); const char kName[] = "annotation 1"; char buffer[1024]; crashpad::Annotation annotation(kType, kName, buffer); EXPECT_FALSE(annotation.is_set()); EXPECT_EQ(0u, AnnotationsCount()); EXPECT_EQ(kType, annotation.type()); EXPECT_EQ(0u, annotation.size()); EXPECT_EQ(std::string(kName), annotation.name()); EXPECT_EQ(buffer, annotation.value()); annotation.SetSize(10); EXPECT_TRUE(annotation.is_set()); EXPECT_EQ(1u, AnnotationsCount()); EXPECT_EQ(10u, annotation.size()); EXPECT_EQ(&annotation, *annotations_.begin()); annotation.Clear(); EXPECT_FALSE(annotation.is_set()); EXPECT_EQ(0u, AnnotationsCount()); EXPECT_EQ(0u, annotation.size()); } TEST_F(Annotation, StringType) { crashpad::StringAnnotation<5> annotation("name"); EXPECT_FALSE(annotation.is_set()); EXPECT_EQ(crashpad::Annotation::Type::kString, annotation.type()); EXPECT_EQ(0u, annotation.size()); EXPECT_EQ(std::string("name"), annotation.name()); EXPECT_EQ(0u, annotation.value().size()); annotation.Set("test"); EXPECT_TRUE(annotation.is_set()); EXPECT_EQ(1u, AnnotationsCount()); EXPECT_EQ(4u, annotation.size()); EXPECT_EQ("test", annotation.value()); annotation.Set(std::string("loooooooooooong")); EXPECT_TRUE(annotation.is_set()); EXPECT_EQ(1u, AnnotationsCount()); EXPECT_EQ(5u, annotation.size()); EXPECT_EQ("loooo", annotation.value()); } TEST_F(Annotation, BaseAnnotationShouldNotSupportSpinGuard) { char buffer[1024]; crashpad::Annotation annotation( crashpad::Annotation::Type::kString, "no-spin-guard", buffer); EXPECT_EQ(annotation.concurrent_access_guard_mode(), crashpad::Annotation::ConcurrentAccessGuardMode::kUnguarded); #if !DCHECK_IS_ON() // This fails a DCHECK() in debug builds, so only test it when DCHECK() // is not on. EXPECT_EQ(std::nullopt, annotation.TryCreateScopedSpinGuard(0)); #endif } TEST_F(Annotation, CustomAnnotationShouldSupportSpinGuardAndSet) { constexpr crashpad::Annotation::Type kType = crashpad::Annotation::UserDefinedType(1); SpinGuardAnnotation spin_guard_annotation(kType, "spin-guard"); EXPECT_EQ(spin_guard_annotation.concurrent_access_guard_mode(), crashpad::Annotation::ConcurrentAccessGuardMode::kScopedSpinGuard); EXPECT_TRUE(spin_guard_annotation.Set(true, 0)); EXPECT_EQ(1U, spin_guard_annotation.size()); } TEST_F(Annotation, CustomAnnotationSetShouldFailIfRunConcurrently) { constexpr crashpad::Annotation::Type kType = crashpad::Annotation::UserDefinedType(1); SpinGuardAnnotation spin_guard_annotation(kType, "spin-guard"); auto guard = spin_guard_annotation.TryCreateScopedSpinGuard(0); EXPECT_NE(std::nullopt, guard); // This should fail, since the guard is already held and the timeout is 0. EXPECT_FALSE(spin_guard_annotation.Set(true, 0)); } TEST_F(Annotation, CustomAnnotationSetShouldSucceedIfSpinGuardUnlockedAsynchronously) { constexpr crashpad::Annotation::Type kType = crashpad::Annotation::UserDefinedType(1); SpinGuardAnnotation spin_guard_annotation(kType, "spin-guard"); auto guard = spin_guard_annotation.TryCreateScopedSpinGuard(0); EXPECT_NE(std::nullopt, guard); // Pass the guard off to a background thread which unlocks it after 1 ms. constexpr uint64_t kSleepTimeNs = 1000000; // 1 ms ScopedSpinGuardUnlockThread unlock_thread(std::move(guard.value()), kSleepTimeNs); unlock_thread.Start(); // Try to set the annotation with a 100 ms timeout. constexpr uint64_t kSpinGuardTimeoutNanos = 100000000; // 100 ms // This should succeed after 1 ms, since the timeout is much larger than the // time the background thread holds the guard. EXPECT_TRUE(spin_guard_annotation.Set(true, kSpinGuardTimeoutNanos)); unlock_thread.Join(); } TEST(StringAnnotation, ArrayOfString) { static crashpad::StringAnnotation<4> annotations[] = { {"test-1", crashpad::StringAnnotation<4>::Tag::kArray}, {"test-2", crashpad::StringAnnotation<4>::Tag::kArray}, {"test-3", crashpad::StringAnnotation<4>::Tag::kArray}, {"test-4", crashpad::StringAnnotation<4>::Tag::kArray}, }; for (auto& annotation : annotations) { EXPECT_FALSE(annotation.is_set()); } } #if DCHECK_IS_ON() TEST(AnnotationDeathTest, EmbeddedNUL) { crashpad::StringAnnotation<5> annotation("name"); EXPECT_DEATH_CHECK(annotation.Set(std::string("te\0st", 5)), "embedded NUL"); } #endif } // namespace } // namespace test } // namespace crashpad