// Copyright (c) 2017 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "gpu/config/gpu_switching.h" #if defined(OS_MAC) #include #endif #include #include #include "base/command_line.h" #include "gpu/config/gpu_driver_bug_workaround_type.h" #include "gpu/config/gpu_info.h" #include "ui/gl/gl_context.h" #include "ui/gl/gl_switches.h" #include "ui/gl/gpu_preference.h" #if defined(OS_MAC) #include "base/mac/mac_util.h" #endif // OS_MAC namespace gpu { namespace { #if defined(OS_MAC) typedef CGLPixelFormatObj PlatformPixelFormatObj; #else typedef void* PlatformPixelFormatObj; #endif // OS_MAC PlatformPixelFormatObj g_discrete_pixel_format_obj = nullptr; bool ContainsWorkaround(const std::vector& workarounds, int32_t workaround) { return (std::find(workarounds.begin(), workarounds.end(), workaround) != workarounds.end()); } void ForceDiscreteGPU() { if (g_discrete_pixel_format_obj) return; #if defined(OS_MAC) CGLPixelFormatAttribute attribs[1]; attribs[0] = static_cast(0); GLint num_pixel_formats = 0; CGLChoosePixelFormat(attribs, &g_discrete_pixel_format_obj, &num_pixel_formats); #endif // OS_MAC } } // namespace anonymous bool SwitchableGPUsSupported(const GPUInfo& gpu_info, const base::CommandLine& command_line) { #if defined(OS_MAC) if (command_line.HasSwitch(switches::kUseGL) && (command_line.GetSwitchValueASCII(switches::kUseGL) != gl::kGLImplementationDesktopName && command_line.GetSwitchValueASCII(switches::kUseGL) != gl::kGLImplementationANGLEName && command_line.GetSwitchValueASCII(switches::kUseGL) != gl::kGLImplementationCoreProfileName)) { return false; } // Always allow offline renderers on ARM-based macs. // https://crbug.com/1131312 switch (base::mac::GetCPUType()) { case base::mac::CPUType::kArm: case base::mac::CPUType::kTranslatedIntel: return true; default: break; } if (gpu_info.secondary_gpus.size() != 1) { return false; } // Only advertise that we have two GPUs to the rest of Chrome's code if we // find an Intel GPU and some other vendor's GPU. Otherwise we don't // understand the configuration and don't deal well with it (an example being // the dual AMD GPUs in recent Mac Pros). Motivation is explained in: // http://crbug.com/380026#c70. const uint32_t kVendorIntel = 0x8086; return ((gpu_info.gpu.vendor_id == kVendorIntel && gpu_info.secondary_gpus[0].vendor_id != kVendorIntel) || (gpu_info.gpu.vendor_id != kVendorIntel && gpu_info.secondary_gpus[0].vendor_id == kVendorIntel)); #else return false; #endif // OS_MAC } void InitializeSwitchableGPUs( const std::vector& driver_bug_workarounds) { gl::GLContext::SetSwitchableGPUsSupported(); if (ContainsWorkaround(driver_bug_workarounds, FORCE_HIGH_PERFORMANCE_GPU)) { gl::GLContext::SetForcedGpuPreference(gl::GpuPreference::kHighPerformance); ForceDiscreteGPU(); } else if (ContainsWorkaround(driver_bug_workarounds, FORCE_LOW_POWER_GPU)) { gl::GLContext::SetForcedGpuPreference(gl::GpuPreference::kLowPower); } } void StopForceDiscreteGPU() { #if defined(OS_MAC) if (g_discrete_pixel_format_obj) { CGLReleasePixelFormat(g_discrete_pixel_format_obj); g_discrete_pixel_format_obj = nullptr; } #endif // OS_MAC } } // namespace gpu