blob: af5331d274cae76b85d69f16b6f9b9e89b8a583c [file] [log] [blame]
// 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