// Copyright (c) 2004-present, Facebook, Inc. // This source code is licensed under the MIT license found in the // LICENSE file in the root directory of this source tree. #pragma once #include #include #include #include #include namespace folly { struct dynamic; } namespace facebook { namespace react { struct InstanceCallback; class JsToNativeBridge; class MessageQueueThread; class ModuleRegistry; class RAMBundleRegistry; // This class manages calls from native code to JS. It also manages // executors and their threads. All functions here can be called from // any thread. // // Except for loadApplicationScriptSync(), all void methods will queue // work to run on the jsQueue passed to the ctor, and return // immediately. class NativeToJsBridge { public: friend class JsToNativeBridge; /** * This must be called on the main JS thread. */ NativeToJsBridge( JSExecutorFactory* jsExecutorFactory, std::shared_ptr registry, std::shared_ptr jsQueue, std::shared_ptr callback); virtual ~NativeToJsBridge(); /** * Executes a function with the module ID and method ID and any additional * arguments in JS. */ void callFunction(std::string&& module, std::string&& method, folly::dynamic&& args); /** * Invokes a callback with the cbID, and optional additional arguments in JS. */ void invokeCallback(double callbackId, folly::dynamic&& args); /** * Starts the JS application. If bundleRegistry is non-null, then it is * used to fetch JavaScript modules as individual scripts. * Otherwise, the script is assumed to include all the modules. */ void loadApplication( std::unique_ptr bundleRegistry, std::unique_ptr startupCode, std::string sourceURL); void loadApplicationSync( std::unique_ptr bundleRegistry, std::unique_ptr startupCode, std::string sourceURL); void registerBundle(uint32_t bundleId, const std::string& bundlePath); void setGlobalVariable(std::string propName, std::unique_ptr jsonValue); void* getJavaScriptContext(); bool isInspectable(); void handleMemoryPressure(int pressureLevel); /** * Synchronously tears down the bridge and the main executor. */ void destroy(); private: void runOnExecutorQueue(std::function task); // This is used to avoid a race condition where a proxyCallback gets queued // after ~NativeToJsBridge(), on the same thread. In that case, the callback // will try to run the task on m_callback which will have been destroyed // within ~NativeToJsBridge(), thus causing a SIGSEGV. std::shared_ptr m_destroyed; std::shared_ptr m_delegate; std::unique_ptr m_executor; std::shared_ptr m_executorMessageQueueThread; // Keep track of whether the JS bundle containing the application logic causes // exception when evaluated initially. If so, more calls to JS will very // likely fail as well, so this flag can help prevent them. bool m_applicationScriptHasFailure = false; #ifdef WITH_FBSYSTRACE std::atomic_uint_least32_t m_systraceCookie = ATOMIC_VAR_INIT(); #endif }; } }