// 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 "base/rand_util.h" #include "client/crashpad_info.h" #include "gtest/gtest.h" #include "util/misc/clock.h" #include "util/thread/thread.h" namespace crashpad { namespace test { namespace { TEST(AnnotationListStatic, Register) { ASSERT_FALSE(AnnotationList::Get()); EXPECT_TRUE(AnnotationList::Register()); EXPECT_TRUE(AnnotationList::Get()); EXPECT_EQ(AnnotationList::Get(), AnnotationList::Register()); // This isn't expected usage of the AnnotationList API, but it is necessary // for testing. AnnotationList* list = AnnotationList::Get(); CrashpadInfo::GetCrashpadInfo()->set_annotations_list(nullptr); delete list; EXPECT_FALSE(AnnotationList::Get()); } class AnnotationList : public testing::Test { public: void SetUp() override { CrashpadInfo::GetCrashpadInfo()->set_annotations_list(&annotations_); } void TearDown() override { CrashpadInfo::GetCrashpadInfo()->set_annotations_list(nullptr); } // NOTE: Annotations should be declared at file-scope, but in order to test // them, they are declared as part of the test. These members are public so // they are accessible from global helpers. crashpad::StringAnnotation<8> one_{"First"}; crashpad::StringAnnotation<256> two_{"Second"}; crashpad::StringAnnotation<101> three_{"First"}; protected: using AllAnnotations = std::vector>; AllAnnotations CollectAnnotations() { AllAnnotations annotations; for (Annotation* curr : annotations_) { if (!curr->is_set()) continue; std::string value(static_cast(curr->value()), curr->size()); annotations.push_back(std::make_pair(curr->name(), value)); } return annotations; } bool ContainsNameValue(const AllAnnotations& annotations, const std::string& name, const std::string& value) { return std::find(annotations.begin(), annotations.end(), std::make_pair(name, value)) != annotations.end(); } crashpad::AnnotationList annotations_; }; TEST_F(AnnotationList, SetAndClear) { one_.Set("this is a value longer than 8 bytes"); AllAnnotations annotations = CollectAnnotations(); EXPECT_EQ(1u, annotations.size()); EXPECT_TRUE(ContainsNameValue(annotations, "First", "this is ")); one_.Clear(); EXPECT_EQ(0u, CollectAnnotations().size()); one_.Set("short"); two_.Set(std::string(500, 'A').data()); annotations = CollectAnnotations(); EXPECT_EQ(2u, annotations.size()); EXPECT_EQ(5u, one_.size()); EXPECT_EQ(256u, two_.size()); EXPECT_TRUE(ContainsNameValue(annotations, "First", "short")); EXPECT_TRUE(ContainsNameValue(annotations, "Second", std::string(256, 'A'))); } TEST_F(AnnotationList, DuplicateKeys) { ASSERT_EQ(0u, CollectAnnotations().size()); one_.Set("1"); three_.Set("2"); AllAnnotations annotations = CollectAnnotations(); EXPECT_EQ(2u, annotations.size()); EXPECT_TRUE(ContainsNameValue(annotations, "First", "1")); EXPECT_TRUE(ContainsNameValue(annotations, "First", "2")); one_.Clear(); annotations = CollectAnnotations(); EXPECT_EQ(1u, annotations.size()); } class RaceThread : public Thread { public: explicit RaceThread(test::AnnotationList* test) : Thread(), test_(test) {} private: void ThreadMain() override { for (int i = 0; i <= 50; ++i) { if (i % 2 == 0) { test_->three_.Set("three"); test_->two_.Clear(); } else { test_->three_.Clear(); } SleepNanoseconds(base::RandInt(1, 1000)); } } test::AnnotationList* test_; }; TEST_F(AnnotationList, MultipleThreads) { ASSERT_EQ(0u, CollectAnnotations().size()); RaceThread other_thread(this); other_thread.Start(); for (int i = 0; i <= 50; ++i) { if (i % 2 == 0) { one_.Set("one"); two_.Set("two"); } else { one_.Clear(); } SleepNanoseconds(base::RandInt(1, 1000)); } other_thread.Join(); AllAnnotations annotations = CollectAnnotations(); EXPECT_GE(annotations.size(), 2u); EXPECT_LE(annotations.size(), 3u); EXPECT_TRUE(ContainsNameValue(annotations, "First", "one")); EXPECT_TRUE(ContainsNameValue(annotations, "First", "three")); if (annotations.size() == 3) { EXPECT_TRUE(ContainsNameValue(annotations, "Second", "two")); } } } // namespace } // namespace test } // namespace crashpad