mirror of
https://github.com/quizhizhe/LiteLoaderBDS-1.16.40.git
synced 2025-06-06 04:03:39 +00:00
234 lines
6.8 KiB
C++
234 lines
6.8 KiB
C++
/*
|
|
* Tencent is pleased to support the open source community by making ScriptX available.
|
|
* Copyright (C) 2021 THL A29 Limited, a Tencent company. All rights reserved.
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <JavaScriptCore/JavaScript.h>
|
|
#include <unordered_map>
|
|
|
|
#include "../../src/Engine.h"
|
|
#include "../../src/Native.h"
|
|
#include "../../src/utils/GlobalWeakBookkeeping.hpp"
|
|
#include "JscHelper.h"
|
|
|
|
namespace script::internal {
|
|
|
|
// forward declare
|
|
// defined in JscEngine.h
|
|
template <typename T, typename Args>
|
|
struct MakeLocalHelper;
|
|
|
|
} // namespace script::internal
|
|
|
|
namespace script::jsc_backend {
|
|
|
|
class JscEngine : public ::script::ScriptEngine {
|
|
struct ClassRegistryData {
|
|
JSClassRef instanceClass{};
|
|
Global<Object> constructor{};
|
|
Global<Object> prototype{};
|
|
};
|
|
|
|
std::shared_ptr<utils::MessageQueue> messageQueue_;
|
|
std::unordered_map<const void*, ClassRegistryData> classRegistry_;
|
|
std::unordered_map<size_t, Global<Value>> keptReference_;
|
|
internal::GlobalWeakBookkeeping globalWeakBookkeeping_;
|
|
size_t keepId_ = 0;
|
|
bool isDestroying_ = false;
|
|
|
|
static JSClassRef globalClass_;
|
|
static JSClassRef externalClass_;
|
|
// >= ios10, >= macOS 10.12
|
|
static bool hasByteBufferAPI_;
|
|
|
|
protected:
|
|
std::shared_ptr<std::recursive_mutex> virtualMachineLock_ =
|
|
std::make_shared<std::recursive_mutex>();
|
|
JSContextGroupRef virtualMachine_ = nullptr;
|
|
JSGlobalContextRef context_ = nullptr;
|
|
Global<Value> internalStorageSymbol_;
|
|
Global<Value> constructorMarkSymbol_;
|
|
Global<Function> getInternalStorageBySymbolFunction_;
|
|
Global<Function> isByteBuffer_;
|
|
|
|
private:
|
|
static void checkException(JSValueRef exception);
|
|
|
|
public:
|
|
explicit JscEngine(std::shared_ptr<utils::MessageQueue> messageQueue = {});
|
|
|
|
void destroy() noexcept override;
|
|
|
|
bool isDestroying() const override { return isDestroying_; }
|
|
|
|
Local<Object> getGlobal();
|
|
|
|
Local<Value> get(const Local<String>& key) override;
|
|
|
|
void set(const Local<String>& key, const Local<Value>& value) override;
|
|
using ScriptEngine::set;
|
|
|
|
Local<Value> eval(const Local<String>& script, const Local<String>& sourceFile) override;
|
|
Local<Value> eval(const Local<String>& script) override;
|
|
using ScriptEngine::eval;
|
|
|
|
Local<Value> loadFile(const Local<String>& scriptFile) override;
|
|
|
|
std::shared_ptr<utils::MessageQueue> messageQueue() override;
|
|
|
|
void gc() override;
|
|
|
|
void adjustAssociatedMemory(int64_t count) override;
|
|
|
|
ScriptLanguage getLanguageType() override;
|
|
|
|
std::string getEngineVersion() override;
|
|
|
|
protected:
|
|
~JscEngine() override;
|
|
|
|
private:
|
|
Local<Value> eval(const Local<String>& script, const Local<Value>& sourceFile);
|
|
|
|
template <typename T>
|
|
bool registerNativeClassImpl(const ClassDefine<T>* classDefine);
|
|
|
|
template <typename T>
|
|
Local<Object> newNativeClassImpl(const ClassDefine<T>* classDefine, size_t size,
|
|
const Local<Value>* args);
|
|
|
|
template <typename T>
|
|
bool isInstanceOfImpl(const Local<Value>& value, const ClassDefine<T>* classDefine);
|
|
|
|
template <typename T>
|
|
T* getNativeInstanceImpl(const Local<Value>& value, const ClassDefine<T>* classDefine);
|
|
|
|
private:
|
|
template <typename T>
|
|
static inline typename RefTypeMap<T>::jscType toJsc(JSGlobalContextRef /*context*/,
|
|
const Local<T>& ref) {
|
|
return ref.val_;
|
|
}
|
|
|
|
static inline typename RefTypeMap<String>::jscType toJsc(JSGlobalContextRef context,
|
|
const Local<String>& ref) {
|
|
return ref.val_.getValue(context);
|
|
}
|
|
|
|
static typename RefTypeMap<Value>::jscType toJsc(JSGlobalContextRef context,
|
|
const Local<Value>& ref);
|
|
|
|
static Arguments newArguments(JscEngine* engine, JSObjectRef thisObject,
|
|
const JSValueRef* arguments, size_t size);
|
|
|
|
template <typename T, typename... Args>
|
|
static T make(Args&&... args) {
|
|
return T(std::forward<Args>(args)...);
|
|
}
|
|
|
|
Local<Function> newStaticFunction(const internal::StaticDefine::FunctionDefine& func);
|
|
|
|
Local<Value> newStaticGetter(const internal::StaticDefine::PropertyDefine& prop);
|
|
|
|
Local<Value> newStaticSetter(const internal::StaticDefine::PropertyDefine& prop);
|
|
|
|
void registerStaticDefine(const internal::StaticDefine& staticDefine,
|
|
const Local<Object>& object);
|
|
|
|
template <typename T>
|
|
void defineInstance(const ClassDefine<T>* classDefine, Local<Value>& object,
|
|
ClassRegistryData& registry);
|
|
|
|
template <typename T>
|
|
JSObjectCallAsConstructorCallback createConstructor();
|
|
|
|
template <typename T>
|
|
Local<Object> defineInstancePrototype(const ClassDefine<T>* classDefine);
|
|
|
|
template <typename T>
|
|
void defineInstanceFunction(const ClassDefine<T>* classDefine, Local<Object>& prototypeObject);
|
|
|
|
template <typename T, typename ConsumeLambda>
|
|
void defineInstanceProperties(const ClassDefine<T>* classDefine, ConsumeLambda consumerLambda);
|
|
|
|
size_t keepReference(const Local<Value>& ref);
|
|
|
|
void removeKeptReference(size_t id);
|
|
|
|
void initInternalSymbols();
|
|
|
|
bool isConstructorMarkSymbol(JSValueRef value);
|
|
|
|
private:
|
|
template <typename T>
|
|
friend class ::script::Local;
|
|
|
|
template <typename T>
|
|
friend class ::script::Global;
|
|
|
|
template <typename T>
|
|
friend class ::script::Weak;
|
|
|
|
friend class ::script::Object;
|
|
|
|
friend class ::script::Array;
|
|
|
|
friend class ::script::Function;
|
|
|
|
friend class ::script::ByteBuffer;
|
|
|
|
friend class ::script::ScriptEngine;
|
|
|
|
friend class ::script::Exception;
|
|
|
|
friend class ::script::Arguments;
|
|
|
|
friend class ::script::ScriptClass;
|
|
|
|
friend class JscStringRefHolder;
|
|
|
|
friend class JscWeakRef;
|
|
|
|
friend JSGlobalContextRef currentEngineContextChecked();
|
|
friend JSContextGroupRef currentEngineContextGroupChecked();
|
|
|
|
friend class JscEngineScope;
|
|
|
|
friend class JscExitEngineScope;
|
|
|
|
friend class StringLocalRef;
|
|
|
|
friend struct JscBookKeepFetcher;
|
|
|
|
template <typename R, typename Fn>
|
|
friend R toJscValues(JSGlobalContextRef context, size_t length, const Local<Value>* args, Fn fn);
|
|
|
|
friend JSObjectRef valueToObj(JSGlobalContextRef context, JSValueRef value);
|
|
|
|
friend struct ::script::jsc_interop;
|
|
|
|
template <typename T, typename Args>
|
|
friend struct ::script::internal::MakeLocalHelper;
|
|
|
|
template <typename Ref>
|
|
static auto& refVal(Ref* ref) {
|
|
return ref->val_;
|
|
}
|
|
};
|
|
|
|
} // namespace script::jsc_backend
|