gpu_data_manager_impl_private.cc revision e5d81f57cb97b3b6b7fccc9c5610d21eb81db09d
1a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)// Copyright (c) 2013 The Chromium Authors. All rights reserved.
2a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)// Use of this source code is governed by a BSD-style license that can be
3a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)// found in the LICENSE file.
4a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
5a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "content/browser/gpu/gpu_data_manager_impl_private.h"
6a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
7a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "base/bind.h"
8a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "base/bind_helpers.h"
9a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "base/command_line.h"
10a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "base/debug/trace_event.h"
11a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "base/metrics/field_trial.h"
12a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "base/metrics/histogram.h"
133240926e260ce088908e02ac07a6cf7b0c0cbf44Ben Murdoch#include "base/metrics/sparse_histogram.h"
14a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "base/strings/string_number_conversions.h"
15868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)#include "base/strings/stringprintf.h"
16a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "base/sys_info.h"
17a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "base/version.h"
1890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#include "cc/base/switches.h"
19a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "content/browser/gpu/gpu_process_host.h"
20a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "content/common/gpu/gpu_messages.h"
21a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "content/public/browser/browser_thread.h"
22a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "content/public/browser/gpu_data_manager_observer.h"
23a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "content/public/common/content_client.h"
24a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "content/public/common/content_constants.h"
25a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "content/public/common/content_switches.h"
26a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "gpu/command_buffer/service/gpu_switches.h"
2790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#include "gpu/config/gpu_control_list_jsons.h"
28868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)#include "gpu/config/gpu_driver_bug_workaround_type.h"
2990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#include "gpu/config/gpu_feature_type.h"
3090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#include "gpu/config/gpu_info_collector.h"
3190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#include "gpu/config/gpu_util.h"
32a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "ui/base/ui_base_switches.h"
33a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "ui/gl/gl_implementation.h"
34a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "ui/gl/gl_switches.h"
35a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "ui/gl/gpu_switching_manager.h"
36868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)#include "webkit/common/webpreferences.h"
37a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
3890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#if defined(OS_MACOSX)
3990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#include <ApplicationServices/ApplicationServices.h>
4090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#endif  // OS_MACOSX
41a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#if defined(OS_WIN)
42a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#include "base/win/windows_version.h"
4390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#endif  // OS_WIN
4490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#if defined(OS_ANDROID)
4590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#include "ui/gfx/android/device_display_info.h"
4690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#endif  // OS_ANDROID
47a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
48a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)namespace content {
4990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)
50a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)namespace {
51a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
5290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)enum GpuFeatureStatus {
5390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    kGpuFeatureEnabled = 0,
5490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    kGpuFeatureBlacklisted = 1,
5590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    kGpuFeatureDisabled = 2,  // disabled by user but not blacklisted
5690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    kGpuFeatureNumStatus
5790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)};
5890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)
5990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#if defined(OS_WIN)
6090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)
6190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)enum WinSubVersion {
6290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  kWinOthers = 0,
6390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  kWinXP,
6490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  kWinVista,
6590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  kWin7,
667dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  kWin8,
6790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  kNumWinSubVersions
6890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)};
6990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)
7090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)int GetGpuBlacklistHistogramValueWin(GpuFeatureStatus status) {
7190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  static WinSubVersion sub_version = kNumWinSubVersions;
7290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  if (sub_version == kNumWinSubVersions) {
7390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    sub_version = kWinOthers;
7490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    std::string version_str = base::SysInfo::OperatingSystemVersion();
7590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    size_t pos = version_str.find_first_not_of("0123456789.");
7690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    if (pos != std::string::npos)
7790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      version_str = version_str.substr(0, pos);
7890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    Version os_version(version_str);
7990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    if (os_version.IsValid() && os_version.components().size() >= 2) {
8090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      const std::vector<uint16>& version_numbers = os_version.components();
8190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      if (version_numbers[0] == 5)
8290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)        sub_version = kWinXP;
8390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      else if (version_numbers[0] == 6 && version_numbers[1] == 0)
8490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)        sub_version = kWinVista;
8590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      else if (version_numbers[0] == 6 && version_numbers[1] == 1)
8690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)        sub_version = kWin7;
877dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      else if (version_numbers[0] == 6 && version_numbers[1] == 2)
887dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch        sub_version = kWin8;
8990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    }
9090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  }
9190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  int entry_index = static_cast<int>(sub_version) * kGpuFeatureNumStatus;
9290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  switch (status) {
9390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    case kGpuFeatureEnabled:
9490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      break;
9590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    case kGpuFeatureBlacklisted:
9690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      entry_index++;
9790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      break;
9890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    case kGpuFeatureDisabled:
9990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      entry_index += 2;
10090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      break;
10190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  }
10290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  return entry_index;
10390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)}
10490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#endif  // OS_WIN
10590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)
1063240926e260ce088908e02ac07a6cf7b0c0cbf44Ben Murdoch// Send UMA histograms about the enabled features and GPU properties.
1073240926e260ce088908e02ac07a6cf7b0c0cbf44Ben Murdochvoid UpdateStats(const gpu::GPUInfo& gpu_info,
1083240926e260ce088908e02ac07a6cf7b0c0cbf44Ben Murdoch                 const gpu::GpuBlacklist* blacklist,
10990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)                 const std::set<int>& blacklisted_features) {
11090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  uint32 max_entry_id = blacklist->max_entry_id();
11190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  if (max_entry_id == 0) {
11290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    // GPU Blacklist was not loaded.  No need to go further.
11390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    return;
11490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  }
11590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)
11690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  const CommandLine& command_line = *CommandLine::ForCurrentProcess();
11790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  bool disabled = false;
1187dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch
1197dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  // Use entry 0 to capture the total number of times that data
1207dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  // was recorded in this histogram in order to have a convenient
1217dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  // denominator to compute blacklist percentages for the rest of the
1227dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  // entries.
1237dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  UMA_HISTOGRAM_ENUMERATION("GPU.BlacklistTestResultsPerEntry",
1247dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      0, max_entry_id + 1);
1257dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch
1267dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  if (blacklisted_features.size() != 0) {
12790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    std::vector<uint32> flag_entries;
12890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    blacklist->GetDecisionEntries(&flag_entries, disabled);
12990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    DCHECK_GT(flag_entries.size(), 0u);
13090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    for (size_t i = 0; i < flag_entries.size(); ++i) {
13190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      UMA_HISTOGRAM_ENUMERATION("GPU.BlacklistTestResultsPerEntry",
13290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)          flag_entries[i], max_entry_id + 1);
13390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    }
13490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  }
13590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)
13690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  // This counts how many users are affected by a disabled entry - this allows
13790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  // us to understand the impact of an entry before enable it.
13890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  std::vector<uint32> flag_disabled_entries;
13990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  disabled = true;
14090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  blacklist->GetDecisionEntries(&flag_disabled_entries, disabled);
14190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  for (size_t i = 0; i < flag_disabled_entries.size(); ++i) {
14290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    UMA_HISTOGRAM_ENUMERATION("GPU.BlacklistTestResultsPerDisabledEntry",
14390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)        flag_disabled_entries[i], max_entry_id + 1);
14490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  }
14590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)
14690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  const gpu::GpuFeatureType kGpuFeatures[] = {
14790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      gpu::GPU_FEATURE_TYPE_ACCELERATED_2D_CANVAS,
14890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      gpu::GPU_FEATURE_TYPE_ACCELERATED_COMPOSITING,
1495d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      gpu::GPU_FEATURE_TYPE_WEBGL
15090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  };
15190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  const std::string kGpuBlacklistFeatureHistogramNames[] = {
15290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      "GPU.BlacklistFeatureTestResults.Accelerated2dCanvas",
15390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      "GPU.BlacklistFeatureTestResults.AcceleratedCompositing",
1547dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      "GPU.BlacklistFeatureTestResults.Webgl",
15590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  };
15690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  const bool kGpuFeatureUserFlags[] = {
15790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      command_line.HasSwitch(switches::kDisableAccelerated2dCanvas),
15890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      command_line.HasSwitch(switches::kDisableAcceleratedCompositing),
1597dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      command_line.HasSwitch(switches::kDisableExperimentalWebGL),
16090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  };
16190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#if defined(OS_WIN)
16290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  const std::string kGpuBlacklistFeatureHistogramNamesWin[] = {
16390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      "GPU.BlacklistFeatureTestResultsWindows.Accelerated2dCanvas",
16490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      "GPU.BlacklistFeatureTestResultsWindows.AcceleratedCompositing",
1657dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      "GPU.BlacklistFeatureTestResultsWindows.Webgl",
16690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  };
16790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#endif
16890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  const size_t kNumFeatures =
16990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      sizeof(kGpuFeatures) / sizeof(gpu::GpuFeatureType);
17090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  for (size_t i = 0; i < kNumFeatures; ++i) {
17190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    // We can't use UMA_HISTOGRAM_ENUMERATION here because the same name is
17290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    // expected if the macro is used within a loop.
17390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    GpuFeatureStatus value = kGpuFeatureEnabled;
17490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    if (blacklisted_features.count(kGpuFeatures[i]))
17590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      value = kGpuFeatureBlacklisted;
17690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    else if (kGpuFeatureUserFlags[i])
17790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      value = kGpuFeatureDisabled;
17890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    base::HistogramBase* histogram_pointer = base::LinearHistogram::FactoryGet(
17990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)        kGpuBlacklistFeatureHistogramNames[i],
18090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)        1, kGpuFeatureNumStatus, kGpuFeatureNumStatus + 1,
18190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)        base::HistogramBase::kUmaTargetedHistogramFlag);
18290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    histogram_pointer->Add(value);
18390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#if defined(OS_WIN)
18490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    histogram_pointer = base::LinearHistogram::FactoryGet(
18590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)        kGpuBlacklistFeatureHistogramNamesWin[i],
18690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)        1, kNumWinSubVersions * kGpuFeatureNumStatus,
18790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)        kNumWinSubVersions * kGpuFeatureNumStatus + 1,
18890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)        base::HistogramBase::kUmaTargetedHistogramFlag);
18990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    histogram_pointer->Add(GetGpuBlacklistHistogramValueWin(value));
19090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#endif
19190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  }
1923240926e260ce088908e02ac07a6cf7b0c0cbf44Ben Murdoch
1933240926e260ce088908e02ac07a6cf7b0c0cbf44Ben Murdoch  UMA_HISTOGRAM_SPARSE_SLOWLY("GPU.GLResetNotificationStrategy",
1943240926e260ce088908e02ac07a6cf7b0c0cbf44Ben Murdoch      gpu_info.gl_reset_notification_strategy);
19590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)}
19690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)
197a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)// Combine the integers into a string, seperated by ','.
198a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)std::string IntSetToString(const std::set<int>& list) {
199a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  std::string rt;
200a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  for (std::set<int>::const_iterator it = list.begin();
201a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)       it != list.end(); ++it) {
202a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    if (!rt.empty())
203a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      rt += ",";
204a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    rt += base::IntToString(*it);
205a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
206a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  return rt;
207a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
208a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
209a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#if defined(OS_MACOSX)
210a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void DisplayReconfigCallback(CGDirectDisplayID display,
211a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                             CGDisplayChangeSummaryFlags flags,
212a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                             void* gpu_data_manager) {
213424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)  if (flags == kCGDisplayBeginConfigurationFlag)
214868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)    return; // This call contains no information about the display change
215868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)
216868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  GpuDataManagerImpl* manager =
217868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)      reinterpret_cast<GpuDataManagerImpl*>(gpu_data_manager);
218868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  DCHECK(manager);
219868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)
220868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  uint32_t displayCount;
221868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  CGGetActiveDisplayList(0, NULL, &displayCount);
222868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)
223868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  bool fireGpuSwitch = flags & kCGDisplayAddFlag;
224868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)
225868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  if (displayCount != manager->GetDisplayCount()) {
226868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)    manager->SetDisplayCount(displayCount);
227868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)    fireGpuSwitch = true;
228a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
229868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)
230868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  if (fireGpuSwitch)
231868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)    manager->HandleGpuSwitch();
232a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
233a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#endif  // OS_MACOSX
234a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
23590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#if defined(OS_ANDROID)
23690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)void ApplyAndroidWorkarounds(const gpu::GPUInfo& gpu_info,
23790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)                             CommandLine* command_line) {
23890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  std::string vendor(StringToLowerASCII(gpu_info.gl_vendor));
23990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  std::string renderer(StringToLowerASCII(gpu_info.gl_renderer));
24090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  bool is_img =
24190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      gpu_info.gl_vendor.find("Imagination") != std::string::npos;
24290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)
24390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  gfx::DeviceDisplayInfo info;
24490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  int default_tile_size = 256;
24590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)
2465d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  // TODO(epenner): Now that this is somewhat generic, maybe we can
2475d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  // unify this for all platforms (http://crbug.com/159524)
2485d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
2495d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  bool real_size_supported = true;
2505d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  int display_width = info.GetPhysicalDisplayWidth();
2515d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  int display_height = info.GetPhysicalDisplayHeight();
2525d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  if (display_width == 0 || display_height == 0) {
2535d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    real_size_supported = false;
2545d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    display_width = info.GetDisplayWidth();
2555d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    display_height = info.GetDisplayHeight();
2565d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
2575d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
2585d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  int portrait_width = std::min(display_width, display_height);
2595d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  int landscape_width = std::max(display_width, display_height);
2605d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
2615d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  if (real_size_supported) {
2625d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    // Maximum HD dimensions should be 768x1280
2635d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    // Maximum FHD dimensions should be 1200x1920
2645d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    if (portrait_width > 768 || landscape_width > 1280)
2655d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)       default_tile_size = 384;
2665d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    if (portrait_width > 1200 || landscape_width > 1920)
2675d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)       default_tile_size = 512;
2685d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)
2695d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    // Adjust for some resolutions that barely straddle an extra
2705d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    // tile when in portrait mode. This helps worst case scroll/raster
2715d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    // by not needing a full extra tile for each row.
2725d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    if (default_tile_size == 256 && portrait_width == 768)
2735d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      default_tile_size += 32;
2745d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    if (default_tile_size == 384 && portrait_width == 1200)
2755d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      default_tile_size += 32;
2765d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  } else {
2775d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    // We don't know the exact resolution due to screen controls etc.
2785d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    // So this just estimates the values above using tile counts.
2795d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    int numTiles = (display_width * display_height) / (256 * 256);
2805d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    if (numTiles > 16)
2815d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      default_tile_size = 384;
2825d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    if (numTiles >= 40)
2835d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)      default_tile_size = 512;
2845d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
28590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)
28690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  // IMG: Fast async texture uploads only work with non-power-of-two,
28790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  // but still multiple-of-eight sizes.
28890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  // http://crbug.com/168099
28990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  if (is_img)
29090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    default_tile_size -= 8;
29190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)
29290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  // Set the command line if it isn't already set and we changed
29390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  // the default tile size.
29490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  if (default_tile_size != 256 &&
29590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      !command_line->HasSwitch(switches::kDefaultTileWidth) &&
29690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      !command_line->HasSwitch(switches::kDefaultTileHeight)) {
29790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    std::stringstream size;
29890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    size << default_tile_size;
29990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    command_line->AppendSwitchASCII(
30090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)        switches::kDefaultTileWidth, size.str());
30190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    command_line->AppendSwitchASCII(
30290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)        switches::kDefaultTileHeight, size.str());
30390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  }
30490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)}
30590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#endif  // OS_ANDROID
30690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)
307a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)// Block all domains' use of 3D APIs for this many milliseconds if
308a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)// approaching a threshold where system stability might be compromised.
309a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)const int64 kBlockAllDomainsMs = 10000;
310a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)const int kNumResetsWithinDuration = 1;
311a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
312a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)// Enums for UMA histograms.
313a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)enum BlockStatusHistogram {
314a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  BLOCK_STATUS_NOT_BLOCKED,
315a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  BLOCK_STATUS_SPECIFIC_DOMAIN_BLOCKED,
316a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  BLOCK_STATUS_ALL_DOMAINS_BLOCKED,
317a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  BLOCK_STATUS_MAX
318a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)};
319a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
320a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}  // namespace anonymous
321a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
322a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::InitializeForTesting(
323a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    const std::string& gpu_blacklist_json,
32490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    const gpu::GPUInfo& gpu_info) {
325a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // This function is for testing only, so disable histograms.
326a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  update_histograms_ = false;
327a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
3284e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)  // Prevent all further initialization.
3294e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)  finalized_ = true;
3304e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)
331f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)  InitializeImpl(gpu_blacklist_json, std::string(), gpu_info);
332a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
333a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
334a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)bool GpuDataManagerImplPrivate::IsFeatureBlacklisted(int feature) const {
3353551c9c881056c480085172ff9840cab31610854Torne (Richard Coles)#if defined(OS_CHROMEOS)
3363551c9c881056c480085172ff9840cab31610854Torne (Richard Coles)  if (feature == gpu::GPU_FEATURE_TYPE_PANEL_FITTING &&
3373551c9c881056c480085172ff9840cab31610854Torne (Richard Coles)      CommandLine::ForCurrentProcess()->HasSwitch(
3383551c9c881056c480085172ff9840cab31610854Torne (Richard Coles)          switches::kDisablePanelFitting)) {
3393551c9c881056c480085172ff9840cab31610854Torne (Richard Coles)    return true;
3403551c9c881056c480085172ff9840cab31610854Torne (Richard Coles)  }
3413551c9c881056c480085172ff9840cab31610854Torne (Richard Coles)#endif  // OS_CHROMEOS
342a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (use_swiftshader_) {
343a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    // Skia's software rendering is probably more efficient than going through
344a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    // software emulation of the GPU, so use that.
34590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    if (feature == gpu::GPU_FEATURE_TYPE_ACCELERATED_2D_CANVAS)
346a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      return true;
347a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    return false;
348a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
349a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
350a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  return (blacklisted_features_.count(feature) == 1);
351a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
352a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
353424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)bool GpuDataManagerImplPrivate::IsDriverBugWorkaroundActive(int feature) const {
354424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)  return (gpu_driver_bugs_.count(feature) == 1);
355424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)}
356424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)
357a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)size_t GpuDataManagerImplPrivate::GetBlacklistedFeatureCount() const {
358a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (use_swiftshader_)
359a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    return 1;
360a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  return blacklisted_features_.size();
361a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
362a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
363868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)void GpuDataManagerImplPrivate::SetDisplayCount(unsigned int display_count) {
364868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  display_count_ = display_count;
365868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)}
366868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)
367868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)unsigned int GpuDataManagerImplPrivate::GetDisplayCount() const {
368868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  return display_count_;
369868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)}
370868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)
37190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)gpu::GPUInfo GpuDataManagerImplPrivate::GetGPUInfo() const {
372a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  return gpu_info_;
373a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
374a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
375a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::GetGpuProcessHandles(
376a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    const GpuDataManager::GetGpuProcessHandlesCallback& callback) const {
377a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  GpuProcessHost::GetProcessHandles(callback);
378a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
379a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
380a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)bool GpuDataManagerImplPrivate::GpuAccessAllowed(
381a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    std::string* reason) const {
382a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (use_swiftshader_)
383a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    return true;
384a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
385eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch  if (!gpu_process_accessible_) {
386a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    if (reason) {
387a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      *reason = "GPU process launch failed.";
388a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    }
389a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    return false;
390a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
391a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
392a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (card_blacklisted_) {
393a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    if (reason) {
394a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      *reason = "GPU access is disabled ";
395a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      CommandLine* command_line = CommandLine::ForCurrentProcess();
396a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      if (command_line->HasSwitch(switches::kDisableGpu))
397a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)        *reason += "through commandline switch --disable-gpu.";
398a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      else
399a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)        *reason += "in chrome://settings.";
400a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    }
401a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    return false;
402a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
403a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
404a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // We only need to block GPU process if more features are disallowed other
405a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // than those in the preliminary gpu feature flags because the latter work
406a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // through renderer commandline switches.
407a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  std::set<int> features = preliminary_blacklisted_features_;
40890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  gpu::MergeFeatureSets(&features, blacklisted_features_);
409a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (features.size() > preliminary_blacklisted_features_.size()) {
410a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    if (reason) {
411a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      *reason = "Features are disabled upon full but not preliminary GPU info.";
412a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    }
413a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    return false;
414a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
415a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
41690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  if (blacklisted_features_.size() == gpu::NUMBER_OF_GPU_FEATURE_TYPES) {
417a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    // On Linux, we use cached GL strings to make blacklist decsions at browser
418a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    // startup time. We need to launch the GPU process to validate these
419a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    // strings even if all features are blacklisted. If all GPU features are
420a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    // disabled, the GPU process will only initialize GL bindings, create a GL
421a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    // context, and collect full GPU info.
422a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#if !defined(OS_LINUX)
423a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    if (reason) {
424a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      *reason = "All GPU features are blacklisted.";
425a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    }
426a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    return false;
427a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#endif
428a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
429a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
430a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  return true;
431a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
432a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
433a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::RequestCompleteGpuInfoIfNeeded() {
434a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (complete_gpu_info_already_requested_ || gpu_info_.finalized)
435a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    return;
436a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  complete_gpu_info_already_requested_ = true;
437a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
438a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  GpuProcessHost::SendOnIO(
439a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#if defined(OS_WIN)
440a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      GpuProcessHost::GPU_PROCESS_KIND_UNSANDBOXED,
441a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#else
442a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      GpuProcessHost::GPU_PROCESS_KIND_SANDBOXED,
443a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#endif
444a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      CAUSE_FOR_GPU_LAUNCH_GPUDATAMANAGER_REQUESTCOMPLETEGPUINFOIFNEEDED,
445a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      new GpuMsg_CollectGraphicsInfo());
446a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
447a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
448a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)bool GpuDataManagerImplPrivate::IsCompleteGpuInfoAvailable() const {
449a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  return gpu_info_.finalized;
450a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
451a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
452a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::RequestVideoMemoryUsageStatsUpdate() const {
453a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  GpuProcessHost::SendOnIO(
454a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      GpuProcessHost::GPU_PROCESS_KIND_SANDBOXED,
455a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      CAUSE_FOR_GPU_LAUNCH_NO_LAUNCH,
456a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      new GpuMsg_GetVideoMemoryUsageStats());
457a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
458a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
459a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)bool GpuDataManagerImplPrivate::ShouldUseSwiftShader() const {
460a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  return use_swiftshader_;
461a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
462a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
463a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::RegisterSwiftShaderPath(
464a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    const base::FilePath& path) {
465a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  swiftshader_path_ = path;
466a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  EnableSwiftShaderIfNecessary();
467a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
468a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
469a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::AddObserver(GpuDataManagerObserver* observer) {
470a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  GpuDataManagerImpl::UnlockedSession session(owner_);
471a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  observer_list_->AddObserver(observer);
472a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
473a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
474a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::RemoveObserver(
475a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    GpuDataManagerObserver* observer) {
476a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  GpuDataManagerImpl::UnlockedSession session(owner_);
477a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  observer_list_->RemoveObserver(observer);
478a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
479a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
480a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::UnblockDomainFrom3DAPIs(const GURL& url) {
481a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // This method must do two things:
482a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  //
483a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  //  1. If the specific domain is blocked, then unblock it.
484a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  //
485a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  //  2. Reset our notion of how many GPU resets have occurred recently.
486a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  //     This is necessary even if the specific domain was blocked.
487a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  //     Otherwise, if we call Are3DAPIsBlocked with the same domain right
488a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  //     after unblocking it, it will probably still be blocked because of
489a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  //     the recent GPU reset caused by that domain.
490a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  //
491a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // These policies could be refined, but at a certain point the behavior
492a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // will become difficult to explain.
493a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  std::string domain = GetDomainFromURL(url);
494a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
495a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  blocked_domains_.erase(domain);
496a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  timestamps_of_gpu_resets_.clear();
497a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
498a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
499a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::DisableGpuWatchdog() {
500a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  GpuProcessHost::SendOnIO(
501a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      GpuProcessHost::GPU_PROCESS_KIND_SANDBOXED,
502a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      CAUSE_FOR_GPU_LAUNCH_NO_LAUNCH,
503a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      new GpuMsg_DisableWatchdog);
504a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
505a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
506a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::SetGLStrings(const std::string& gl_vendor,
507a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                                             const std::string& gl_renderer,
508a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                                             const std::string& gl_version) {
509a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (gl_vendor.empty() && gl_renderer.empty() && gl_version.empty())
510a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    return;
511a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
512a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // If GPUInfo already got GL strings, do nothing.  This is for the rare
513a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // situation where GPU process collected GL strings before this call.
514a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (!gpu_info_.gl_vendor.empty() ||
515a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      !gpu_info_.gl_renderer.empty() ||
516a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      !gpu_info_.gl_version_string.empty())
517a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    return;
518a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
51990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  gpu::GPUInfo gpu_info = gpu_info_;
520a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
521a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  gpu_info.gl_vendor = gl_vendor;
522a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  gpu_info.gl_renderer = gl_renderer;
523a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  gpu_info.gl_version_string = gl_version;
524a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
52590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  gpu::CollectDriverInfoGL(&gpu_info);
526a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
527a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  UpdateGpuInfo(gpu_info);
528a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  UpdateGpuSwitchingManager(gpu_info);
529a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  UpdatePreliminaryBlacklistedFeatures();
530a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
531a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
532a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::GetGLStrings(std::string* gl_vendor,
533a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                                             std::string* gl_renderer,
534a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                                             std::string* gl_version) {
535a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  DCHECK(gl_vendor && gl_renderer && gl_version);
536a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
537a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  *gl_vendor = gpu_info_.gl_vendor;
538a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  *gl_renderer = gpu_info_.gl_renderer;
539a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  *gl_version = gpu_info_.gl_version_string;
540a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
541a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
542a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::Initialize() {
543a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  TRACE_EVENT0("startup", "GpuDataManagerImpl::Initialize");
5444e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)  if (finalized_) {
545f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)    DVLOG(0) << "GpuDataManagerImpl marked as finalized; skipping Initialize";
5464e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)    return;
5474e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)  }
5484e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)
5494e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)  const CommandLine* command_line = CommandLine::ForCurrentProcess();
5504e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)  if (command_line->HasSwitch(switches::kSkipGpuDataLoading))
551a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    return;
552a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
55390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  gpu::GPUInfo gpu_info;
5544e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)  if (command_line->GetSwitchValueASCII(
5554e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)          switches::kUseGL) == gfx::kGLImplementationOSMesaName) {
5564e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)    // If using the OSMesa GL implementation, use fake vendor and device ids to
5574e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)    // make sure it never gets blacklisted. This is better than simply
5584e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)    // cancelling GPUInfo gathering as it allows us to proceed with loading the
5594e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)    // blacklist below which may have non-device specific entries we want to
5604e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)    // apply anyways (e.g., OS version blacklisting).
5614e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)    gpu_info.gpu.vendor_id = 0xffff;
5624e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)    gpu_info.gpu.device_id = 0xffff;
5634e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)
5644e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)    // Also declare the driver_vendor to be osmesa to be able to specify
5654e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)    // exceptions based on driver_vendor==osmesa for some blacklist rules.
5664e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)    gpu_info.driver_vendor = gfx::kGLImplementationOSMesaName;
5674e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)  } else {
568a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    TRACE_EVENT0("startup",
569a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      "GpuDataManagerImpl::Initialize:CollectBasicGraphicsInfo");
57090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    gpu::CollectBasicGraphicsInfo(&gpu_info);
571a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
572a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#if defined(ARCH_CPU_X86_FAMILY)
573a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (!gpu_info.gpu.vendor_id || !gpu_info.gpu.device_id)
574a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    gpu_info.finalized = true;
575a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#endif
576a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
577a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  std::string gpu_blacklist_string;
578a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  std::string gpu_driver_bug_list_string;
5797dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  if (!command_line->HasSwitch(switches::kIgnoreGpuBlacklist) &&
5807dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      !command_line->HasSwitch(switches::kUseGpuInTests)) {
58190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    gpu_blacklist_string = gpu::kSoftwareRenderingListJson;
582a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
583a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (!command_line->HasSwitch(switches::kDisableGpuDriverBugWorkarounds)) {
58490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    gpu_driver_bug_list_string = gpu::kGpuDriverBugListJson;
585a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
586a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  InitializeImpl(gpu_blacklist_string,
587a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                 gpu_driver_bug_list_string,
588a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                 gpu_info);
589a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
590a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
591e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdochvoid GpuDataManagerImplPrivate::UpdateGpuInfoHelper() {
592a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  GetContentClient()->SetGpuInfo(gpu_info_);
593a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
594a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (gpu_blacklist_) {
595a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    std::set<int> features = gpu_blacklist_->MakeDecision(
59690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)        gpu::GpuControlList::kOsAny, std::string(), gpu_info_);
597a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    if (update_histograms_)
5983240926e260ce088908e02ac07a6cf7b0c0cbf44Ben Murdoch      UpdateStats(gpu_info_, gpu_blacklist_.get(), features);
599a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
600a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    UpdateBlacklistedFeatures(features);
601a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
602e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  if (gpu_driver_bug_list_) {
603a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    gpu_driver_bugs_ = gpu_driver_bug_list_->MakeDecision(
60490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)        gpu::GpuControlList::kOsAny, std::string(), gpu_info_);
60590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  }
606e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  gpu::GpuDriverBugList::AppendWorkaroundsFromCommandLine(
607e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch      &gpu_driver_bugs_, *CommandLine::ForCurrentProcess());
608a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
609a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // We have to update GpuFeatureType before notify all the observers.
610a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  NotifyGpuInfoUpdate();
611a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
612a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
613e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdochvoid GpuDataManagerImplPrivate::UpdateGpuInfo(const gpu::GPUInfo& gpu_info) {
614e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  // No further update of gpu_info if falling back to SwiftShader.
615e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  if (use_swiftshader_)
616e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch    return;
617e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch
618e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  gpu::MergeGPUInfo(&gpu_info_, gpu_info);
619e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  complete_gpu_info_already_requested_ =
620e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch      complete_gpu_info_already_requested_ || gpu_info_.finalized;
621e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch
622e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  UpdateGpuInfoHelper();
623e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch}
624e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch
625a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::UpdateVideoMemoryUsageStats(
626a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    const GPUVideoMemoryUsageStats& video_memory_usage_stats) {
627a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  GpuDataManagerImpl::UnlockedSession session(owner_);
628a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  observer_list_->Notify(&GpuDataManagerObserver::OnVideoMemoryUsageStatsUpdate,
629a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                         video_memory_usage_stats);
630a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
631a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
632a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::AppendRendererCommandLine(
633a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    CommandLine* command_line) const {
634a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  DCHECK(command_line);
635a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
63690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  if (IsFeatureBlacklisted(gpu::GPU_FEATURE_TYPE_ACCELERATED_COMPOSITING) &&
637a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      !command_line->HasSwitch(switches::kDisableAcceleratedCompositing))
638a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    command_line->AppendSwitch(switches::kDisableAcceleratedCompositing);
63990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  if (IsFeatureBlacklisted(gpu::GPU_FEATURE_TYPE_ACCELERATED_VIDEO_DECODE) &&
640a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      !command_line->HasSwitch(switches::kDisableAcceleratedVideoDecode))
641a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    command_line->AppendSwitch(switches::kDisableAcceleratedVideoDecode);
642a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)#if defined(ENABLE_WEBRTC)
643a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)  if (IsFeatureBlacklisted(gpu::GPU_FEATURE_TYPE_ACCELERATED_VIDEO_ENCODE) &&
644a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)      !command_line->HasSwitch(switches::kDisableWebRtcHWEncoding))
645a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)    command_line->AppendSwitch(switches::kDisableWebRtcHWEncoding);
646a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)#endif
647a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)
648a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)  if (use_software_compositor_ &&
649a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)      !command_line->HasSwitch(switches::kEnableSoftwareCompositing))
650a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)    command_line->AppendSwitch(switches::kEnableSoftwareCompositing);
651a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)
652a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)#if defined(USE_AURA)
6535d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  if (!CanUseGpuBrowserCompositor())
654a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)    command_line->AppendSwitch(switches::kDisableGpuCompositing);
655a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)#endif
656a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
657a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
658a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::AppendGpuCommandLine(
659a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    CommandLine* command_line) const {
660a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  DCHECK(command_line);
661a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
662a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  std::string use_gl =
663a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      CommandLine::ForCurrentProcess()->GetSwitchValueASCII(switches::kUseGL);
664a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  base::FilePath swiftshader_path =
665a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      CommandLine::ForCurrentProcess()->GetSwitchValuePath(
666a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)          switches::kSwiftShaderPath);
6677dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  if (gpu_driver_bugs_.find(gpu::DISABLE_D3D11) != gpu_driver_bugs_.end())
6687dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch    command_line->AppendSwitch(switches::kDisableD3D11);
669a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (use_swiftshader_) {
670a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    command_line->AppendSwitchASCII(switches::kUseGL, "swiftshader");
671a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    if (swiftshader_path.empty())
672a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      swiftshader_path = swiftshader_path_;
67390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  } else if ((IsFeatureBlacklisted(gpu::GPU_FEATURE_TYPE_WEBGL) ||
67490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)              IsFeatureBlacklisted(
67590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)                  gpu::GPU_FEATURE_TYPE_ACCELERATED_COMPOSITING) ||
67690dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)              IsFeatureBlacklisted(
67790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)                  gpu::GPU_FEATURE_TYPE_ACCELERATED_2D_CANVAS)) &&
678a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      (use_gl == "any")) {
679a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    command_line->AppendSwitchASCII(
680a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)        switches::kUseGL, gfx::kGLImplementationOSMesaName);
681a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  } else if (!use_gl.empty()) {
682a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    command_line->AppendSwitchASCII(switches::kUseGL, use_gl);
683a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
684f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)  if (ui::GpuSwitchingManager::GetInstance()->SupportsDualGpus())
685a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    command_line->AppendSwitchASCII(switches::kSupportsDualGpus, "true");
686f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)  else
687a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    command_line->AppendSwitchASCII(switches::kSupportsDualGpus, "false");
688a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
6897d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  if (!swiftshader_path.empty()) {
690a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    command_line->AppendSwitchPath(switches::kSwiftShaderPath,
691a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                                   swiftshader_path);
6927d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  }
693a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
694a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (!gpu_driver_bugs_.empty()) {
695a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    command_line->AppendSwitchASCII(switches::kGpuDriverBugWorkarounds,
696a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                                    IntSetToString(gpu_driver_bugs_));
697a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
698a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
6993551c9c881056c480085172ff9840cab31610854Torne (Richard Coles)  if (IsFeatureBlacklisted(gpu::GPU_FEATURE_TYPE_ACCELERATED_VIDEO_DECODE) &&
7003551c9c881056c480085172ff9840cab31610854Torne (Richard Coles)      !command_line->HasSwitch(switches::kDisableAcceleratedVideoDecode)) {
7013551c9c881056c480085172ff9840cab31610854Torne (Richard Coles)    command_line->AppendSwitch(switches::kDisableAcceleratedVideoDecode);
7023551c9c881056c480085172ff9840cab31610854Torne (Richard Coles)  }
703a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)#if defined(ENABLE_WEBRTC)
704a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)  if (IsFeatureBlacklisted(gpu::GPU_FEATURE_TYPE_ACCELERATED_VIDEO_ENCODE) &&
705a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)      !command_line->HasSwitch(switches::kDisableWebRtcHWEncoding)) {
706a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)    command_line->AppendSwitch(switches::kDisableWebRtcHWEncoding);
707a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)  }
708a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)#endif
7093551c9c881056c480085172ff9840cab31610854Torne (Richard Coles)
710a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // Pass GPU and driver information to GPU process. We try to avoid full GPU
711a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // info collection at GPU process startup, but we need gpu vendor_id,
712a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // device_id, driver_vendor, driver_version for deciding whether we need to
713a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // collect full info (on Linux) and for crash reporting purpose.
714a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  command_line->AppendSwitchASCII(switches::kGpuVendorID,
715a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      base::StringPrintf("0x%04x", gpu_info_.gpu.vendor_id));
716a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  command_line->AppendSwitchASCII(switches::kGpuDeviceID,
717a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      base::StringPrintf("0x%04x", gpu_info_.gpu.device_id));
718a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  command_line->AppendSwitchASCII(switches::kGpuDriverVendor,
719a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      gpu_info_.driver_vendor);
720a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  command_line->AppendSwitchASCII(switches::kGpuDriverVersion,
721a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      gpu_info_.driver_version);
722a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
723a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
724a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::AppendPluginCommandLine(
725a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    CommandLine* command_line) const {
726a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  DCHECK(command_line);
727a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
728a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#if defined(OS_MACOSX)
729a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // TODO(jbauman): Add proper blacklist support for core animation plugins so
730a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // special-casing this video card won't be necessary. See
731a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // http://crbug.com/134015
73290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  if (IsFeatureBlacklisted(gpu::GPU_FEATURE_TYPE_ACCELERATED_COMPOSITING) ||
733a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      CommandLine::ForCurrentProcess()->HasSwitch(
734a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)          switches::kDisableAcceleratedCompositing)) {
735a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    if (!command_line->HasSwitch(
736a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)           switches::kDisableCoreAnimationPlugins))
737a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      command_line->AppendSwitch(
738a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)          switches::kDisableCoreAnimationPlugins);
739a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
740a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#endif
741a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
742a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
743a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::UpdateRendererWebPrefs(
744a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    WebPreferences* prefs) const {
745a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  DCHECK(prefs);
746a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
74790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  if (IsFeatureBlacklisted(gpu::GPU_FEATURE_TYPE_ACCELERATED_COMPOSITING))
748a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    prefs->accelerated_compositing_enabled = false;
7495d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  if (IsFeatureBlacklisted(gpu::GPU_FEATURE_TYPE_WEBGL)) {
750a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    prefs->experimental_webgl_enabled = false;
7515d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    prefs->pepper_3d_enabled = false;
7525d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
75390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  if (IsFeatureBlacklisted(gpu::GPU_FEATURE_TYPE_FLASH3D))
754a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    prefs->flash_3d_enabled = false;
75590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  if (IsFeatureBlacklisted(gpu::GPU_FEATURE_TYPE_FLASH_STAGE3D)) {
756a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    prefs->flash_stage3d_enabled = false;
757a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    prefs->flash_stage3d_baseline_enabled = false;
758a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
75990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  if (IsFeatureBlacklisted(gpu::GPU_FEATURE_TYPE_FLASH_STAGE3D_BASELINE))
760a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    prefs->flash_stage3d_baseline_enabled = false;
76190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  if (IsFeatureBlacklisted(gpu::GPU_FEATURE_TYPE_ACCELERATED_2D_CANVAS))
762a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    prefs->accelerated_2d_canvas_enabled = false;
7635d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  if (IsDriverBugWorkaroundActive(gpu::DISABLE_MULTISAMPLING) ||
764424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)      (IsDriverBugWorkaroundActive(gpu::DISABLE_MULTIMONITOR_MULTISAMPLING) &&
765424c4d7b64af9d0d8fd9624f381f469654d5e3d2Torne (Richard Coles)          display_count_ > 1))
766a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    prefs->gl_multisampling_enabled = false;
76790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  if (IsFeatureBlacklisted(gpu::GPU_FEATURE_TYPE_3D_CSS)) {
768a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    prefs->accelerated_compositing_for_3d_transforms_enabled = false;
769a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    prefs->accelerated_compositing_for_animation_enabled = false;
770a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
77190dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  if (IsFeatureBlacklisted(gpu::GPU_FEATURE_TYPE_ACCELERATED_VIDEO))
772a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    prefs->accelerated_compositing_for_video_enabled = false;
773a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
774a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // Accelerated video and animation are slower than regular when using
7755d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  // SwiftShader. 3D CSS or Pepper 3D may also be too slow to be worthwhile.
776a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (ShouldUseSwiftShader()) {
777a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    prefs->accelerated_compositing_for_video_enabled = false;
778a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    prefs->accelerated_compositing_for_animation_enabled = false;
779a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    prefs->accelerated_compositing_for_3d_transforms_enabled = false;
780a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    prefs->accelerated_compositing_for_plugins_enabled = false;
7815d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    prefs->pepper_3d_enabled = false;
782a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
783a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)
784a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)  if (use_software_compositor_) {
785a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)    prefs->force_compositing_mode = true;
786a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)    prefs->accelerated_compositing_enabled = true;
787a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)    prefs->accelerated_compositing_for_3d_transforms_enabled = true;
788a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    prefs->accelerated_compositing_for_animation_enabled = true;
789a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)    prefs->accelerated_compositing_for_plugins_enabled = true;
7903551c9c881056c480085172ff9840cab31610854Torne (Richard Coles)    prefs->accelerated_compositing_for_video_enabled = true;
791a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)  }
792a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)
793a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)#if defined(USE_AURA)
7945d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  if (!CanUseGpuBrowserCompositor()) {
795a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)    prefs->accelerated_2d_canvas_enabled = false;
7965d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    prefs->pepper_3d_enabled = false;
7975d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
798a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)#endif
799a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)
800a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  if (!IsFeatureBlacklisted(gpu::GPU_FEATURE_TYPE_ACCELERATED_VIDEO_DECODE) &&
801a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      !CommandLine::ForCurrentProcess()->HasSwitch(
802a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)          switches::kDisableAcceleratedVideoDecode)) {
803a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    prefs->pepper_accelerated_video_decode_enabled = true;
804a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  }
805a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
806a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
807a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::DisableHardwareAcceleration() {
808a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  card_blacklisted_ = true;
809a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
81090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)  for (int i = 0; i < gpu::NUMBER_OF_GPU_FEATURE_TYPES; ++i)
811a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    blacklisted_features_.insert(i);
812a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
813a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  EnableSwiftShaderIfNecessary();
814a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  NotifyGpuInfoUpdate();
815a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
816a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
817a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)std::string GpuDataManagerImplPrivate::GetBlacklistVersion() const {
818a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (gpu_blacklist_)
819a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    return gpu_blacklist_->version();
820a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  return "0";
821a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
822a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
823868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)std::string GpuDataManagerImplPrivate::GetDriverBugListVersion() const {
824868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  if (gpu_driver_bug_list_)
825868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)    return gpu_driver_bug_list_->version();
826868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  return "0";
827868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)}
828868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)
829868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)void GpuDataManagerImplPrivate::GetBlacklistReasons(
830868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)    base::ListValue* reasons) const {
831a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (gpu_blacklist_)
832a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    gpu_blacklist_->GetReasons(reasons, "disabledFeatures");
833a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  if (gpu_driver_bug_list_)
834a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)    gpu_driver_bug_list_->GetReasons(reasons, "workarounds");
835868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)}
836868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)
837868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)void GpuDataManagerImplPrivate::GetDriverBugWorkarounds(
838868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)    base::ListValue* workarounds) const {
839868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  for (std::set<int>::const_iterator it = gpu_driver_bugs_.begin();
840868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)       it != gpu_driver_bugs_.end(); ++it) {
841868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)    workarounds->AppendString(
842868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)        gpu::GpuDriverBugWorkaroundTypeToString(
843868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)            static_cast<gpu::GpuDriverBugWorkaroundType>(*it)));
844868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  }
845a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
846a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
847a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::AddLogMessage(
848a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    int level, const std::string& header, const std::string& message) {
8495d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  log_messages_.push_back(LogMessage(level, header, message));
850a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
851a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
852a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::ProcessCrashed(
853a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    base::TerminationStatus exit_code) {
854a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (!BrowserThread::CurrentlyOn(BrowserThread::UI)) {
855a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    // Unretained is ok, because it's posted to UI thread, the thread
856a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    // where the singleton GpuDataManagerImpl lives until the end.
857a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    BrowserThread::PostTask(
858a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)        BrowserThread::UI,
859a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)        FROM_HERE,
860a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)        base::Bind(&GpuDataManagerImpl::ProcessCrashed,
861a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                   base::Unretained(owner_),
862a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                   exit_code));
863a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    return;
864a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
865a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  {
866a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    GpuDataManagerImpl::UnlockedSession session(owner_);
867a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    observer_list_->Notify(
868a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)        &GpuDataManagerObserver::OnGpuProcessCrashed, exit_code);
869a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
870a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
871a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
872a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)base::ListValue* GpuDataManagerImplPrivate::GetLogMessages() const {
8735d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  base::ListValue* value = new base::ListValue;
8745d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  for (size_t ii = 0; ii < log_messages_.size(); ++ii) {
8755d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    base::DictionaryValue* dict = new base::DictionaryValue();
8765d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    dict->SetInteger("level", log_messages_[ii].level);
8775d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    dict->SetString("header", log_messages_[ii].header);
8785d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    dict->SetString("message", log_messages_[ii].message);
8795d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)    value->Append(dict);
8805d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  }
881a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  return value;
882a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
883a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
884a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::HandleGpuSwitch() {
885e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  // Check if the active gpu has changed.
886e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  uint32 vendor_id, device_id;
887e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  gpu::GPUInfo::GPUDevice* active = NULL;
888e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  gpu::GPUInfo::GPUDevice* old_active = NULL;
889e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  if (gpu::CollectGpuID(&vendor_id, &device_id) == gpu::kGpuIDSuccess) {
890e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch    if (gpu_info_.gpu.active)
891e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch      old_active = &gpu_info_.gpu;
892e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch    if (gpu_info_.gpu.vendor_id == vendor_id &&
893e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch        gpu_info_.gpu.device_id == device_id)
894e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch      active = &gpu_info_.gpu;
895e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch    for (size_t ii = 0; ii < gpu_info_.secondary_gpus.size(); ++ii) {
896e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch      gpu::GPUInfo::GPUDevice& gpu = gpu_info_.secondary_gpus[ii];
897e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch      if (gpu.active)
898e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch        old_active = &gpu;
899e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch      if (gpu.vendor_id == vendor_id && gpu.device_id == device_id)
900e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch        active = &gpu;
901e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch    }
902e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch    DCHECK(active && old_active);
903e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch    if (active != old_active) {  // A different GPU is used.
904e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch      old_active->active = false;
905e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch      active->active = true;
906e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch      UpdateGpuInfoHelper();
907e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch    }
908e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  }
909e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  {
910e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch    GpuDataManagerImpl::UnlockedSession session(owner_);
911e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch    observer_list_->Notify(&GpuDataManagerObserver::OnGpuSwitching);
912e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch  }
913a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
914a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
9157dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdochbool GpuDataManagerImplPrivate::CanUseGpuBrowserCompositor() const {
916a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)  return !ShouldUseSwiftShader() &&
917a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)         !IsFeatureBlacklisted(gpu::GPU_FEATURE_TYPE_ACCELERATED_COMPOSITING) &&
918a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)         !IsFeatureBlacklisted(gpu::GPU_FEATURE_TYPE_FORCE_COMPOSITING_MODE);
9197dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch}
9207dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch
921a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::BlockDomainFrom3DAPIs(
922a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    const GURL& url, GpuDataManagerImpl::DomainGuilt guilt) {
923a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  BlockDomainFrom3DAPIsAtTime(url, guilt, base::Time::Now());
924a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
925a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
926a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)bool GpuDataManagerImplPrivate::Are3DAPIsBlocked(const GURL& url,
927a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                                                 int render_process_id,
928a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                                                 int render_view_id,
929a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                                                 ThreeDAPIType requester) {
930a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  bool blocked = Are3DAPIsBlockedAtTime(url, base::Time::Now()) !=
931a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      GpuDataManagerImpl::DOMAIN_BLOCK_STATUS_NOT_BLOCKED;
932a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (blocked) {
933a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    // Unretained is ok, because it's posted to UI thread, the thread
934a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    // where the singleton GpuDataManagerImpl lives until the end.
935a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    BrowserThread::PostTask(
936a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)        BrowserThread::UI, FROM_HERE,
937a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)        base::Bind(&GpuDataManagerImpl::Notify3DAPIBlocked,
938a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                   base::Unretained(owner_), url, render_process_id,
939a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                   render_view_id, requester));
940a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
941a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
942a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  return blocked;
943a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
944a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
945a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::DisableDomainBlockingFor3DAPIsForTesting() {
946a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  domain_blocking_enabled_ = false;
947a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
948a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
949a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)// static
950a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)GpuDataManagerImplPrivate* GpuDataManagerImplPrivate::Create(
951a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    GpuDataManagerImpl* owner) {
952a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  return new GpuDataManagerImplPrivate(owner);
953a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
954a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
955a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)GpuDataManagerImplPrivate::GpuDataManagerImplPrivate(
956a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    GpuDataManagerImpl* owner)
957a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    : complete_gpu_info_already_requested_(false),
958a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      observer_list_(new GpuDataManagerObserverList),
959a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      use_swiftshader_(false),
960a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      card_blacklisted_(false),
961a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      update_histograms_(true),
962a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      window_count_(0),
963a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      domain_blocking_enabled_(true),
964868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)      owner_(owner),
965eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch      display_count_(0),
966a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)      gpu_process_accessible_(true),
9674e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)      use_software_compositor_(false),
9684e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)      finalized_(false) {
969a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  DCHECK(owner_);
970a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  CommandLine* command_line = CommandLine::ForCurrentProcess();
971a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (command_line->HasSwitch(switches::kDisableAcceleratedCompositing)) {
972a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    command_line->AppendSwitch(switches::kDisableAccelerated2dCanvas);
973a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    command_line->AppendSwitch(switches::kDisableAcceleratedLayers);
974a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
975a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (command_line->HasSwitch(switches::kDisableGpu))
976a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    DisableHardwareAcceleration();
977a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)  if (command_line->HasSwitch(switches::kEnableSoftwareCompositing))
978a36e5920737c6adbddd3e43b760e5de8431db6e0Torne (Richard Coles)    use_software_compositor_ = true;
979e5d81f57cb97b3b6b7fccc9c5610d21eb81db09dBen Murdoch#if defined(USE_AURA) || defined(OS_MACOSX)
9803240926e260ce088908e02ac07a6cf7b0c0cbf44Ben Murdoch  use_software_compositor_ = true;
9813240926e260ce088908e02ac07a6cf7b0c0cbf44Ben Murdoch#endif
982a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
983a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#if defined(OS_MACOSX)
984868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)  CGGetActiveDisplayList (0, NULL, &display_count_);
985a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  CGDisplayRegisterReconfigurationCallback(DisplayReconfigCallback, owner_);
986a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#endif  // OS_MACOSX
9878bcbed890bc3ce4d7a057a8f32cab53fa534672eTorne (Richard Coles)
9888bcbed890bc3ce4d7a057a8f32cab53fa534672eTorne (Richard Coles)  // For testing only.
9898bcbed890bc3ce4d7a057a8f32cab53fa534672eTorne (Richard Coles)  if (command_line->HasSwitch(switches::kDisableDomainBlockingFor3DAPIs)) {
9908bcbed890bc3ce4d7a057a8f32cab53fa534672eTorne (Richard Coles)    domain_blocking_enabled_ = false;
9918bcbed890bc3ce4d7a057a8f32cab53fa534672eTorne (Richard Coles)  }
992a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
993a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
994a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)GpuDataManagerImplPrivate::~GpuDataManagerImplPrivate() {
995a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#if defined(OS_MACOSX)
996a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  CGDisplayRemoveReconfigurationCallback(DisplayReconfigCallback, owner_);
997a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)#endif
998a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
999a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1000a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::InitializeImpl(
1001a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    const std::string& gpu_blacklist_json,
1002a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    const std::string& gpu_driver_bug_list_json,
100390dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    const gpu::GPUInfo& gpu_info) {
10044e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)  const bool log_gpu_control_list_decisions =
10054e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)      CommandLine::ForCurrentProcess()->HasSwitch(
10064e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)          switches::kLogGpuControlListDecisions);
10074e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)
1008a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (!gpu_blacklist_json.empty()) {
100990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    gpu_blacklist_.reset(gpu::GpuBlacklist::Create());
10104e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)    if (log_gpu_control_list_decisions)
10114e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)      gpu_blacklist_->enable_control_list_logging("gpu_blacklist");
10124e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)    bool success = gpu_blacklist_->LoadList(
1013a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)        gpu_blacklist_json, gpu::GpuControlList::kCurrentOsOnly);
10144e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)    DCHECK(success);
1015a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
1016a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (!gpu_driver_bug_list_json.empty()) {
101790dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    gpu_driver_bug_list_.reset(gpu::GpuDriverBugList::Create());
10184e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)    if (log_gpu_control_list_decisions)
10194e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)      gpu_driver_bug_list_->enable_control_list_logging("gpu_driver_bug_list");
10204e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)    bool success = gpu_driver_bug_list_->LoadList(
1021a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)        gpu_driver_bug_list_json, gpu::GpuControlList::kCurrentOsOnly);
10224e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)    DCHECK(success);
1023a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
1024a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1025a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  gpu_info_ = gpu_info;
1026a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  UpdateGpuInfo(gpu_info);
1027a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  UpdateGpuSwitchingManager(gpu_info);
1028a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  UpdatePreliminaryBlacklistedFeatures();
102990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)
103090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#if defined(OS_ANDROID)
10315d1f7b1de12d16ceb2c938c56701a3e8bfa558f7Torne (Richard Coles)  ApplyAndroidWorkarounds(gpu_info, CommandLine::ForCurrentProcess());
103290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#endif  // OS_ANDROID
1033a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
1034a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1035a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::UpdateBlacklistedFeatures(
1036a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    const std::set<int>& features) {
1037a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  CommandLine* command_line = CommandLine::ForCurrentProcess();
1038a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  blacklisted_features_ = features;
1039a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1040a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // Force disable using the GPU for these features, even if they would
1041a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // otherwise be allowed.
1042a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (card_blacklisted_ ||
1043a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      command_line->HasSwitch(switches::kBlacklistAcceleratedCompositing)) {
104490dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    blacklisted_features_.insert(
104590dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)        gpu::GPU_FEATURE_TYPE_ACCELERATED_COMPOSITING);
1046a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
1047a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (card_blacklisted_ ||
1048a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      command_line->HasSwitch(switches::kBlacklistWebGL)) {
104990dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    blacklisted_features_.insert(gpu::GPU_FEATURE_TYPE_WEBGL);
1050a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
1051a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1052a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  EnableSwiftShaderIfNecessary();
1053a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
1054a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1055a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::UpdatePreliminaryBlacklistedFeatures() {
1056a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  preliminary_blacklisted_features_ = blacklisted_features_;
1057a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
1058a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1059a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::UpdateGpuSwitchingManager(
106090dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)    const gpu::GPUInfo& gpu_info) {
1061a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  ui::GpuSwitchingManager::GetInstance()->SetGpuCount(
1062a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      gpu_info.secondary_gpus.size() + 1);
1063a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1064a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (ui::GpuSwitchingManager::GetInstance()->SupportsDualGpus()) {
1065f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)    if (gpu_driver_bugs_.count(gpu::FORCE_DISCRETE_GPU) == 1)
1066f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)      ui::GpuSwitchingManager::GetInstance()->ForceUseOfDiscreteGpu();
1067f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)    else if (gpu_driver_bugs_.count(gpu::FORCE_INTEGRATED_GPU) == 1)
1068f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)      ui::GpuSwitchingManager::GetInstance()->ForceUseOfIntegratedGpu();
1069a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
1070a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
1071a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1072a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::NotifyGpuInfoUpdate() {
1073a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  observer_list_->Notify(&GpuDataManagerObserver::OnGpuInfoUpdate);
1074a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
1075a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1076a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::EnableSwiftShaderIfNecessary() {
1077a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (!GpuAccessAllowed(NULL) ||
107890dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)      blacklisted_features_.count(gpu::GPU_FEATURE_TYPE_WEBGL)) {
1079a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    if (!swiftshader_path_.empty() &&
1080a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)        !CommandLine::ForCurrentProcess()->HasSwitch(
1081a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)             switches::kDisableSoftwareRasterizer))
1082a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      use_swiftshader_ = true;
1083a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
1084a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
1085a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1086a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)std::string GpuDataManagerImplPrivate::GetDomainFromURL(
1087a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    const GURL& url) const {
1088a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // For the moment, we just use the host, or its IP address, as the
1089a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // entry in the set, rather than trying to figure out the top-level
1090a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // domain. This does mean that a.foo.com and b.foo.com will be
1091a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // treated independently in the blocking of a given domain, but it
1092a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // would require a third-party library to reliably figure out the
1093a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // top-level domain from a URL.
1094a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (!url.has_host()) {
1095a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    return std::string();
1096a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
1097a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1098a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  return url.host();
1099a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
1100a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1101a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::BlockDomainFrom3DAPIsAtTime(
1102a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    const GURL& url,
1103a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    GpuDataManagerImpl::DomainGuilt guilt,
1104a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    base::Time at_time) {
1105a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (!domain_blocking_enabled_)
1106a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    return;
1107a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1108a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  std::string domain = GetDomainFromURL(url);
1109a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1110a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  DomainBlockEntry& entry = blocked_domains_[domain];
1111a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  entry.last_guilt = guilt;
1112a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  timestamps_of_gpu_resets_.push_back(at_time);
1113a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
1114a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1115a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)GpuDataManagerImpl::DomainBlockStatus
1116a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)GpuDataManagerImplPrivate::Are3DAPIsBlockedAtTime(
1117a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    const GURL& url, base::Time at_time) const {
1118a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (!domain_blocking_enabled_)
1119a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    return GpuDataManagerImpl::DOMAIN_BLOCK_STATUS_NOT_BLOCKED;
1120a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1121a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // Note: adjusting the policies in this code will almost certainly
1122a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // require adjusting the associated unit tests.
1123a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  std::string domain = GetDomainFromURL(url);
1124a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1125a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  DomainBlockMap::const_iterator iter = blocked_domains_.find(domain);
1126a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  if (iter != blocked_domains_.end()) {
1127a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    // Err on the side of caution, and assume that if a particular
1128a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    // domain shows up in the block map, it's there for a good
1129a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    // reason and don't let its presence there automatically expire.
1130a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1131a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    UMA_HISTOGRAM_ENUMERATION("GPU.BlockStatusForClient3DAPIs",
1132a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                              BLOCK_STATUS_SPECIFIC_DOMAIN_BLOCKED,
1133a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                              BLOCK_STATUS_MAX);
1134a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1135a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    return GpuDataManagerImpl::DOMAIN_BLOCK_STATUS_BLOCKED;
1136a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
1137a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1138a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // Look at the timestamps of the recent GPU resets to see if there are
1139a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // enough within the threshold which would cause us to blacklist all
1140a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // domains. This doesn't need to be overly precise -- if time goes
1141a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // backward due to a system clock adjustment, that's fine.
1142a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  //
1143a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // TODO(kbr): make this pay attention to the TDR thresholds in the
1144a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  // Windows registry, but make sure it continues to be testable.
1145a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  {
1146a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    std::list<base::Time>::iterator iter = timestamps_of_gpu_resets_.begin();
1147a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    int num_resets_within_timeframe = 0;
1148a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    while (iter != timestamps_of_gpu_resets_.end()) {
1149a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      base::Time time = *iter;
1150a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      base::TimeDelta delta_t = at_time - time;
1151a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1152a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      // If this entry has "expired", just remove it.
1153a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      if (delta_t.InMilliseconds() > kBlockAllDomainsMs) {
1154a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)        iter = timestamps_of_gpu_resets_.erase(iter);
1155a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)        continue;
1156a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      }
1157a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1158a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      ++num_resets_within_timeframe;
1159a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      ++iter;
1160a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    }
1161a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1162a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    if (num_resets_within_timeframe >= kNumResetsWithinDuration) {
1163a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      UMA_HISTOGRAM_ENUMERATION("GPU.BlockStatusForClient3DAPIs",
1164a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                                BLOCK_STATUS_ALL_DOMAINS_BLOCKED,
1165a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                                BLOCK_STATUS_MAX);
1166a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1167a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)      return GpuDataManagerImpl::DOMAIN_BLOCK_STATUS_ALL_DOMAINS_BLOCKED;
1168a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)    }
1169a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  }
1170a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1171a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  UMA_HISTOGRAM_ENUMERATION("GPU.BlockStatusForClient3DAPIs",
1172a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                            BLOCK_STATUS_NOT_BLOCKED,
1173a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                            BLOCK_STATUS_MAX);
1174a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1175a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  return GpuDataManagerImpl::DOMAIN_BLOCK_STATUS_NOT_BLOCKED;
1176a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
1177a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1178a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)int64 GpuDataManagerImplPrivate::GetBlockAllDomainsDurationInMs() const {
1179a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  return kBlockAllDomainsMs;
1180a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
1181a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1182a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)void GpuDataManagerImplPrivate::Notify3DAPIBlocked(const GURL& url,
1183a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                                                   int render_process_id,
1184a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                                                   int render_view_id,
1185a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                                                   ThreeDAPIType requester) {
1186a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  GpuDataManagerImpl::UnlockedSession session(owner_);
1187a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)  observer_list_->Notify(&GpuDataManagerObserver::DidBlock3DAPIs,
1188a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)                         url, render_process_id, render_view_id, requester);
1189a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}
1190a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)
1191eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdochvoid GpuDataManagerImplPrivate::OnGpuProcessInitFailure() {
1192eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch  gpu_process_accessible_ = false;
1193eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch  gpu_info_.finalized = true;
1194eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch  complete_gpu_info_already_requested_ = true;
1195eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch  // Some observers might be waiting.
1196eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch  NotifyGpuInfoUpdate();
1197eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch}
1198eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch
1199a93a17c8d99d686bd4a1511e5504e5e6cc9fcadfTorne (Richard Coles)}  // namespace content
1200