2014-10-31 18:17:05 +00:00
|
|
|
// Copyright (c) 2013 GitHub, Inc.
|
2014-04-25 09:49:37 +00:00
|
|
|
// Use of this source code is governed by the MIT license that can be
|
2013-07-22 08:05:35 +00:00
|
|
|
// found in the LICENSE file.
|
|
|
|
|
2019-06-19 20:46:59 +00:00
|
|
|
#include "shell/common/node_bindings_win.h"
|
2013-07-22 08:05:35 +00:00
|
|
|
|
2013-07-22 08:43:58 +00:00
|
|
|
#include <windows.h>
|
|
|
|
|
|
|
|
#include "base/logging.h"
|
2019-01-10 18:14:04 +00:00
|
|
|
#include "base/system/sys_info.h"
|
2013-07-23 11:19:11 +00:00
|
|
|
|
2019-06-19 21:23:04 +00:00
|
|
|
namespace electron {
|
2013-07-22 08:05:35 +00:00
|
|
|
|
2017-03-08 08:33:44 +00:00
|
|
|
NodeBindingsWin::NodeBindingsWin(BrowserEnvironment browser_env)
|
2018-12-11 02:07:35 +00:00
|
|
|
: NodeBindings(browser_env) {
|
|
|
|
// on single-core the io comp port NumberOfConcurrentThreads needs to be 2
|
|
|
|
// to avoid cpu pegging likely caused by a busy loop in PollEvents
|
|
|
|
if (base::SysInfo::NumberOfProcessors() == 1) {
|
|
|
|
// the expectation is the uv_loop_ has just been initialized
|
|
|
|
// which makes iocp replacement safe
|
|
|
|
CHECK_EQ(0u, uv_loop_->active_handles);
|
|
|
|
CHECK_EQ(0u, uv_loop_->active_reqs.count);
|
|
|
|
|
|
|
|
if (uv_loop_->iocp && uv_loop_->iocp != INVALID_HANDLE_VALUE)
|
|
|
|
CloseHandle(uv_loop_->iocp);
|
|
|
|
uv_loop_->iocp = CreateIoCompletionPort(INVALID_HANDLE_VALUE, NULL, 0, 2);
|
|
|
|
}
|
|
|
|
}
|
2013-07-22 08:05:35 +00:00
|
|
|
|
2021-06-04 04:16:13 +00:00
|
|
|
NodeBindingsWin::~NodeBindingsWin() = default;
|
2013-07-22 08:05:35 +00:00
|
|
|
|
2022-03-21 07:42:22 +00:00
|
|
|
void NodeBindingsWin::PrepareMessageLoop() {
|
|
|
|
// IOCP does not change for the process until the loop is recreated,
|
|
|
|
// we ensure that there is only a single polling thread satisfying
|
|
|
|
// the concurrency limit set from CreateIoCompletionPort call by
|
|
|
|
// uv_loop_init for the lifetime of this process.
|
|
|
|
if (initialized_)
|
|
|
|
return;
|
|
|
|
|
|
|
|
NodeBindings::PrepareMessageLoop();
|
|
|
|
}
|
|
|
|
|
|
|
|
void NodeBindingsWin::RunMessageLoop() {
|
|
|
|
// Avoid calling UvRunOnce if the loop is already active,
|
|
|
|
// otherwise it can lead to situations were the number of active
|
|
|
|
// threads processing on IOCP is greater than the concurrency limit.
|
|
|
|
if (initialized_)
|
|
|
|
return;
|
|
|
|
|
|
|
|
initialized_ = true;
|
|
|
|
|
|
|
|
NodeBindings::RunMessageLoop();
|
|
|
|
}
|
|
|
|
|
2013-07-22 08:05:35 +00:00
|
|
|
void NodeBindingsWin::PollEvents() {
|
2015-08-21 01:59:58 +00:00
|
|
|
// If there are other kinds of events pending, uv_backend_timeout will
|
|
|
|
// instruct us not to wait.
|
|
|
|
DWORD bytes, timeout;
|
|
|
|
ULONG_PTR key;
|
|
|
|
OVERLAPPED* overlapped;
|
|
|
|
|
|
|
|
timeout = uv_backend_timeout(uv_loop_);
|
|
|
|
|
2018-04-18 01:55:30 +00:00
|
|
|
GetQueuedCompletionStatus(uv_loop_->iocp, &bytes, &key, &overlapped, timeout);
|
2015-08-21 01:59:58 +00:00
|
|
|
|
|
|
|
// Give the event back so libuv can deal with it.
|
|
|
|
if (overlapped != NULL)
|
2018-04-18 01:55:30 +00:00
|
|
|
PostQueuedCompletionStatus(uv_loop_->iocp, bytes, key, overlapped);
|
2013-07-22 08:05:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
2017-03-08 08:33:44 +00:00
|
|
|
NodeBindings* NodeBindings::Create(BrowserEnvironment browser_env) {
|
|
|
|
return new NodeBindingsWin(browser_env);
|
2013-07-22 08:05:35 +00:00
|
|
|
}
|
|
|
|
|
2019-06-19 21:23:04 +00:00
|
|
|
} // namespace electron
|