Remove unused functionality in CxxModuleWrapper

Reviewed By: mhorowitz

Differential Revision: D4409789

fbshipit-source-id: 91e70d8333141e1e4dcba0e2620ef2c744d0c9d3
This commit is contained in:
Pieter De Baets 2017-01-30 06:39:49 -08:00 committed by Facebook Github Bot
parent 59226f022c
commit 2a638c2ee7
3 changed files with 12 additions and 222 deletions

View File

@ -15,6 +15,8 @@ import com.facebook.soloader.SoLoader;
/** /**
* A Java Object which represents a cross-platform C++ module * A Java Object which represents a cross-platform C++ module
* *
* This module implements the NativeModule interface but will never be invoked from Java,
* instead the underlying Cxx module will be extracted by the bridge and called directly.
*/ */
@DoNotStrip @DoNotStrip
public class CxxModuleWrapper implements NativeModule public class CxxModuleWrapper implements NativeModule
@ -26,28 +28,6 @@ public class CxxModuleWrapper implements NativeModule
@DoNotStrip @DoNotStrip
private HybridData mHybridData; private HybridData mHybridData;
@DoNotStrip
private static class MethodWrapper implements NativeMethod
{
@DoNotStrip
HybridData mHybridData;
MethodWrapper() {
mHybridData = initHybrid();
}
public native HybridData initHybrid();
@Override
public native void invoke(
CatalystInstance catalystInstance,
ExecutorToken executorToken,
ReadableNativeArray args);
@Override
public native String getType();
}
public CxxModuleWrapper(String library, String factory) { public CxxModuleWrapper(String library, String factory) {
SoLoader.loadLibrary(library); SoLoader.loadLibrary(library);
mHybridData = mHybridData =
@ -58,9 +38,9 @@ public class CxxModuleWrapper implements NativeModule
public native String getName(); public native String getName();
@Override @Override
public native Map<String, NativeMethod> getMethods(); public Map<String, NativeMethod> getMethods() {
throw new UnsupportedOperationException();
public native String getConstantsJson(); }
@Override @Override
public void initialize() { public void initialize() {

View File

@ -4,171 +4,22 @@
#include <fb/fbjni.h> #include <fb/fbjni.h>
#include <fb/Environment.h> #include <fb/Environment.h>
#include <jni/LocalString.h>
#include <jni/Registration.h>
#include <android/log.h>
#include <folly/json.h>
#include <folly/ScopeGuard.h> #include <folly/ScopeGuard.h>
#include <iterator> #include <cxxreact/CxxModule.h>
#include <unordered_set>
#include <dlfcn.h> #include <dlfcn.h>
#include <cxxreact/JsArgumentHelpers.h>
#include "ReadableNativeArray.h"
using namespace facebook::jni; using namespace facebook::jni;
using namespace facebook::xplat::module; using namespace facebook::xplat::module;
using namespace facebook::react; using namespace facebook::react;
namespace {
class ExecutorToken : public HybridClass<ExecutorToken> {
public:
constexpr static const char *const kJavaDescriptor = "Lcom/facebook/react/bridge/ExecutorToken;";
};
class CxxMethodWrapper : public HybridClass<CxxMethodWrapper> {
public:
constexpr static const char *const kJavaDescriptor =
"Lcom/facebook/react/cxxbridge/CxxModuleWrapper$MethodWrapper;";
static local_ref<jhybriddata> initHybrid(alias_ref<jhybridobject>) {
return makeCxxInstance();
}
static void registerNatives() {
registerHybrid({
makeNativeMethod("initHybrid", CxxMethodWrapper::initHybrid),
makeNativeMethod("getType", CxxMethodWrapper::getType),
makeNativeMethod("invoke",
"(Lcom/facebook/react/bridge/CatalystInstance;Lcom/facebook/react/bridge/ExecutorToken;Lcom/facebook/react/bridge/ReadableNativeArray;)V",
CxxMethodWrapper::invoke),
});
}
std::string getType() {
if (method_->func) {
return "async";
} else {
return "sync";
}
}
void invoke(jobject catalystinstance, ExecutorToken::jhybridobject executorToken, NativeArray* args);
const CxxModule::Method* method_;
};
void CxxMethodWrapper::invoke(jobject jCatalystInstance, ExecutorToken::jhybridobject jExecutorToken, NativeArray* arguments) {
CxxModule::Callback first;
CxxModule::Callback second;
if (!method_->func) {
throw std::runtime_error(
folly::to<std::string>("Method ", method_->name,
" is synchronous but invoked asynchronously"));
}
if (method_->callbacks >= 1) {
auto catalystInstance = make_global(adopt_local(jCatalystInstance));
global_ref<ExecutorToken::jhybridobject> executorToken = make_global(jExecutorToken);
// TODO(10184774): Support ExecutorToken in CxxModules
static auto sCatalystInstanceInvokeCallback =
catalystInstance->getClass()->getMethod<void(ExecutorToken::jhybridobject, jint, NativeArray::jhybridobject)>(
"invokeCallback");
int id1;
if (!arguments->array[arguments->array.size() - 1].isInt()) {
throwNewJavaException(gJavaLangIllegalArgumentException,
"Expected callback as last argument");
}
if (method_->callbacks == 2) {
if (!arguments->array[arguments->array.size() - 2].isInt()) {
throwNewJavaException(gJavaLangIllegalArgumentException,
"Expected callback as penultimate argument");
return;
}
id1 = arguments->array[arguments->array.size() - 2].getInt();
int id2 = arguments->array[arguments->array.size() - 1].getInt();
second = [catalystInstance, executorToken, id2](std::vector<folly::dynamic> args) mutable {
folly::dynamic argsArray(std::make_move_iterator(args.begin()),
std::make_move_iterator(args.end()));
ThreadScope guard;
sCatalystInstanceInvokeCallback(
catalystInstance.get(), executorToken.get(), id2,
ReadableNativeArray::newObjectCxxArgs(std::move(argsArray)).get());
catalystInstance.reset();
executorToken.reset();
};
} else {
id1 = arguments->array[arguments->array.size() - 1].getInt();
}
first = [catalystInstance, executorToken, id1](std::vector<folly::dynamic> args) mutable {
folly::dynamic argsArray(std::make_move_iterator(args.begin()),
std::make_move_iterator(args.end()));
ThreadScope guard;
sCatalystInstanceInvokeCallback(
catalystInstance.get(), executorToken.get(), id1,
ReadableNativeArray::newObjectCxxArgs(std::move(argsArray)).get());
// This is necessary because by the time the lambda's dtor runs,
// the guard has been destroyed, and it may not be possible to
// get a JNIEnv* to clean up the captured global_ref.
catalystInstance.reset();
executorToken.reset();
};
}
// I've got a few flawed options here. I can catch C++ exceptions
// here, and log/convert them to java exceptions. This lets all the
// java handling work ok, but the only info I can capture about the
// C++ exception is the what() string, not the stack. I can let the
// C++ exception escape, crashing the app. This causes the full,
// accurate C++ stack trace to be added to logcat by debuggerd. The
// java state is lost, but in practice, the java stack is always the
// same in this case since the javascript stack is not visible. The
// what() value is also lost. Finally, I can catch, log the java
// stack, then rethrow the C++ exception. In this case I get java
// and C++ stack data, but the C++ stack is as of the rethrow, not
// the original throw, both the C++ and java stacks always look the
// same.
//
// I am going with option 2, since that seems like the most useful
// choice. It would be nice to be able to get what() and the C++
// stack. I'm told that will be possible in the future. TODO
// mhorowitz #7128529: convert C++ exceptions to Java
folly::dynamic dargs = arguments->array;
dargs.resize(arguments->array.size() - method_->callbacks);
try {
method_->func(dargs, first, second);
} catch (const facebook::xplat::JsArgumentException& ex) {
throwNewJavaException(gJavaLangIllegalArgumentException, ex.what());
}
}
}
// CxxModuleWrapper
void CxxModuleWrapper::registerNatives() { void CxxModuleWrapper::registerNatives() {
registerHybrid({ registerHybrid({
makeNativeMethod("initHybrid", CxxModuleWrapper::initHybrid), makeNativeMethod("initHybrid", CxxModuleWrapper::initHybrid),
makeNativeMethod("getName", CxxModuleWrapper::getName), makeNativeMethod("getName", CxxModuleWrapper::getName)
makeNativeMethod("getConstantsJson", CxxModuleWrapper::getConstantsJson),
makeNativeMethod("getMethods", "()Ljava/util/Map;", CxxModuleWrapper::getMethods),
}); });
CxxMethodWrapper::registerNatives();
} }
CxxModuleWrapper::CxxModuleWrapper(const std::string& soPath, const std::string& fname) { CxxModuleWrapper::CxxModuleWrapper(const std::string& soPath, const std::string& fname) {
@ -196,45 +47,8 @@ CxxModuleWrapper::CxxModuleWrapper(const std::string& soPath, const std::string&
} }
auto factory = reinterpret_cast<CxxModule* (*)()>(sym); auto factory = reinterpret_cast<CxxModule* (*)()>(sym);
module_.reset((*factory)()); module_.reset((*factory)());
methods_ = module_->getMethods();
} }
std::string CxxModuleWrapper::getName() { std::string CxxModuleWrapper::getName() {
return module_->getName(); return module_->getName();
} }
std::string CxxModuleWrapper::getConstantsJson() {
std::map<std::string, folly::dynamic> constants = module_->getConstants();
folly::dynamic constsobject = folly::dynamic::object;
for (auto& c : constants) {
constsobject.insert(std::move(c.first), std::move(c.second));
}
return folly::toJson(constsobject);
}
jobject CxxModuleWrapper::getMethods() {
static auto hashmap = findClassStatic("java/util/HashMap");
static auto hashmap_put = hashmap->getMethod<jobject(jobject, jobject)>("put");
auto methods = hashmap->newObject(hashmap->getConstructor<jobject()>());
std::unordered_set<std::string> names;
for (const auto& m : methods_) {
if (names.find(m.name) != names.end()) {
throwNewJavaException(gJavaLangIllegalArgumentException,
"C++ Module %s method name already registered: %s",
module_->getName().c_str(), m.name.c_str());
}
names.insert(m.name);
auto name = make_jstring(m.name);
static auto ctor =
CxxMethodWrapper::javaClassStatic()->getConstructor<CxxMethodWrapper::jhybridobject()>();
auto method = CxxMethodWrapper::javaClassStatic()->newObject(ctor);
cthis(method)->method_ = &m;
hashmap_put(methods.get(), name.get(), method.get());
}
return methods.release();
}

View File

@ -2,12 +2,13 @@
#pragma once #pragma once
#include <cxxreact/CxxModule.h>
#include <fb/fbjni.h>
#include <memory> #include <memory>
#include <string> #include <string>
#include <vector> #include <vector>
#include <cxxreact/CxxModule.h>
#include <fb/fbjni.h>
namespace facebook { namespace facebook {
namespace react { namespace react {
@ -27,14 +28,11 @@ public:
// JNI methods // JNI methods
std::string getName(); std::string getName();
std::string getConstantsJson();
jobject getMethods();
// This steals ownership of the underlying module for use by the C++ bridge // This steals ownership of the underlying module for use by the C++ bridge
std::unique_ptr<xplat::module::CxxModule> getModule() { std::unique_ptr<xplat::module::CxxModule> getModule() {
// TODO mhorowitz: remove this (and a lot of other code) once the java // TODO mhorowitz: remove this (and a lot of other code) once the java
// bridge is dead. // bridge is dead.
methods_.clear();
return std::move(module_); return std::move(module_);
} }
@ -42,11 +40,9 @@ protected:
friend HybridBase; friend HybridBase;
explicit CxxModuleWrapper(std::unique_ptr<xplat::module::CxxModule> module) explicit CxxModuleWrapper(std::unique_ptr<xplat::module::CxxModule> module)
: module_(std::move(module)) : module_(std::move(module)) {}
, methods_(module_->getMethods()) {}
std::unique_ptr<xplat::module::CxxModule> module_; std::unique_ptr<xplat::module::CxxModule> module_;
std::vector<xplat::module::CxxModule::Method> methods_;
}; };
} }