2010-10-03 23:04:50 +02:00
|
|
|
// Copyright (c) 2010 The Chromium Embedded Framework Authors.
|
|
|
|
// Portions copyright (c) 2010 The Chromium Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
|
|
|
#include "cef_process_ui_thread.h"
|
|
|
|
#include "browser_webkit_glue.h"
|
|
|
|
#include "browser_webkit_init.h"
|
|
|
|
#include "cef_context.h"
|
|
|
|
|
|
|
|
#include "base/command_line.h"
|
|
|
|
#include "base/i18n/icu_util.h"
|
2010-10-15 20:37:25 +02:00
|
|
|
#include "base/metrics/stats_table.h"
|
2010-10-03 23:04:50 +02:00
|
|
|
#include "base/rand_util.h"
|
|
|
|
#include "base/string_number_conversions.h"
|
|
|
|
#include "build/build_config.h"
|
|
|
|
#include "net/base/net_module.h"
|
2011-04-05 18:17:33 +02:00
|
|
|
#include "net/url_request/url_request.h"
|
2011-05-18 17:49:52 +02:00
|
|
|
#include "third_party/WebKit/Source/WebKit/chromium/public/WebNetworkStateNotifier.h"
|
2011-04-05 18:17:33 +02:00
|
|
|
#include "ui/gfx/gl/gl_implementation.h"
|
2010-10-03 23:04:50 +02:00
|
|
|
#include "webkit/blob/blob_storage_controller.h"
|
|
|
|
#include "webkit/blob/blob_url_request_job.h"
|
|
|
|
#include "webkit/extensions/v8/gc_extension.h"
|
2011-04-05 18:17:33 +02:00
|
|
|
#include "webkit/fileapi/file_system_context.h"
|
|
|
|
#include "webkit/fileapi/file_system_dir_url_request_job.h"
|
|
|
|
#include "webkit/fileapi/file_system_url_request_job.h"
|
2011-01-07 22:34:20 +01:00
|
|
|
#include "webkit/plugins/npapi/plugin_list.h"
|
2010-10-03 23:04:50 +02:00
|
|
|
|
|
|
|
#if defined(OS_WIN)
|
|
|
|
#include <commctrl.h>
|
|
|
|
#include <Objbase.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static const char* kStatsFilePrefix = "libcef_";
|
|
|
|
static int kStatsFileThreads = 20;
|
|
|
|
static int kStatsFileCounters = 200;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2010-12-16 22:54:42 +01:00
|
|
|
net::URLRequestJob* BlobURLRequestJobFactory(net::URLRequest* request,
|
|
|
|
const std::string& scheme) {
|
2010-10-03 23:04:50 +02:00
|
|
|
webkit_blob::BlobStorageController* blob_storage_controller =
|
|
|
|
static_cast<BrowserRequestContext*>(request->context())->
|
|
|
|
blob_storage_controller();
|
|
|
|
return new webkit_blob::BlobURLRequestJob(
|
|
|
|
request,
|
|
|
|
blob_storage_controller->GetBlobDataFromUrl(request->url()),
|
2011-04-05 18:17:33 +02:00
|
|
|
CefThread::GetMessageLoopProxyForThread(CefThread::FILE));
|
|
|
|
}
|
|
|
|
|
|
|
|
net::URLRequestJob* FileSystemURLRequestJobFactory(net::URLRequest* request,
|
|
|
|
const std::string& scheme) {
|
|
|
|
fileapi::FileSystemContext* fs_context =
|
|
|
|
static_cast<BrowserRequestContext*>(request->context())
|
|
|
|
->file_system_context();
|
|
|
|
if (!fs_context) {
|
|
|
|
LOG(WARNING) << "No FileSystemContext found, ignoring filesystem: URL";
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the path ends with a /, we know it's a directory. If the path refers
|
|
|
|
// to a directory and gets dispatched to FileSystemURLRequestJob, that class
|
|
|
|
// redirects back here, by adding a / to the URL.
|
|
|
|
const std::string path = request->url().path();
|
|
|
|
if (!path.empty() && path[path.size() - 1] == '/') {
|
|
|
|
return new fileapi::FileSystemDirURLRequestJob(
|
|
|
|
request,
|
2011-05-16 18:56:12 +02:00
|
|
|
fs_context,
|
2011-04-05 18:17:33 +02:00
|
|
|
CefThread::GetMessageLoopProxyForThread(CefThread::FILE));
|
|
|
|
}
|
|
|
|
return new fileapi::FileSystemURLRequestJob(
|
|
|
|
request,
|
2011-05-16 18:56:12 +02:00
|
|
|
fs_context,
|
2011-04-05 18:17:33 +02:00
|
|
|
CefThread::GetMessageLoopProxyForThread(CefThread::FILE));
|
2010-10-03 23:04:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
|
|
|
|
CefProcessUIThread::CefProcessUIThread()
|
|
|
|
: CefThread(CefThread::UI), statstable_(NULL), webkit_init_(NULL) {}
|
|
|
|
|
|
|
|
CefProcessUIThread::CefProcessUIThread(MessageLoop* message_loop)
|
|
|
|
: CefThread(CefThread::UI, message_loop), statstable_(NULL),
|
|
|
|
webkit_init_(NULL) {}
|
|
|
|
|
|
|
|
CefProcessUIThread::~CefProcessUIThread() {
|
|
|
|
// We cannot rely on our base class to stop the thread since we want our
|
|
|
|
// CleanUp function to run.
|
|
|
|
Stop();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CefProcessUIThread::Init() {
|
2010-11-15 16:39:56 +01:00
|
|
|
PlatformInit();
|
2010-10-03 23:04:50 +02:00
|
|
|
|
|
|
|
// Initialize the global CommandLine object.
|
|
|
|
CommandLine::Init(0, NULL);
|
|
|
|
|
2011-01-24 21:41:07 +01:00
|
|
|
const CefSettings& settings = _Context->settings();
|
|
|
|
|
|
|
|
// Initialize logging.
|
|
|
|
logging::LoggingDestination logging_dest;
|
|
|
|
if (settings.log_severity == LOGSEVERITY_DISABLE) {
|
|
|
|
logging_dest = logging::LOG_NONE;
|
|
|
|
} else {
|
|
|
|
#if defined(OS_WIN)
|
|
|
|
logging_dest = logging::LOG_ONLY_TO_FILE;
|
|
|
|
#else
|
|
|
|
logging_dest = logging::LOG_TO_BOTH_FILE_AND_SYSTEM_DEBUG_LOG;
|
|
|
|
#endif
|
|
|
|
logging::SetMinLogLevel(settings.log_severity);
|
|
|
|
}
|
|
|
|
|
|
|
|
FilePath log_file = FilePath(CefString(&settings.log_file));
|
|
|
|
logging::InitLogging(log_file.value().c_str(), logging_dest,
|
|
|
|
logging::DONT_LOCK_LOG_FILE, logging::APPEND_TO_OLD_LOG_FILE,
|
|
|
|
logging::DISABLE_DCHECK_FOR_NON_OFFICIAL_RELEASE_BUILDS);
|
|
|
|
|
2010-10-03 23:04:50 +02:00
|
|
|
// Initialize WebKit.
|
|
|
|
webkit_init_ = new BrowserWebKitInit();
|
|
|
|
|
|
|
|
// Initialize WebKit encodings
|
|
|
|
webkit_glue::InitializeTextEncoding();
|
|
|
|
|
|
|
|
// Load ICU data tables.
|
|
|
|
bool ret = icu_util::Initialize();
|
|
|
|
if(!ret) {
|
|
|
|
#if defined(OS_WIN)
|
|
|
|
MessageBox(NULL, L"Failed to load the required icudt38 library",
|
|
|
|
L"CEF Initialization Error", MB_ICONERROR | MB_OK);
|
|
|
|
#endif
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Config the network module so it has access to a limited set of resources.
|
|
|
|
net::NetModule::SetResourceProvider(webkit_glue::NetResourceProvider);
|
|
|
|
|
|
|
|
// Load and initialize the stats table. Attempt to construct a somewhat
|
|
|
|
// unique name to isolate separate instances from each other.
|
2010-10-15 20:37:25 +02:00
|
|
|
statstable_ = new base::StatsTable(
|
2010-10-03 23:04:50 +02:00
|
|
|
kStatsFilePrefix + base::Uint64ToString(base::RandUint64()),
|
|
|
|
kStatsFileThreads,
|
|
|
|
kStatsFileCounters);
|
2010-10-15 20:37:25 +02:00
|
|
|
base::StatsTable::set_current(statstable_);
|
2010-10-03 23:04:50 +02:00
|
|
|
|
|
|
|
// CEF always exposes the GC.
|
|
|
|
webkit_glue::SetJavaScriptFlags("--expose-gc");
|
|
|
|
// Expose GCController to JavaScript.
|
|
|
|
WebKit::WebScriptController::registerExtension(
|
|
|
|
extensions_v8::GCExtension::Get());
|
|
|
|
|
2011-05-16 18:56:12 +02:00
|
|
|
gfx::InitializeGLBindings(gfx::kGLImplementationEGLGLES2);
|
2010-10-03 23:04:50 +02:00
|
|
|
|
2010-12-16 22:54:42 +01:00
|
|
|
net::URLRequest::RegisterProtocolFactory("blob", &BlobURLRequestJobFactory);
|
2011-04-05 18:17:33 +02:00
|
|
|
net::URLRequest::RegisterProtocolFactory("filesystem",
|
|
|
|
&FileSystemURLRequestJobFactory);
|
2010-11-17 18:28:32 +01:00
|
|
|
|
2010-11-18 22:05:25 +01:00
|
|
|
if (!_Context->cache_path().empty()) {
|
2010-11-17 18:28:32 +01:00
|
|
|
// Create the storage context object.
|
|
|
|
_Context->set_storage_context(new DOMStorageContext());
|
|
|
|
}
|
2010-11-18 22:05:25 +01:00
|
|
|
|
2010-11-22 18:49:46 +01:00
|
|
|
if (settings.user_agent.length > 0)
|
|
|
|
webkit_glue::SetUserAgent(CefString(&settings.user_agent));
|
2010-11-18 22:05:25 +01:00
|
|
|
|
|
|
|
if (settings.extra_plugin_paths) {
|
|
|
|
cef_string_t str;
|
2011-01-07 22:34:20 +01:00
|
|
|
memset(&str, 0, sizeof(str));
|
|
|
|
|
2010-11-18 22:05:25 +01:00
|
|
|
FilePath path;
|
|
|
|
int size = cef_string_list_size(settings.extra_plugin_paths);
|
|
|
|
for(int i = 0; i < size; ++i) {
|
2010-11-22 18:49:46 +01:00
|
|
|
if (!cef_string_list_value(settings.extra_plugin_paths, i, &str))
|
|
|
|
continue;
|
|
|
|
path = FilePath(CefString(&str));
|
2011-01-07 22:34:20 +01:00
|
|
|
webkit::npapi::PluginList::Singleton()->AddExtraPluginPath(path);
|
2010-11-18 22:05:25 +01:00
|
|
|
}
|
|
|
|
}
|
2011-05-18 17:49:52 +02:00
|
|
|
|
|
|
|
// Create a network change notifier before starting the IO & File threads.
|
|
|
|
network_change_notifier_.reset(net::NetworkChangeNotifier::Create());
|
|
|
|
|
|
|
|
// Add a listener for OnOnlineStateChanged to notify WebKit of changes.
|
|
|
|
net::NetworkChangeNotifier::AddOnlineStateObserver(this);
|
|
|
|
|
|
|
|
// Initialize WebKit with the current state.
|
|
|
|
WebKit::WebNetworkStateNotifier::setOnLine(
|
|
|
|
!net::NetworkChangeNotifier::IsOffline());
|
2010-10-03 23:04:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void CefProcessUIThread::CleanUp() {
|
|
|
|
// Flush any remaining messages. This ensures that any accumulated
|
|
|
|
// Task objects get destroyed before we exit, which avoids noise in
|
|
|
|
// purify leak-test results.
|
|
|
|
MessageLoop::current()->RunAllPending();
|
|
|
|
|
2010-11-17 18:28:32 +01:00
|
|
|
// Destroy the storage context object.
|
|
|
|
_Context->set_storage_context(NULL);
|
|
|
|
|
2010-10-03 23:04:50 +02:00
|
|
|
// Tear down the shared StatsTable.
|
2010-10-15 20:37:25 +02:00
|
|
|
base::StatsTable::set_current(NULL);
|
2010-10-03 23:04:50 +02:00
|
|
|
delete statstable_;
|
|
|
|
statstable_ = NULL;
|
|
|
|
|
|
|
|
// Shut down WebKit.
|
|
|
|
delete webkit_init_;
|
|
|
|
webkit_init_ = NULL;
|
|
|
|
|
2011-05-18 17:49:52 +02:00
|
|
|
// Release the network change notifier after all other threads end.
|
|
|
|
net::NetworkChangeNotifier::RemoveOnlineStateObserver(this);
|
|
|
|
network_change_notifier_.reset();
|
|
|
|
|
2010-11-15 16:39:56 +01:00
|
|
|
PlatformCleanUp();
|
2010-10-03 23:04:50 +02:00
|
|
|
}
|
2011-05-18 17:49:52 +02:00
|
|
|
|
|
|
|
void CefProcessUIThread::OnOnlineStateChanged(bool online) {
|
|
|
|
DCHECK(CefThread::CurrentlyOn(CefThread::UI));
|
|
|
|
WebKit::WebNetworkStateNotifier::setOnLine(online);
|
|
|
|
}
|
|
|
|
|