#ifndef __JS_PLUGINX_SPIDERMONKEY_SPECIFICS_H__ #define __JS_PLUGINX_SPIDERMONKEY_SPECIFICS_H__ #include "cocos2d.h" #include #include #include #include #include "jsapi.h" #include "uthash.h" #include #ifdef ANDROID #include #endif #ifndef CCASSERT #define CCASSERT(a,b) assert(a) #endif #define PLUGINX_JSB_DEBUG 0 namespace pluginx { #if PLUGINX_JSB_DEBUG #ifdef ANDROID #define LOG_TAG "jsb_pluginx" #define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG,LOG_TAG,__VA_ARGS__) #else #define LOGD(...) printf(__VA_ARGS__) #endif #else #define LOGD(...) do {} while (0) #endif #define JSB_PRECONDITION( condition, ...) do { \ if( ! (condition) ) { \ LOGD("jsb: ERROR: File %s: Line: %d, Function: %s", __FILE__, __LINE__, __FUNCTION__ ); \ LOGD(__VA_ARGS__); \ JSContext* globalContext = ScriptingCore::getInstance()->getGlobalContext(); \ if( ! JS_IsExceptionPending( globalContext ) ) { \ JS_ReportError( globalContext, __VA_ARGS__ ); \ } \ return false; \ } \ } while(0) #define JSB_PRECONDITION2( condition, context, ret_value, ...) do { \ if( ! (condition) ) { \ LOGD("jsb: ERROR: File %s: Line: %d, Function: %s", __FILE__, __LINE__, __FUNCTION__ ); \ LOGD(__VA_ARGS__); \ if( ! JS_IsExceptionPending( context ) ) { \ JS_ReportError( context, __VA_ARGS__ ); \ } \ return ret_value; \ } \ } while(0) typedef struct js_proxy { void *ptr; JS::Heap obj; UT_hash_handle hh; } js_proxy_t; extern js_proxy_t *_native_js_global_ht; extern js_proxy_t *_js_native_global_ht; typedef struct js_type_class { JSClass *jsclass; JS::Heap proto; JS::Heap parentProto; } js_type_class_t; extern std::unordered_map _js_global_type_map; unsigned int getHashCodeByString(const char *key); template< typename DERIVED > class TypeTest { public: static int s_id() { // return id unique for DERIVED // NOT SURE IT WILL BE REALLY UNIQUE FOR EACH CLASS!! /* Commented by James Chen Using 'getHashCodeByString(typeid(*native_obj).name())' instead of 'reinterpret_cast(typeid(*native_obj).name());'. Since on win32 platform, 'reinterpret_cast(typeid(*native_obj).name());' invoking in cocos2d.dll and outside cocos2d.dll(in TestJavascript.exe) will return different address. But the return string from typeid(*native_obj).name() is the same string, so we must convert the string to hash id to make sure we can get unique id. */ // static const long id = reinterpret_cast(typeid( DERIVED ).name()); static const long id = getHashCodeByString(typeid( DERIVED ).name()); return id; } static const char* s_name() { // return id unique for DERIVED // ALWAYS VALID BUT STRING, NOT INT - BUT VALID AND CROSS-PLATFORM/CROSS-VERSION COMPATBLE // AS FAR AS YOU KEEP THE CLASS NAME return typeid( DERIVED ).name(); } }; #define JS_NEW_PROXY(p, native_obj, js_obj) \ do { \ p = (js_proxy_t *)malloc(sizeof(js_proxy_t)); \ assert(p); \ js_proxy_t* native_obj##js_obj##tmp = NULL; \ HASH_FIND_PTR(_native_js_global_ht, &native_obj, native_obj##js_obj##tmp); \ assert(!native_obj##js_obj##tmp); \ p->ptr = native_obj; \ p->obj = js_obj; \ HASH_ADD_PTR(_native_js_global_ht, ptr, p); \ p = (js_proxy_t *)malloc(sizeof(js_proxy_t)); \ assert(p); \ native_obj##js_obj##tmp = NULL; \ HASH_FIND_PTR(_js_native_global_ht, &js_obj, native_obj##js_obj##tmp); \ assert(!native_obj##js_obj##tmp); \ p->ptr = native_obj; \ p->obj = js_obj; \ HASH_ADD_PTR(_js_native_global_ht, obj, p); \ } while(0) \ #define JS_GET_PROXY(p, native_obj) \ do { \ HASH_FIND_PTR(_native_js_global_ht, &native_obj, p); \ } while (0) #define JS_GET_NATIVE_PROXY(p, js_obj) \ do { \ HASH_FIND_PTR(_js_native_global_ht, &js_obj, p); \ } while (0) #define JS_REMOVE_PROXY(nproxy, jsproxy) \ do { \ if (nproxy) { HASH_DEL(_native_js_global_ht, nproxy); free(nproxy); } \ if (jsproxy) { HASH_DEL(_js_native_global_ht, jsproxy); free(jsproxy); } \ } while (0) #define JS_TEST_NATIVE_OBJECT(cx, native_obj) \ if (!native_obj) { \ JS_ReportError(cx, "Invalid Native Object"); \ return false; \ } js_proxy_t* jsb_new_proxy(void* nativeObj, JSObject* jsObj); js_proxy_t* jsb_get_native_proxy(void* nativeObj); js_proxy_t* jsb_get_js_proxy(JSObject* jsObj); void jsb_remove_proxy(js_proxy_t* nativeProxy, js_proxy_t* jsProxy); void get_or_create_js_obj(JSContext* cx, JS::HandleObject obj, const std::string &name, JS::MutableHandleObject jsObj); /** * You don't need to manage the returned pointer. They live for the whole life of * the app. */ template inline js_type_class_t *js_get_type_from_native(T* native_obj) { bool found = false; std::string typeName = typeid(*native_obj).name(); auto typeProxyIter = _js_global_type_map.find(typeName); if (typeProxyIter == _js_global_type_map.end()) { typeName = typeid(T).name(); typeProxyIter = _js_global_type_map.find(typeName); if (typeProxyIter != _js_global_type_map.end()) { found = true; } } else { found = true; } return found ? typeProxyIter->second : nullptr; } /** * The returned pointer should be deleted using jsb_remove_proxy. Most of the * time you do that in the C++ destructor. */ template inline js_proxy_t *js_get_or_create_proxy(JSContext *cx, T *native_obj) { js_proxy_t *proxy; HASH_FIND_PTR(_native_js_global_ht, &native_obj, proxy); if (!proxy) { js_type_class_t *typeProxy = js_get_type_from_native(native_obj); // Return NULL if can't find its type rather than making an assert. // assert(typeProxy); if (!typeProxy) { CCLOGINFO("Could not find the type of native object."); return NULL; } //JSB_AUTOCOMPARTMENT_WITH_GLOBAL_OBJCET JSObject* js_obj = JS_NewObject(cx, typeProxy->jsclass, JS::RootedObject(cx, typeProxy->proto), JS::RootedObject(cx,typeProxy->parentProto)); proxy = jsb_new_proxy(native_obj, js_obj); #ifdef DEBUG JS::AddNamedObjectRoot(cx, &proxy->obj, typeid(*native_obj).name()); #else JS::AddObjectRoot(cx, &proxy->obj); #endif return proxy; } else { return proxy; } return NULL; } } // namespace pluginx { #endif /* __JS_PLUGINX_SPIDERMONKEY_SPECIFICS_H__ */