mirror of
https://github.com/chromium/crashpad.git
synced 2025-01-01 19:05:20 +08:00
184 lines
5.0 KiB
C++
184 lines
5.0 KiB
C++
|
// 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
|