2016-05-03 19:29:58 -07:00
|
|
|
// Copyright 2004-present Facebook. All Rights Reserved.
|
|
|
|
|
|
|
|
#include "JSCWebWorker.h"
|
|
|
|
|
|
|
|
#include <condition_variable>
|
|
|
|
#include <mutex>
|
|
|
|
#include <unordered_map>
|
|
|
|
|
|
|
|
#include <folly/Memory.h>
|
2016-11-22 06:05:36 -08:00
|
|
|
#include <glog/logging.h>
|
2016-05-03 19:29:58 -07:00
|
|
|
|
2016-11-01 11:38:43 -07:00
|
|
|
#include <jschelpers/JSCHelpers.h>
|
|
|
|
#include <jschelpers/Value.h>
|
2016-11-22 06:05:36 -08:00
|
|
|
|
2016-05-03 19:29:58 -07:00
|
|
|
#include "MessageQueueThread.h"
|
|
|
|
#include "Platform.h"
|
2016-11-01 11:38:43 -07:00
|
|
|
#include "JSCUtils.h"
|
2016-05-03 19:29:58 -07:00
|
|
|
|
|
|
|
namespace facebook {
|
|
|
|
namespace react {
|
|
|
|
|
|
|
|
// TODO(9604425): thread safety
|
|
|
|
static std::unordered_map<JSContextRef, JSCWebWorker*> s_globalContextRefToJSCWebWorker;
|
|
|
|
|
|
|
|
JSCWebWorker::JSCWebWorker(int id, JSCWebWorkerOwner *owner, std::string scriptSrc) :
|
|
|
|
id_(id),
|
|
|
|
scriptName_(std::move(scriptSrc)),
|
|
|
|
owner_(owner) {
|
|
|
|
ownerMessageQueueThread_ = owner->getMessageQueueThread();
|
|
|
|
CHECK(ownerMessageQueueThread_) << "Owner MessageQueue must not be null";
|
|
|
|
workerMessageQueueThread_ = WebWorkerUtil::createWebWorkerThread(id, ownerMessageQueueThread_.get());
|
|
|
|
CHECK(workerMessageQueueThread_) << "Failed to create worker thread";
|
|
|
|
|
|
|
|
workerMessageQueueThread_->runOnQueue([this] () {
|
|
|
|
initJSVMAndLoadScript();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
JSCWebWorker::~JSCWebWorker() {
|
|
|
|
CHECK(isTerminated()) << "Didn't terminate the web worker before releasing it!";;
|
|
|
|
}
|
|
|
|
|
|
|
|
void JSCWebWorker::postMessage(JSValueRef msg) {
|
|
|
|
std::string msgString = Value(owner_->getContext(), msg).toJSONString();
|
|
|
|
|
|
|
|
workerMessageQueueThread_->runOnQueue([this, msgString] () {
|
|
|
|
if (isTerminated()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSValueRef args[] = { createMessageObject(context_, msgString) };
|
|
|
|
Value onmessageValue = Object::getGlobalObject(context_).getProperty("onmessage");
|
|
|
|
onmessageValue.asObject().callAsFunction(1, args);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void JSCWebWorker::terminate() {
|
|
|
|
if (isTerminated()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
isTerminated_.store(true, std::memory_order_release);
|
|
|
|
|
|
|
|
workerMessageQueueThread_->runOnQueueSync([this] {
|
|
|
|
terminateOnWorkerThread();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void JSCWebWorker::terminateOnWorkerThread() {
|
|
|
|
s_globalContextRefToJSCWebWorker.erase(context_);
|
2016-11-22 06:05:38 -08:00
|
|
|
JSC_JSGlobalContextRelease(context_);
|
2016-05-03 19:29:58 -07:00
|
|
|
context_ = nullptr;
|
|
|
|
workerMessageQueueThread_->quitSynchronous();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool JSCWebWorker::isTerminated() {
|
|
|
|
return isTerminated_.load(std::memory_order_acquire);
|
|
|
|
}
|
|
|
|
|
|
|
|
void JSCWebWorker::initJSVMAndLoadScript() {
|
|
|
|
CHECK(!isTerminated()) << "Worker was already finished!";
|
|
|
|
CHECK(!context_) << "Worker JS VM was already created!";
|
|
|
|
|
2016-11-22 06:05:38 -08:00
|
|
|
context_ = JSC_JSGlobalContextCreateInGroup(
|
|
|
|
false, // no support required for custom JSC
|
2016-05-03 19:29:58 -07:00
|
|
|
NULL, // use default JS 'global' object
|
|
|
|
NULL // create new group (i.e. new VM)
|
|
|
|
);
|
|
|
|
s_globalContextRefToJSCWebWorker[context_] = this;
|
|
|
|
|
|
|
|
// TODO(9604438): Protect against script does not exist
|
2016-05-13 17:15:06 -07:00
|
|
|
std::unique_ptr<const JSBigString> script = WebWorkerUtil::loadScriptFromAssets(scriptName_);
|
2016-11-18 06:25:29 -08:00
|
|
|
evaluateScript(context_, jsStringFromBigString(context_, *script), String(context_, scriptName_.c_str()));
|
2016-05-03 19:29:58 -07:00
|
|
|
|
|
|
|
installGlobalFunction(context_, "postMessage", nativePostMessage);
|
|
|
|
}
|
|
|
|
|
|
|
|
void JSCWebWorker::postMessageToOwner(JSValueRef msg) {
|
|
|
|
std::string msgString = Value(context_, msg).toJSONString();
|
|
|
|
ownerMessageQueueThread_->runOnQueue([this, msgString] () {
|
|
|
|
owner_->onMessageReceived(id_, msgString);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
JSValueRef JSCWebWorker::nativePostMessage(
|
|
|
|
JSContextRef ctx,
|
|
|
|
JSObjectRef function,
|
|
|
|
JSObjectRef thisObject,
|
|
|
|
size_t argumentCount,
|
|
|
|
const JSValueRef arguments[],
|
|
|
|
JSValueRef *exception) {
|
|
|
|
if (argumentCount != 1) {
|
2016-11-18 06:25:24 -08:00
|
|
|
*exception = Value::makeError(ctx, "postMessage got wrong number of arguments");
|
|
|
|
return Value::makeUndefined(ctx);
|
2016-05-03 19:29:58 -07:00
|
|
|
}
|
|
|
|
JSValueRef msg = arguments[0];
|
2016-11-22 06:05:38 -08:00
|
|
|
JSCWebWorker *webWorker = s_globalContextRefToJSCWebWorker.at(JSC_JSContextGetGlobalContext(ctx));
|
2016-05-03 19:29:58 -07:00
|
|
|
|
2016-11-18 06:25:24 -08:00
|
|
|
if (!webWorker->isTerminated()) {
|
|
|
|
webWorker->postMessageToOwner(msg);
|
2016-05-03 19:29:58 -07:00
|
|
|
}
|
|
|
|
|
2016-11-18 06:25:24 -08:00
|
|
|
return Value::makeUndefined(ctx);
|
2016-05-03 19:29:58 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*static*/
|
|
|
|
Object JSCWebWorker::createMessageObject(JSContextRef context, const std::string& msgJson) {
|
2016-11-18 06:25:29 -08:00
|
|
|
Value rebornJSMsg = Value::fromJSON(context, String(context, msgJson.c_str()));
|
2016-05-03 19:29:58 -07:00
|
|
|
Object messageObject = Object::create(context);
|
|
|
|
messageObject.setProperty("data", rebornJSMsg);
|
|
|
|
return messageObject;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|