Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(456)

Side by Side Diff: chrome/browser/gpu_blacklist.cc

Issue 5612002: Blacklist bad GPU drivers: currenly we disable all gpu related features if th... (Closed) Base URL: svn://chrome-svn/chrome/trunk/src/
Patch Set: Created 10 years ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
(Empty)
1 // Copyright (c) 2010 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "chrome/browser/gpu_blacklist.h"
6
7 #include "base/json/json_reader.h"
8 #include "base/logging.h"
9 #include "base/stringprintf.h"
10 #include "base/string_number_conversions.h"
11 #include "base/sys_info.h"
12
13 GpuBlacklist::VersionInfo::VersionInfo(const std::string& version_op,
14 const std::string& version_string,
15 const std::string& version_string2) {
16 op_ = StringToOp(version_op);
17 if (op_ == kUnknown || op_ == kAny)
18 return;
19 version_.reset(Version::GetVersionFromString(version_string));
20 if (version_.get() == NULL) {
21 op_ = kUnknown;
22 return;
23 }
24 if (op_ == kBetween) {
25 version2_.reset(Version::GetVersionFromString(version_string2));
26 if (version2_.get() == NULL)
27 op_ = kUnknown;
28 }
29 }
30
31 bool GpuBlacklist::VersionInfo::Contains(const Version& version) const {
32 if (op_ == kUnknown)
33 return false;
34 if (op_ == kAny)
35 return true;
36 if (op_ == kEQ) {
37 // Handles cases where 10.6 is considered as containing 10.6.*.
38 const std::vector<uint16>& components_reference = version_->components();
39 const std::vector<uint16>& components = version.components();
40 for (size_t i = 0; i < components_reference.size(); ++i) {
41 if (i >= components.size() && components_reference[i] != 0)
42 return false;
43 if (components[i] != components_reference[i])
44 return false;
45 }
46 return true;
47 }
48 int relation = version.CompareTo(*version_);
49 if (op_ == kEQ)
50 return (relation == 0);
51 else if (op_ == kLT)
52 return (relation < 0);
53 else if (op_ == kLE)
54 return (relation <= 0);
55 else if (op_ == kGT)
56 return (relation > 0);
57 else if (op_ == kGE)
58 return (relation >= 0);
59 // op_ == kBetween
60 if (relation < 0)
61 return false;
62 return version.CompareTo(*version2_) <= 0;
63 }
64
65 bool GpuBlacklist::VersionInfo::IsValid() const {
66 return op_ != kUnknown;
67 }
68
69 GpuBlacklist::VersionInfo::Op GpuBlacklist::VersionInfo::StringToOp(
70 const std::string& version_op) {
71 if (version_op == "=")
72 return kEQ;
73 else if (version_op == "<")
74 return kLT;
75 else if (version_op == "<=")
76 return kLE;
77 else if (version_op == ">")
78 return kGT;
79 else if (version_op == ">=")
80 return kGE;
81 else if (version_op == "any")
82 return kAny;
83 else if (version_op == "between")
84 return kBetween;
85 return kUnknown;
86 }
87
88 GpuBlacklist::OsInfo::OsInfo(const std::string& os,
89 const std::string& version_op,
90 const std::string& version_string,
91 const std::string& version_string2) {
92 type_ = StringToOsType(os);
93 if (type_ != kOsUnknown) {
94 version_info_.reset(
95 new VersionInfo(version_op, version_string, version_string2));
96 }
97 }
98
99 bool GpuBlacklist::OsInfo::Contains(OsType type,
100 const Version& version) const {
101 if (!IsValid())
102 return false;
103 if (type_ != type && type_ != kOsAny)
104 return false;
105 return version_info_->Contains(version);
106 }
107
108 bool GpuBlacklist::OsInfo::IsValid() const {
109 return type_ != kOsUnknown && version_info_->IsValid();
110 }
111
112 GpuBlacklist::OsType GpuBlacklist::OsInfo::type() const {
113 return type_;
114 }
115
116 GpuBlacklist::GpuBlacklistEntry*
117 GpuBlacklist::GpuBlacklistEntry::GetGpuBlacklistEntryFromValue(
118 DictionaryValue* value) {
119 if (value == NULL)
120 return NULL;
121
122 GpuBlacklistEntry* entry = new GpuBlacklistEntry();
123
124 DictionaryValue* os_value = NULL;
125 if (value->GetDictionary("os", &os_value)) {
126 std::string os_type = "";
127 std::string os_version_op = "any";
128 std::string os_version_string = "";
129 std::string os_version_string2 = "";
130 os_value->GetString("type", &os_type);
131 DictionaryValue* os_version_value = NULL;
132 if (os_value->GetDictionary("version", &os_version_value)) {
133 os_version_value->GetString("op", &os_version_op);
134 os_version_value->GetString("number", &os_version_string);
135 os_version_value->GetString("number2", &os_version_string2);
136 }
137 if (!entry->SetOsInfo(os_type, os_version_op, os_version_string,
138 os_version_string2)) {
139 delete entry;
140 return NULL;
141 }
142 }
143
144 std::string vendor_id = "";
145 if (value->GetString("vendor_id", &vendor_id)) {
146 if (!entry->SetVendorId(vendor_id)) {
147 delete entry;
148 return NULL;
149 }
150 }
151
152 std::string device_id = "";
153 if (value->GetString("device_id", &device_id)) {
154 if (!entry->SetDeviceId(device_id)) {
155 delete entry;
156 return NULL;
157 }
158 }
159
160 DictionaryValue* driver_version_value = NULL;
161 if (value->GetDictionary("driver_version", &driver_version_value)) {
162 std::string driver_version_op = "any";
163 std::string driver_version_string = "";
164 std::string driver_version_string2 = "";
165 driver_version_value->GetString("op", &driver_version_op);
166 driver_version_value->GetString("number", &driver_version_string);
167 driver_version_value->GetString("number2", &driver_version_string2);
168 if (!entry->SetDriverVersionInfo(driver_version_op, driver_version_string,
169 driver_version_string2)) {
170 delete entry;
171 return NULL;
172 }
173 }
174
175 ListValue* blacklist_value = NULL;
176 if (!value->GetList("blacklist", &blacklist_value)) {
177 delete entry;
178 return NULL;
179 }
180 std::vector<std::string> blacklist;
181 for (size_t i = 0; i < blacklist_value->GetSize(); ++i) {
182 std::string feature = "";
183 if (blacklist_value->GetString(i, &feature)) {
184 blacklist.push_back(feature);
185 } else {
186 delete entry;
187 return NULL;
188 }
189 }
190 if (!entry->SetBlacklistedFeatures(blacklist)) {
191 delete entry;
192 return NULL;
193 }
194
195 return entry;
196 }
197
198 GpuBlacklist::GpuBlacklistEntry::GpuBlacklistEntry()
199 : vendor_id_(0),
200 device_id_(0) {
201 }
202
203 bool GpuBlacklist::GpuBlacklistEntry::SetOsInfo(
204 const std::string& os,
205 const std::string& version_op,
206 const std::string& version_string,
207 const std::string& version_string2) {
208 os_info_.reset(new OsInfo(os, version_op, version_string, version_string2));
209 return os_info_->IsValid();
210 }
211
212 bool GpuBlacklist::GpuBlacklistEntry::SetVendorId(
213 const std::string& vendor_id_string) {
214 vendor_id_ = 0;
215 return base::HexStringToInt(vendor_id_string,
216 reinterpret_cast<int*>(&vendor_id_));
217 }
218
219 bool GpuBlacklist::GpuBlacklistEntry::SetDeviceId(
220 const std::string& device_id_string) {
221 device_id_ = 0;
222 return base::HexStringToInt(device_id_string,
223 reinterpret_cast<int*>(&device_id_));
224 }
225
226 bool GpuBlacklist::GpuBlacklistEntry::SetDriverVersionInfo(
227 const std::string& version_op,
228 const std::string& version_string,
229 const std::string& version_string2) {
230 driver_version_info_.reset(
231 new VersionInfo(version_op, version_string, version_string2));
232 return driver_version_info_->IsValid();
233 }
234
235 bool GpuBlacklist::GpuBlacklistEntry::SetBlacklistedFeatures(
236 const std::vector<std::string>& blacklisted_features) {
237 size_t size = blacklisted_features.size();
238 if (size == 0)
239 return false;
240 bool accelerated_2d_canvas = false;
241 bool accelerated_compositing = false;
242 bool webgl = false;
243 for (size_t i = 0; i < size; ++i) {
244 GpuFeatureFlags::GpuFeatureType type =
245 GpuFeatureFlags::StringToGpuFeatureType(blacklisted_features[i]);
246 switch (type) {
247 case GpuFeatureFlags::kGpuFeatureAny:
248 if (size == 1) {
249 feature_flags_.reset(new GpuFeatureFlags());
250 feature_flags_->SetFlags(true, true, true);
251 return true;
252 }
253 return false;
254 case GpuFeatureFlags::kGpuFeatureAccelerated2dCanvas:
255 accelerated_2d_canvas = true;
256 break;
257 case GpuFeatureFlags::kGpuFeatureAcceleratedCompositing:
258 accelerated_compositing = true;
259 break;
260 case GpuFeatureFlags::kGpuFeatureWebgl:
261 webgl = true;
262 break;
263 default:
264 return false;
265 }
266 }
267 feature_flags_.reset(new GpuFeatureFlags());
268 feature_flags_->SetFlags(accelerated_2d_canvas,
269 accelerated_compositing,
270 webgl);
271 return true;
272 }
273
274 bool GpuBlacklist::GpuBlacklistEntry::Contains(
275 OsType os_type, const Version& os_version,
276 uint32 vendor_id, uint32 device_id,
277 const Version& driver_version) const {
278 DCHECK(os_type != kOsAny);
279 if (os_info_.get() != NULL && !os_info_->Contains(os_type, os_version))
280 return false;
281 if (vendor_id_ != 0 && vendor_id_ != vendor_id)
282 return false;
283 if (device_id_ != 0 && device_id_ != device_id)
284 return false;
285 if (driver_version_info_.get() == NULL)
286 return true;
287 return driver_version_info_->Contains(driver_version);
288 }
289
290 GpuBlacklist::OsType GpuBlacklist::GpuBlacklistEntry::GetOsType() const {
291 if (os_info_.get() == NULL)
292 return kOsUnknown;
293 return os_info_->type();
294 }
295
296 GpuFeatureFlags GpuBlacklist::GpuBlacklistEntry::GetGpuFeatureFlags() const {
297 return *feature_flags_;
298 }
299
300 GpuBlacklist::GpuBlacklist() {
301 }
302
303 GpuBlacklist::~GpuBlacklist() {
304 Clear();
305 }
306
307 bool GpuBlacklist::LoadGpuBlacklist(const std::string& json_context) {
308 std::vector<GpuBlacklistEntry*> entries;
309 scoped_ptr<Value> root;
310 root.reset(base::JSONReader::Read(json_context, false));
311 if (root.get() == NULL || !root->IsType(Value::TYPE_DICTIONARY))
312 return false;
313
314 ListValue* list = NULL;
315 static_cast<DictionaryValue*>(root.get())->GetList("entries", &list);
316 if (list == NULL)
317 return false;
318
319 for (size_t i = 0; i < list->GetSize(); ++i) {
320 DictionaryValue* list_item = NULL;
321 bool valid = list->GetDictionary(i, &list_item);
322 if (!valid)
323 break;
324 GpuBlacklistEntry* entry =
325 GpuBlacklistEntry::GetGpuBlacklistEntryFromValue(list_item);
326 if (entry == NULL)
327 break;
328 entries.push_back(entry);
329 }
330
331 if (entries.size() < list->GetSize()) {
332 for (size_t i = 0; i < entries.size(); ++i)
333 delete entries[i];
334 return false;
335 }
336
337 // Only saves entries that affect the current OS (MacOsx/Linux/Win only).
338 OsType os_filter = GetOsType();
339 Clear();
340 for (size_t i = 0; i < entries.size(); ++i) {
341 if (os_filter != kOsUnknown && entries[i]->GetOsType() == os_filter)
342 blacklist_.push_back(entries[i]);
343 }
344 return true;
345 }
346
347 GpuFeatureFlags GpuBlacklist::DetermineGpuFeatureFlags(
348 GpuBlacklist::OsType os,
349 Version* os_version,
350 const GPUInfo& gpu_info) const {
351 GpuFeatureFlags flags;
352 if (os == kOsAny)
353 os = GetOsType();
354 scoped_ptr<Version> my_os_version;
355 if (os_version == NULL) {
356 std::string version_string;
357 #if defined(OS_MACOSX)
358 // Seems like base::SysInfo::OperatingSystemVersion() returns the wrong
359 // version in MacOsx.
360 int32 version_major, version_minor, version_bugfix;
361 base::SysInfo::OperatingSystemVersionNumbers(
362 &version_major, &version_minor, &version_bugfix);
363 version_string = base::StringPrintf("%d.%d.%d",
364 version_major,
365 version_minor,
366 version_bugfix);
367 #else
368 version_string = base::SysInfo::OperatingSystemVersion();
369 #endif
370 my_os_version.reset(Version::GetVersionFromString(version_string));
371 os_version = my_os_version.get();
372 }
373 for (size_t i = 0; i < blacklist_.size(); ++i) {
374 scoped_ptr<Version> driver_version(
375 Version::GetVersionFromString(gpu_info.driver_version()));
376 if (blacklist_[i]->Contains(os, *os_version,
377 gpu_info.vendor_id(), gpu_info.device_id(),
378 *driver_version)) {
379 flags.Combine(blacklist_[i]->GetGpuFeatureFlags());
380 }
381 }
382 return flags;
383 }
384
385 GpuBlacklist::OsType GpuBlacklist::StringToOsType(const std::string& os) {
386 if (os == "win")
387 return kOsWin;
388 else if (os == "macosx")
389 return kOsMacosx;
390 else if (os == "linux")
391 return kOsLinux;
392 else if (os == "any")
393 return kOsAny;
394 return kOsUnknown;
395 }
396
397 GpuBlacklist::OsType GpuBlacklist::GetOsType() {
398 #if defined(OS_WIN)
399 return kOsWin;
400 #elif defined(OS_LINUX)
401 return kOsLinux;
402 #elif defined(OS_MACOSX)
403 return kOsMacosx;
404 #else
405 return kOsUnknown;
406 #endif
407 }
408
409 void GpuBlacklist::Clear() {
410 for (size_t i = 0; i < blacklist_.size(); ++i)
411 delete blacklist_[i];
412 blacklist_.clear();
413 }
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698