|  | // 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. | 
|  |  | 
|  | #include "client/annotation.h" | 
|  |  | 
|  | #include <string> | 
|  | #include <vector> | 
|  |  | 
|  | #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<std::pair<std::string, std::string>>; | 
|  |  | 
|  | AllAnnotations CollectAnnotations() { | 
|  | AllAnnotations annotations; | 
|  |  | 
|  | for (Annotation* curr : annotations_) { | 
|  | if (!curr->is_set()) | 
|  | continue; | 
|  | std::string value(static_cast<const char*>(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 |