| // Copyright (c) 2013 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_driver_bug_list.h" |
| |
| #include "base/logging.h" |
| #include "base/stl_util.h" |
| #include "gpu/config/gpu_driver_bug_list_autogen.h" |
| #include "gpu/config/gpu_driver_bug_workaround_type.h" |
| #include "gpu/config/gpu_switches.h" |
| #include "gpu/config/gpu_util.h" |
| |
| namespace gpu { |
| |
| namespace { |
| |
| struct GpuDriverBugWorkaroundInfo { |
| GpuDriverBugWorkaroundType type; |
| const char* name; |
| }; |
| |
| const GpuDriverBugWorkaroundInfo kFeatureList[] = { |
| #define GPU_OP(type, name) { type, #name }, |
| GPU_DRIVER_BUG_WORKAROUNDS(GPU_OP) |
| #undef GPU_OP |
| }; |
| |
| } // namespace anonymous |
| |
| GpuDriverBugList::GpuDriverBugList(const GpuControlListData& data) |
| : GpuControlList(data) {} |
| |
| GpuDriverBugList::~GpuDriverBugList() = default; |
| |
| // static |
| std::unique_ptr<GpuDriverBugList> GpuDriverBugList::Create() { |
| GpuControlListData data(kGpuDriverBugListEntryCount, |
| kGpuDriverBugListEntries); |
| return Create(data); |
| } |
| |
| // static |
| std::unique_ptr<GpuDriverBugList> GpuDriverBugList::Create( |
| const GpuControlListData& data) { |
| std::unique_ptr<GpuDriverBugList> list(new GpuDriverBugList(data)); |
| |
| DCHECK_EQ(static_cast<int>(base::size(kFeatureList)), |
| NUMBER_OF_GPU_DRIVER_BUG_WORKAROUND_TYPES); |
| for (int i = 0; i < NUMBER_OF_GPU_DRIVER_BUG_WORKAROUND_TYPES; ++i) { |
| list->AddSupportedFeature(kFeatureList[i].name, |
| kFeatureList[i].type); |
| } |
| return list; |
| } |
| |
| std::string GpuDriverBugWorkaroundTypeToString( |
| GpuDriverBugWorkaroundType type) { |
| if (type < NUMBER_OF_GPU_DRIVER_BUG_WORKAROUND_TYPES) |
| return kFeatureList[type].name; |
| else |
| return "unknown"; |
| } |
| |
| // static |
| void GpuDriverBugList::AppendWorkaroundsFromCommandLine( |
| std::set<int>* workarounds, |
| const base::CommandLine& command_line) { |
| DCHECK(workarounds); |
| for (int i = 0; i < NUMBER_OF_GPU_DRIVER_BUG_WORKAROUND_TYPES; i++) { |
| if (command_line.HasSwitch(kFeatureList[i].name)) { |
| // Check for disabling workaround flag. |
| if (command_line.GetSwitchValueASCII(kFeatureList[i].name) == "0") { |
| workarounds->erase(kFeatureList[i].type); |
| continue; |
| } |
| |
| // Removing conflicting workarounds. |
| switch (kFeatureList[i].type) { |
| case FORCE_DISCRETE_GPU: |
| workarounds->erase(FORCE_INTEGRATED_GPU); |
| workarounds->insert(FORCE_DISCRETE_GPU); |
| break; |
| case FORCE_INTEGRATED_GPU: |
| workarounds->erase(FORCE_DISCRETE_GPU); |
| workarounds->insert(FORCE_INTEGRATED_GPU); |
| break; |
| default: |
| workarounds->insert(kFeatureList[i].type); |
| break; |
| } |
| } |
| } |
| } |
| |
| // static |
| void GpuDriverBugList::AppendAllWorkarounds( |
| std::vector<const char*>* workarounds) { |
| static_assert(std::extent<decltype(kFeatureList)>::value == |
| NUMBER_OF_GPU_DRIVER_BUG_WORKAROUND_TYPES, |
| "Expected kFeatureList to include all gpu workarounds"); |
| |
| DCHECK(workarounds->empty()); |
| workarounds->resize(NUMBER_OF_GPU_DRIVER_BUG_WORKAROUND_TYPES); |
| size_t i = 0; |
| for (const GpuDriverBugWorkaroundInfo& feature : kFeatureList) |
| (*workarounds)[i++] = feature.name; |
| } |
| |
| // static |
| bool GpuDriverBugList::AreEntryIndicesValid( |
| const std::vector<uint32_t>& entry_indices) { |
| return GpuControlList::AreEntryIndicesValid(entry_indices, |
| kGpuDriverBugListEntryCount); |
| } |
| |
| } // namespace gpu |