fix: use render client id to track deleted render process hosts (backport: 3-0-x) (#14557)

* fix: use render client id to track deleted render process hosts

* fix: use webContentsId with contextId together (#13749)
This commit is contained in:
Robo 2018-09-12 08:26:20 +05:30 committed by Shelley Vohr
parent 5e81d8dad9
commit 8d27657fa5
8 changed files with 41 additions and 47 deletions

View file

@ -56,7 +56,6 @@
#include "atom/common/native_mate_converters/value_converter.h" #include "atom/common/native_mate_converters/value_converter.h"
#include "atom/common/options_switches.h" #include "atom/common/options_switches.h"
#include "base/message_loop/message_loop.h" #include "base/message_loop/message_loop.h"
#include "base/process/process_handle.h"
#include "base/strings/utf_string_conversions.h" #include "base/strings/utf_string_conversions.h"
#include "base/threading/thread_task_runner_handle.h" #include "base/threading/thread_task_runner_handle.h"
#include "base/values.h" #include "base/values.h"
@ -770,8 +769,7 @@ void WebContents::RenderViewCreated(content::RenderViewHost* render_view_host) {
} }
void WebContents::RenderViewDeleted(content::RenderViewHost* render_view_host) { void WebContents::RenderViewDeleted(content::RenderViewHost* render_view_host) {
Emit("render-view-deleted", render_view_host->GetProcess()->GetID(), Emit("render-view-deleted", render_view_host->GetProcess()->GetID());
base::GetProcId(render_view_host->GetProcess()->GetHandle()));
} }
void WebContents::RenderProcessGone(base::TerminationStatus status) { void WebContents::RenderProcessGone(base::TerminationStatus status) {

View file

@ -345,10 +345,6 @@ void AtomBrowserClient::AppendExtraCommandLineSwitches(
web_preferences->AppendCommandLineSwitches(command_line); web_preferences->AppendCommandLineSwitches(command_line);
SessionPreferences::AppendExtraCommandLineSwitches( SessionPreferences::AppendExtraCommandLineSwitches(
web_contents->GetBrowserContext(), command_line); web_contents->GetBrowserContext(), command_line);
auto context_id = atom::api::WebContents::GetIDForContents(web_contents);
command_line->AppendSwitchASCII(switches::kContextId,
base::IntToString(context_id));
} }
} }

View file

@ -189,9 +189,6 @@ const char kAppUserModelId[] = "app-user-model-id";
// The application path // The application path
const char kAppPath[] = "app-path"; const char kAppPath[] = "app-path";
// The context ID for this process
const char kContextId[] = "context-id";
// The command line switch versions of the options. // The command line switch versions of the options.
const char kBackgroundColor[] = "background-color"; const char kBackgroundColor[] = "background-color";
const char kPreloadScript[] = "preload"; const char kPreloadScript[] = "preload";

View file

@ -93,7 +93,6 @@ extern const char kRegisterServiceWorkerSchemes[];
extern const char kSecureSchemes[]; extern const char kSecureSchemes[];
extern const char kAppUserModelId[]; extern const char kAppUserModelId[];
extern const char kAppPath[]; extern const char kAppPath[];
extern const char kContextId[];
extern const char kBackgroundColor[]; extern const char kBackgroundColor[];
extern const char kPreloadScript[]; extern const char kPreloadScript[];

View file

@ -16,13 +16,13 @@
#include "atom/renderer/content_settings_observer.h" #include "atom/renderer/content_settings_observer.h"
#include "atom/renderer/preferences_manager.h" #include "atom/renderer/preferences_manager.h"
#include "base/command_line.h" #include "base/command_line.h"
#include "base/process/process.h"
#include "base/strings/string_split.h" #include "base/strings/string_split.h"
#include "base/strings/stringprintf.h" #include "base/strings/stringprintf.h"
#include "chrome/renderer/pepper/pepper_helper.h" #include "chrome/renderer/pepper/pepper_helper.h"
#include "chrome/renderer/printing/print_web_view_helper.h" #include "chrome/renderer/printing/print_web_view_helper.h"
#include "chrome/renderer/tts_dispatcher.h" #include "chrome/renderer/tts_dispatcher.h"
#include "content/public/common/content_constants.h" #include "content/public/common/content_constants.h"
#include "content/public/common/content_switches.h"
#include "content/public/renderer/render_frame.h" #include "content/public/renderer/render_frame.h"
#include "content/public/renderer/render_view.h" #include "content/public/renderer/render_view.h"
#include "native_mate/dictionary.h" #include "native_mate/dictionary.h"
@ -46,14 +46,6 @@
#include "atom/common/atom_constants.h" #include "atom/common/atom_constants.h"
#endif // defined(ENABLE_PDF_VIEWER) #endif // defined(ENABLE_PDF_VIEWER)
// This is defined in later versions of Chromium, remove this if you see
// compiler complaining duplicate defines.
#if defined(OS_WIN) || defined(OS_FUCHSIA)
#define CrPRIdPid "ld"
#else
#define CrPRIdPid "d"
#endif
namespace atom { namespace atom {
namespace { namespace {
@ -67,8 +59,8 @@ v8::Local<v8::Value> GetRenderProcessPreferences(
return v8::Null(isolate); return v8::Null(isolate);
} }
std::vector<std::string> ParseSchemesCLISwitch(const char* switch_name) { std::vector<std::string> ParseSchemesCLISwitch(base::CommandLine* command_line,
base::CommandLine* command_line = base::CommandLine::ForCurrentProcess(); const char* switch_name) {
std::string custom_schemes = command_line->GetSwitchValueASCII(switch_name); std::string custom_schemes = command_line->GetSwitchValueASCII(switch_name);
return base::SplitString(custom_schemes, ",", base::TRIM_WHITESPACE, return base::SplitString(custom_schemes, ",", base::TRIM_WHITESPACE,
base::SPLIT_WANT_NONEMPTY); base::SPLIT_WANT_NONEMPTY);
@ -78,12 +70,20 @@ std::vector<std::string> ParseSchemesCLISwitch(const char* switch_name) {
RendererClientBase::RendererClientBase() { RendererClientBase::RendererClientBase() {
// Parse --standard-schemes=scheme1,scheme2 // Parse --standard-schemes=scheme1,scheme2
auto* command_line = base::CommandLine::ForCurrentProcess();
std::vector<std::string> standard_schemes_list = std::vector<std::string> standard_schemes_list =
ParseSchemesCLISwitch(switches::kStandardSchemes); ParseSchemesCLISwitch(command_line, switches::kStandardSchemes);
for (const std::string& scheme : standard_schemes_list) for (const std::string& scheme : standard_schemes_list)
url::AddStandardScheme(scheme.c_str(), url::SCHEME_WITHOUT_PORT); url::AddStandardScheme(scheme.c_str(), url::SCHEME_WITHOUT_PORT);
isolated_world_ = base::CommandLine::ForCurrentProcess()->HasSwitch( isolated_world_ = base::CommandLine::ForCurrentProcess()->HasSwitch(
switches::kContextIsolation); switches::kContextIsolation);
// We rely on the unique process host id which is notified to the
// renderer process via command line switch from the content layer,
// if this switch is removed from the content layer for some reason,
// we should define our own.
DCHECK(command_line->HasSwitch(::switches::kRendererClientId));
renderer_client_id_ =
command_line->GetSwitchValueASCII(::switches::kRendererClientId);
} }
RendererClientBase::~RendererClientBase() {} RendererClientBase::~RendererClientBase() {}
@ -91,10 +91,9 @@ RendererClientBase::~RendererClientBase() {}
void RendererClientBase::DidCreateScriptContext( void RendererClientBase::DidCreateScriptContext(
v8::Handle<v8::Context> context, v8::Handle<v8::Context> context,
content::RenderFrame* render_frame) { content::RenderFrame* render_frame) {
// global.setHidden("contextId", `${processId}-${++nextContextId}`) // global.setHidden("contextId", `${processHostId}-${++next_context_id_}`)
std::string context_id = base::StringPrintf( auto context_id = base::StringPrintf(
"%" CrPRIdPid "-%d", base::GetProcId(base::Process::Current().Handle()), "%s-%" PRId64, renderer_client_id_.c_str(), ++next_context_id_);
++next_context_id_);
v8::Isolate* isolate = context->GetIsolate(); v8::Isolate* isolate = context->GetIsolate();
v8::Local<v8::String> key = mate::StringToSymbol(isolate, "contextId"); v8::Local<v8::String> key = mate::StringToSymbol(isolate, "contextId");
v8::Local<v8::Private> private_key = v8::Private::ForApi(isolate, key); v8::Local<v8::Private> private_key = v8::Private::ForApi(isolate, key);
@ -112,6 +111,8 @@ void RendererClientBase::AddRenderBindings(
} }
void RendererClientBase::RenderThreadStarted() { void RendererClientBase::RenderThreadStarted() {
auto* command_line = base::CommandLine::ForCurrentProcess();
blink::WebCustomElement::AddEmbedderCustomElementName("webview"); blink::WebCustomElement::AddEmbedderCustomElementName("webview");
blink::WebCustomElement::AddEmbedderCustomElementName("browserplugin"); blink::WebCustomElement::AddEmbedderCustomElementName("browserplugin");
@ -133,7 +134,7 @@ void RendererClientBase::RenderThreadStarted() {
// Parse --secure-schemes=scheme1,scheme2 // Parse --secure-schemes=scheme1,scheme2
std::vector<std::string> secure_schemes_list = std::vector<std::string> secure_schemes_list =
ParseSchemesCLISwitch(switches::kSecureSchemes); ParseSchemesCLISwitch(command_line, switches::kSecureSchemes);
for (const std::string& scheme : secure_schemes_list) for (const std::string& scheme : secure_schemes_list)
blink::SchemeRegistry::RegisterURLSchemeAsSecure( blink::SchemeRegistry::RegisterURLSchemeAsSecure(
WTF::String::FromUTF8(scheme.data(), scheme.length())); WTF::String::FromUTF8(scheme.data(), scheme.length()));
@ -147,7 +148,6 @@ void RendererClientBase::RenderThreadStarted() {
#if defined(OS_WIN) #if defined(OS_WIN)
// Set ApplicationUserModelID in renderer process. // Set ApplicationUserModelID in renderer process.
base::CommandLine* command_line = base::CommandLine::ForCurrentProcess();
base::string16 app_id = base::string16 app_id =
command_line->GetSwitchValueNative(switches::kAppUserModelId); command_line->GetSwitchValueNative(switches::kAppUserModelId);
if (!app_id.empty()) { if (!app_id.empty()) {

View file

@ -58,8 +58,9 @@ class RendererClientBase : public content::ContentRendererClient {
ChromeKeySystemsProvider key_systems_provider_; ChromeKeySystemsProvider key_systems_provider_;
bool isolated_world_; bool isolated_world_;
std::string renderer_client_id_;
// An increasing ID used for indentifying an V8 context in this process. // An increasing ID used for indentifying an V8 context in this process.
int next_context_id_ = 0; int64_t next_context_id_ = 0;
}; };
} // namespace atom } // namespace atom

View file

@ -2,6 +2,10 @@
const v8Util = process.atomBinding('v8_util') const v8Util = process.atomBinding('v8_util')
const getOwnerKey = (webContents, contextId) => {
return `${webContents.id}-${contextId}`
}
class ObjectsRegistry { class ObjectsRegistry {
constructor () { constructor () {
this.nextId = 0 this.nextId = 0
@ -11,7 +15,7 @@ class ObjectsRegistry {
this.storage = {} this.storage = {}
// Stores the IDs of objects referenced by WebContents. // Stores the IDs of objects referenced by WebContents.
// (webContentsContextId) => [id] // (ownerKey) => [id]
this.owners = {} this.owners = {}
} }
@ -22,10 +26,10 @@ class ObjectsRegistry {
const id = this.saveToStorage(obj) const id = this.saveToStorage(obj)
// Add object to the set of referenced objects. // Add object to the set of referenced objects.
const webContentsContextId = `${webContents.id}-${contextId}` const ownerKey = getOwnerKey(webContents, contextId)
let owner = this.owners[webContentsContextId] let owner = this.owners[ownerKey]
if (!owner) { if (!owner) {
owner = this.owners[webContentsContextId] = new Set() owner = this.owners[ownerKey] = new Set()
this.registerDeleteListener(webContents, contextId) this.registerDeleteListener(webContents, contextId)
} }
if (!owner.has(id)) { if (!owner.has(id)) {
@ -46,8 +50,8 @@ class ObjectsRegistry {
// Note that an object may be double-freed (cleared when page is reloaded, and // Note that an object may be double-freed (cleared when page is reloaded, and
// then garbage collected in old page). // then garbage collected in old page).
remove (webContents, contextId, id) { remove (webContents, contextId, id) {
const webContentsContextId = `${webContents.id}-${contextId}` const ownerKey = getOwnerKey(webContents, contextId)
let owner = this.owners[webContentsContextId] let owner = this.owners[ownerKey]
if (owner) { if (owner) {
// Remove the reference in owner. // Remove the reference in owner.
owner.delete(id) owner.delete(id)
@ -58,13 +62,13 @@ class ObjectsRegistry {
// Clear all references to objects refrenced by the WebContents. // Clear all references to objects refrenced by the WebContents.
clear (webContents, contextId) { clear (webContents, contextId) {
const webContentsContextId = `${webContents.id}-${contextId}` const ownerKey = getOwnerKey(webContents, contextId)
let owner = this.owners[webContentsContextId] let owner = this.owners[ownerKey]
if (!owner) return if (!owner) return
for (let id of owner) this.dereference(id) for (let id of owner) this.dereference(id)
delete this.owners[webContentsContextId] delete this.owners[ownerKey]
} }
// Private: Saves the object into storage and assigns an ID for it. // Private: Saves the object into storage and assigns an ID for it.
@ -83,8 +87,6 @@ class ObjectsRegistry {
// Private: Dereference the object from store. // Private: Dereference the object from store.
dereference (id) { dereference (id) {
if (process.env.ELECTRON_DISABLE_REMOTE_DEREFERENCING) return
let pointer = this.storage[id] let pointer = this.storage[id]
if (pointer == null) { if (pointer == null) {
return return
@ -96,12 +98,13 @@ class ObjectsRegistry {
} }
} }
// Private: Clear the storage when renderer process is destoryed. // Private: Clear the storage when renderer process is destroyed.
registerDeleteListener (webContents, contextId) { registerDeleteListener (webContents, contextId) {
// contextId => ${OSProcessId}-${contextCount} // contextId => ${processHostId}-${contextCount}
const OSProcessId = contextId.split('-')[0] const processHostId = contextId.split('-')[0]
const listener = (event, deletedProcessId, deletedOSProcessId) => { const listener = (event, deletedProcessHostId) => {
if (deletedOSProcessId && deletedOSProcessId.toString() === OSProcessId) { if (deletedProcessHostId &&
deletedProcessHostId.toString() === processHostId) {
webContents.removeListener('render-view-deleted', listener) webContents.removeListener('render-view-deleted', listener)
this.clear(webContents, contextId) this.clear(webContents, contextId)
} }

View file

@ -17,7 +17,7 @@
} }
// This should trigger a dereference and a remote getURL call should fail // This should trigger a dereference and a remote getURL call should fail
contents.emit('render-view-deleted', {}, '', contents.getOSProcessId()) contents.emit('render-view-deleted', {}, contents.getProcessId())
try { try {
contents.getURL() contents.getURL()
ipcRenderer.send('error-message', 'No error thrown') ipcRenderer.send('error-message', 'No error thrown')