2016-05-04 02:29:58 +00:00
|
|
|
// Copyright 2004-present Facebook. All Rights Reserved.
|
|
|
|
|
|
|
|
#include <atomic>
|
|
|
|
#include <functional>
|
|
|
|
#include <mutex>
|
|
|
|
#include <string>
|
|
|
|
#include <thread>
|
|
|
|
#include <queue>
|
|
|
|
|
|
|
|
#include <JavaScriptCore/JSValueRef.h>
|
|
|
|
|
2016-11-01 18:38:43 +00:00
|
|
|
#include <jschelpers/Value.h>
|
2016-05-04 02:29:58 +00:00
|
|
|
|
|
|
|
namespace facebook {
|
|
|
|
namespace react {
|
|
|
|
|
|
|
|
class MessageQueueThread;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A class that can own the lifecycle, receive messages from, and dispatch messages
|
|
|
|
* to JSCWebWorkers.
|
|
|
|
*/
|
|
|
|
class JSCWebWorkerOwner {
|
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* Called when a worker has posted a message with `postMessage`.
|
|
|
|
*/
|
|
|
|
virtual void onMessageReceived(int workerId, const std::string& message) = 0;
|
|
|
|
virtual JSGlobalContextRef getContext() = 0;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Should return the owner's MessageQueueThread. Calls to onMessageReceived will be enqueued
|
|
|
|
* on this thread.
|
|
|
|
*/
|
|
|
|
virtual std::shared_ptr<MessageQueueThread> getMessageQueueThread() = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Implementation of a web worker for JSC. The web worker should be created from the owner's
|
|
|
|
* (e.g., owning JSCExecutor instance) JS MessageQueueThread. The worker is responsible for
|
|
|
|
* creating its own MessageQueueThread.
|
|
|
|
*
|
|
|
|
* During operation, the JSCExecutor should call postMessage **from its own MessageQueueThread**
|
|
|
|
* to send messages to the worker. The worker will handle enqueueing those messages on its own
|
|
|
|
* MessageQueueThread as appropriate. When the worker has a message to post to the owner, it will
|
|
|
|
* enqueue a call to owner->onMessageReceived on the owner's MessageQueueThread.
|
|
|
|
*/
|
|
|
|
class JSCWebWorker {
|
|
|
|
public:
|
|
|
|
explicit JSCWebWorker(int id, JSCWebWorkerOwner *owner, std::string script);
|
|
|
|
~JSCWebWorker();
|
|
|
|
|
|
|
|
/**
|
2016-11-01 18:38:43 +00:00
|
|
|
* Post a message to be received by the worker on its thread. This must be called from
|
2016-05-04 02:29:58 +00:00
|
|
|
* ownerMessageQueueThread_.
|
|
|
|
*/
|
|
|
|
void postMessage(JSValueRef msg);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Synchronously quits the current worker and cleans up its VM.
|
|
|
|
*/
|
|
|
|
void terminate();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether terminate() has been called on this worker.
|
|
|
|
*/
|
|
|
|
bool isTerminated();
|
|
|
|
|
|
|
|
static Object createMessageObject(JSContextRef context, const std::string& msgData);
|
|
|
|
private:
|
|
|
|
void initJSVMAndLoadScript();
|
|
|
|
void postRunnableToEventLoop(std::function<void()>&& runnable);
|
|
|
|
void postMessageToOwner(JSValueRef result);
|
|
|
|
void terminateOnWorkerThread();
|
|
|
|
|
|
|
|
int id_;
|
|
|
|
std::atomic_bool isTerminated_ = ATOMIC_VAR_INIT(false);
|
|
|
|
std::string scriptName_;
|
|
|
|
JSCWebWorkerOwner *owner_ = nullptr;
|
|
|
|
std::shared_ptr<MessageQueueThread> ownerMessageQueueThread_;
|
|
|
|
std::unique_ptr<MessageQueueThread> workerMessageQueueThread_;
|
|
|
|
JSGlobalContextRef context_ = nullptr;
|
|
|
|
|
|
|
|
static JSValueRef nativePostMessage(
|
|
|
|
JSContextRef ctx,
|
|
|
|
JSObjectRef function,
|
|
|
|
JSObjectRef thisObject,
|
|
|
|
size_t argumentCount,
|
|
|
|
const JSValueRef arguments[],
|
|
|
|
JSValueRef *exception);
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|