feat: Implementation of getGPUInfo API. (#13486)

* Implementation of getGPUInfo API.

* Clear promise set

* Changes to promise usage

* Minor fixes

* Fix linux build

* Update spec

* Fix lint (linter didn't run on windows locally)

* Test running single test for CI

* Update spec
This commit is contained in:
Nitish Sakhawalkar 2018-09-27 07:59:23 -07:00 committed by Samuel Attard
parent 638311b6b3
commit 5c108728d6
13 changed files with 529 additions and 127 deletions

View file

@ -10,6 +10,7 @@
#include "atom/browser/api/atom_api_menu.h" #include "atom/browser/api/atom_api_menu.h"
#include "atom/browser/api/atom_api_session.h" #include "atom/browser/api/atom_api_session.h"
#include "atom/browser/api/atom_api_web_contents.h" #include "atom/browser/api/atom_api_web_contents.h"
#include "atom/browser/api/gpuinfo_manager.h"
#include "atom/browser/atom_browser_context.h" #include "atom/browser/atom_browser_context.h"
#include "atom/browser/atom_browser_main_parts.h" #include "atom/browser/atom_browser_main_parts.h"
#include "atom/browser/login_handler.h" #include "atom/browser/login_handler.h"
@ -548,6 +549,7 @@ App::App(v8::Isolate* isolate) {
static_cast<AtomBrowserClient*>(AtomBrowserClient::Get())->set_delegate(this); static_cast<AtomBrowserClient*>(AtomBrowserClient::Get())->set_delegate(this);
Browser::Get()->AddObserver(this); Browser::Get()->AddObserver(this);
content::GpuDataManager::GetInstance()->AddObserver(this); content::GpuDataManager::GetInstance()->AddObserver(this);
base::ProcessId pid = base::GetCurrentProcId(); base::ProcessId pid = base::GetCurrentProcId();
auto process_metric = std::make_unique<atom::ProcessMetric>( auto process_metric = std::make_unique<atom::ProcessMetric>(
content::PROCESS_TYPE_BROWSER, pid, content::PROCESS_TYPE_BROWSER, pid,
@ -1148,6 +1150,25 @@ v8::Local<v8::Value> App::GetGPUFeatureStatus(v8::Isolate* isolate) {
return mate::ConvertToV8(isolate, status ? *status : temp); return mate::ConvertToV8(isolate, status ? *status : temp);
} }
v8::Local<v8::Promise> App::GetGPUInfo(v8::Isolate* isolate,
const std::string& info_type) {
auto* const gpu_data_manager = content::GpuDataManagerImpl::GetInstance();
scoped_refptr<util::Promise> promise = new util::Promise(isolate);
if ((info_type != "basic" && info_type != "complete") ||
!gpu_data_manager->GpuAccessAllowed(nullptr)) {
promise->Reject("Error fetching GPU Info");
return promise->GetHandle();
}
auto* const info_mgr = GPUInfoManager::GetInstance();
if (info_type == "complete") {
info_mgr->FetchCompleteInfo(promise);
} else /* (info_type == "basic") */ {
info_mgr->FetchBasicInfo(promise);
}
return promise->GetHandle();
}
void App::EnableMixedSandbox(mate::Arguments* args) { void App::EnableMixedSandbox(mate::Arguments* args) {
if (Browser::Get()->is_ready()) { if (Browser::Get()->is_ready()) {
args->ThrowError( args->ThrowError(
@ -1270,6 +1291,7 @@ void App::BuildPrototype(v8::Isolate* isolate,
.SetMethod("getFileIcon", &App::GetFileIcon) .SetMethod("getFileIcon", &App::GetFileIcon)
.SetMethod("getAppMetrics", &App::GetAppMetrics) .SetMethod("getAppMetrics", &App::GetAppMetrics)
.SetMethod("getGPUFeatureStatus", &App::GetGPUFeatureStatus) .SetMethod("getGPUFeatureStatus", &App::GetGPUFeatureStatus)
.SetMethod("getGPUInfo", &App::GetGPUInfo)
// TODO(juturu): Remove in 2.0, deprecate before then with warnings // TODO(juturu): Remove in 2.0, deprecate before then with warnings
#if defined(OS_MACOSX) #if defined(OS_MACOSX)
.SetMethod("moveToApplicationsFolder", &App::MoveToApplicationsFolder) .SetMethod("moveToApplicationsFolder", &App::MoveToApplicationsFolder)

View file

@ -199,6 +199,8 @@ class App : public AtomBrowserClient::Delegate,
std::vector<mate::Dictionary> GetAppMetrics(v8::Isolate* isolate); std::vector<mate::Dictionary> GetAppMetrics(v8::Isolate* isolate);
v8::Local<v8::Value> GetGPUFeatureStatus(v8::Isolate* isolate); v8::Local<v8::Value> GetGPUFeatureStatus(v8::Isolate* isolate);
v8::Local<v8::Promise> GetGPUInfo(v8::Isolate* isolate,
const std::string& info_type);
void EnableMixedSandbox(mate::Arguments* args); void EnableMixedSandbox(mate::Arguments* args);
#if defined(OS_MACOSX) #if defined(OS_MACOSX)

View file

@ -0,0 +1,102 @@
// Copyright (c) 2018 GitHub, Inc.
// Use of this source code is governed by the MIT license that can be
// found in the LICENSE file.
#include "atom/browser/api/gpu_info_enumerator.h"
#include <utility>
namespace atom {
GPUInfoEnumerator::GPUInfoEnumerator()
: value_stack(), current(std::make_unique<base::DictionaryValue>()) {}
GPUInfoEnumerator::~GPUInfoEnumerator() {}
void GPUInfoEnumerator::AddInt64(const char* name, int64_t value) {
current->SetInteger(name, value);
}
void GPUInfoEnumerator::AddInt(const char* name, int value) {
current->SetInteger(name, value);
}
void GPUInfoEnumerator::AddString(const char* name, const std::string& value) {
if (!value.empty())
current->SetString(name, value);
}
void GPUInfoEnumerator::AddBool(const char* name, bool value) {
current->SetBoolean(name, value);
}
void GPUInfoEnumerator::AddTimeDeltaInSecondsF(const char* name,
const base::TimeDelta& value) {
current->SetInteger(name, value.InMilliseconds());
}
void GPUInfoEnumerator::BeginGPUDevice() {
value_stack.push(std::move(current));
current = std::make_unique<base::DictionaryValue>();
}
void GPUInfoEnumerator::EndGPUDevice() {
auto& top_value = value_stack.top();
// GPUDevice can be more than one. So create a list of all.
// The first one is the active GPU device.
if (top_value->HasKey(kGPUDeviceKey)) {
base::ListValue* list;
top_value->GetList(kGPUDeviceKey, &list);
list->Append(std::move(current));
} else {
auto gpus = std::make_unique<base::ListValue>();
gpus->Append(std::move(current));
top_value->SetList(kGPUDeviceKey, std::move(gpus));
}
current = std::move(top_value);
value_stack.pop();
}
void GPUInfoEnumerator::BeginVideoDecodeAcceleratorSupportedProfile() {
value_stack.push(std::move(current));
current = std::make_unique<base::DictionaryValue>();
}
void GPUInfoEnumerator::EndVideoDecodeAcceleratorSupportedProfile() {
auto& top_value = value_stack.top();
top_value->SetDictionary(kVideoDecodeAcceleratorSupportedProfileKey,
std::move(current));
current = std::move(top_value);
value_stack.pop();
}
void GPUInfoEnumerator::BeginVideoEncodeAcceleratorSupportedProfile() {
value_stack.push(std::move(current));
current = std::make_unique<base::DictionaryValue>();
}
void GPUInfoEnumerator::EndVideoEncodeAcceleratorSupportedProfile() {
auto& top_value = value_stack.top();
top_value->SetDictionary(kVideoEncodeAcceleratorSupportedProfileKey,
std::move(current));
current = std::move(top_value);
value_stack.pop();
}
void GPUInfoEnumerator::BeginAuxAttributes() {
value_stack.push(std::move(current));
current = std::make_unique<base::DictionaryValue>();
}
void GPUInfoEnumerator::EndAuxAttributes() {
auto& top_value = value_stack.top();
top_value->SetDictionary(kAuxAttributesKey, std::move(current));
current = std::move(top_value);
value_stack.pop();
}
std::unique_ptr<base::DictionaryValue> GPUInfoEnumerator::GetDictionary() {
return std::move(current);
}
} // namespace atom

View file

@ -0,0 +1,53 @@
// Copyright (c) 2018 GitHub, Inc.
// Use of this source code is governed by the MIT license that can be
// found in the LICENSE file.
#ifndef ATOM_BROWSER_API_GPU_INFO_ENUMERATOR_H_
#define ATOM_BROWSER_API_GPU_INFO_ENUMERATOR_H_
#include <memory>
#include <stack>
#include <string>
#include "base/values.h"
#include "gpu/config/gpu_info.h"
namespace atom {
// This class implements the enumerator for reading all the attributes in
// GPUInfo into a dictionary.
class GPUInfoEnumerator final : public gpu::GPUInfo::Enumerator {
const char* kGPUDeviceKey = "gpuDevice";
const char* kVideoDecodeAcceleratorSupportedProfileKey =
"videoDecodeAcceleratorSupportedProfile";
const char* kVideoEncodeAcceleratorSupportedProfileKey =
"videoEncodeAcceleratorSupportedProfile";
const char* kAuxAttributesKey = "auxAttributes";
public:
GPUInfoEnumerator();
~GPUInfoEnumerator() override;
void AddInt64(const char* name, int64_t value) override;
void AddInt(const char* name, int value) override;
void AddString(const char* name, const std::string& value) override;
void AddBool(const char* name, bool value) override;
void AddTimeDeltaInSecondsF(const char* name,
const base::TimeDelta& value) override;
void BeginGPUDevice() override;
void EndGPUDevice() override;
void BeginVideoDecodeAcceleratorSupportedProfile() override;
void EndVideoDecodeAcceleratorSupportedProfile() override;
void BeginVideoEncodeAcceleratorSupportedProfile() override;
void EndVideoEncodeAcceleratorSupportedProfile() override;
void BeginAuxAttributes() override;
void EndAuxAttributes() override;
std::unique_ptr<base::DictionaryValue> GetDictionary();
private:
// The stack is used to manage nested values
std::stack<std::unique_ptr<base::DictionaryValue>> value_stack;
std::unique_ptr<base::DictionaryValue> current;
};
} // namespace atom
#endif // ATOM_BROWSER_API_GPU_INFO_ENUMERATOR_H_

View file

@ -0,0 +1,93 @@
// Copyright (c) 2018 GitHub, Inc.
// Use of this source code is governed by the MIT license that can be
// found in the LICENSE file.
#include "atom/browser/api/gpuinfo_manager.h"
#include "atom/browser/api/gpu_info_enumerator.h"
#include "base/memory/singleton.h"
#include "base/threading/thread_task_runner_handle.h"
#include "content/public/browser/browser_thread.h"
#include "gpu/config/gpu_info_collector.h"
namespace atom {
GPUInfoManager* GPUInfoManager::GetInstance() {
return base::Singleton<GPUInfoManager>::get();
}
GPUInfoManager::GPUInfoManager()
: gpu_data_manager_(content::GpuDataManagerImpl::GetInstance()) {
gpu_data_manager_->AddObserver(this);
}
GPUInfoManager::~GPUInfoManager() {
content::GpuDataManagerImpl::GetInstance()->RemoveObserver(this);
}
// Based on
// https://chromium.googlesource.com/chromium/src.git/+/66.0.3359.181/content/browser/gpu/gpu_data_manager_impl_private.cc#810
bool GPUInfoManager::NeedsCompleteGpuInfoCollection() {
#if defined(OS_MACOSX)
return gpu_data_manager_->GetGPUInfo().gl_vendor.empty();
#elif defined(OS_WIN)
const auto& gpu_info = gpu_data_manager_->GetGPUInfo();
return (gpu_info.dx_diagnostics.values.empty() &&
gpu_info.dx_diagnostics.children.empty());
#else
return false;
#endif
}
// Should be posted to the task runner
void GPUInfoManager::ProcessCompleteInfo() {
const auto result = EnumerateGPUInfo(gpu_data_manager_->GetGPUInfo());
// We have received the complete information, resolve all promises that
// were waiting for this info.
for (const auto& promise : complete_info_promise_set_) {
promise->Resolve(*result);
}
complete_info_promise_set_.clear();
}
void GPUInfoManager::OnGpuInfoUpdate() {
// Ignore if called when not asked for complete GPUInfo
if (NeedsCompleteGpuInfoCollection())
return;
base::ThreadTaskRunnerHandle::Get()->PostTask(
FROM_HERE, base::BindOnce(&GPUInfoManager::ProcessCompleteInfo,
base::Unretained(this)));
}
// Should be posted to the task runner
void GPUInfoManager::CompleteInfoFetcher(scoped_refptr<util::Promise> promise) {
complete_info_promise_set_.push_back(promise);
if (NeedsCompleteGpuInfoCollection()) {
gpu_data_manager_->RequestCompleteGpuInfoIfNeeded();
} else {
GPUInfoManager::OnGpuInfoUpdate();
}
}
void GPUInfoManager::FetchCompleteInfo(scoped_refptr<util::Promise> promise) {
base::ThreadTaskRunnerHandle::Get()->PostTask(
FROM_HERE, base::BindOnce(&GPUInfoManager::CompleteInfoFetcher,
base::Unretained(this), promise));
}
// This fetches the info synchronously, so no need to post to the task queue.
// There cannot be multiple promises as they are resolved synchronously.
void GPUInfoManager::FetchBasicInfo(scoped_refptr<util::Promise> promise) {
gpu::GPUInfo gpu_info;
CollectBasicGraphicsInfo(&gpu_info);
promise->Resolve(*EnumerateGPUInfo(gpu_info));
}
std::unique_ptr<base::DictionaryValue> GPUInfoManager::EnumerateGPUInfo(
gpu::GPUInfo gpu_info) const {
GPUInfoEnumerator enumerator;
gpu_info.EnumerateFields(&enumerator);
return enumerator.GetDictionary();
}
} // namespace atom

View file

@ -0,0 +1,49 @@
// Copyright (c) 2018 GitHub, Inc.
// Use of this source code is governed by the MIT license that can be
// found in the LICENSE file.
#ifndef ATOM_BROWSER_API_GPUINFO_MANAGER_H_
#define ATOM_BROWSER_API_GPUINFO_MANAGER_H_
#include <memory>
#include <unordered_set>
#include <vector>
#include "atom/common/native_mate_converters/value_converter.h"
#include "atom/common/promise_util.h"
#include "content/browser/gpu/gpu_data_manager_impl.h"
#include "content/public/browser/gpu_data_manager.h"
#include "content/public/browser/gpu_data_manager_observer.h"
namespace atom {
// GPUInfoManager is a singleton used to manage and fetch GPUInfo
class GPUInfoManager : public content::GpuDataManagerObserver {
public:
static GPUInfoManager* GetInstance();
GPUInfoManager();
~GPUInfoManager() override;
bool NeedsCompleteGpuInfoCollection();
void FetchCompleteInfo(scoped_refptr<util::Promise> promise);
void FetchBasicInfo(scoped_refptr<util::Promise> promise);
void OnGpuInfoUpdate() override;
private:
std::unique_ptr<base::DictionaryValue> EnumerateGPUInfo(
gpu::GPUInfo gpu_info) const;
// These should be posted to the task queue
void CompleteInfoFetcher(scoped_refptr<util::Promise> promise);
void ProcessCompleteInfo();
// This set maintains all the promises that should be fulfilled
// once we have the complete information data
std::vector<scoped_refptr<util::Promise>> complete_info_promise_set_;
content::GpuDataManager* gpu_data_manager_;
DISALLOW_COPY_AND_ASSIGN(GPUInfoManager);
};
} // namespace atom
#endif // ATOM_BROWSER_API_GPUINFO_MANAGER_H_

View file

@ -25,7 +25,7 @@ v8::Maybe<bool> Promise::RejectWithErrorMessage(const std::string& string) {
mate::ConvertToV8(isolate(), error)); mate::ConvertToV8(isolate(), error));
} }
v8::Local<v8::Object> Promise::GetHandle() const { v8::Local<v8::Promise> Promise::GetHandle() const {
return GetInner()->GetPromise(); return GetInner()->GetPromise();
} }

View file

@ -14,14 +14,13 @@ namespace atom {
namespace util { namespace util {
class Promise { class Promise : public base::RefCounted<Promise> {
public: public:
explicit Promise(v8::Isolate* isolate); explicit Promise(v8::Isolate* isolate);
~Promise();
v8::Isolate* isolate() const { return isolate_; } v8::Isolate* isolate() const { return isolate_; }
virtual v8::Local<v8::Object> GetHandle() const; virtual v8::Local<v8::Promise> GetHandle() const;
v8::Maybe<bool> Resolve() { v8::Maybe<bool> Resolve() {
return GetInner()->Resolve(isolate()->GetCurrentContext(), return GetInner()->Resolve(isolate()->GetCurrentContext(),
@ -34,13 +33,13 @@ class Promise {
} }
template <typename T> template <typename T>
v8::Maybe<bool> Resolve(T* value) { v8::Maybe<bool> Resolve(const T& value) {
return GetInner()->Resolve(isolate()->GetCurrentContext(), return GetInner()->Resolve(isolate()->GetCurrentContext(),
mate::ConvertToV8(isolate(), value)); mate::ConvertToV8(isolate(), value));
} }
template <typename T> template <typename T>
v8::Maybe<bool> Reject(T* value) { v8::Maybe<bool> Reject(const T& value) {
return GetInner()->Reject(isolate()->GetCurrentContext(), return GetInner()->Reject(isolate()->GetCurrentContext(),
mate::ConvertToV8(isolate(), value)); mate::ConvertToV8(isolate(), value));
} }
@ -48,6 +47,8 @@ class Promise {
v8::Maybe<bool> RejectWithErrorMessage(const std::string& error); v8::Maybe<bool> RejectWithErrorMessage(const std::string& error);
protected: protected:
virtual ~Promise();
friend class base::RefCounted<Promise>;
v8::Isolate* isolate_; v8::Isolate* isolate_;
private: private:

View file

@ -905,6 +905,41 @@ Returns [`ProcessMetric[]`](structures/process-metric.md): Array of `ProcessMetr
Returns [`GPUFeatureStatus`](structures/gpu-feature-status.md) - The Graphics Feature Status from `chrome://gpu/`. Returns [`GPUFeatureStatus`](structures/gpu-feature-status.md) - The Graphics Feature Status from `chrome://gpu/`.
### `app.getGPUInfo(infoType)`
* `infoType` String - Values can be either `basic` for basic info or `complete` for complete info.
Returns `Promise`
For `infoType` equal to `complete`:
Promise is fulfilled with `Object` containing all the GPU Information as in [chromium's GPUInfo object](https://chromium.googlesource.com/chromium/src.git/+/66.0.3359.181/gpu/config/gpu_info.cc). This includes the version and driver information that's shown on `chrome://gpu` page.
For `infoType` equal to `basic`:
Promise is fulfilled with `Object` containing fewer attributes than when requested with `complete`. Here's an example of basic response:
```js
{ auxAttributes:
{ amdSwitchable: true,
canSupportThreadedTextureMailbox: false,
directComposition: false,
directRendering: true,
glResetNotificationStrategy: 0,
inProcessGpu: true,
initializationTime: 0,
jpegDecodeAcceleratorSupported: false,
optimus: false,
passthroughCmdDecoder: false,
sandboxed: false,
softwareRendering: false,
supportsOverlays: false,
videoDecodeAcceleratorFlags: 0 },
gpuDevice:
[ { active: true, deviceId: 26657, vendorId: 4098 },
{ active: false, deviceId: 3366, vendorId: 32902 } ],
machineModelName: 'MacBookPro',
machineModelVersion: '11.5' }
```
Using `basic` should be preferred if only basic information like `vendorId` or `driverId` is needed.
### `app.setBadgeCount(count)` _Linux_ _macOS_ ### `app.setBadgeCount(count)` _Linux_ _macOS_
* `count` Integer * `count` Integer

View file

@ -191,6 +191,10 @@ filenames = {
"atom/browser/api/trackable_object.h", "atom/browser/api/trackable_object.h",
"atom/browser/api/frame_subscriber.cc", "atom/browser/api/frame_subscriber.cc",
"atom/browser/api/frame_subscriber.h", "atom/browser/api/frame_subscriber.h",
"atom/browser/api/gpu_info_enumerator.cc",
"atom/browser/api/gpu_info_enumerator.h",
"atom/browser/api/gpuinfo_manager.cc",
"atom/browser/api/gpuinfo_manager.h",
"atom/browser/api/save_page_handler.cc", "atom/browser/api/save_page_handler.cc",
"atom/browser/api/save_page_handler.h", "atom/browser/api/save_page_handler.h",
"atom/browser/auto_updater.cc", "atom/browser/auto_updater.cc",

View file

@ -15,6 +15,7 @@ using v8::Isolate;
using v8::Local; using v8::Local;
using v8::Number; using v8::Number;
using v8::Object; using v8::Object;
using v8::Promise;
using v8::String; using v8::String;
using v8::Value; using v8::Value;
@ -33,11 +34,12 @@ bool Converter<bool>::FromV8(Isolate* isolate, Local<Value> val, bool* out) {
#if !defined(OS_LINUX) && !defined(OS_FREEBSD) #if !defined(OS_LINUX) && !defined(OS_FREEBSD)
Local<Value> Converter<unsigned long>::ToV8(Isolate* isolate, Local<Value> Converter<unsigned long>::ToV8(Isolate* isolate,
unsigned long val) { unsigned long val) {
return v8::Integer::New(isolate, val); return v8::Integer::New(isolate, val);
} }
bool Converter<unsigned long>::FromV8(Isolate* isolate, Local<Value> val, bool Converter<unsigned long>::FromV8(Isolate* isolate,
Local<Value> val,
unsigned long* out) { unsigned long* out) {
if (!val->IsNumber()) if (!val->IsNumber())
return false; return false;
@ -50,7 +52,8 @@ Local<Value> Converter<int32_t>::ToV8(Isolate* isolate, int32_t val) {
return v8::Integer::New(isolate, val); return v8::Integer::New(isolate, val);
} }
bool Converter<int32_t>::FromV8(Isolate* isolate, Local<Value> val, bool Converter<int32_t>::FromV8(Isolate* isolate,
Local<Value> val,
int32_t* out) { int32_t* out) {
if (!val->IsInt32()) if (!val->IsInt32())
return false; return false;
@ -62,7 +65,8 @@ Local<Value> Converter<uint32_t>::ToV8(Isolate* isolate, uint32_t val) {
return v8::Integer::NewFromUnsigned(isolate, val); return v8::Integer::NewFromUnsigned(isolate, val);
} }
bool Converter<uint32_t>::FromV8(Isolate* isolate, Local<Value> val, bool Converter<uint32_t>::FromV8(Isolate* isolate,
Local<Value> val,
uint32_t* out) { uint32_t* out) {
if (!val->IsUint32()) if (!val->IsUint32())
return false; return false;
@ -74,7 +78,8 @@ Local<Value> Converter<int64_t>::ToV8(Isolate* isolate, int64_t val) {
return v8::Number::New(isolate, static_cast<double>(val)); return v8::Number::New(isolate, static_cast<double>(val));
} }
bool Converter<int64_t>::FromV8(Isolate* isolate, Local<Value> val, bool Converter<int64_t>::FromV8(Isolate* isolate,
Local<Value> val,
int64_t* out) { int64_t* out) {
if (!val->IsNumber()) if (!val->IsNumber())
return false; return false;
@ -88,7 +93,8 @@ Local<Value> Converter<uint64_t>::ToV8(Isolate* isolate, uint64_t val) {
return v8::Number::New(isolate, static_cast<double>(val)); return v8::Number::New(isolate, static_cast<double>(val));
} }
bool Converter<uint64_t>::FromV8(Isolate* isolate, Local<Value> val, bool Converter<uint64_t>::FromV8(Isolate* isolate,
Local<Value> val,
uint64_t* out) { uint64_t* out) {
if (!val->IsNumber()) if (!val->IsNumber())
return false; return false;
@ -100,8 +106,7 @@ Local<Value> Converter<float>::ToV8(Isolate* isolate, float val) {
return v8::Number::New(isolate, val); return v8::Number::New(isolate, val);
} }
bool Converter<float>::FromV8(Isolate* isolate, Local<Value> val, bool Converter<float>::FromV8(Isolate* isolate, Local<Value> val, float* out) {
float* out) {
if (!val->IsNumber()) if (!val->IsNumber())
return false; return false;
*out = static_cast<float>(val->NumberValue()); *out = static_cast<float>(val->NumberValue());
@ -112,7 +117,8 @@ Local<Value> Converter<double>::ToV8(Isolate* isolate, double val) {
return v8::Number::New(isolate, val); return v8::Number::New(isolate, val);
} }
bool Converter<double>::FromV8(Isolate* isolate, Local<Value> val, bool Converter<double>::FromV8(Isolate* isolate,
Local<Value> val,
double* out) { double* out) {
if (!val->IsNumber()) if (!val->IsNumber())
return false; return false;
@ -120,25 +126,23 @@ bool Converter<double>::FromV8(Isolate* isolate, Local<Value> val,
return true; return true;
} }
Local<Value> Converter<const char*>::ToV8( Local<Value> Converter<const char*>::ToV8(Isolate* isolate, const char* val) {
Isolate* isolate, const char* val) {
return v8::String::NewFromUtf8(isolate, val); return v8::String::NewFromUtf8(isolate, val);
} }
Local<Value> Converter<base::StringPiece>::ToV8( Local<Value> Converter<base::StringPiece>::ToV8(Isolate* isolate,
Isolate* isolate, const base::StringPiece& val) { const base::StringPiece& val) {
return v8::String::NewFromUtf8(isolate, return v8::String::NewFromUtf8(isolate, val.data(), v8::String::kNormalString,
val.data(),
v8::String::kNormalString,
static_cast<uint32_t>(val.length())); static_cast<uint32_t>(val.length()));
} }
Local<Value> Converter<std::string>::ToV8(Isolate* isolate, Local<Value> Converter<std::string>::ToV8(Isolate* isolate,
const std::string& val) { const std::string& val) {
return Converter<base::StringPiece>::ToV8(isolate, val); return Converter<base::StringPiece>::ToV8(isolate, val);
} }
bool Converter<std::string>::FromV8(Isolate* isolate, Local<Value> val, bool Converter<std::string>::FromV8(Isolate* isolate,
Local<Value> val,
std::string* out) { std::string* out) {
if (!val->IsString()) if (!val->IsString())
return false; return false;
@ -154,83 +158,89 @@ Local<Value> Converter<Local<Function>>::ToV8(Isolate* isolate,
return val; return val;
} }
bool Converter<Local<Function> >::FromV8(Isolate* isolate, Local<Value> val, bool Converter<Local<Function>>::FromV8(Isolate* isolate,
Local<Function>* out) { Local<Value> val,
Local<Function>* out) {
if (!val->IsFunction()) if (!val->IsFunction())
return false; return false;
*out = Local<Function>::Cast(val); *out = Local<Function>::Cast(val);
return true; return true;
} }
Local<Value> Converter<Local<Object> >::ToV8(Isolate* isolate, Local<Value> Converter<Local<Object>>::ToV8(Isolate* isolate,
Local<Object> val) { Local<Object> val) {
return val; return val;
} }
bool Converter<Local<Object> >::FromV8(Isolate* isolate, Local<Value> val, bool Converter<Local<Object>>::FromV8(Isolate* isolate,
Local<Object>* out) { Local<Value> val,
Local<Object>* out) {
if (!val->IsObject()) if (!val->IsObject())
return false; return false;
*out = Local<Object>::Cast(val); *out = Local<Object>::Cast(val);
return true; return true;
} }
Local<Value> Converter<Local<String> >::ToV8(Isolate* isolate, Local<Value> Converter<Local<String>>::ToV8(Isolate* isolate,
Local<String> val) { Local<String> val) {
return val; return val;
} }
bool Converter<Local<String> >::FromV8(Isolate* isolate, Local<Value> val, bool Converter<Local<String>>::FromV8(Isolate* isolate,
Local<String>* out) { Local<Value> val,
Local<String>* out) {
if (!val->IsString()) if (!val->IsString())
return false; return false;
*out = Local<String>::Cast(val); *out = Local<String>::Cast(val);
return true; return true;
} }
Local<Value> Converter<Local<External> >::ToV8(Isolate* isolate, Local<Value> Converter<Local<External>>::ToV8(Isolate* isolate,
Local<External> val) { Local<External> val) {
return val; return val;
} }
bool Converter<Local<External> >::FromV8(Isolate* isolate, bool Converter<Local<External>>::FromV8(Isolate* isolate,
v8::Local<Value> val, v8::Local<Value> val,
Local<External>* out) { Local<External>* out) {
if (!val->IsExternal()) if (!val->IsExternal())
return false; return false;
*out = Local<External>::Cast(val); *out = Local<External>::Cast(val);
return true; return true;
} }
Local<Value> Converter<Local<Array> >::ToV8(Isolate* isolate, Local<Value> Converter<Local<Array>>::ToV8(Isolate* isolate, Local<Array> val) {
Local<Array> val) {
return val; return val;
} }
bool Converter<Local<Array> >::FromV8(Isolate* isolate, bool Converter<Local<Array>>::FromV8(Isolate* isolate,
v8::Local<Value> val, v8::Local<Value> val,
Local<Array>* out) { Local<Array>* out) {
if (!val->IsArray()) if (!val->IsArray())
return false; return false;
*out = Local<Array>::Cast(val); *out = Local<Array>::Cast(val);
return true; return true;
} }
Local<Value> Converter<Local<Value> >::ToV8(Isolate* isolate, Local<Value> Converter<Local<Value>>::ToV8(Isolate* isolate, Local<Value> val) {
Local<Value> val) {
return val; return val;
} }
bool Converter<Local<Value> >::FromV8(Isolate* isolate, Local<Value> val, Local<Promise> Converter<Local<Promise>>::ToV8(Isolate* isolate,
Local<Value>* out) { Local<Promise> val) {
return val;
}
bool Converter<Local<Value>>::FromV8(Isolate* isolate,
Local<Value> val,
Local<Value>* out) {
*out = val; *out = val;
return true; return true;
} }
v8::Local<v8::String> StringToSymbol(v8::Isolate* isolate, v8::Local<v8::String> StringToSymbol(v8::Isolate* isolate,
const base::StringPiece& val) { const base::StringPiece& val) {
return v8::String::NewFromUtf8(isolate, return v8::String::NewFromUtf8(isolate, val.data(),
val.data(),
v8::String::kInternalizedString, v8::String::kInternalizedString,
static_cast<uint32_t>(val.length())); static_cast<uint32_t>(val.length()));
} }

View file

@ -6,16 +6,16 @@
#define NATIVE_MATE_CONVERTER_H_ #define NATIVE_MATE_CONVERTER_H_
#include <map> #include <map>
#include <set>
#include <string> #include <string>
#include <vector> #include <vector>
#include <set>
#include "base/strings/string_piece.h" #include "base/strings/string_piece.h"
#include "v8/include/v8.h" #include "v8/include/v8.h"
namespace mate { namespace mate {
template<typename KeyType> template <typename KeyType>
bool SetProperty(v8::Isolate* isolate, bool SetProperty(v8::Isolate* isolate,
v8::Local<v8::Object> object, v8::Local<v8::Object> object,
KeyType key, KeyType key,
@ -24,188 +24,187 @@ bool SetProperty(v8::Isolate* isolate,
return !maybe.IsNothing() && maybe.FromJust(); return !maybe.IsNothing() && maybe.FromJust();
} }
template<typename T> template <typename T>
struct ToV8ReturnsMaybe { struct ToV8ReturnsMaybe {
static const bool value = false; static const bool value = false;
}; };
template<typename T, typename Enable = void> template <typename T, typename Enable = void>
struct Converter {}; struct Converter {};
template<> template <>
struct Converter<void*> { struct Converter<void*> {
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, void* val) { static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, void* val) {
return v8::Undefined(isolate); return v8::Undefined(isolate);
} }
}; };
template<> template <>
struct Converter<std::nullptr_t> { struct Converter<std::nullptr_t> {
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, std::nullptr_t val) { static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, std::nullptr_t val) {
return v8::Null(isolate); return v8::Null(isolate);
} }
}; };
template<> template <>
struct Converter<bool> { struct Converter<bool> {
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, bool val);
bool val); static bool FromV8(v8::Isolate* isolate, v8::Local<v8::Value> val, bool* out);
static bool FromV8(v8::Isolate* isolate,
v8::Local<v8::Value> val,
bool* out);
}; };
#if !defined(OS_LINUX) && !defined(OS_FREEBSD) #if !defined(OS_LINUX) && !defined(OS_FREEBSD)
template<> template <>
struct Converter<unsigned long> { struct Converter<unsigned long> {
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, unsigned long val);
unsigned long val);
static bool FromV8(v8::Isolate* isolate, static bool FromV8(v8::Isolate* isolate,
v8::Local<v8::Value> val, v8::Local<v8::Value> val,
unsigned long* out); unsigned long* out);
}; };
#endif #endif
template<> template <>
struct Converter<int32_t> { struct Converter<int32_t> {
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, int32_t val);
int32_t val);
static bool FromV8(v8::Isolate* isolate, static bool FromV8(v8::Isolate* isolate,
v8::Local<v8::Value> val, v8::Local<v8::Value> val,
int32_t* out); int32_t* out);
}; };
template<> template <>
struct Converter<uint32_t> { struct Converter<uint32_t> {
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, uint32_t val);
uint32_t val);
static bool FromV8(v8::Isolate* isolate, static bool FromV8(v8::Isolate* isolate,
v8::Local<v8::Value> val, v8::Local<v8::Value> val,
uint32_t* out); uint32_t* out);
}; };
template<> template <>
struct Converter<int64_t> { struct Converter<int64_t> {
// Warning: JavaScript cannot represent 64 integers precisely. // Warning: JavaScript cannot represent 64 integers precisely.
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, int64_t val);
int64_t val);
static bool FromV8(v8::Isolate* isolate, static bool FromV8(v8::Isolate* isolate,
v8::Local<v8::Value> val, v8::Local<v8::Value> val,
int64_t* out); int64_t* out);
}; };
template<> template <>
struct Converter<uint64_t> { struct Converter<uint64_t> {
// Warning: JavaScript cannot represent 64 integers precisely. // Warning: JavaScript cannot represent 64 integers precisely.
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, uint64_t val);
uint64_t val);
static bool FromV8(v8::Isolate* isolate, static bool FromV8(v8::Isolate* isolate,
v8::Local<v8::Value> val, v8::Local<v8::Value> val,
uint64_t* out); uint64_t* out);
}; };
template<> template <>
struct Converter<float> { struct Converter<float> {
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, float val);
float val);
static bool FromV8(v8::Isolate* isolate, static bool FromV8(v8::Isolate* isolate,
v8::Local<v8::Value> val, v8::Local<v8::Value> val,
float* out); float* out);
}; };
template<> template <>
struct Converter<double> { struct Converter<double> {
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, double val);
double val);
static bool FromV8(v8::Isolate* isolate, static bool FromV8(v8::Isolate* isolate,
v8::Local<v8::Value> val, v8::Local<v8::Value> val,
double* out); double* out);
}; };
template<> template <>
struct Converter<const char*> { struct Converter<const char*> {
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, const char* val); static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, const char* val);
}; };
template<> template <>
struct Converter<base::StringPiece> { struct Converter<base::StringPiece> {
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, static v8::Local<v8::Value> ToV8(v8::Isolate* isolate,
const base::StringPiece& val); const base::StringPiece& val);
// No conversion out is possible because StringPiece does not contain storage. // No conversion out is possible because StringPiece does not contain storage.
}; };
template<> template <>
struct Converter<std::string> { struct Converter<std::string> {
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, static v8::Local<v8::Value> ToV8(v8::Isolate* isolate,
const std::string& val); const std::string& val);
static bool FromV8(v8::Isolate* isolate, static bool FromV8(v8::Isolate* isolate,
v8::Local<v8::Value> val, v8::Local<v8::Value> val,
std::string* out); std::string* out);
}; };
v8::Local<v8::String> StringToSymbol(v8::Isolate* isolate, v8::Local<v8::String> StringToSymbol(v8::Isolate* isolate,
const base::StringPiece& input); const base::StringPiece& input);
std::string V8ToString(v8::Local<v8::Value> value); std::string V8ToString(v8::Local<v8::Value> value);
template<> template <>
struct Converter<v8::Local<v8::Function> > { struct Converter<v8::Local<v8::Function>> {
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, static v8::Local<v8::Value> ToV8(v8::Isolate* isolate,
v8::Local<v8::Function> val); v8::Local<v8::Function> val);
static bool FromV8(v8::Isolate* isolate, static bool FromV8(v8::Isolate* isolate,
v8::Local<v8::Value> val, v8::Local<v8::Value> val,
v8::Local<v8::Function>* out); v8::Local<v8::Function>* out);
}; };
template<> template <>
struct Converter<v8::Local<v8::Object> > { struct Converter<v8::Local<v8::Object>> {
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, static v8::Local<v8::Value> ToV8(v8::Isolate* isolate,
v8::Local<v8::Object> val); v8::Local<v8::Object> val);
static bool FromV8(v8::Isolate* isolate, static bool FromV8(v8::Isolate* isolate,
v8::Local<v8::Value> val, v8::Local<v8::Value> val,
v8::Local<v8::Object>* out); v8::Local<v8::Object>* out);
}; };
template<> template <>
struct Converter<v8::Local<v8::String> > { struct Converter<v8::Local<v8::String>> {
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, static v8::Local<v8::Value> ToV8(v8::Isolate* isolate,
v8::Local<v8::String> val); v8::Local<v8::String> val);
static bool FromV8(v8::Isolate* isolate, static bool FromV8(v8::Isolate* isolate,
v8::Local<v8::Value> val, v8::Local<v8::Value> val,
v8::Local<v8::String>* out); v8::Local<v8::String>* out);
}; };
template<> template <>
struct Converter<v8::Local<v8::External> > { struct Converter<v8::Local<v8::External>> {
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, static v8::Local<v8::Value> ToV8(v8::Isolate* isolate,
v8::Local<v8::External> val); v8::Local<v8::External> val);
static bool FromV8(v8::Isolate* isolate, static bool FromV8(v8::Isolate* isolate,
v8::Local<v8::Value> val, v8::Local<v8::Value> val,
v8::Local<v8::External>* out); v8::Local<v8::External>* out);
}; };
template<> template <>
struct Converter<v8::Local<v8::Array> > { struct Converter<v8::Local<v8::Array>> {
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, static v8::Local<v8::Value> ToV8(v8::Isolate* isolate,
v8::Local<v8::Array> val); v8::Local<v8::Array> val);
static bool FromV8(v8::Isolate* isolate, static bool FromV8(v8::Isolate* isolate,
v8::Local<v8::Value> val, v8::Local<v8::Value> val,
v8::Local<v8::Array>* out); v8::Local<v8::Array>* out);
}; };
template<> template <>
struct Converter<v8::Local<v8::Value> > { struct Converter<v8::Local<v8::Value>> {
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, static v8::Local<v8::Value> ToV8(v8::Isolate* isolate,
v8::Local<v8::Value> val); v8::Local<v8::Value> val);
static bool FromV8(v8::Isolate* isolate, static bool FromV8(v8::Isolate* isolate,
v8::Local<v8::Value> val, v8::Local<v8::Value> val,
v8::Local<v8::Value>* out); v8::Local<v8::Value>* out);
}; };
template<typename T> template <>
struct Converter<std::vector<T> > { struct Converter<v8::Local<v8::Promise>> {
static v8::Local<v8::Promise> ToV8(v8::Isolate* isolate,
v8::Local<v8::Promise> val);
// static bool FromV8(v8::Isolate* isolate,
// v8::Local<v8::Value> val,
// v8::Local<v8::Value>* out);
};
template <typename T>
struct Converter<std::vector<T>> {
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, static v8::Local<v8::Value> ToV8(v8::Isolate* isolate,
const std::vector<T>& val) { const std::vector<T>& val) {
v8::Local<v8::Array> result( v8::Local<v8::Array> result(
v8::Array::New(isolate, static_cast<int>(val.size()))); v8::Array::New(isolate, static_cast<int>(val.size())));
for (size_t i = 0; i < val.size(); ++i) { for (size_t i = 0; i < val.size(); ++i) {
@ -235,10 +234,10 @@ struct Converter<std::vector<T> > {
} }
}; };
template<typename T> template <typename T>
struct Converter<std::set<T> > { struct Converter<std::set<T>> {
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, static v8::Local<v8::Value> ToV8(v8::Isolate* isolate,
const std::set<T>& val) { const std::set<T>& val) {
v8::Local<v8::Array> result( v8::Local<v8::Array> result(
v8::Array::New(isolate, static_cast<int>(val.size()))); v8::Array::New(isolate, static_cast<int>(val.size())));
typename std::set<T>::const_iterator it; typename std::set<T>::const_iterator it;
@ -269,11 +268,11 @@ struct Converter<std::set<T> > {
} }
}; };
template<typename T> template <typename T>
struct Converter<std::map<std::string, T> > { struct Converter<std::map<std::string, T>> {
static bool FromV8(v8::Isolate* isolate, static bool FromV8(v8::Isolate* isolate,
v8::Local<v8::Value> val, v8::Local<v8::Value> val,
std::map<std::string, T> * out) { std::map<std::string, T>* out) {
if (!val->IsObject()) if (!val->IsObject())
return false; return false;
@ -288,18 +287,18 @@ struct Converter<std::map<std::string, T> > {
return true; return true;
} }
static v8::Local<v8::Value> ToV8(v8::Isolate* isolate, static v8::Local<v8::Value> ToV8(v8::Isolate* isolate,
const std::map<std::string, T>& val) { const std::map<std::string, T>& val) {
v8::Local<v8::Object> result = v8::Object::New(isolate); v8::Local<v8::Object> result = v8::Object::New(isolate);
for (auto i = val.begin(); i != val.end(); i++) { for (auto i = val.begin(); i != val.end(); i++) {
result->Set(Converter<T>::ToV8(isolate, i->first), result->Set(Converter<T>::ToV8(isolate, i->first),
Converter<T>::ToV8(isolate, i->second)); Converter<T>::ToV8(isolate, i->second));
} }
return result; return result;
} }
}; };
// Convenience functions that deduce T. // Convenience functions that deduce T.
template<typename T> template <typename T>
v8::Local<v8::Value> ConvertToV8(v8::Isolate* isolate, const T& input) { v8::Local<v8::Value> ConvertToV8(v8::Isolate* isolate, const T& input) {
return Converter<T>::ToV8(isolate, input); return Converter<T>::ToV8(isolate, input);
} }
@ -309,13 +308,14 @@ inline v8::Local<v8::Value> ConvertToV8(v8::Isolate* isolate,
return Converter<const char*>::ToV8(isolate, input); return Converter<const char*>::ToV8(isolate, input);
} }
template<typename T> template <typename T>
v8::MaybeLocal<v8::Value> ConvertToV8(v8::Local<v8::Context> context, v8::MaybeLocal<v8::Value> ConvertToV8(v8::Local<v8::Context> context,
const T& input) { const T& input) {
return Converter<T>::ToV8(context, input); return Converter<T>::ToV8(context, input);
} }
template<typename T, bool = ToV8ReturnsMaybe<T>::value> struct ToV8Traits; template <typename T, bool = ToV8ReturnsMaybe<T>::value>
struct ToV8Traits;
template <typename T> template <typename T>
struct ToV8Traits<T, true> { struct ToV8Traits<T, true> {
@ -347,15 +347,15 @@ bool TryConvertToV8(v8::Isolate* isolate,
return ToV8Traits<T>::TryConvertToV8(isolate, input, output); return ToV8Traits<T>::TryConvertToV8(isolate, input, output);
} }
template<typename T> template <typename T>
bool ConvertFromV8(v8::Isolate* isolate, v8::Local<v8::Value> input, bool ConvertFromV8(v8::Isolate* isolate,
v8::Local<v8::Value> input,
T* result) { T* result) {
return Converter<T>::FromV8(isolate, input, result); return Converter<T>::FromV8(isolate, input, result);
} }
inline v8::Local<v8::String> StringToV8( inline v8::Local<v8::String> StringToV8(v8::Isolate* isolate,
v8::Isolate* isolate, const base::StringPiece& input) {
const base::StringPiece& input) {
return ConvertToV8(isolate, input).As<v8::String>(); return ConvertToV8(isolate, input).As<v8::String>();
} }

View file

@ -804,6 +804,37 @@ describe('app module', () => {
}) })
}) })
describe('getGPUInfo() API', () => {
it('succeeds with basic GPUInfo', (done) => {
app.getGPUInfo('basic').then((gpuInfo) => {
// Devices information is always present in the available info
expect(gpuInfo.gpuDevice).to.be.an('array')
expect(gpuInfo.gpuDevice.length).to.be.greaterThan(0)
const device = gpuInfo.gpuDevice[0]
expect(device).to.be.an('object')
expect(device)
.to.have.property('deviceId')
.that.is.a('number')
.not.lessThan(0)
done()
})
})
it('succeeds with complete GPUInfo', (done) => {
app.getGPUInfo('complete').then((completeInfo) => {
// Driver version is present in the complete info
expect(completeInfo.auxAttributes.glVersion).to.be.a('string').that.has.length.greaterThan(0)
done()
})
})
it('fails for invalid info_type', (done) => {
app.getGPUInfo('invalid').catch(() => {
done()
})
})
})
describe('mixed sandbox option', () => { describe('mixed sandbox option', () => {
let appProcess = null let appProcess = null
let server = null let server = null