| Index: chrome/browser/performance_monitor/performance_monitor.cc
|
| diff --git a/chrome/browser/performance_monitor/performance_monitor.cc b/chrome/browser/performance_monitor/performance_monitor.cc
|
| index b800e7f902af12739662b6916f136d0d7605313b..55e5df146176ae939cab29a41640edf41e57062b 100644
|
| --- a/chrome/browser/performance_monitor/performance_monitor.cc
|
| +++ b/chrome/browser/performance_monitor/performance_monitor.cc
|
| @@ -10,6 +10,7 @@
|
| #include "base/bind.h"
|
| #include "base/command_line.h"
|
| #include "base/logging.h"
|
| +#include "base/memory/singleton.h"
|
| #include "base/process/process_iterator.h"
|
| #include "base/stl_util.h"
|
| #include "base/strings/string_number_conversions.h"
|
| @@ -29,9 +30,10 @@
|
| #include "chrome/common/chrome_version_info.h"
|
| #include "chrome/common/extensions/extension.h"
|
| #include "chrome/common/extensions/extension_constants.h"
|
| -#include "chrome/test/base/chrome_process_util.h"
|
| #include "content/public/browser/browser_child_process_host.h"
|
| +#include "content/public/browser/browser_child_process_host_iterator.h"
|
| #include "content/public/browser/browser_thread.h"
|
| +#include "content/public/browser/child_process_data.h"
|
| #include "content/public/browser/load_notification_details.h"
|
| #include "content/public/browser/notification_service.h"
|
| #include "content/public/browser/notification_types.h"
|
| @@ -53,8 +55,6 @@ const uint32 kAccessFlags = base::kProcessAccessDuplicateHandle |
|
| base::kProcessAccessTerminate |
|
| base::kProcessAccessWaitForTermination;
|
|
|
| -bool g_started_initialization = false;
|
| -
|
| std::string TimeToString(base::Time time) {
|
| int64 time_int64 = time.ToInternalValue();
|
| return base::Int64ToString(time_int64);
|
| @@ -99,7 +99,15 @@ PerformanceMonitor::PerformanceDataForIOThread::PerformanceDataForIOThread()
|
| : network_bytes_read(0) {
|
| }
|
|
|
| -PerformanceMonitor::PerformanceMonitor() : metrics_map_(new MetricsMap) {}
|
| +PerformanceMonitor::PerformanceMonitor()
|
| + : gather_interval_in_seconds_(kDefaultGatherIntervalInSeconds),
|
| + database_logging_enabled_(false),
|
| + timer_(FROM_HERE,
|
| + base::TimeDelta::FromSeconds(kSampleIntervalInSeconds),
|
| + this,
|
| + &PerformanceMonitor::DoTimedCollections),
|
| + disable_timer_autostart_for_testing_(false) {
|
| +}
|
|
|
| PerformanceMonitor::~PerformanceMonitor() {
|
| BrowserThread::PostBlockingPoolSequencedTask(
|
| @@ -126,11 +134,39 @@ PerformanceMonitor* PerformanceMonitor::GetInstance() {
|
| void PerformanceMonitor::Start() {
|
| CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
|
|
|
| - // Avoid responding to multiple calls to Start().
|
| - if (g_started_initialization)
|
| + static bool has_initialized = false;
|
| + if (has_initialized) {
|
| + NOTREACHED();
|
| return;
|
| + }
|
| + has_initialized = true;
|
| +
|
| + if (CommandLine::ForCurrentProcess()->HasSwitch(
|
| + switches::kPerformanceMonitorGathering)) {
|
| + database_logging_enabled_ = true;
|
| +
|
| + std::string switch_value = CommandLine::ForCurrentProcess()->
|
| + GetSwitchValueASCII(switches::kPerformanceMonitorGathering);
|
| +
|
| + if (!switch_value.empty()) {
|
| + int specified_interval = 0;
|
| + if (!base::StringToInt(switch_value, &specified_interval) ||
|
| + specified_interval <= 0) {
|
| + LOG(ERROR) << "Invalid value for switch: '"
|
| + << switches::kPerformanceMonitorGathering
|
| + << "'; please use an integer greater than 0.";
|
| + } else {
|
| + gather_interval_in_seconds_ = std::max(specified_interval,
|
| + kSampleIntervalInSeconds);
|
| + }
|
| + }
|
| + }
|
| +
|
| + DCHECK(gather_interval_in_seconds_ >= kSampleIntervalInSeconds);
|
| +
|
| + next_collection_time_ = base::Time::Now() +
|
| + base::TimeDelta::FromSeconds(gather_interval_in_seconds_);
|
|
|
| - g_started_initialization = true;
|
| util::PostTaskToDatabaseThreadAndReply(
|
| FROM_HERE,
|
| base::Bind(&PerformanceMonitor::InitOnBackgroundThread,
|
| @@ -141,59 +177,42 @@ void PerformanceMonitor::Start() {
|
|
|
| void PerformanceMonitor::InitOnBackgroundThread() {
|
| CHECK(!BrowserThread::CurrentlyOn(BrowserThread::UI));
|
| - database_ = Database::Create(database_path_);
|
| - if (!database_) {
|
| - LOG(ERROR) << "Could not initialize database; aborting initialization.";
|
| - return;
|
| - }
|
|
|
| - // Initialize the io thread's performance data to the value in the database;
|
| - // if there isn't a recording in the database, the value stays at 0.
|
| - Metric metric;
|
| - if (database_->GetRecentStatsForActivityAndMetric(METRIC_NETWORK_BYTES_READ,
|
| - &metric)) {
|
| - performance_data_for_io_thread_.network_bytes_read = metric.value;
|
| + if (database_logging_enabled_) {
|
| + database_ = Database::Create(database_path_);
|
| + if (!database_) {
|
| + LOG(ERROR) << "Could not initialize database; aborting initialization.";
|
| + database_logging_enabled_ = false;
|
| + return;
|
| + }
|
| +
|
| + // Initialize the io thread's performance data to the value in the database;
|
| + // if there isn't a recording in the database, the value stays at 0.
|
| + Metric metric;
|
| + if (database_->GetRecentStatsForActivityAndMetric(METRIC_NETWORK_BYTES_READ,
|
| + &metric)) {
|
| + performance_data_for_io_thread_.network_bytes_read = metric.value;
|
| + }
|
| }
|
| }
|
|
|
| void PerformanceMonitor::FinishInit() {
|
| CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
|
| - // Short-circuit the initialization process if the database wasn't properly
|
| - // created. This will prevent PerformanceMonitor from performing any actions,
|
| - // including observing events.
|
| - if (!database_)
|
| - return;
|
| -
|
| - RegisterForNotifications();
|
| - CheckForUncleanExits();
|
| - BrowserThread::PostBlockingPoolSequencedTask(
|
| - Database::kDatabaseSequenceToken,
|
| - FROM_HERE,
|
| - base::Bind(&PerformanceMonitor::CheckForVersionUpdateOnBackgroundThread,
|
| - base::Unretained(this)));
|
|
|
| - int gather_interval_in_seconds = kDefaultGatherIntervalInSeconds;
|
| - if (CommandLine::ForCurrentProcess()->HasSwitch(
|
| - switches::kPerformanceMonitorGathering) &&
|
| - !CommandLine::ForCurrentProcess()->GetSwitchValueASCII(
|
| - switches::kPerformanceMonitorGathering).empty()) {
|
| - int specified_interval = 0;
|
| - if (!base::StringToInt(
|
| - CommandLine::ForCurrentProcess()->GetSwitchValueASCII(
|
| - switches::kPerformanceMonitorGathering),
|
| - &specified_interval) || specified_interval <= 0) {
|
| - LOG(ERROR) << "Invalid value for switch: '"
|
| - << switches::kPerformanceMonitorGathering
|
| - << "'; please use an integer greater than 0.";
|
| - } else {
|
| - gather_interval_in_seconds = specified_interval;
|
| - }
|
| + // Events and notifications are only useful if we're logging to the database.
|
| + if (database_logging_enabled_) {
|
| + RegisterForNotifications();
|
| + CheckForUncleanExits();
|
| + BrowserThread::PostBlockingPoolSequencedTask(
|
| + Database::kDatabaseSequenceToken,
|
| + FROM_HERE,
|
| + base::Bind(&PerformanceMonitor::CheckForVersionUpdateOnBackgroundThread,
|
| + base::Unretained(this)));
|
| }
|
|
|
| - timer_.Start(FROM_HERE,
|
| - base::TimeDelta::FromSeconds(gather_interval_in_seconds),
|
| - this,
|
| - &PerformanceMonitor::DoTimedCollections);
|
| + // Start our periodic gathering of metrics.
|
| + if (!disable_timer_autostart_for_testing_)
|
| + timer_.Reset();
|
|
|
| // Post a task to the background thread to a function which does nothing.
|
| // This will force any tasks the database is performing to finish prior to
|
| @@ -210,6 +229,8 @@ void PerformanceMonitor::FinishInit() {
|
| }
|
|
|
| void PerformanceMonitor::RegisterForNotifications() {
|
| + DCHECK(database_logging_enabled_);
|
| +
|
| // Extensions
|
| registrar_.Add(this, chrome::NOTIFICATION_EXTENSION_INSTALLED,
|
| content::NotificationService::AllSources());
|
| @@ -241,6 +262,8 @@ void PerformanceMonitor::RegisterForNotifications() {
|
| // loaded prior to PerformanceMonitor's initialization. Later profiles will be
|
| // checked through the PROFILE_ADDED notification.
|
| void PerformanceMonitor::CheckForUncleanExits() {
|
| + DCHECK(database_logging_enabled_);
|
| +
|
| std::vector<Profile*> profiles =
|
| g_browser_process->profile_manager()->GetLoadedProfiles();
|
|
|
| @@ -258,7 +281,8 @@ void PerformanceMonitor::CheckForUncleanExits() {
|
| }
|
|
|
| void PerformanceMonitor::AddUncleanExitEventOnBackgroundThread(
|
| - const std::string& profile_name) {
|
| + const std::string& profile_name) {
|
| + DCHECK(database_logging_enabled_);
|
| std::string database_key = kStateProfilePrefix + profile_name;
|
| std::string last_active_string = database_->GetStateValue(database_key);
|
|
|
| @@ -279,6 +303,7 @@ void PerformanceMonitor::AddUncleanExitEventOnBackgroundThread(
|
|
|
| void PerformanceMonitor::CheckForVersionUpdateOnBackgroundThread() {
|
| CHECK(!BrowserThread::CurrentlyOn(BrowserThread::UI));
|
| + DCHECK(database_logging_enabled_);
|
|
|
| chrome::VersionInfo version;
|
| DCHECK(version.is_valid());
|
| @@ -305,6 +330,8 @@ void PerformanceMonitor::CheckForVersionUpdateOnBackgroundThread() {
|
| }
|
|
|
| void PerformanceMonitor::AddEvent(scoped_ptr<Event> event) {
|
| + DCHECK(database_logging_enabled_);
|
| +
|
| BrowserThread::PostBlockingPoolSequencedTask(
|
| Database::kDatabaseSequenceToken,
|
| FROM_HERE,
|
| @@ -319,6 +346,8 @@ void PerformanceMonitor::AddEventOnBackgroundThread(scoped_ptr<Event> event) {
|
|
|
| void PerformanceMonitor::AddMetricOnBackgroundThread(const Metric& metric) {
|
| DCHECK(!BrowserThread::CurrentlyOn(BrowserThread::UI));
|
| + DCHECK(database_logging_enabled_);
|
| +
|
| database_->AddMetric(metric);
|
| }
|
|
|
| @@ -331,88 +360,62 @@ void PerformanceMonitor::NotifyInitialized() {
|
| initialized_ = true;
|
| }
|
|
|
| -void PerformanceMonitor::GatherStatisticsOnBackgroundThread() {
|
| - CHECK(!BrowserThread::CurrentlyOn(BrowserThread::UI));
|
| -
|
| - // Because the CPU usage is gathered as an average since the last time the
|
| - // function was called, while the memory usage is gathered as an instantaneous
|
| - // usage, the CPU usage is gathered before the metrics map is wiped.
|
| - GatherCPUUsageOnBackgroundThread();
|
| - UpdateMetricsMapOnBackgroundThread();
|
| - GatherMemoryUsageOnBackgroundThread();
|
| -}
|
| -
|
| -void PerformanceMonitor::GatherCPUUsageOnBackgroundThread() {
|
| - if (metrics_map_->size()) {
|
| - double cpu_usage = 0;
|
| - for (MetricsMap::const_iterator iter = metrics_map_->begin();
|
| - iter != metrics_map_->end(); ++iter) {
|
| - cpu_usage += iter->second->GetCPUUsage();
|
| - }
|
| +void PerformanceMonitor::DoTimedCollections() {
|
| +#if !defined(OS_ANDROID)
|
| + // The profile list is only useful for the logged events.
|
| + if (database_logging_enabled_)
|
| + UpdateLiveProfiles();
|
| +#endif
|
|
|
| - database_->AddMetric(Metric(METRIC_CPU_USAGE,
|
| - base::Time::Now(),
|
| - cpu_usage));
|
| - }
|
| + GatherMetricsMapOnUIThread();
|
| }
|
|
|
| -void PerformanceMonitor::GatherMemoryUsageOnBackgroundThread() {
|
| - size_t private_memory_sum = 0;
|
| - size_t shared_memory_sum = 0;
|
| - for (MetricsMap::const_iterator iter = metrics_map_->begin();
|
| - iter != metrics_map_->end(); ++iter) {
|
| - size_t private_memory = 0;
|
| - size_t shared_memory = 0;
|
| - if (iter->second->GetMemoryBytes(&private_memory, &shared_memory)) {
|
| - private_memory_sum += private_memory;
|
| - shared_memory_sum += shared_memory;
|
| - } else {
|
| - LOG(WARNING) << "GetMemoryBytes returned NULL (platform-specific error)";
|
| - }
|
| +void PerformanceMonitor::GatherMetricsMapOnUIThread() {
|
| + CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
|
| + static int current_update_sequence = 0;
|
| + // Even in the "somewhat" unlikely event this wraps around,
|
| + // it doesn't matter. We just check it for inequality.
|
| + current_update_sequence++;
|
| +
|
| + // Find all render child processes; has to be done on the UI thread.
|
| + for (content::RenderProcessHost::iterator rph_iter =
|
| + content::RenderProcessHost::AllHostsIterator();
|
| + !rph_iter.IsAtEnd(); rph_iter.Advance()) {
|
| + base::ProcessHandle handle = rph_iter.GetCurrentValue()->GetHandle();
|
| + MarkProcessAsAlive(handle, content::PROCESS_TYPE_RENDERER,
|
| + current_update_sequence);
|
| }
|
|
|
| - database_->AddMetric(Metric(METRIC_PRIVATE_MEMORY_USAGE,
|
| - base::Time::Now(),
|
| - static_cast<double>(private_memory_sum)));
|
| - database_->AddMetric(Metric(METRIC_SHARED_MEMORY_USAGE,
|
| - base::Time::Now(),
|
| - static_cast<double>(shared_memory_sum)));
|
| -}
|
| -
|
| -void PerformanceMonitor::UpdateMetricsMapOnBackgroundThread() {
|
| - MetricsMap* new_map = new MetricsMap;
|
| -
|
| - base::ProcessId browser_pid = base::GetCurrentProcId();
|
| - ChromeProcessList chrome_processes(GetRunningChromeProcesses(browser_pid));
|
| - for (ChromeProcessList::const_iterator pid_iter = chrome_processes.begin();
|
| - pid_iter != chrome_processes.end(); ++pid_iter) {
|
| - base::ProcessHandle handle;
|
| - if (base::OpenProcessHandleWithAccess(*pid_iter, kAccessFlags, &handle)) {
|
| - // If we were already watching this process, transfer it to the new map.
|
| - if (ContainsKey(*metrics_map_, handle)) {
|
| - (*new_map)[handle] = (*metrics_map_)[handle];
|
| - continue;
|
| - }
|
| -
|
| - // Otherwise, gather information and prime the CPU usage to be gathered.
|
| -#if defined (OS_MACOSX)
|
| - linked_ptr<base::ProcessMetrics> process_metrics(
|
| - base::ProcessMetrics::CreateProcessMetrics(
|
| - handle, content::BrowserChildProcessHost::GetPortProvider()));
|
| -#else
|
| - linked_ptr<base::ProcessMetrics> process_metrics(
|
| - base::ProcessMetrics::CreateProcessMetrics(handle));
|
| -#endif
|
| + BrowserThread::PostTask(
|
| + BrowserThread::IO,
|
| + FROM_HERE,
|
| + base::Bind(&PerformanceMonitor::GatherMetricsMapOnIOThread,
|
| + base::Unretained(this),
|
| + current_update_sequence));
|
| +}
|
|
|
| - process_metrics->GetCPUUsage();
|
| +void PerformanceMonitor::MarkProcessAsAlive(const base::ProcessHandle& handle,
|
| + int process_type,
|
| + int current_update_sequence) {
|
|
|
| - (*new_map)[handle] = process_metrics;
|
| - }
|
| + if (handle == 0) {
|
| + // Process may not be valid yet.
|
| + return;
|
| }
|
|
|
| - metrics_map_.reset(new_map);
|
| + MetricsMap::iterator process_metrics_iter = metrics_map_.find(handle);
|
| + if (process_metrics_iter == metrics_map_.end()) {
|
| + // If we're not already watching the process, let's initialize it.
|
| + metrics_map_[handle]
|
| + .Initialize(handle, process_type, current_update_sequence);
|
| + } else {
|
| + // If we are watching the process, touch it to keep it alive.
|
| + ProcessMetricsHistory& process_metrics = process_metrics_iter->second;
|
| + process_metrics.set_last_update_sequence(current_update_sequence);
|
| + }
|
| }
|
|
|
| +#if !defined(OS_ANDROID)
|
| void PerformanceMonitor::UpdateLiveProfiles() {
|
| std::string time = TimeToString(base::Time::Now());
|
| scoped_ptr<std::set<std::string> > active_profiles(
|
| @@ -434,48 +437,119 @@ void PerformanceMonitor::UpdateLiveProfilesHelper(
|
| scoped_ptr<std::set<std::string> > active_profiles,
|
| std::string time) {
|
| CHECK(!BrowserThread::CurrentlyOn(BrowserThread::UI));
|
| + DCHECK(database_logging_enabled_);
|
|
|
| for (std::set<std::string>::const_iterator iter = active_profiles->begin();
|
| iter != active_profiles->end(); ++iter) {
|
| database_->AddStateValue(kStateProfilePrefix + *iter, time);
|
| }
|
| }
|
| +#endif
|
|
|
| -void PerformanceMonitor::DoTimedCollections() {
|
| - UpdateLiveProfiles();
|
| -
|
| - BrowserThread::PostBlockingPoolSequencedTask(
|
| - Database::kDatabaseSequenceToken,
|
| - FROM_HERE,
|
| - base::Bind(&PerformanceMonitor::GatherStatisticsOnBackgroundThread,
|
| - base::Unretained(this)));
|
| +void PerformanceMonitor::GatherMetricsMapOnIOThread(
|
| + int current_update_sequence) {
|
| + CHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
|
|
|
| - BrowserThread::PostTask(
|
| - BrowserThread::IO,
|
| - FROM_HERE,
|
| - base::Bind(&PerformanceMonitor::CallInsertIOData,
|
| - base::Unretained(this)));
|
| -}
|
| + // Find all child processes (does not include renderers), which has to be
|
| + // done on the IO thread.
|
| + for (content::BrowserChildProcessHostIterator iter; !iter.Done(); ++iter) {
|
| + const content::ChildProcessData& child_process_data = iter.GetData();
|
| + base::ProcessHandle handle = child_process_data.handle;
|
| + MarkProcessAsAlive(handle, child_process_data.process_type,
|
| + current_update_sequence);
|
| + }
|
|
|
| -void PerformanceMonitor::CallInsertIOData() {
|
| - CHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
|
| + // Add the current (browser) process.
|
| + MarkProcessAsAlive(base::GetCurrentProcessHandle(),
|
| + content::PROCESS_TYPE_BROWSER, current_update_sequence);
|
|
|
| BrowserThread::PostBlockingPoolSequencedTask(
|
| Database::kDatabaseSequenceToken,
|
| FROM_HERE,
|
| - base::Bind(&PerformanceMonitor::InsertIOData,
|
| - base::Unretained(this),
|
| + base::Bind(&PerformanceMonitor::StoreMetricsOnBackgroundThread,
|
| + base::Unretained(this), current_update_sequence,
|
| performance_data_for_io_thread_));
|
| }
|
|
|
| -void PerformanceMonitor::InsertIOData(
|
| +void PerformanceMonitor::StoreMetricsOnBackgroundThread(
|
| + int current_update_sequence,
|
| const PerformanceDataForIOThread& performance_data_for_io_thread) {
|
| CHECK(!BrowserThread::CurrentlyOn(BrowserThread::UI));
|
| - database_->AddMetric(
|
| - Metric(METRIC_NETWORK_BYTES_READ,
|
| - base::Time::Now(),
|
| - static_cast<double>(
|
| - performance_data_for_io_thread.network_bytes_read)));
|
| +
|
| + base::Time time_now = base::Time::Now();
|
| +
|
| + // The timing can be off by kSampleIntervalInSeconds during any one particular
|
| + // run, but will be correct over time.
|
| + bool end_of_cycle = time_now >= next_collection_time_;
|
| + if (end_of_cycle) {
|
| + next_collection_time_ +=
|
| + base::TimeDelta::FromSeconds(gather_interval_in_seconds_);
|
| + }
|
| +
|
| + double cpu_usage = 0.0;
|
| + size_t private_memory_sum = 0;
|
| + size_t shared_memory_sum = 0;
|
| +
|
| + // Update metrics for all watched processes; remove dead entries from the map.
|
| + MetricsMap::iterator iter = metrics_map_.begin();
|
| + while (iter != metrics_map_.end()) {
|
| + ProcessMetricsHistory& process_metrics = iter->second;
|
| + if (process_metrics.last_update_sequence() != current_update_sequence) {
|
| + // Not touched this iteration; let's get rid of it.
|
| + metrics_map_.erase(iter++);
|
| + } else {
|
| + process_metrics.SampleMetrics();
|
| +
|
| + if (end_of_cycle) {
|
| + // Gather averages of previously sampled metrics.
|
| + cpu_usage += process_metrics.GetAverageCPUUsage();
|
| +
|
| + size_t private_memory = 0;
|
| + size_t shared_memory = 0;
|
| + process_metrics.GetAverageMemoryBytes(&private_memory, &shared_memory);
|
| + private_memory_sum += private_memory;
|
| + shared_memory_sum += shared_memory;
|
| +
|
| + process_metrics.EndOfCycle();
|
| + }
|
| +
|
| + ++iter;
|
| + }
|
| + }
|
| +
|
| + // Store previously-sampled metrics.
|
| + if (end_of_cycle && database_logging_enabled_) {
|
| + if (!metrics_map_.empty()) {
|
| + database_->AddMetric(Metric(METRIC_CPU_USAGE, time_now, cpu_usage));
|
| + database_->AddMetric(Metric(METRIC_PRIVATE_MEMORY_USAGE,
|
| + time_now,
|
| + static_cast<double>(private_memory_sum)));
|
| + database_->AddMetric(Metric(METRIC_SHARED_MEMORY_USAGE,
|
| + time_now,
|
| + static_cast<double>(shared_memory_sum)));
|
| + }
|
| +
|
| + database_->AddMetric(
|
| + Metric(METRIC_NETWORK_BYTES_READ,
|
| + time_now,
|
| + static_cast<double>(
|
| + performance_data_for_io_thread.network_bytes_read)));
|
| + }
|
| +
|
| + BrowserThread::PostTask(
|
| + BrowserThread::UI,
|
| + FROM_HERE,
|
| + base::Bind(&PerformanceMonitor::ResetMetricsTimerOnUIThread,
|
| + base::Unretained(this)));
|
| +}
|
| +
|
| +void PerformanceMonitor::ResetMetricsTimerOnUIThread() {
|
| + CHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
|
| +
|
| + // Start up the timer again; we avoid the use of a repeating timer
|
| + // to not have concurrent metrics gathering running.
|
| + if (!disable_timer_autostart_for_testing_)
|
| + timer_.Reset();
|
| }
|
|
|
| void PerformanceMonitor::BytesReadOnIOThread(const net::URLRequest& request,
|
| @@ -489,6 +563,8 @@ void PerformanceMonitor::BytesReadOnIOThread(const net::URLRequest& request,
|
| void PerformanceMonitor::Observe(int type,
|
| const content::NotificationSource& source,
|
| const content::NotificationDetails& details) {
|
| + DCHECK(database_logging_enabled_);
|
| +
|
| switch (type) {
|
| case chrome::NOTIFICATION_EXTENSION_INSTALLED: {
|
| AddExtensionEvent(
|
|
|