This source file includes following definitions.
- crash_cb_
- ShutDown
- DoPostMessage
- StartupInitializationComplete
- PostMessage_MainThreadContinuation
- EnumerateManifestKeys
- OpenManifestEntry
- OpenManifestEntry_MainThreadContinuation
- StreamAsFile_MainThreadContinuation
- CloseManifestEntry
- CloseManifestEntry_MainThreadContinuation
- ReportCrash
- ReportExitStatus
- RequestQuotaForWrite
- AddTempQuotaManagedFile
- callback_factory_
- SetupCommandChannel
- LoadModule
- InitReverseService
- StartModule
- StartSelLdr
- StartSelLdrContinuation
- WaitForSelLdrStart
- SignalStartSelLdrDone
- LoadNexeAndStart
- SetupAppChannel
- Log
- Shutdown
- set_exit_status
- GetCrashLogOutput
#define NACL_LOG_MODULE_NAME "Plugin_ServiceRuntime"
#include "ppapi/native_client/src/trusted/plugin/service_runtime.h"
#include <string.h>
#include <set>
#include <string>
#include <utility>
#include "base/compiler_specific.h"
#include "native_client/src/include/checked_cast.h"
#include "native_client/src/include/portability_io.h"
#include "native_client/src/include/portability_string.h"
#include "native_client/src/include/nacl_macros.h"
#include "native_client/src/include/nacl_scoped_ptr.h"
#include "native_client/src/include/nacl_string.h"
#include "native_client/src/shared/platform/nacl_check.h"
#include "native_client/src/shared/platform/nacl_log.h"
#include "native_client/src/shared/platform/nacl_sync.h"
#include "native_client/src/shared/platform/nacl_sync_checked.h"
#include "native_client/src/shared/platform/nacl_sync_raii.h"
#include "native_client/src/shared/platform/scoped_ptr_refcount.h"
#include "native_client/src/trusted/desc/nacl_desc_imc.h"
#include "native_client/src/trusted/desc/nacl_desc_io.h"
#include "native_client/src/trusted/desc/nrd_xfer.h"
#include "native_client/src/trusted/nonnacl_util/sel_ldr_launcher.h"
#ifdef PostMessage
#undef PostMessage
#endif
#include "native_client/src/public/imc_types.h"
#include "native_client/src/trusted/service_runtime/nacl_error_code.h"
#include "native_client/src/trusted/validator/nacl_file_info.h"
#include "ppapi/c/pp_errors.h"
#include "ppapi/cpp/core.h"
#include "ppapi/cpp/completion_callback.h"
#include "ppapi/native_client/src/trusted/plugin/manifest.h"
#include "ppapi/native_client/src/trusted/plugin/plugin.h"
#include "ppapi/native_client/src/trusted/plugin/plugin_error.h"
#include "ppapi/native_client/src/trusted/plugin/pnacl_options.h"
#include "ppapi/native_client/src/trusted/plugin/pnacl_resources.h"
#include "ppapi/native_client/src/trusted/plugin/sel_ldr_launcher_chrome.h"
#include "ppapi/native_client/src/trusted/plugin/srpc_client.h"
#include "ppapi/native_client/src/trusted/weak_ref/call_on_main_thread.h"
namespace {
const int64_t kMaxTempQuota = 0x8000000;
}
namespace plugin {
PluginReverseInterface::PluginReverseInterface(
nacl::WeakRefAnchor* anchor,
Plugin* plugin,
const Manifest* manifest,
ServiceRuntime* service_runtime,
pp::CompletionCallback init_done_cb,
pp::CompletionCallback crash_cb)
: anchor_(anchor),
plugin_(plugin),
manifest_(manifest),
service_runtime_(service_runtime),
shutting_down_(false),
init_done_cb_(init_done_cb),
crash_cb_(crash_cb) {
NaClXMutexCtor(&mu_);
NaClXCondVarCtor(&cv_);
}
PluginReverseInterface::~PluginReverseInterface() {
NaClCondVarDtor(&cv_);
NaClMutexDtor(&mu_);
}
void PluginReverseInterface::ShutDown() {
NaClLog(4, "PluginReverseInterface::Shutdown: entered\n");
nacl::MutexLocker take(&mu_);
shutting_down_ = true;
NaClXCondVarBroadcast(&cv_);
NaClLog(4, "PluginReverseInterface::Shutdown: broadcasted, exiting\n");
}
void PluginReverseInterface::DoPostMessage(nacl::string message) {
PostMessageResource* continuation = new PostMessageResource(message);
CHECK(continuation != NULL);
NaClLog(4, "PluginReverseInterface::DoPostMessage(%s)\n", message.c_str());
plugin::WeakRefCallOnMainThread(
anchor_,
0,
this,
&plugin::PluginReverseInterface::PostMessage_MainThreadContinuation,
continuation);
}
void PluginReverseInterface::StartupInitializationComplete() {
NaClLog(4, "PluginReverseInterface::StartupInitializationComplete\n");
if (init_done_cb_.pp_completion_callback().func != NULL) {
NaClLog(4,
"PluginReverseInterface::StartupInitializationComplete:"
" invoking CB\n");
pp::Module::Get()->core()->CallOnMainThread(0, init_done_cb_, PP_OK);
} else {
NaClLog(1,
"PluginReverseInterface::StartupInitializationComplete:"
" init_done_cb_ not valid, skipping.\n");
}
}
void PluginReverseInterface::PostMessage_MainThreadContinuation(
PostMessageResource* p,
int32_t err) {
UNREFERENCED_PARAMETER(err);
NaClLog(4,
"PluginReverseInterface::PostMessage_MainThreadContinuation(%s)\n",
p->message.c_str());
plugin_->PostMessage(std::string("DEBUG_POSTMESSAGE:") + p->message);
}
bool PluginReverseInterface::EnumerateManifestKeys(
std::set<nacl::string>* out_keys) {
return manifest_->GetFileKeys(out_keys);
}
bool PluginReverseInterface::OpenManifestEntry(nacl::string url_key,
struct NaClFileInfo* info) {
bool op_complete = false;
OpenManifestEntryResource* to_open =
new OpenManifestEntryResource(url_key, info, &op_complete);
CHECK(to_open != NULL);
NaClLog(4, "PluginReverseInterface::OpenManifestEntry: %s\n",
url_key.c_str());
plugin::WeakRefCallOnMainThread(
anchor_,
0,
this,
&plugin::PluginReverseInterface::OpenManifestEntry_MainThreadContinuation,
to_open);
NaClLog(4,
"PluginReverseInterface::OpenManifestEntry:"
" waiting on main thread\n");
{
nacl::MutexLocker take(&mu_);
while (!shutting_down_ && !op_complete)
NaClXCondVarWait(&cv_, &mu_);
NaClLog(4, "PluginReverseInterface::OpenManifestEntry: done!\n");
if (shutting_down_) {
NaClLog(4,
"PluginReverseInterface::OpenManifestEntry:"
" plugin is shutting down\n");
return false;
}
}
NaClLog(4,
"PluginReverseInterface::OpenManifestEntry: info->desc = %d\n",
info->desc);
if (info->desc == -1) {
NaClLog(4, "OpenManifestEntry: failed for key %s", url_key.c_str());
}
return true;
}
void PluginReverseInterface::OpenManifestEntry_MainThreadContinuation(
OpenManifestEntryResource* p,
int32_t err) {
UNREFERENCED_PARAMETER(err);
NaClLog(4, "Entered OpenManifestEntry_MainThreadContinuation\n");
std::string mapped_url;
PnaclOptions pnacl_options;
ErrorInfo error_info;
if (!manifest_->ResolveKey(p->url, &mapped_url,
&pnacl_options, &error_info)) {
NaClLog(4, "OpenManifestEntry_MainThreadContinuation: ResolveKey failed\n");
NaClLog(4,
"Error code %d, string %s\n",
error_info.error_code(),
error_info.message().c_str());
nacl::MutexLocker take(&mu_);
*p->op_complete_ptr = true;
p->file_info->desc = -1;
NaClXCondVarBroadcast(&cv_);
return;
}
NaClLog(4,
"OpenManifestEntry_MainThreadContinuation: "
"ResolveKey: %s -> %s (pnacl_translate(%d))\n",
p->url.c_str(), mapped_url.c_str(), pnacl_options.translate());
if (pnacl_options.translate()) {
NaClLog(4,
"OpenManifestEntry_MainThreadContinuation: "
"Requires PNaCl translation -- not supported\n");
nacl::MutexLocker take(&mu_);
*p->op_complete_ptr = true;
p->file_info->desc = -1;
NaClXCondVarBroadcast(&cv_);
return;
}
if (PnaclUrls::IsPnaclComponent(mapped_url)) {
int32_t fd = PnaclResources::GetPnaclFD(
plugin_,
PnaclUrls::PnaclComponentURLToFilename(mapped_url).c_str());
if (fd < 0) {
NaClLog(4,
"OpenManifestEntry_MainThreadContinuation: "
"GetReadonlyPnaclFd failed\n");
}
nacl::MutexLocker take(&mu_);
*p->op_complete_ptr = true;
p->file_info->desc = fd;
NaClXCondVarBroadcast(&cv_);
NaClLog(4,
"OpenManifestEntry_MainThreadContinuation: GetPnaclFd okay\n");
return;
}
OpenManifestEntryResource* open_cont = new OpenManifestEntryResource(*p);
open_cont->url = mapped_url;
pp::CompletionCallback stream_cc = WeakRefNewCallback(
anchor_,
this,
&PluginReverseInterface::StreamAsFile_MainThreadContinuation,
open_cont);
if (!plugin_->StreamAsFile(mapped_url, stream_cc)) {
NaClLog(4,
"OpenManifestEntry_MainThreadContinuation: "
"StreamAsFile failed\n");
stream_cc.Run(PP_ERROR_FAILED);
return;
}
NaClLog(4, "OpenManifestEntry_MainThreadContinuation: StreamAsFile okay\n");
}
void PluginReverseInterface::StreamAsFile_MainThreadContinuation(
OpenManifestEntryResource* p,
int32_t result) {
NaClLog(4,
"Entered StreamAsFile_MainThreadContinuation\n");
nacl::MutexLocker take(&mu_);
if (result == PP_OK) {
NaClLog(4, "StreamAsFile_MainThreadContinuation: GetFileInfo(%s)\n",
p->url.c_str());
*p->file_info = plugin_->GetFileInfo(p->url);
NaClLog(4,
"StreamAsFile_MainThreadContinuation: PP_OK, desc %d\n",
p->file_info->desc);
} else {
NaClLog(4,
"StreamAsFile_MainThreadContinuation: !PP_OK, setting desc -1\n");
p->file_info->desc = -1;
}
*p->op_complete_ptr = true;
NaClXCondVarBroadcast(&cv_);
}
bool PluginReverseInterface::CloseManifestEntry(int32_t desc) {
bool op_complete = false;
bool op_result;
CloseManifestEntryResource* to_close =
new CloseManifestEntryResource(desc, &op_complete, &op_result);
plugin::WeakRefCallOnMainThread(
anchor_,
0,
this,
&plugin::PluginReverseInterface::
CloseManifestEntry_MainThreadContinuation,
to_close);
{
nacl::MutexLocker take(&mu_);
while (!shutting_down_ && !op_complete)
NaClXCondVarWait(&cv_, &mu_);
if (shutting_down_)
return false;
}
return op_result;
}
void PluginReverseInterface::CloseManifestEntry_MainThreadContinuation(
CloseManifestEntryResource* cls,
int32_t err) {
UNREFERENCED_PARAMETER(err);
nacl::MutexLocker take(&mu_);
*cls->op_result_ptr = true;
*cls->op_complete_ptr = true;
NaClXCondVarBroadcast(&cv_);
}
void PluginReverseInterface::ReportCrash() {
NaClLog(4, "PluginReverseInterface::ReportCrash\n");
if (crash_cb_.pp_completion_callback().func != NULL) {
NaClLog(4, "PluginReverseInterface::ReportCrash: invoking CB\n");
pp::Module::Get()->core()->CallOnMainThread(0, crash_cb_, PP_OK);
} else {
NaClLog(1,
"PluginReverseInterface::ReportCrash:"
" crash_cb_ not valid, skipping\n");
}
}
void PluginReverseInterface::ReportExitStatus(int exit_status) {
service_runtime_->set_exit_status(exit_status);
}
int64_t PluginReverseInterface::RequestQuotaForWrite(
nacl::string file_id, int64_t offset, int64_t bytes_to_write) {
NaClLog(4,
"PluginReverseInterface::RequestQuotaForWrite:"
" (file_id='%s', offset=%" NACL_PRId64 ", bytes_to_write=%"
NACL_PRId64 ")\n", file_id.c_str(), offset, bytes_to_write);
uint64_t file_key = STRTOULL(file_id.c_str(), NULL, 10);
nacl::MutexLocker take(&mu_);
if (quota_files_.count(file_key) == 0) {
NaClLog(4, "PluginReverseInterface::RequestQuotaForWrite: failed...\n");
return 0;
}
if (offset + bytes_to_write >= kMaxTempQuota)
return 0;
return bytes_to_write;
}
void PluginReverseInterface::AddTempQuotaManagedFile(
const nacl::string& file_id) {
NaClLog(4, "PluginReverseInterface::AddTempQuotaManagedFile: "
"(file_id='%s')\n", file_id.c_str());
uint64_t file_key = STRTOULL(file_id.c_str(), NULL, 10);
nacl::MutexLocker take(&mu_);
quota_files_.insert(file_key);
}
ServiceRuntime::ServiceRuntime(Plugin* plugin,
const Manifest* manifest,
bool main_service_runtime,
bool uses_nonsfi_mode,
pp::CompletionCallback init_done_cb,
pp::CompletionCallback crash_cb)
: plugin_(plugin),
main_service_runtime_(main_service_runtime),
uses_nonsfi_mode_(uses_nonsfi_mode),
reverse_service_(NULL),
anchor_(new nacl::WeakRefAnchor()),
rev_interface_(new PluginReverseInterface(anchor_, plugin,
manifest,
this,
init_done_cb, crash_cb)),
exit_status_(-1),
start_sel_ldr_done_(false),
callback_factory_(this) {
NaClSrpcChannelInitialize(&command_channel_);
NaClXMutexCtor(&mu_);
NaClXCondVarCtor(&cond_);
}
bool ServiceRuntime::SetupCommandChannel(ErrorInfo* error_info) {
NaClLog(4, "ServiceRuntime::SetupCommand (this=%p, subprocess=%p)\n",
static_cast<void*>(this),
static_cast<void*>(subprocess_.get()));
if (!subprocess_->SetupCommand(&command_channel_)) {
error_info->SetReport(PP_NACL_ERROR_SEL_LDR_COMMUNICATION_CMD_CHANNEL,
"ServiceRuntime: command channel creation failed");
return false;
}
return true;
}
bool ServiceRuntime::LoadModule(nacl::DescWrapper* nacl_desc,
ErrorInfo* error_info) {
NaClLog(4, "ServiceRuntime::LoadModule"
" (this=%p, subprocess=%p)\n",
static_cast<void*>(this),
static_cast<void*>(subprocess_.get()));
CHECK(nacl_desc);
if (!subprocess_->LoadModule(&command_channel_, nacl_desc)) {
error_info->SetReport(PP_NACL_ERROR_SEL_LDR_COMMUNICATION_CMD_CHANNEL,
"ServiceRuntime: load module failed");
return false;
}
return true;
}
bool ServiceRuntime::InitReverseService(ErrorInfo* error_info) {
if (uses_nonsfi_mode_) {
rev_interface_->StartupInitializationComplete();
return true;
}
NaClDesc* out_conn_cap;
NaClSrpcResultCodes rpc_result =
NaClSrpcInvokeBySignature(&command_channel_,
"reverse_setup::h",
&out_conn_cap);
if (NACL_SRPC_RESULT_OK != rpc_result) {
error_info->SetReport(PP_NACL_ERROR_SEL_LDR_COMMUNICATION_REV_SETUP,
"ServiceRuntime: reverse setup rpc failed");
return false;
}
NaClLog(4, "ServiceRuntime: got 0x%" NACL_PRIxPTR "\n",
(uintptr_t) out_conn_cap);
nacl::DescWrapper* conn_cap = plugin_->wrapper_factory()->MakeGenericCleanup(
out_conn_cap);
if (conn_cap == NULL) {
error_info->SetReport(PP_NACL_ERROR_SEL_LDR_COMMUNICATION_WRAPPER,
"ServiceRuntime: wrapper allocation failure");
return false;
}
out_conn_cap = NULL;
NaClLog(4, "ServiceRuntime::InitReverseService: starting reverse service\n");
reverse_service_ = new nacl::ReverseService(conn_cap, rev_interface_->Ref());
if (!reverse_service_->Start()) {
error_info->SetReport(PP_NACL_ERROR_SEL_LDR_COMMUNICATION_REV_SERVICE,
"ServiceRuntime: starting reverse services failed");
return false;
}
return true;
}
bool ServiceRuntime::StartModule(ErrorInfo* error_info) {
int load_status = -1;
if (uses_nonsfi_mode_) {
load_status = LOAD_OK;
} else {
NaClSrpcResultCodes rpc_result =
NaClSrpcInvokeBySignature(&command_channel_,
"start_module::i",
&load_status);
if (NACL_SRPC_RESULT_OK != rpc_result) {
error_info->SetReport(PP_NACL_ERROR_SEL_LDR_START_MODULE,
"ServiceRuntime: could not start nacl module");
return false;
}
}
NaClLog(4, "ServiceRuntime::StartModule (load_status=%d)\n", load_status);
if (main_service_runtime_) {
plugin_->ReportSelLdrLoadStatus(load_status);
}
if (LOAD_OK != load_status) {
error_info->SetReport(
PP_NACL_ERROR_SEL_LDR_START_STATUS,
NaClErrorString(static_cast<NaClErrorCode>(load_status)));
return false;
}
return true;
}
void ServiceRuntime::StartSelLdr(const SelLdrStartParams& params,
pp::CompletionCallback callback) {
NaClLog(4, "ServiceRuntime::Start\n");
nacl::scoped_ptr<SelLdrLauncherChrome>
tmp_subprocess(new SelLdrLauncherChrome());
if (NULL == tmp_subprocess.get()) {
NaClLog(LOG_ERROR, "ServiceRuntime::Start (subprocess create failed)\n");
if (main_service_runtime_) {
ErrorInfo error_info;
error_info.SetReport(
PP_NACL_ERROR_SEL_LDR_CREATE_LAUNCHER,
"ServiceRuntime: failed to create sel_ldr launcher");
plugin_->ReportLoadError(error_info);
}
pp::Module::Get()->core()->CallOnMainThread(0, callback, PP_ERROR_FAILED);
return;
}
pp::CompletionCallback internal_callback =
callback_factory_.NewCallback(&ServiceRuntime::StartSelLdrContinuation,
callback);
tmp_subprocess->Start(plugin_->pp_instance(),
params.url.c_str(),
params.uses_irt,
params.uses_ppapi,
params.uses_nonsfi_mode,
params.enable_dev_interfaces,
params.enable_dyncode_syscalls,
params.enable_exception_handling,
params.enable_crash_throttling,
&start_sel_ldr_error_message_,
internal_callback);
subprocess_.reset(tmp_subprocess.release());
}
void ServiceRuntime::StartSelLdrContinuation(int32_t pp_error,
pp::CompletionCallback callback) {
if (pp_error != PP_OK) {
NaClLog(LOG_ERROR, "ServiceRuntime::StartSelLdrContinuation "
" (start failed)\n");
if (main_service_runtime_) {
std::string error_message;
pp::Var var_error_message_cpp(pp::PASS_REF, start_sel_ldr_error_message_);
if (var_error_message_cpp.is_string()) {
error_message = var_error_message_cpp.AsString();
}
ErrorInfo error_info;
error_info.SetReportWithConsoleOnlyError(
PP_NACL_ERROR_SEL_LDR_LAUNCH,
"ServiceRuntime: failed to start",
error_message);
plugin_->ReportLoadError(error_info);
}
}
pp::Module::Get()->core()->CallOnMainThread(0, callback, pp_error);
}
void ServiceRuntime::WaitForSelLdrStart() {
nacl::MutexLocker take(&mu_);
while(!start_sel_ldr_done_) {
NaClXCondVarWait(&cond_, &mu_);
}
}
void ServiceRuntime::SignalStartSelLdrDone() {
nacl::MutexLocker take(&mu_);
start_sel_ldr_done_ = true;
NaClXCondVarSignal(&cond_);
}
bool ServiceRuntime::LoadNexeAndStart(nacl::DescWrapper* nacl_desc,
const pp::CompletionCallback& crash_cb) {
NaClLog(4, "ServiceRuntime::LoadNexeAndStart (nacl_desc=%p)\n",
reinterpret_cast<void*>(nacl_desc));
ErrorInfo error_info;
bool ok = SetupCommandChannel(&error_info) &&
InitReverseService(&error_info) &&
LoadModule(nacl_desc, &error_info) &&
StartModule(&error_info);
if (!ok) {
if (main_service_runtime_) {
plugin_->ReportLoadError(error_info);
}
Log(LOG_FATAL, "reap logs");
if (NULL == reverse_service_) {
NaClLog(LOG_ERROR, "scheduling to get crash log\n");
pp::Module::Get()->core()->CallOnMainThread(0, crash_cb, PP_OK);
NaClLog(LOG_ERROR, "should fire soon\n");
} else {
NaClLog(LOG_ERROR, "Reverse service thread will pick up crash log\n");
}
return false;
}
NaClLog(4, "ServiceRuntime::LoadNexeAndStart (return 1)\n");
return true;
}
SrpcClient* ServiceRuntime::SetupAppChannel() {
NaClLog(4, "ServiceRuntime::SetupAppChannel (subprocess_=%p)\n",
reinterpret_cast<void*>(subprocess_.get()));
nacl::DescWrapper* connect_desc = subprocess_->socket_addr()->Connect();
if (NULL == connect_desc) {
NaClLog(LOG_ERROR, "ServiceRuntime::SetupAppChannel (connect failed)\n");
return NULL;
} else {
NaClLog(4, "ServiceRuntime::SetupAppChannel (conect_desc=%p)\n",
static_cast<void*>(connect_desc));
SrpcClient* srpc_client = SrpcClient::New(connect_desc);
NaClLog(4, "ServiceRuntime::SetupAppChannel (srpc_client=%p)\n",
static_cast<void*>(srpc_client));
delete connect_desc;
return srpc_client;
}
}
bool ServiceRuntime::Log(int severity, const nacl::string& msg) {
NaClSrpcResultCodes rpc_result =
NaClSrpcInvokeBySignature(&command_channel_,
"log:is:",
severity,
strdup(msg.c_str()));
return (NACL_SRPC_RESULT_OK == rpc_result);
}
void ServiceRuntime::Shutdown() {
rev_interface_->ShutDown();
anchor_->Abandon();
subprocess_.reset(NULL);
NaClSrpcDtor(&command_channel_);
if (reverse_service_ != NULL) {
reverse_service_->WaitForServiceThreadsToExit();
reverse_service_->Unref();
reverse_service_ = NULL;
}
}
ServiceRuntime::~ServiceRuntime() {
NaClLog(4, "ServiceRuntime::~ServiceRuntime (this=%p)\n",
static_cast<void*>(this));
subprocess_.reset(NULL);
if (reverse_service_ != NULL) {
reverse_service_->Unref();
}
rev_interface_->Unref();
anchor_->Unref();
NaClCondVarDtor(&cond_);
NaClMutexDtor(&mu_);
}
void ServiceRuntime::set_exit_status(int exit_status) {
nacl::MutexLocker take(&mu_);
if (main_service_runtime_)
plugin_->set_exit_status(exit_status & 0xff);
}
nacl::string ServiceRuntime::GetCrashLogOutput() {
if (NULL != subprocess_.get()) {
return subprocess_->GetCrashLogOutput();
} else {
return std::string();
}
}
}