blob: 5991bdff52c32d09e429067cdfa5a0915931c704 [file] [log] [blame]
[email protected]4d3c27f32012-03-31 09:11:591// Copyright (c) 2012 The Chromium Authors. All rights reserved.
[email protected]df353ab2011-12-16 23:57:572// 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/component_updater/swiftshader_component_installer.h"
6
[email protected]df353ab2011-12-16 23:57:577#include "base/base_paths.h"
[email protected]57999812013-02-24 05:40:528#include "base/bind.h"
[email protected]df353ab2011-12-16 23:57:579#include "base/compiler_specific.h"
[email protected]4d3c27f32012-03-31 09:11:5910#include "base/cpu.h"
[email protected]df353ab2011-12-16 23:57:5711#include "base/file_util.h"
[email protected]25a4c1c2013-06-08 04:53:3612#include "base/files/file_enumerator.h"
[email protected]57999812013-02-24 05:40:5213#include "base/files/file_path.h"
[email protected]df353ab2011-12-16 23:57:5714#include "base/logging.h"
15#include "base/path_service.h"
[email protected]e7463412013-06-10 22:53:4616#include "base/strings/string_util.h"
[email protected]df353ab2011-12-16 23:57:5717#include "base/values.h"
18#include "chrome/browser/component_updater/component_updater_service.h"
19#include "chrome/common/chrome_paths.h"
20#include "content/public/browser/browser_thread.h"
[email protected]79078df2012-02-16 01:22:3221#include "content/public/browser/gpu_data_manager.h"
22#include "content/public/browser/gpu_data_manager_observer.h"
[email protected]d7b5cc72013-05-23 20:05:0023#include "gpu/config/gpu_feature_type.h"
[email protected]df353ab2011-12-16 23:57:5724
25using content::BrowserThread;
[email protected]79078df2012-02-16 01:22:3226using content::GpuDataManager;
[email protected]df353ab2011-12-16 23:57:5727
[email protected]055981f2014-01-17 20:22:3228namespace component_updater {
29
[email protected]df353ab2011-12-16 23:57:5730namespace {
31
32// CRX hash. The extension id is: nhfgdggnnopgbfdlpeoalgcjdgfafocg.
[email protected]d4261bea2013-04-22 05:29:3933const uint8 kSha2Hash[] = {0xd7, 0x56, 0x36, 0x6d, 0xde, 0xf6, 0x15, 0x3b,
[email protected]df353ab2011-12-16 23:57:5734 0xf4, 0xe0, 0xb6, 0x29, 0x36, 0x50, 0x5e, 0x26,
35 0xbd, 0x77, 0x8b, 0x8e, 0x35, 0xc2, 0x7e, 0x43,
36 0x52, 0x47, 0x62, 0xed, 0x12, 0xca, 0xcc, 0x6a};
37
38// File name of the internal SwiftShader plugin on different platforms.
[email protected]650b2d52013-02-10 03:41:4539const base::FilePath::CharType kSwiftShaderEglName[] =
[email protected]df353ab2011-12-16 23:57:5740 FILE_PATH_LITERAL("libegl.dll");
[email protected]650b2d52013-02-10 03:41:4541const base::FilePath::CharType kSwiftShaderGlesName[] =
[email protected]df353ab2011-12-16 23:57:5742 FILE_PATH_LITERAL("libglesv2.dll");
43
44const char kSwiftShaderManifestName[] = "SwiftShader";
45
[email protected]650b2d52013-02-10 03:41:4546const base::FilePath::CharType kSwiftShaderBaseDirectory[] =
[email protected]df353ab2011-12-16 23:57:5747 FILE_PATH_LITERAL("SwiftShader");
48
49// If we don't have a SwiftShader component, this is the version we claim.
50const char kNullVersion[] = "0.0.0.0";
51
52// The base directory on windows looks like:
53// <profile>\AppData\Local\Google\Chrome\User Data\SwiftShader\.
[email protected]650b2d52013-02-10 03:41:4554base::FilePath GetSwiftShaderBaseDirectory() {
55 base::FilePath result;
[email protected]df353ab2011-12-16 23:57:5756 PathService::Get(chrome::DIR_USER_DATA, &result);
57 return result.Append(kSwiftShaderBaseDirectory);
58}
59
60// SwiftShader has version encoded in the path itself
61// so we need to enumerate the directories to find the full path.
62// On success it returns something like:
63// <profile>\AppData\Local\Google\Chrome\User Data\SwiftShader\10.3.44.555\.
[email protected]650b2d52013-02-10 03:41:4564bool GetLatestSwiftShaderDirectory(base::FilePath* result,
[email protected]c5e4a2222014-01-03 16:06:1365 Version* latest,
[email protected]650b2d52013-02-10 03:41:4566 std::vector<base::FilePath>* older_dirs) {
67 base::FilePath base_dir = GetSwiftShaderBaseDirectory();
[email protected]df353ab2011-12-16 23:57:5768 bool found = false;
[email protected]25a4c1c2013-06-08 04:53:3669 base::FileEnumerator
70 file_enumerator(base_dir, false, base::FileEnumerator::DIRECTORIES);
[email protected]650b2d52013-02-10 03:41:4571 for (base::FilePath path = file_enumerator.Next(); !path.value().empty();
[email protected]df353ab2011-12-16 23:57:5772 path = file_enumerator.Next()) {
[email protected]c5e4a2222014-01-03 16:06:1373 Version version(path.BaseName().MaybeAsASCII());
[email protected]df353ab2011-12-16 23:57:5774 if (!version.IsValid())
75 continue;
76 if (version.CompareTo(*latest) > 0 &&
[email protected]7567484142013-07-11 17:36:0777 base::PathExists(path.Append(kSwiftShaderEglName)) &&
78 base::PathExists(path.Append(kSwiftShaderGlesName))) {
[email protected]56177432012-06-06 00:37:4979 if (found && older_dirs)
80 older_dirs->push_back(*result);
[email protected]df353ab2011-12-16 23:57:5781 *latest = version;
82 *result = path;
83 found = true;
[email protected]56177432012-06-06 00:37:4984 } else {
85 if (older_dirs)
86 older_dirs->push_back(path);
[email protected]df353ab2011-12-16 23:57:5787 }
88 }
89 return found;
90}
91
[email protected]650b2d52013-02-10 03:41:4592void RegisterSwiftShaderWithChrome(const base::FilePath& path) {
[email protected]df353ab2011-12-16 23:57:5793 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
94 GpuDataManager::GetInstance()->RegisterSwiftShaderPath(path);
95}
96
[email protected]df353ab2011-12-16 23:57:5797class SwiftShaderComponentInstaller : public ComponentInstaller {
98 public:
[email protected]c5e4a2222014-01-03 16:06:1399 explicit SwiftShaderComponentInstaller(const Version& version);
[email protected]df353ab2011-12-16 23:57:57100
101 virtual ~SwiftShaderComponentInstaller() {}
102
103 virtual void OnUpdateError(int error) OVERRIDE;
104
[email protected]14ca8872013-05-02 01:51:06105 virtual bool Install(const base::DictionaryValue& manifest,
[email protected]650b2d52013-02-10 03:41:45106 const base::FilePath& unpack_path) OVERRIDE;
[email protected]df353ab2011-12-16 23:57:57107
[email protected]e3e696d32013-06-21 20:41:36108 virtual bool GetInstalledFile(const std::string& file,
109 base::FilePath* installed_file) OVERRIDE;
110
[email protected]df353ab2011-12-16 23:57:57111 private:
[email protected]c5e4a2222014-01-03 16:06:13112 Version current_version_;
[email protected]df353ab2011-12-16 23:57:57113};
114
115SwiftShaderComponentInstaller::SwiftShaderComponentInstaller(
[email protected]c5e4a2222014-01-03 16:06:13116 const Version& version) : current_version_(version) {
[email protected]df353ab2011-12-16 23:57:57117 DCHECK(version.IsValid());
118}
119
120void SwiftShaderComponentInstaller::OnUpdateError(int error) {
121 NOTREACHED() << "SwiftShader update error: " << error;
122}
123
[email protected]14ca8872013-05-02 01:51:06124bool SwiftShaderComponentInstaller::Install(
125 const base::DictionaryValue& manifest,
126 const base::FilePath& unpack_path) {
[email protected]df353ab2011-12-16 23:57:57127 std::string name;
[email protected]14ca8872013-05-02 01:51:06128 manifest.GetStringASCII("name", &name);
[email protected]df353ab2011-12-16 23:57:57129 if (name != kSwiftShaderManifestName)
130 return false;
131 std::string proposed_version;
[email protected]14ca8872013-05-02 01:51:06132 manifest.GetStringASCII("version", &proposed_version);
[email protected]c5e4a2222014-01-03 16:06:13133 Version version(proposed_version.c_str());
[email protected]df353ab2011-12-16 23:57:57134 if (!version.IsValid())
135 return false;
136 if (current_version_.CompareTo(version) >= 0)
137 return false;
[email protected]7567484142013-07-11 17:36:07138 if (!base::PathExists(unpack_path.Append(kSwiftShaderEglName)) ||
139 !base::PathExists(unpack_path.Append(kSwiftShaderGlesName)))
[email protected]df353ab2011-12-16 23:57:57140 return false;
141 // Passed the basic tests. Time to install it.
[email protected]650b2d52013-02-10 03:41:45142 base::FilePath path =
[email protected]df353ab2011-12-16 23:57:57143 GetSwiftShaderBaseDirectory().AppendASCII(version.GetString());
[email protected]7567484142013-07-11 17:36:07144 if (base::PathExists(path))
[email protected]df353ab2011-12-16 23:57:57145 return false;
[email protected]5553d5b2013-07-01 23:07:36146 if (!base::Move(unpack_path, path))
[email protected]df353ab2011-12-16 23:57:57147 return false;
148 // Installation is done. Now tell the rest of chrome.
149 current_version_ = version;
150 BrowserThread::PostTask(BrowserThread::UI, FROM_HERE,
151 base::Bind(&RegisterSwiftShaderWithChrome, path));
152 return true;
153}
154
[email protected]e3e696d32013-06-21 20:41:36155bool SwiftShaderComponentInstaller::GetInstalledFile(
156 const std::string& file, base::FilePath* installed_file) {
157 return false;
158}
159
[email protected]df353ab2011-12-16 23:57:57160void FinishSwiftShaderUpdateRegistration(ComponentUpdateService* cus,
[email protected]c5e4a2222014-01-03 16:06:13161 const Version& version) {
[email protected]df353ab2011-12-16 23:57:57162 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
163
164 CrxComponent swiftshader;
165 swiftshader.name = "Swift Shader";
166 swiftshader.installer = new SwiftShaderComponentInstaller(version);
167 swiftshader.version = version;
[email protected]d4261bea2013-04-22 05:29:39168 swiftshader.pk_hash.assign(kSha2Hash, &kSha2Hash[sizeof(kSha2Hash)]);
[email protected]df353ab2011-12-16 23:57:57169 if (cus->RegisterComponent(swiftshader) != ComponentUpdateService::kOk) {
170 NOTREACHED() << "SwiftShader component registration fail";
171 }
172}
173
[email protected]79078df2012-02-16 01:22:32174class UpdateChecker : public content::GpuDataManagerObserver {
[email protected]df353ab2011-12-16 23:57:57175 public:
176 explicit UpdateChecker(ComponentUpdateService* cus);
177
178 virtual void OnGpuInfoUpdate() OVERRIDE;
179
180 private:
181 ComponentUpdateService* cus_;
182};
183
184UpdateChecker::UpdateChecker(ComponentUpdateService* cus)
185 : cus_(cus) {
186}
187
188void UpdateChecker::OnGpuInfoUpdate() {
189 GpuDataManager *gpu_data_manager = GpuDataManager::GetInstance();
190
[email protected]a0401382013-05-07 00:12:55191 if (!gpu_data_manager->GpuAccessAllowed(NULL) ||
[email protected]d7b5cc72013-05-23 20:05:00192 gpu_data_manager->IsFeatureBlacklisted(gpu::GPU_FEATURE_TYPE_WEBGL) ||
[email protected]7dcb85832013-04-23 19:39:00193 gpu_data_manager->ShouldUseSwiftShader()) {
[email protected]df353ab2011-12-16 23:57:57194 gpu_data_manager->RemoveObserver(this);
195 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE));
[email protected]650b2d52013-02-10 03:41:45196 base::FilePath path = GetSwiftShaderBaseDirectory();
[email protected]df353ab2011-12-16 23:57:57197
[email protected]c5e4a2222014-01-03 16:06:13198 Version version(kNullVersion);
[email protected]56177432012-06-06 00:37:49199 GetLatestSwiftShaderDirectory(&path, &version, NULL);
[email protected]df353ab2011-12-16 23:57:57200
201 BrowserThread::PostTask(BrowserThread::UI, FROM_HERE,
202 base::Bind(&FinishSwiftShaderUpdateRegistration, cus_, version));
203 }
204}
205
[email protected]3462b2512014-01-22 16:56:39206#if defined(ENABLE_SWIFTSHADER)
207
[email protected]df353ab2011-12-16 23:57:57208// Check if there already is a version of swiftshader installed,
209// and if so register it.
210void RegisterSwiftShaderPath(ComponentUpdateService* cus) {
211 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE));
[email protected]650b2d52013-02-10 03:41:45212 base::FilePath path = GetSwiftShaderBaseDirectory();
[email protected]7567484142013-07-11 17:36:07213 if (!base::PathExists(path)) {
[email protected]426d1c92013-12-03 20:08:54214 if (!base::CreateDirectory(path)) {
[email protected]df353ab2011-12-16 23:57:57215 NOTREACHED() << "Could not create SwiftShader directory.";
216 return;
217 }
218 }
219
[email protected]c5e4a2222014-01-03 16:06:13220 Version version(kNullVersion);
[email protected]650b2d52013-02-10 03:41:45221 std::vector<base::FilePath> older_dirs;
[email protected]56177432012-06-06 00:37:49222 if (GetLatestSwiftShaderDirectory(&path, &version, &older_dirs))
[email protected]df353ab2011-12-16 23:57:57223 BrowserThread::PostTask(BrowserThread::UI, FROM_HERE,
224 base::Bind(&RegisterSwiftShaderWithChrome, path));
225
226 UpdateChecker *update_checker = new UpdateChecker(cus);
227 GpuDataManager::GetInstance()->AddObserver(update_checker);
228 update_checker->OnGpuInfoUpdate();
229 // We leak update_checker here, because it has to stick around for the life
230 // of the GpuDataManager.
[email protected]56177432012-06-06 00:37:49231
232 // Remove older versions of SwiftShader.
[email protected]650b2d52013-02-10 03:41:45233 for (std::vector<base::FilePath>::iterator iter = older_dirs.begin();
[email protected]56177432012-06-06 00:37:49234 iter != older_dirs.end(); ++iter) {
[email protected]dd3aa792013-07-16 19:10:23235 base::DeleteFile(*iter, true);
[email protected]56177432012-06-06 00:37:49236 }
[email protected]df353ab2011-12-16 23:57:57237}
238
[email protected]3462b2512014-01-22 16:56:39239#endif // ENABLE_SWIFTSHADER
240
241} // namespace
242
[email protected]df353ab2011-12-16 23:57:57243void RegisterSwiftShaderComponent(ComponentUpdateService* cus) {
244#if defined(ENABLE_SWIFTSHADER)
[email protected]4d3c27f32012-03-31 09:11:59245 base::CPU cpu;
246
247 if (!cpu.has_sse2())
248 return;
[email protected]df353ab2011-12-16 23:57:57249 BrowserThread::PostTask(BrowserThread::FILE, FROM_HERE,
250 base::Bind(&RegisterSwiftShaderPath, cus));
251#endif
252}
[email protected]055981f2014-01-17 20:22:32253
254} // namespace component_updater
255