| // Copyright 2016 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 "ui/gl/init/gl_factory.h" |
| |
| #include <algorithm> |
| #include <string> |
| #include <vector> |
| |
| #include "base/command_line.h" |
| #include "base/containers/contains.h" |
| #include "base/logging.h" |
| #include "base/metrics/histogram_macros.h" |
| #include "base/trace_event/trace_event.h" |
| #include "build/build_config.h" |
| #include "ui/gl/gl_share_group.h" |
| #include "ui/gl/gl_surface.h" |
| #include "ui/gl/gl_utils.h" |
| #include "ui/gl/gl_version_info.h" |
| #include "ui/gl/init/gl_initializer.h" |
| |
| #if defined(USE_OZONE) |
| #include "ui/base/ui_base_features.h" |
| #include "ui/ozone/public/ozone_platform.h" |
| #endif |
| |
| namespace gl { |
| namespace init { |
| |
| namespace { |
| |
| bool g_is_angle_enabled = true; |
| |
| bool ShouldFallbackToSoftwareGL() { |
| const base::CommandLine* cmd = base::CommandLine::ForCurrentProcess(); |
| std::string requested_implementation_gl_name = |
| cmd->GetSwitchValueASCII(switches::kUseGL); |
| |
| if (cmd->HasSwitch(switches::kUseGL) && |
| requested_implementation_gl_name == "any") { |
| return true; |
| } else { |
| return false; |
| } |
| } |
| |
| GLImplementationParts GetRequestedGLImplementation( |
| bool* fallback_to_software_gl) { |
| const base::CommandLine* cmd = base::CommandLine::ForCurrentProcess(); |
| std::string requested_implementation_gl_name = |
| cmd->GetSwitchValueASCII(switches::kUseGL); |
| std::string requested_implementation_angle_name = |
| cmd->GetSwitchValueASCII(switches::kUseANGLE); |
| |
| // If --use-angle was specified but --use-gl was not, assume --use-gl=angle |
| if (cmd->HasSwitch(switches::kUseANGLE) && |
| !cmd->HasSwitch(switches::kUseGL)) { |
| requested_implementation_gl_name = "angle"; |
| } |
| |
| if (requested_implementation_gl_name == kGLImplementationDisabledName) { |
| return GLImplementationParts(kGLImplementationDisabled); |
| } |
| |
| std::vector<GLImplementationParts> allowed_impls = |
| GetAllowedGLImplementations(); |
| |
| if (cmd->HasSwitch(switches::kDisableES3GLContext)) { |
| auto iter = |
| std::find(allowed_impls.begin(), allowed_impls.end(), |
| GLImplementationParts(kGLImplementationDesktopGLCoreProfile)); |
| if (iter != allowed_impls.end()) |
| allowed_impls.erase(iter); |
| } |
| |
| if (cmd->HasSwitch(switches::kDisableES3GLContextForTesting)) { |
| GLVersionInfo::DisableES3ForTesting(); |
| } |
| |
| // If the passthrough command decoder is enabled, put ANGLE first if allowed |
| if (g_is_angle_enabled && gl::UsePassthroughCommandDecoder(cmd)) { |
| std::vector<GLImplementationParts> angle_impls = {}; |
| bool software_gl_allowed = false; |
| bool legacy_software_gl_allowed = false; |
| auto iter = allowed_impls.begin(); |
| while (iter != allowed_impls.end()) { |
| if ((*iter) == GetSoftwareGLImplementation()) { |
| software_gl_allowed = true; |
| allowed_impls.erase(iter); |
| } else if ((*iter) == GetLegacySoftwareGLImplementation()) { |
| legacy_software_gl_allowed = true; |
| allowed_impls.erase(iter); |
| } else if (iter->gl == kGLImplementationEGLANGLE) { |
| angle_impls.emplace_back(*iter); |
| allowed_impls.erase(iter); |
| } else { |
| iter++; |
| } |
| } |
| allowed_impls.insert(allowed_impls.begin(), angle_impls.begin(), |
| angle_impls.end()); |
| // Insert software implementations at the end, after all other hardware |
| // implementations |
| if (software_gl_allowed) { |
| allowed_impls.emplace_back(GetSoftwareGLImplementation()); |
| } |
| if (legacy_software_gl_allowed) { |
| allowed_impls.emplace_back(GetLegacySoftwareGLImplementation()); |
| } |
| } |
| |
| if (allowed_impls.empty()) { |
| LOG(ERROR) << "List of allowed GL implementations is empty."; |
| return GLImplementationParts(kGLImplementationNone); |
| } |
| |
| // The default implementation is always the first one in list. |
| GLImplementationParts impl = allowed_impls[0]; |
| |
| *fallback_to_software_gl = false; |
| if (cmd->HasSwitch(switches::kOverrideUseSoftwareGLForHeadless)) { |
| impl = GetSoftwareGLImplementationForPlatform(); |
| } else if (cmd->HasSwitch(switches::kOverrideUseSoftwareGLForTests)) { |
| impl = GetSoftwareGLImplementationForPlatform(); |
| } else if (cmd->HasSwitch(switches::kUseGL) || |
| cmd->HasSwitch(switches::kUseANGLE)) { |
| if (requested_implementation_gl_name == "any") { |
| *fallback_to_software_gl = true; |
| } else if ((requested_implementation_gl_name == |
| kGLImplementationSwiftShaderName) || |
| (requested_implementation_gl_name == |
| kGLImplementationSwiftShaderForWebGLName)) { |
| impl = GLImplementationParts(kGLImplementationSwiftShaderGL); |
| } else if ((requested_implementation_gl_name == |
| kGLImplementationANGLEName) && |
| ((requested_implementation_angle_name == |
| kANGLEImplementationSwiftShaderName) || |
| (requested_implementation_angle_name == |
| kANGLEImplementationSwiftShaderForWebGLName))) { |
| impl = GLImplementationParts(ANGLEImplementation::kSwiftShader); |
| } else { |
| impl = GetNamedGLImplementation(requested_implementation_gl_name, |
| requested_implementation_angle_name); |
| if (!impl.IsAllowed(allowed_impls)) { |
| LOG(ERROR) << "Requested GL implementation is not available."; |
| return GLImplementationParts(kGLImplementationNone); |
| } |
| } |
| } |
| |
| return impl; |
| } |
| |
| bool InitializeGLOneOffPlatformHelper(bool init_extensions) { |
| TRACE_EVENT1("gpu,startup", "gl::init::InitializeGLOneOffPlatformHelper", |
| "init_extensions", init_extensions); |
| |
| bool fallback_to_software_gl = ShouldFallbackToSoftwareGL(); |
| const base::CommandLine* cmd = base::CommandLine::ForCurrentProcess(); |
| bool disable_gl_drawing = cmd->HasSwitch(switches::kDisableGLDrawingForTests); |
| |
| return InitializeGLOneOffPlatformImplementation( |
| fallback_to_software_gl, disable_gl_drawing, init_extensions); |
| } |
| |
| } // namespace |
| |
| GLImplementationParts GetSoftwareGLImplementationForPlatform() { |
| #if defined(OS_WIN) || defined(OS_LINUX) |
| return GetSoftwareGLImplementation(); |
| #else |
| return GetLegacySoftwareGLImplementation(); |
| #endif |
| } |
| |
| bool InitializeGLOneOff() { |
| TRACE_EVENT0("gpu,startup", "gl::init::InitializeOneOff"); |
| |
| if (!InitializeStaticGLBindingsOneOff()) |
| return false; |
| if (GetGLImplementation() == kGLImplementationDisabled) |
| return true; |
| |
| return InitializeGLOneOffPlatformHelper(true); |
| } |
| |
| bool InitializeGLNoExtensionsOneOff(bool init_bindings) { |
| TRACE_EVENT1("gpu,startup", "gl::init::InitializeNoExtensionsOneOff", |
| "init_bindings", init_bindings); |
| if (init_bindings) { |
| if (!InitializeStaticGLBindingsOneOff()) |
| return false; |
| if (GetGLImplementation() == kGLImplementationDisabled) |
| return true; |
| } |
| |
| return InitializeGLOneOffPlatformHelper(false); |
| } |
| |
| bool InitializeStaticGLBindingsOneOff() { |
| DCHECK_EQ(kGLImplementationNone, GetGLImplementation()); |
| |
| bool fallback_to_software_gl = false; |
| GLImplementationParts impl = |
| GetRequestedGLImplementation(&fallback_to_software_gl); |
| if (impl.gl == gl::kGLImplementationDisabled) { |
| gl::SetGLImplementation(gl::kGLImplementationDisabled); |
| return true; |
| } else if (impl.gl == gl::kGLImplementationNone) { |
| return false; |
| } |
| |
| return InitializeStaticGLBindingsImplementation(impl, |
| fallback_to_software_gl); |
| } |
| |
| bool InitializeStaticGLBindingsImplementation(GLImplementationParts impl, |
| bool fallback_to_software_gl) { |
| if (IsSoftwareGLImplementation(impl)) |
| fallback_to_software_gl = false; |
| |
| bool initialized = InitializeStaticGLBindings(impl); |
| if (!initialized && fallback_to_software_gl) { |
| ShutdownGL(/*due_to_fallback*/ true); |
| initialized = |
| InitializeStaticGLBindings(GetSoftwareGLImplementationForPlatform()); |
| } |
| if (!initialized) { |
| ShutdownGL(/*due_to_fallback*/ false); |
| return false; |
| } |
| return true; |
| } |
| |
| bool InitializeGLOneOffPlatformImplementation(bool fallback_to_software_gl, |
| bool disable_gl_drawing, |
| bool init_extensions) { |
| if (IsSoftwareGLImplementation(GetGLImplementationParts())) |
| fallback_to_software_gl = false; |
| |
| bool initialized = InitializeGLOneOffPlatform(); |
| if (!initialized && fallback_to_software_gl) { |
| ShutdownGL(/*due_to_fallback*/ true); |
| initialized = |
| InitializeStaticGLBindings(GetSoftwareGLImplementationForPlatform()) && |
| InitializeGLOneOffPlatform(); |
| } |
| if (initialized && init_extensions) { |
| initialized = InitializeExtensionSettingsOneOffPlatform(); |
| } |
| |
| if (!initialized) |
| ShutdownGL(false); |
| |
| if (initialized) { |
| DVLOG(1) << "Using " |
| << GetGLImplementationGLName(GetGLImplementationParts()) |
| << " GL implementation."; |
| if (disable_gl_drawing) |
| InitializeNullDrawGLBindings(); |
| } |
| return initialized; |
| } |
| |
| void ShutdownGL(bool due_to_fallback) { |
| ShutdownGLPlatform(); |
| |
| UnloadGLNativeLibraries(due_to_fallback); |
| SetGLImplementation(kGLImplementationNone); |
| } |
| |
| scoped_refptr<GLSurface> CreateOffscreenGLSurface(const gfx::Size& size) { |
| return CreateOffscreenGLSurfaceWithFormat(size, GLSurfaceFormat()); |
| } |
| |
| void DisableANGLE() { |
| DCHECK_NE(GetGLImplementation(), kGLImplementationEGLANGLE); |
| g_is_angle_enabled = false; |
| } |
| |
| } // namespace init |
| } // namespace gl |