You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
249 lines
18 KiB
249 lines
18 KiB
/****************************************************************************
|
|
Copyright (c) 2016 Chukong Technologies Inc.
|
|
Copyright (c) 2017-2023 Xiamen Yaji Software Co., Ltd.
|
|
|
|
http://www.cocos.com
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
in the Software without restriction, including without limitation the rights to
|
|
use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
|
|
of the Software, and to permit persons to whom the Software is furnished to do so,
|
|
subject to the following conditions:
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
all copies or substantial portions of the Software.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
THE SOFTWARE.
|
|
****************************************************************************/
|
|
|
|
#pragma once
|
|
|
|
#include "../ValueArrayPool.h"
|
|
#include "../config.h"
|
|
|
|
//#define RECORD_JSB_INVOKING
|
|
|
|
#ifndef CC_DEBUG
|
|
#undef RECORD_JSB_INVOKING
|
|
#endif
|
|
|
|
#if SCRIPT_ENGINE_TYPE == SCRIPT_ENGINE_SM
|
|
|
|
#if defined(RECORD_JSB_INVOKING)
|
|
|
|
class JsbInvokeScopeT {
|
|
public:
|
|
JsbInvokeScopeT(const char *functionName);
|
|
~JsbInvokeScopeT();
|
|
|
|
private:
|
|
const char *_functionName;
|
|
std::chrono::time_point<std::chrono::high_resolution_clock> _start;
|
|
};
|
|
#define JsbInvokeScope(arg) JsbInvokeScopeT invokeScope(arg); // NOLINT(readability-identifier-naming)
|
|
|
|
#else
|
|
// NOLINTNEXTLINE(readability-identifier-naming)
|
|
#define JsbInvokeScope(arg) \
|
|
do { \
|
|
} while (0)
|
|
|
|
#endif
|
|
|
|
template <typename T, typename STATE>
|
|
constexpr inline T *SE_THIS_OBJECT(STATE &s) { // NOLINT(readability-identifier-naming)
|
|
return reinterpret_cast<T *>(s.nativeThisObject());
|
|
}
|
|
|
|
template <typename T>
|
|
constexpr typename std::enable_if<std::is_enum<T>::value, char *>::type SE_UNDERLYING_TYPE_NAME() { // NOLINT(readability-identifier-naming)
|
|
return typeid(std::underlying_type_t<T>).name();
|
|
}
|
|
|
|
template <typename T>
|
|
constexpr typename std::enable_if<!std::is_enum<T>::value, char *>::type SE_UNDERLYING_TYPE_NAME() { // NOLINT(readability-identifier-naming)
|
|
return typeid(T).name();
|
|
}
|
|
|
|
void clearRecordJSBInvoke();
|
|
|
|
void printJSBInvoke();
|
|
|
|
void printJSBInvokeAtFrame(int n);
|
|
|
|
#define SAFE_INC_REF(obj) \
|
|
if (obj != nullptr) obj->incRef()
|
|
#define SAFE_DEC_REF(obj) \
|
|
if ((obj) != nullptr) { \
|
|
(obj)->decRef(); \
|
|
(obj) = nullptr; \
|
|
}
|
|
|
|
#define _SE(name) name##Registry
|
|
|
|
#define SE_DECLARE_FUNC(funcName) \
|
|
bool funcName##Registry(JSContext *_cx, unsigned argc, JS::Value *_vp)
|
|
|
|
#define SE_BIND_FUNC(funcName) \
|
|
bool funcName##Registry(JSContext *_cx, unsigned argc, JS::Value *_vp) { \
|
|
JsbInvokeScope(#funcName); \
|
|
bool ret = false; \
|
|
JS::CallArgs _argv = JS::CallArgsFromVp(argc, _vp); \
|
|
JS::RootedObject _thizObj(_cx); \
|
|
_argv.computeThis(_cx, &_thizObj); \
|
|
bool needDeleteValueArray{false}; \
|
|
se::ValueArray &args = se::gValueArrayPool.get(argc, needDeleteValueArray); \
|
|
se::CallbackDepthGuard depthGuard{args, se::gValueArrayPool._depth, needDeleteValueArray}; \
|
|
se::internal::jsToSeArgs(_cx, argc, _argv, args); \
|
|
se::PrivateObjectBase *privateObject = static_cast<se::PrivateObjectBase *>(se::internal::getPrivate(_cx, _thizObj, 0)); \
|
|
se::Object *thisObject = reinterpret_cast<se::Object *>(se::internal::getPrivate(_cx, _thizObj, 1)); \
|
|
se::State state(thisObject, privateObject, args); \
|
|
ret = funcName(state); \
|
|
if (!ret) { \
|
|
SE_LOGE("[ERROR] Failed to invoke %s, location: %s:%d\n", #funcName, __FILE__, __LINE__); \
|
|
} \
|
|
se::internal::setReturnValue(_cx, state.rval(), _argv); \
|
|
return ret; \
|
|
}
|
|
|
|
#define SE_BIND_FUNC_FAST(funcName) \
|
|
bool funcName##Registry(JSContext *_cx, unsigned argc, JS::Value *_vp) { \
|
|
JS::CallArgs _argv = JS::CallArgsFromVp(argc, _vp); \
|
|
JS::RootedObject _thizObj(_cx); \
|
|
_argv.computeThis(_cx, &_thizObj); \
|
|
auto *privateObject = static_cast<se::PrivateObjectBase *>(se::internal::SE_JS_GetPrivate(_thizObj, 0)); \
|
|
funcName(privateObject->getRaw()); \
|
|
_argv.rval().setUndefined(); \
|
|
return true; \
|
|
}
|
|
|
|
#define SE_DECLARE_FINALIZE_FUNC(funcName) \
|
|
void funcName##Registry(JSFreeOp *_fop, JSObject *_obj);
|
|
|
|
#define SE_BIND_FINALIZE_FUNC(funcName) \
|
|
void funcName##Registry(JSFreeOp *_fop, JSObject *_obj) { \
|
|
JsbInvokeScope(#funcName); \
|
|
se::PrivateObjectBase *privateObject = static_cast<se::PrivateObjectBase *>(se::internal::SE_JS_GetPrivate(_obj, 0)); \
|
|
se::Object *seObj = static_cast<se::Object *>(se::internal::SE_JS_GetPrivate(_obj, 1)); \
|
|
bool ret = false; \
|
|
if (privateObject == nullptr) \
|
|
return; \
|
|
se::State state(privateObject); \
|
|
ret = funcName(state); \
|
|
if (!ret) { \
|
|
SE_LOGE("[ERROR] Failed to invoke %s, location: %s:%d\n", #funcName, __FILE__, __LINE__); \
|
|
} \
|
|
if (seObj->isClearMappingInFinalizer() && privateObject != nullptr) { \
|
|
void *nativeObj = privateObject->getRaw(); \
|
|
auto iter = se::NativePtrToObjectMap::find(nativeObj); \
|
|
if (iter != se::NativePtrToObjectMap::end()) { \
|
|
se::NativePtrToObjectMap::erase(iter); \
|
|
} \
|
|
} \
|
|
seObj->decRef(); \
|
|
}
|
|
|
|
#define SE_BIND_CTOR(funcName, cls, finalizeCb) \
|
|
bool funcName##Registry(JSContext *_cx, unsigned argc, JS::Value *_vp) { \
|
|
JsbInvokeScope(#funcName); \
|
|
bool ret = false; \
|
|
JS::CallArgs _argv = JS::CallArgsFromVp(argc, _vp); \
|
|
bool needDeleteValueArray{false}; \
|
|
se::ValueArray &args = se::gValueArrayPool.get(argc, needDeleteValueArray); \
|
|
se::CallbackDepthGuard depthGuard{args, se::gValueArrayPool._depth, needDeleteValueArray}; \
|
|
se::internal::jsToSeArgs(_cx, argc, _argv, args); \
|
|
se::Object *thisObject = se::Object::_createJSObjectForConstructor(cls, _argv); \
|
|
thisObject->_setFinalizeCallback(finalizeCb##Registry); \
|
|
_argv.rval().setObject(*thisObject->_getJSObject()); \
|
|
se::State state(thisObject, args); \
|
|
ret = funcName(state); \
|
|
if (ret) { \
|
|
se::Value _property; \
|
|
bool _found = false; \
|
|
_found = thisObject->getProperty("_ctor", &_property); \
|
|
if (_found) _property.toObject()->call(args, thisObject); \
|
|
} else { \
|
|
SE_LOGE("[ERROR] Failed to invoke %s, location: %s:%d\n", #funcName, __FILE__, __LINE__); \
|
|
} \
|
|
return ret; \
|
|
}
|
|
|
|
#define SE_BIND_PROP_GET_IMPL(funcName, postFix) \
|
|
bool funcName##postFix##Registry(JSContext *_cx, unsigned argc, JS::Value *_vp) { \
|
|
JsbInvokeScope(#funcName); \
|
|
bool ret = false; \
|
|
JS::CallArgs _argv = JS::CallArgsFromVp(argc, _vp); \
|
|
JS::RootedObject _thizObj(_cx); \
|
|
_argv.computeThis(_cx, &_thizObj); \
|
|
se::PrivateObjectBase *privateObject = static_cast<se::PrivateObjectBase *>(se::internal::getPrivate(_cx, _thizObj, 0)); \
|
|
se::Object *thisObject = reinterpret_cast<se::Object *>(se::internal::getPrivate(_cx, _thizObj, 1)); \
|
|
se::State state(thisObject, privateObject); \
|
|
ret = funcName(state); \
|
|
if (!ret) { \
|
|
SE_LOGE("[ERROR] Failed to invoke %s, location: %s:%d\n", #funcName, __FILE__, __LINE__); \
|
|
} \
|
|
se::internal::setReturnValue(_cx, state.rval(), _argv); \
|
|
return ret; \
|
|
}
|
|
|
|
#define SE_BIND_PROP_GET(funcName) SE_BIND_PROP_GET_IMPL(funcName, )
|
|
#define SE_BIND_FUNC_AS_PROP_GET(funcName) SE_BIND_PROP_GET_IMPL(funcName, _asGetter)
|
|
|
|
#define SE_BIND_PROP_SET_IMPL(funcName, postFix) \
|
|
bool funcName##postFix##Registry(JSContext *_cx, unsigned _argc, JS::Value *_vp) { \
|
|
JsbInvokeScope(#funcName); \
|
|
bool ret = false; \
|
|
JS::CallArgs _argv = JS::CallArgsFromVp(_argc, _vp); \
|
|
JS::RootedObject _thizObj(_cx); \
|
|
_argv.computeThis(_cx, &_thizObj); \
|
|
se::PrivateObjectBase *privateObject = static_cast<se::PrivateObjectBase *>(se::internal::getPrivate(_cx, _thizObj, 0)); \
|
|
se::Object *thisObject = reinterpret_cast<se::Object *>(se::internal::getPrivate(_cx, _thizObj, 1)); \
|
|
bool needDeleteValueArray{false}; \
|
|
se::ValueArray &args = se::gValueArrayPool.get(1, needDeleteValueArray); \
|
|
se::CallbackDepthGuard depthGuard{args, se::gValueArrayPool._depth, needDeleteValueArray}; \
|
|
se::Value &data{args[0]}; \
|
|
se::internal::jsToSeValue(_cx, _argv[0], &data); \
|
|
se::State state(thisObject, privateObject, args); \
|
|
ret = funcName(state); \
|
|
if (!ret) { \
|
|
SE_LOGE("[ERROR] Failed to invoke %s, location: %s:%d\n", #funcName, __FILE__, __LINE__); \
|
|
} \
|
|
return ret; \
|
|
}
|
|
|
|
#define SE_BIND_PROP_SET(funcName) SE_BIND_PROP_SET_IMPL(funcName, )
|
|
#define SE_BIND_FUNC_AS_PROP_SET(funcName) SE_BIND_PROP_SET_IMPL(funcName, _asSetter)
|
|
|
|
#define SE_TYPE_NAME(t) typeid(t).name()
|
|
|
|
#define SE_QUOTEME_(x) #x
|
|
#define SE_QUOTEME(x) SE_QUOTEME_(x)
|
|
|
|
#define SE_REPORT_ERROR(fmt, ...) \
|
|
SE_LOGD("ERROR (" __FILE__ ", " SE_QUOTEME(__LINE__) "): " fmt "\n", ##__VA_ARGS__); \
|
|
JS_ReportErrorUTF8(se::ScriptEngine::getInstance()->_getContext(), fmt, ##__VA_ARGS__)
|
|
|
|
#if CC_DEBUG > 0
|
|
|
|
#define SE_ASSERT(cond, fmt, ...) \
|
|
do { \
|
|
if (!(cond)) { \
|
|
SE_LOGE("ASSERT (" __FILE__ ", " SE_QUOTEME(__LINE__) "): " fmt "\n", ##__VA_ARGS__); \
|
|
assert(false); \
|
|
} \
|
|
} while (false)
|
|
|
|
#else
|
|
|
|
#define SE_ASSERT(cond, fmt, ...)
|
|
|
|
#endif // #if CC_DEBUG > 0
|
|
|
|
#endif // #if SCRIPT_ENGINE_TYPE == SCRIPT_ENGINE_SM
|
|
|