| OLD | NEW |
| 1 // Copyright 2015 The Chromium Authors. All rights reserved. | 1 // Copyright 2015 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "components/metrics/call_stack_profile_metrics_provider.h" | 5 #include "components/metrics/call_stack_profile_metrics_provider.h" |
| 6 | 6 |
| 7 #include "base/profiler/stack_sampling_profiler.h" | 7 #include "base/profiler/stack_sampling_profiler.h" |
| 8 #include "base/strings/string_number_conversions.h" | 8 #include "base/strings/string_number_conversions.h" |
| 9 #include "components/metrics/proto/chrome_user_metrics_extension.pb.h" | 9 #include "components/metrics/proto/chrome_user_metrics_extension.pb.h" |
| 10 #include "testing/gtest/include/gtest/gtest.h" | 10 #include "testing/gtest/include/gtest/gtest.h" |
| 11 | 11 |
| 12 using base::StackSamplingProfiler; | 12 using base::StackSamplingProfiler; |
| 13 using Frame = StackSamplingProfiler::Frame; | 13 using Frame = StackSamplingProfiler::Frame; |
| 14 using Module = StackSamplingProfiler::Module; | 14 using Module = StackSamplingProfiler::Module; |
| 15 using Profile = StackSamplingProfiler::Profile; | 15 using Profile = StackSamplingProfiler::CallStackProfile; |
| 16 using Sample = StackSamplingProfiler::Sample; | 16 using Sample = StackSamplingProfiler::Sample; |
| 17 | 17 |
| 18 namespace metrics { | 18 namespace metrics { |
| 19 | 19 |
| 20 // Checks that all properties from multiple profiles are filled as expected. | 20 // Checks that all properties from multiple profiles are filled as expected. |
| 21 TEST(CallStackProfileMetricsProviderTest, MultipleProfiles) { | 21 TEST(CallStackProfileMetricsProviderTest, MultipleProfiles) { |
| 22 const uintptr_t module1_base_address = 0x1000; | 22 const uintptr_t module1_base_address = 0x1000; |
| 23 const uintptr_t module2_base_address = 0x2000; | 23 const uintptr_t module2_base_address = 0x2000; |
| 24 const uintptr_t module3_base_address = 0x3000; | 24 const uintptr_t module3_base_address = 0x3000; |
| 25 | 25 |
| (...skipping 168 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 194 ASSERT_TRUE(entry.has_address()); | 194 ASSERT_TRUE(entry.has_address()); |
| 195 const char* instruction_pointer = reinterpret_cast<const char*>( | 195 const char* instruction_pointer = reinterpret_cast<const char*>( |
| 196 profile_sample_frames[i][j][k].instruction_pointer); | 196 profile_sample_frames[i][j][k].instruction_pointer); |
| 197 const char* module_base_address = reinterpret_cast<const char*>( | 197 const char* module_base_address = reinterpret_cast<const char*>( |
| 198 profile_modules[i][profile_sample_frames[i][j][k].module_index] | 198 profile_modules[i][profile_sample_frames[i][j][k].module_index] |
| 199 .base_address); | 199 .base_address); |
| 200 EXPECT_EQ(static_cast<uint64>(instruction_pointer - | 200 EXPECT_EQ(static_cast<uint64>(instruction_pointer - |
| 201 module_base_address), entry.address()); | 201 module_base_address), entry.address()); |
| 202 ASSERT_TRUE(entry.has_module_id_index()); | 202 ASSERT_TRUE(entry.has_module_id_index()); |
| 203 EXPECT_EQ(profile_sample_frames[i][j][k].module_index, | 203 EXPECT_EQ(profile_sample_frames[i][j][k].module_index, |
| 204 entry.module_id_index()); | 204 static_cast<size_t>(entry.module_id_index())); |
| 205 } | 205 } |
| 206 } | 206 } |
| 207 | 207 |
| 208 ASSERT_EQ(static_cast<int>(arraysize(profile_modules[i])), | 208 ASSERT_EQ(static_cast<int>(arraysize(profile_modules[i])), |
| 209 call_stack_profile.module_id().size()); | 209 call_stack_profile.module_id().size()); |
| 210 for (size_t j = 0; j < arraysize(profile_modules[i]); ++j) { | 210 for (size_t j = 0; j < arraysize(profile_modules[i]); ++j) { |
| 211 SCOPED_TRACE("module " + base::IntToString(j)); | 211 SCOPED_TRACE("module " + base::IntToString(j)); |
| 212 const CallStackProfile::ModuleIdentifier& module_identifier = | 212 const CallStackProfile::ModuleIdentifier& module_identifier = |
| 213 call_stack_profile.module_id().Get(j); | 213 call_stack_profile.module_id().Get(j); |
| 214 ASSERT_TRUE(module_identifier.has_build_id()); | 214 ASSERT_TRUE(module_identifier.has_build_id()); |
| (...skipping 76 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 291 SCOPED_TRACE("frame " + base::IntToString(j)); | 291 SCOPED_TRACE("frame " + base::IntToString(j)); |
| 292 const CallStackProfile::Entry& entry = proto_sample.entry().Get(j); | 292 const CallStackProfile::Entry& entry = proto_sample.entry().Get(j); |
| 293 ASSERT_TRUE(entry.has_address()); | 293 ASSERT_TRUE(entry.has_address()); |
| 294 const char* instruction_pointer = reinterpret_cast<const char*>( | 294 const char* instruction_pointer = reinterpret_cast<const char*>( |
| 295 sample_frames[i][j].instruction_pointer); | 295 sample_frames[i][j].instruction_pointer); |
| 296 const char* module_base_address = reinterpret_cast<const char*>( | 296 const char* module_base_address = reinterpret_cast<const char*>( |
| 297 modules[sample_frames[i][j].module_index].base_address); | 297 modules[sample_frames[i][j].module_index].base_address); |
| 298 EXPECT_EQ(static_cast<uint64>(instruction_pointer - module_base_address), | 298 EXPECT_EQ(static_cast<uint64>(instruction_pointer - module_base_address), |
| 299 entry.address()); | 299 entry.address()); |
| 300 ASSERT_TRUE(entry.has_module_id_index()); | 300 ASSERT_TRUE(entry.has_module_id_index()); |
| 301 EXPECT_EQ(sample_frames[i][j].module_index, entry.module_id_index()); | 301 EXPECT_EQ(sample_frames[i][j].module_index, |
| 302 static_cast<size_t>(entry.module_id_index())); |
| 302 } | 303 } |
| 303 } | 304 } |
| 304 } | 305 } |
| 305 | 306 |
| 306 // Checks that only contiguous duplicate samples are collapsed with | 307 // Checks that only contiguous duplicate samples are collapsed with |
| 307 // preserve_sample_ordering = true. | 308 // preserve_sample_ordering = true. |
| 308 TEST(CallStackProfileMetricsProviderTest, RepeatedStacksOrdered) { | 309 TEST(CallStackProfileMetricsProviderTest, RepeatedStacksOrdered) { |
| 309 const uintptr_t module_base_address = 0x1000; | 310 const uintptr_t module_base_address = 0x1000; |
| 310 | 311 |
| 311 const Module modules[] = { | 312 const Module modules[] = { |
| (...skipping 55 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 367 SCOPED_TRACE("frame " + base::IntToString(j)); | 368 SCOPED_TRACE("frame " + base::IntToString(j)); |
| 368 const CallStackProfile::Entry& entry = proto_sample.entry().Get(j); | 369 const CallStackProfile::Entry& entry = proto_sample.entry().Get(j); |
| 369 ASSERT_TRUE(entry.has_address()); | 370 ASSERT_TRUE(entry.has_address()); |
| 370 const char* instruction_pointer = reinterpret_cast<const char*>( | 371 const char* instruction_pointer = reinterpret_cast<const char*>( |
| 371 sample_frames[i][j].instruction_pointer); | 372 sample_frames[i][j].instruction_pointer); |
| 372 const char* module_base_address = reinterpret_cast<const char*>( | 373 const char* module_base_address = reinterpret_cast<const char*>( |
| 373 modules[sample_frames[i][j].module_index].base_address); | 374 modules[sample_frames[i][j].module_index].base_address); |
| 374 EXPECT_EQ(static_cast<uint64>(instruction_pointer - module_base_address), | 375 EXPECT_EQ(static_cast<uint64>(instruction_pointer - module_base_address), |
| 375 entry.address()); | 376 entry.address()); |
| 376 ASSERT_TRUE(entry.has_module_id_index()); | 377 ASSERT_TRUE(entry.has_module_id_index()); |
| 377 EXPECT_EQ(sample_frames[i][j].module_index, entry.module_id_index()); | 378 EXPECT_EQ(sample_frames[i][j].module_index, |
| 379 static_cast<size_t>(entry.module_id_index())); |
| 378 } | 380 } |
| 379 } | 381 } |
| 380 } | 382 } |
| 381 | 383 |
| 382 | 384 |
| 383 // Checks that unknown modules produce an empty Entry. | 385 // Checks that unknown modules produce an empty Entry. |
| 384 TEST(CallStackProfileMetricsProviderTest, UnknownModule) { | 386 TEST(CallStackProfileMetricsProviderTest, UnknownModule) { |
| 385 // -1 indicates an unknown module. | 387 const Frame frame(reinterpret_cast<const void*>(0x1000), |
| 386 const Frame frame(reinterpret_cast<const void*>(0x1000), -1); | 388 Frame::kUnknownModuleIndex); |
| 387 | 389 |
| 388 Profile profile; | 390 Profile profile; |
| 389 | 391 |
| 390 profile.samples.push_back(Sample(1, frame)); | 392 profile.samples.push_back(Sample(1, frame)); |
| 391 | 393 |
| 392 profile.profile_duration = base::TimeDelta::FromMilliseconds(100); | 394 profile.profile_duration = base::TimeDelta::FromMilliseconds(100); |
| 393 profile.sampling_period = base::TimeDelta::FromMilliseconds(10); | 395 profile.sampling_period = base::TimeDelta::FromMilliseconds(10); |
| 394 profile.preserve_sample_ordering = false; | 396 profile.preserve_sample_ordering = false; |
| 395 | 397 |
| 396 CallStackProfileMetricsProvider provider; | 398 CallStackProfileMetricsProvider provider; |
| (...skipping 12 matching lines...) Expand all Loading... |
| 409 call_stack_profile.sample().Get(0); | 411 call_stack_profile.sample().Get(0); |
| 410 ASSERT_EQ(1, proto_sample.entry().size()); | 412 ASSERT_EQ(1, proto_sample.entry().size()); |
| 411 ASSERT_TRUE(proto_sample.has_count()); | 413 ASSERT_TRUE(proto_sample.has_count()); |
| 412 EXPECT_EQ(1u, proto_sample.count()); | 414 EXPECT_EQ(1u, proto_sample.count()); |
| 413 const CallStackProfile::Entry& entry = proto_sample.entry().Get(0); | 415 const CallStackProfile::Entry& entry = proto_sample.entry().Get(0); |
| 414 EXPECT_FALSE(entry.has_address()); | 416 EXPECT_FALSE(entry.has_address()); |
| 415 EXPECT_FALSE(entry.has_module_id_index()); | 417 EXPECT_FALSE(entry.has_module_id_index()); |
| 416 } | 418 } |
| 417 | 419 |
| 418 } // namespace metrics | 420 } // namespace metrics |
| OLD | NEW |