From 166f7223d1a9fe59c01145f3a70e8eb7a32fc65f Mon Sep 17 00:00:00 2001 From: Karol Suprynowicz Date: Fri, 18 Aug 2023 12:08:36 +0200 Subject: [PATCH] Fix most of the crash causes on script engine reload/shutdown --- interface/src/Application.cpp | 5 +- .../controllers/src/controllers/Actions.cpp | 2 +- .../src/controllers/InputDevice.cpp | 2 +- .../src/controllers/StateController.cpp | 2 +- .../src/controllers/UserInputMapper.cpp | 31 +++++---- .../src/controllers/UserInputMapper.h | 5 +- .../src/controllers/impl/Endpoint.h | 30 +++++--- .../impl/endpoints/ActionEndpoint.h | 6 +- .../controllers/impl/endpoints/AnyEndpoint.h | 7 +- .../impl/endpoints/ArrayEndpoint.h | 6 +- .../impl/endpoints/CompositeEndpoint.h | 7 +- .../impl/endpoints/InputEndpoint.h | 10 ++- .../controllers/impl/endpoints/JSEndpoint.h | 12 ++-- .../impl/endpoints/ScriptEndpoint.cpp | 5 +- .../impl/endpoints/ScriptEndpoint.h | 9 ++- .../src/EntityTreeRenderer.cpp | 8 +++ libraries/script-engine/src/ScriptEngine.h | 5 ++ libraries/script-engine/src/ScriptEngines.cpp | 69 +++++++++++-------- libraries/script-engine/src/ScriptManager.cpp | 30 ++++++-- libraries/script-engine/src/ScriptManager.h | 9 +++ .../script-engine/src/v8/ScriptEngineV8.cpp | 18 +++++ .../script-engine/src/v8/ScriptEngineV8.h | 13 ++++ .../src/v8/ScriptObjectV8Proxy.cpp | 9 +++ .../src/v8/ScriptObjectV8Proxy.h | 3 + .../src/v8/ScriptValueV8Wrapper.cpp | 3 + 25 files changed, 222 insertions(+), 84 deletions(-) diff --git a/interface/src/Application.cpp b/interface/src/Application.cpp index 10574f330d..642f183cfc 100644 --- a/interface/src/Application.cpp +++ b/interface/src/Application.cpp @@ -7556,8 +7556,9 @@ void Application::registerScriptEngineWithApplicationServices(ScriptManagerPoint *connection = scriptManager->connect(scriptManager.get(), &ScriptManager::scriptEnding, [scriptManager, connection]() { // Request removal of controller routes with callbacks to a given script engine auto userInputMapper = DependencyManager::get(); - userInputMapper->scheduleScriptEndpointCleanup(scriptManager->engine().get()); - // V8TODO: Maybe we should wait until endpoint cleanup is finished before deleting the script engine if there are still crashes + // scheduleScriptEndpointCleanup will have the last instance of shared pointer to script manager + // so script manager will get deleted as soon as cleanup is done + userInputMapper->scheduleScriptEndpointCleanup(scriptManager); QObject::disconnect(*connection); }); } diff --git a/libraries/controllers/src/controllers/Actions.cpp b/libraries/controllers/src/controllers/Actions.cpp index 5f95cff76b..8632fe2241 100644 --- a/libraries/controllers/src/controllers/Actions.cpp +++ b/libraries/controllers/src/controllers/Actions.cpp @@ -30,7 +30,7 @@ namespace controller { } EndpointPointer ActionsDevice::createEndpoint(const Input& input) const { - return std::make_shared(input); + return ActionEndpoint::newEndpoint(input); } /*@jsdoc diff --git a/libraries/controllers/src/controllers/InputDevice.cpp b/libraries/controllers/src/controllers/InputDevice.cpp index b28e7cfc82..58f7a805fa 100644 --- a/libraries/controllers/src/controllers/InputDevice.cpp +++ b/libraries/controllers/src/controllers/InputDevice.cpp @@ -91,7 +91,7 @@ namespace controller { } EndpointPointer InputDevice::createEndpoint(const Input& input) const { - return std::make_shared(input); + return InputEndpoint::newEndpoint(input); } } diff --git a/libraries/controllers/src/controllers/StateController.cpp b/libraries/controllers/src/controllers/StateController.cpp index 9b0301eb14..1397bd1f77 100644 --- a/libraries/controllers/src/controllers/StateController.cpp +++ b/libraries/controllers/src/controllers/StateController.cpp @@ -51,7 +51,7 @@ Input::NamedVector StateController::getAvailableInputs() const { EndpointPointer StateController::createEndpoint(const Input& input) const { auto name = stateVariables[input.getChannel()]; ReadLambda& readLambda = const_cast&>(_namedReadLambdas)[name]; - return std::make_shared(readLambda); + return LambdaRefEndpoint::newEndpoint(readLambda); } } \ No newline at end of file diff --git a/libraries/controllers/src/controllers/UserInputMapper.cpp b/libraries/controllers/src/controllers/UserInputMapper.cpp index 0a23f6e386..40c388ac74 100644 --- a/libraries/controllers/src/controllers/UserInputMapper.cpp +++ b/libraries/controllers/src/controllers/UserInputMapper.cpp @@ -32,6 +32,7 @@ #include #include #include +#include #include "impl/conditionals/AndConditional.h" #include "impl/conditionals/NotConditional.h" @@ -92,9 +93,9 @@ void UserInputMapper::registerDevice(InputDevice::Pointer device) { if (input.device == STANDARD_DEVICE) { endpoint = std::make_shared(input); } else if (input.device == ACTIONS_DEVICE) { - endpoint = std::make_shared(input); + endpoint = ActionEndpoint::newEndpoint(input); } else { - endpoint = std::make_shared(input); + endpoint = InputEndpoint::newEndpoint(input); } } _inputsByEndpoint[endpoint] = input; @@ -663,7 +664,7 @@ Endpoint::Pointer UserInputMapper::endpointFor(const QJSValue& endpoint) { } if (endpoint.isCallable()) { - auto result = std::make_shared(endpoint); + auto result = JSEndpoint::newEndpoint(endpoint); return result; } @@ -677,7 +678,7 @@ Endpoint::Pointer UserInputMapper::endpointFor(const ScriptValue& endpoint) { } if (endpoint.isFunction()) { - auto result = std::make_shared(endpoint); + auto result = ScriptEndpoint::newEndpoint(endpoint); return result; } @@ -692,7 +693,7 @@ Endpoint::Pointer UserInputMapper::endpointFor(const ScriptValue& endpoint) { } children.push_back(destination); } - return std::make_shared(children); + return AnyEndpoint::newEndpoint(children); } @@ -715,7 +716,7 @@ Endpoint::Pointer UserInputMapper::compositeEndpointFor(Endpoint::Pointer first, Endpoint::Pointer result; auto iterator = _compositeEndpoints.find(pair); if (_compositeEndpoints.end() == iterator) { - result = std::make_shared(first, second); + result = CompositeEndpoint::newEndpoint(first, second); _compositeEndpoints[pair] = result; } else { result = iterator->second; @@ -858,9 +859,9 @@ void UserInputMapper::unloadMapping(const QString& jsonFile) { } } -void UserInputMapper::scheduleScriptEndpointCleanup(ScriptEngine* engine) { +void UserInputMapper::scheduleScriptEndpointCleanup(std::shared_ptr manager) { _lock.lock(); - scriptEnginesRequestingCleanup.enqueue(engine); + scriptManagersRequestingCleanup.enqueue(manager); _lock.unlock(); } @@ -1025,7 +1026,7 @@ Filter::List UserInputMapper::parseFilters(const QJsonValue& value) { Endpoint::Pointer UserInputMapper::parseDestination(const QJsonValue& value) { if (value.isArray()) { - ArrayEndpoint::Pointer result = std::make_shared(); + ArrayEndpoint::Pointer result = std::dynamic_pointer_cast(ArrayEndpoint::newEndpoint()); auto array = value.toArray(); for (const auto& arrayItem : array) { Endpoint::Pointer destination = parseEndpoint(arrayItem); @@ -1052,7 +1053,7 @@ Endpoint::Pointer UserInputMapper::parseAxis(const QJsonValue& value) { Endpoint::Pointer first = parseEndpoint(axisArray.first()); Endpoint::Pointer second = parseEndpoint(axisArray.last()); if (first && second) { - return std::make_shared(first, second); + return CompositeEndpoint::newEndpoint(first, second); } } } @@ -1072,7 +1073,7 @@ Endpoint::Pointer UserInputMapper::parseAny(const QJsonValue& value) { } children.push_back(destination); } - return std::make_shared(children); + return AnyEndpoint::newEndpoint(children); } return Endpoint::Pointer(); } @@ -1261,8 +1262,8 @@ void UserInputMapper::disableMapping(const Mapping::Pointer& mapping) { void UserInputMapper::runScriptEndpointCleanup() { _lock.lock(); QList routesToRemove; - while (!scriptEnginesRequestingCleanup.empty()){ - auto engine = scriptEnginesRequestingCleanup.dequeue(); + while (!scriptManagersRequestingCleanup.empty()){ + auto manager = scriptManagersRequestingCleanup.dequeue(); QList routeLists = {&_deviceRoutes, &_standardRoutes}; auto iterator = _mappingsByName.begin(); while (iterator != _mappingsByName.end()) { @@ -1274,12 +1275,12 @@ void UserInputMapper::runScriptEndpointCleanup() { for (auto routeList: routeLists) { for (auto route: *routeList) { auto source = std::dynamic_pointer_cast(route->source); - if (source && source->getEngine() == engine) { + if (source && source->getEngine() == manager->engine().get()) { qDebug() << "UserInputMapper::runScriptEndpointCleanup source"; routesToRemove.append(route); } auto destination = std::dynamic_pointer_cast(route->destination); - if (destination && destination->getEngine() == engine) { + if (destination && destination->getEngine() == manager->engine().get()) { qDebug() << "UserInputMapper::runScriptEndpointCleanup destination"; routesToRemove.append(route); } diff --git a/libraries/controllers/src/controllers/UserInputMapper.h b/libraries/controllers/src/controllers/UserInputMapper.h index 401f77a74c..c20b040a85 100644 --- a/libraries/controllers/src/controllers/UserInputMapper.h +++ b/libraries/controllers/src/controllers/UserInputMapper.h @@ -35,6 +35,7 @@ #include "StateController.h" class ScriptEngine; +class ScriptManager; class ScriptValue; namespace controller { @@ -136,7 +137,7 @@ namespace controller { * * @param engine Pointer to the script engine that will be shut down */ - void scheduleScriptEndpointCleanup(ScriptEngine* engine); + void scheduleScriptEndpointCleanup(std::shared_ptr manager); AxisValue getValue(const Input& input) const; Pose getPose(const Input& input) const; @@ -223,7 +224,7 @@ namespace controller { InputCalibrationData inputCalibrationData; // Contains pointers to script engines that are requesting callback cleanup during their shutdown process - QQueue scriptEnginesRequestingCleanup; + QQueue> scriptManagersRequestingCleanup; mutable std::recursive_mutex _lock; }; diff --git a/libraries/controllers/src/controllers/impl/Endpoint.h b/libraries/controllers/src/controllers/impl/Endpoint.h index 44a8c254ff..0eb0ec89a6 100644 --- a/libraries/controllers/src/controllers/impl/Endpoint.h +++ b/libraries/controllers/src/controllers/impl/Endpoint.h @@ -27,7 +27,9 @@ namespace controller { * Encapsulates a particular input / output, * i.e. Hydra.Button0, Standard.X, Action.Yaw */ - class Endpoint : public QObject { + +// All the derived classes need to have private constructors +class Endpoint : public QObject, public std::enable_shared_from_this { Q_OBJECT; public: using Pointer = std::shared_ptr; @@ -36,7 +38,6 @@ namespace controller { using ReadLambda = std::function; using WriteLambda = std::function; - Endpoint(const Input& input) : _input(input) {} virtual AxisValue value() { return peek(); } virtual AxisValue peek() const = 0; virtual void apply(AxisValue value, const Pointer& source) = 0; @@ -51,19 +52,21 @@ namespace controller { const Input& getInput() { return _input; } protected: + Endpoint(const Input& input) : _input(input) {} Input _input; }; class LambdaEndpoint : public Endpoint { public: using Endpoint::apply; - LambdaEndpoint(ReadLambda readLambda, WriteLambda writeLambda = [](float) {}) - : Endpoint(Input::INVALID_INPUT), _readLambda(readLambda), _writeLambda(writeLambda) { } virtual AxisValue peek() const override { return AxisValue(_readLambda(), 0); } virtual void apply(AxisValue value, const Pointer& source) override { _writeLambda(value.value); } private: + LambdaEndpoint(ReadLambda readLambda, WriteLambda writeLambda = [](float) {}) + : Endpoint(Input::INVALID_INPUT), _readLambda(readLambda), _writeLambda(writeLambda) { } + ReadLambda _readLambda; WriteLambda _writeLambda; }; @@ -72,15 +75,20 @@ namespace controller { class LambdaRefEndpoint : public Endpoint { public: + static std::shared_ptr newEndpoint(const ReadLambda& readLambda, const WriteLambda& writeLambda = DEFAULT_WRITE_LAMBDA) { + return std::shared_ptr(new LambdaRefEndpoint(readLambda, writeLambda)); + }; + using Endpoint::apply; - LambdaRefEndpoint(const ReadLambda& readLambda, const WriteLambda& writeLambda = DEFAULT_WRITE_LAMBDA) - : Endpoint(Input::INVALID_INPUT), _readLambda(readLambda), _writeLambda(writeLambda) { - } virtual AxisValue peek() const override { return AxisValue(_readLambda(), 0); } virtual void apply(AxisValue value, const Pointer& source) override { _writeLambda(value.value); } private: + LambdaRefEndpoint(const ReadLambda& readLambda, const WriteLambda& writeLambda = DEFAULT_WRITE_LAMBDA) + : Endpoint(Input::INVALID_INPUT), _readLambda(readLambda), _writeLambda(writeLambda) { + } + const ReadLambda& _readLambda; const WriteLambda& _writeLambda; }; @@ -88,10 +96,6 @@ namespace controller { class VirtualEndpoint : public Endpoint { public: - VirtualEndpoint(const Input& id = Input::INVALID_INPUT) - : Endpoint(id) { - } - virtual AxisValue peek() const override { return _currentValue; } virtual void apply(AxisValue value, const Pointer& source) override { _currentValue = value; } @@ -100,6 +104,10 @@ namespace controller { _currentPose = value; } protected: + VirtualEndpoint(const Input& id = Input::INVALID_INPUT) + : Endpoint(id) { + } + AxisValue _currentValue { 0.0f, 0, false }; Pose _currentPose {}; }; diff --git a/libraries/controllers/src/controllers/impl/endpoints/ActionEndpoint.h b/libraries/controllers/src/controllers/impl/endpoints/ActionEndpoint.h index 7ab21031a7..898c7759d2 100644 --- a/libraries/controllers/src/controllers/impl/endpoints/ActionEndpoint.h +++ b/libraries/controllers/src/controllers/impl/endpoints/ActionEndpoint.h @@ -21,7 +21,9 @@ namespace controller { class ActionEndpoint : public Endpoint { public: - ActionEndpoint(const Input& id = Input::INVALID_INPUT) : Endpoint(id) { } + static std::shared_ptr newEndpoint(const Input& id = Input::INVALID_INPUT) { + return std::shared_ptr(new ActionEndpoint(id)); + }; virtual AxisValue peek() const override { return _currentValue; } virtual void apply(AxisValue newValue, const Pointer& source) override; @@ -32,6 +34,8 @@ public: virtual void reset() override; private: + ActionEndpoint(const Input& id = Input::INVALID_INPUT) : Endpoint(id) { } + AxisValue _currentValue { 0.0f, 0, false }; Pose _currentPose{}; }; diff --git a/libraries/controllers/src/controllers/impl/endpoints/AnyEndpoint.h b/libraries/controllers/src/controllers/impl/endpoints/AnyEndpoint.h index a9634f10d8..191995118a 100644 --- a/libraries/controllers/src/controllers/impl/endpoints/AnyEndpoint.h +++ b/libraries/controllers/src/controllers/impl/endpoints/AnyEndpoint.h @@ -17,8 +17,11 @@ namespace controller { class AnyEndpoint : public Endpoint { friend class UserInputMapper; public: + static std::shared_ptr newEndpoint(Endpoint::List children) { + return std::shared_ptr(new AnyEndpoint(children)); + }; + using Endpoint::apply; - AnyEndpoint(Endpoint::List children); virtual AxisValue peek() const override; virtual AxisValue value() override; virtual void apply(AxisValue newValue, const Endpoint::Pointer& source) override; @@ -26,6 +29,8 @@ public: virtual bool readable() const override; private: + AnyEndpoint(Endpoint::List children); + Endpoint::List _children; }; diff --git a/libraries/controllers/src/controllers/impl/endpoints/ArrayEndpoint.h b/libraries/controllers/src/controllers/impl/endpoints/ArrayEndpoint.h index f928a4f568..0f97848e33 100644 --- a/libraries/controllers/src/controllers/impl/endpoints/ArrayEndpoint.h +++ b/libraries/controllers/src/controllers/impl/endpoints/ArrayEndpoint.h @@ -17,9 +17,12 @@ namespace controller { class ArrayEndpoint : public Endpoint { friend class UserInputMapper; public: + static std::shared_ptr newEndpoint() { + return std::shared_ptr(new ArrayEndpoint()); + }; + using Endpoint::apply; using Pointer = std::shared_ptr; - ArrayEndpoint() : Endpoint(Input::INVALID_INPUT) { } virtual AxisValue peek() const override { return AxisValue(); } @@ -34,6 +37,7 @@ public: virtual bool readable() const override { return false; } private: + ArrayEndpoint() : Endpoint(Input::INVALID_INPUT) { } Endpoint::List _children; }; diff --git a/libraries/controllers/src/controllers/impl/endpoints/CompositeEndpoint.h b/libraries/controllers/src/controllers/impl/endpoints/CompositeEndpoint.h index 004df36273..c8c99eba7b 100644 --- a/libraries/controllers/src/controllers/impl/endpoints/CompositeEndpoint.h +++ b/libraries/controllers/src/controllers/impl/endpoints/CompositeEndpoint.h @@ -15,14 +15,19 @@ namespace controller { class CompositeEndpoint : public Endpoint, Endpoint::Pair { public: + static std::shared_ptr newEndpoint(Endpoint::Pointer first, Endpoint::Pointer second) { + return std::shared_ptr(new CompositeEndpoint(first, second)); + }; + using Endpoint::apply; - CompositeEndpoint(Endpoint::Pointer first, Endpoint::Pointer second); virtual AxisValue peek() const override; virtual AxisValue value() override; virtual void apply(AxisValue newValue, const Pointer& source) override; virtual bool readable() const override; + private: + CompositeEndpoint(Endpoint::Pointer first, Endpoint::Pointer second); }; } diff --git a/libraries/controllers/src/controllers/impl/endpoints/InputEndpoint.h b/libraries/controllers/src/controllers/impl/endpoints/InputEndpoint.h index 9581228cef..65a6da84ea 100644 --- a/libraries/controllers/src/controllers/impl/endpoints/InputEndpoint.h +++ b/libraries/controllers/src/controllers/impl/endpoints/InputEndpoint.h @@ -16,9 +16,9 @@ namespace controller { class InputEndpoint : public Endpoint { public: - InputEndpoint(const Input& id = Input::INVALID_INPUT) - : Endpoint(id) { - } + static std::shared_ptr newEndpoint(const Input& id = Input::INVALID_INPUT) { + return std::shared_ptr(new InputEndpoint(id)); + }; virtual AxisValue peek() const override; virtual AxisValue value() override; @@ -34,6 +34,10 @@ public: virtual void reset() override { _read = false; } private: + InputEndpoint(const Input& id = Input::INVALID_INPUT) + : Endpoint(id) { + } + bool _read { false }; }; diff --git a/libraries/controllers/src/controllers/impl/endpoints/JSEndpoint.h b/libraries/controllers/src/controllers/impl/endpoints/JSEndpoint.h index 6c953399dd..e3a36763bf 100644 --- a/libraries/controllers/src/controllers/impl/endpoints/JSEndpoint.h +++ b/libraries/controllers/src/controllers/impl/endpoints/JSEndpoint.h @@ -19,15 +19,19 @@ namespace controller { class JSEndpoint : public Endpoint { public: - using Endpoint::apply; - JSEndpoint(const QJSValue& callable) - : Endpoint(Input::INVALID_INPUT), _callable(callable) { - } + static std::shared_ptr newEndpoint(const QJSValue& callable) { + return std::shared_ptr(new JSEndpoint(callable)); + }; + using Endpoint::apply; virtual AxisValue peek() const override; virtual void apply(AxisValue newValue, const Pointer& source) override; private: + JSEndpoint(const QJSValue& callable) + : Endpoint(Input::INVALID_INPUT), _callable(callable) { + } + mutable QJSValue _callable; }; diff --git a/libraries/controllers/src/controllers/impl/endpoints/ScriptEndpoint.cpp b/libraries/controllers/src/controllers/impl/endpoints/ScriptEndpoint.cpp index 954fc175c6..1bdad0a884 100644 --- a/libraries/controllers/src/controllers/impl/endpoints/ScriptEndpoint.cpp +++ b/libraries/controllers/src/controllers/impl/endpoints/ScriptEndpoint.cpp @@ -45,7 +45,10 @@ AxisValue ScriptEndpoint::peek() const { void ScriptEndpoint::updateValue() { if (QThread::currentThread() != thread()) { - QMetaObject::invokeMethod(this, "updateValue", Qt::QueuedConnection); + auto pointer = shared_from_this(); + QMetaObject::invokeMethod(this, [pointer]{ + std::dynamic_pointer_cast(pointer)->updateValue(); + }); return; } diff --git a/libraries/controllers/src/controllers/impl/endpoints/ScriptEndpoint.h b/libraries/controllers/src/controllers/impl/endpoints/ScriptEndpoint.h index 240da44716..24a1574652 100644 --- a/libraries/controllers/src/controllers/impl/endpoints/ScriptEndpoint.h +++ b/libraries/controllers/src/controllers/impl/endpoints/ScriptEndpoint.h @@ -22,9 +22,9 @@ class ScriptEndpoint : public Endpoint { Q_OBJECT; public: using Endpoint::apply; - ScriptEndpoint(const ScriptValue& callable) - : Endpoint(Input::INVALID_INPUT), _callable(callable) { - } + static std::shared_ptr newEndpoint(const ScriptValue& callable) { + return std::shared_ptr(new ScriptEndpoint(callable)); + }; virtual AxisValue peek() const override; virtual void apply(AxisValue newValue, const Pointer& source) override; @@ -42,6 +42,9 @@ protected: Q_INVOKABLE void updatePose(); Q_INVOKABLE virtual void internalApply(const Pose& newValue, int sourceID); private: + ScriptEndpoint(const ScriptValue& callable) + : Endpoint(Input::INVALID_INPUT), _callable(callable) { + } ScriptValue _callable; float _lastValueRead { 0.0f }; AxisValue _lastValueWritten { 0.0f, 0, false }; diff --git a/libraries/entities-renderer/src/EntityTreeRenderer.cpp b/libraries/entities-renderer/src/EntityTreeRenderer.cpp index e667f6cf9d..9c565153cd 100644 --- a/libraries/entities-renderer/src/EntityTreeRenderer.cpp +++ b/libraries/entities-renderer/src/EntityTreeRenderer.cpp @@ -226,6 +226,10 @@ void EntityTreeRenderer::resetPersistentEntitiesScriptEngine() { manager->stop(); manager->waitTillDoneRunning(); manager->disconnectNonEssentialSignals(); + // TODO: script manager pointer is still in use somewhere after the cleanup in lambda. + // To prevent memory leaks on multiple reloads we would need to find all the usages and remove them. + // Script engines are correctly deleted later during shutdown currently. + qDebug() << "_nonPersistentEntitiesScriptManager lambda finished, script manager pointer use count: " << manager.use_count(); }); } _persistentEntitiesScriptManager = scriptManagerFactory(ScriptManager::ENTITY_CLIENT_SCRIPT, NO_SCRIPT, @@ -248,6 +252,10 @@ void EntityTreeRenderer::resetNonPersistentEntitiesScriptEngine() { manager->stop(); manager->waitTillDoneRunning(); manager->disconnectNonEssentialSignals(); + // TODO: script manager pointer is still in use somewhere after the cleanup in lambda. + // To prevent memory leaks on multiple reloads we would need to find all the usages and remove them. + // Script engines are correctly deleted later during shutdown currently. + qDebug() << "_nonPersistentEntitiesScriptManager lambda finished, script manager pointer use count: " << manager.use_count(); }); } _nonPersistentEntitiesScriptManager = scriptManagerFactory(ScriptManager::ENTITY_CLIENT_SCRIPT, NO_SCRIPT, diff --git a/libraries/script-engine/src/ScriptEngine.h b/libraries/script-engine/src/ScriptEngine.h index 29594a63b8..0e9cc90c22 100644 --- a/libraries/script-engine/src/ScriptEngine.h +++ b/libraries/script-engine/src/ScriptEngine.h @@ -423,6 +423,11 @@ public: */ virtual void stopProfilingAndSave() = 0; + /** + * @brief Cleanup function that disconnects signals connected to script proxies to avoid use-after-delete crash when shutting down script engine. + */ + virtual void disconnectSignalProxies() = 0; + public: // helper to detect and log warnings when other code invokes QScriptEngine/BaseScriptEngine in thread-unsafe ways bool IS_THREADSAFE_INVOCATION(const QString& method); diff --git a/libraries/script-engine/src/ScriptEngines.cpp b/libraries/script-engine/src/ScriptEngines.cpp index 400a2fc415..b47255ed24 100644 --- a/libraries/script-engine/src/ScriptEngines.cpp +++ b/libraries/script-engine/src/ScriptEngines.cpp @@ -21,6 +21,7 @@ #include #include #include +#include #include "ScriptCache.h" #include "ScriptEngine.h" @@ -404,42 +405,49 @@ QStringList ScriptEngines::getRunningScripts() { } void ScriptEngines::stopAllScripts(bool restart) { - QReadLocker lock(&_scriptManagersHashLock); + QtConcurrent::run([this, restart] { + QHash scriptManagersHashCopy; - if (_isReloading) { - return; - } - - for (QHash::const_iterator it = _scriptManagersHash.constBegin(); - it != _scriptManagersHash.constEnd(); it++) { - ScriptManagerPointer scriptManager = it.value(); - // skip already stopped scripts - if (scriptManager->isFinished() || scriptManager->isStopping()) { - continue; + { + QReadLocker lock(&_scriptManagersHashLock); + scriptManagersHashCopy = _scriptManagersHash; } - bool isOverrideScript = it.key().toString().compare(this->_defaultScriptsOverride.toString()) == 0; - // queue user scripts if restarting - if (restart && (scriptManager->isUserLoaded() || isOverrideScript)) { - _isReloading = true; - ScriptManager::Type type = scriptManager->getType(); - - connect(scriptManager.get(), &ScriptManager::finished, this, [this, type, isOverrideScript](QString scriptName) { - reloadScript(scriptName, !isOverrideScript)->setType(type); - }); + if (_isReloading) { + return; } - // stop all scripts - scriptManager->stop(); - } + for (QHash::const_iterator it = scriptManagersHashCopy.constBegin(); + it != scriptManagersHashCopy.constEnd(); it++) { + ScriptManagerPointer scriptManager = it.value(); + // skip already stopped scripts + if (scriptManager->isFinished() || scriptManager->isStopping()) { + continue; + } - if (restart) { - qCDebug(scriptengine) << "stopAllScripts -- emitting scriptsReloading"; - QTimer::singleShot(RELOAD_ALL_SCRIPTS_TIMEOUT, this, [&] { - _isReloading = false; - }); - emit scriptsReloading(); - } + bool isOverrideScript = it.key().toString().compare(this->_defaultScriptsOverride.toString()) == 0; + // queue user scripts if restarting + if (restart && (scriptManager->isUserLoaded() || isOverrideScript)) { + _isReloading = true; + ScriptManager::Type type = scriptManager->getType(); + + connect(scriptManager.get(), &ScriptManager::finished, this, + [this, type, isOverrideScript](QString scriptName) { + reloadScript(scriptName, !isOverrideScript)->setType(type); + }); + } + + // stop all scripts + scriptManager->stop(); + scriptManager->waitTillDoneRunning(); + } + + if (restart) { + qCDebug(scriptengine) << "stopAllScripts -- emitting scriptsReloading"; + QTimer::singleShot(RELOAD_ALL_SCRIPTS_TIMEOUT, this, [&] { _isReloading = false; }); + emit scriptsReloading(); + } + }); } bool ScriptEngines::stopScript(const QString& rawScriptURL, bool restart) { @@ -612,6 +620,7 @@ void ScriptEngines::onScriptFinished(const QString& rawScriptURL, ScriptManagerP } } + // Could this cause deadlocks when script engine invokes a blocking method on main thread? manager->waitTillDoneRunning(); removeScriptEngine(manager); diff --git a/libraries/script-engine/src/ScriptManager.cpp b/libraries/script-engine/src/ScriptManager.cpp index 9bde4c1781..a3279eee1e 100644 --- a/libraries/script-engine/src/ScriptManager.cpp +++ b/libraries/script-engine/src/ScriptManager.cpp @@ -67,6 +67,7 @@ #include #include #include +#include const QString ScriptManager::_SETTINGS_ENABLE_EXTENDED_EXCEPTIONS { "com.highfidelity.experimental.enableExtendedJSExceptions" @@ -366,7 +367,12 @@ bool ScriptManager::isDebugMode() const { #endif } -ScriptManager::~ScriptManager() {} +ScriptManager::~ScriptManager() { + qDebug() << "ScriptManager::~ScriptManager() : Script manager deleted, type: " << _type << " name: " << _fileNameString; + if (_type == ScriptManager::Type::ENTITY_CLIENT) { + printf("ScriptManager::~ScriptManager"); + } +} void ScriptManager::disconnectNonEssentialSignals() { disconnect(); @@ -464,11 +470,14 @@ void ScriptManager::waitTillDoneRunning(bool shutdown) { } #else auto startedWaiting = usecTimestampNow(); - while (workerThread->isRunning()) { + while (!_isDoneRunning) { // If the final evaluation takes too long, then tell the script engine to stop running auto elapsedUsecs = usecTimestampNow() - startedWaiting; + // TODO: This part was very unsafe and was causing crashes all the time. + // I disabled it for now until we find a safer solution. + // With it disabled now we get clean shutdowns and restarts. // V8TODO: temporarily increased script timeout. Maybe use different timeouts for release and unoptimized debug? - static const auto MAX_SCRIPT_EVALUATION_TIME = USECS_PER_SECOND; + /*static const auto MAX_SCRIPT_EVALUATION_TIME = 10 * USECS_PER_SECOND; if (elapsedUsecs > MAX_SCRIPT_EVALUATION_TIME) { workerThread->quit(); @@ -485,12 +494,12 @@ void ScriptManager::waitTillDoneRunning(bool shutdown) { // Wait for the scripting thread to stop running, as // flooding it with aborts/exceptions will persist it longer - static const auto MAX_SCRIPT_QUITTING_TIME = 0.5 * MSECS_PER_SECOND; + static const auto MAX_SCRIPT_QUITTING_TIME = 50 * MSECS_PER_SECOND; if (!workerThread->wait(MAX_SCRIPT_QUITTING_TIME)) { Q_ASSERT(false); workerThread->terminate(); } - } + }*/ if (shutdown) { // NOTE: This will be called on the main application thread (among other threads) from stopAllScripts. @@ -502,7 +511,7 @@ void ScriptManager::waitTillDoneRunning(bool shutdown) { } // Avoid a pure busy wait - QThread::yieldCurrentThread(); + QThread::msleep(1); } #endif @@ -985,6 +994,7 @@ void ScriptManager::run() { PROFILE_RANGE(script, "ScriptMainLoop"); +//#define SCRIPT_DELAY_DEBUG #ifdef SCRIPT_DELAY_DEBUG { auto actuallySleptUntil = clock::now(); @@ -1064,6 +1074,13 @@ void ScriptManager::run() { _isRunning = false; emit runningStateChanged(); emit doneRunning(); + _engine->disconnectSignalProxies(); + // Process all remaining events + { + QEventLoop loop; + loop.processEvents(); + } + _isDoneRunning = true; } // NOTE: This is private because it must be called on the same thread that created the timers, which is why @@ -1121,6 +1138,7 @@ void ScriptManager::timerFired() { return; // bail early } +//#define SCRIPT_TIMER_PERFORMANCE_STATISTICS #ifdef SCRIPT_TIMER_PERFORMANCE_STATISTICS _timerCallCounter++; if (_timerCallCounter % 100 == 0) { diff --git a/libraries/script-engine/src/ScriptManager.h b/libraries/script-engine/src/ScriptManager.h index 73fbb77e7d..af95e1dd79 100644 --- a/libraries/script-engine/src/ScriptManager.h +++ b/libraries/script-engine/src/ScriptManager.h @@ -1197,6 +1197,14 @@ public: */ void setAbortOnUncaughtException(bool value) { _abortOnUncaughtException = value; } + /** + * @brief Returns true after script finished running and doneRunning signal was called + * + * @return true If the script and doneRunning signal was called + * @return false If the script has not finished running yet + */ + bool isDoneRunning() { return _isDoneRunning; }; + public slots: /** @@ -1527,6 +1535,7 @@ protected: std::atomic _isFinished { false }; std::atomic _isRunning { false }; std::atomic _isStopping { false }; + std::atomic _isDoneRunning { false }; bool _areMetaTypesInitialized { false }; bool _isInitialized { false }; QHash _timerFunctionMap; diff --git a/libraries/script-engine/src/v8/ScriptEngineV8.cpp b/libraries/script-engine/src/v8/ScriptEngineV8.cpp index 5d8066646a..bf8ebc7cf6 100644 --- a/libraries/script-engine/src/v8/ScriptEngineV8.cpp +++ b/libraries/script-engine/src/v8/ScriptEngineV8.cpp @@ -248,12 +248,27 @@ ScriptEngineV8::ScriptEngineV8(ScriptManager *manager) : ScriptEngine(manager), ScriptEngineV8::~ScriptEngineV8() { deleteUnusedValueWrappers(); +#ifdef OVERTE_SCRIPT_USE_AFTER_DELETE_GUARD + _wasDestroyed = true; +#endif + qDebug() << "ScriptEngineV8::~ScriptEngineV8: script engine destroyed"; } void ScriptEngineV8::perManagerLoopIterationCleanup() { deleteUnusedValueWrappers(); } +void ScriptEngineV8::disconnectSignalProxies() { + _signalProxySetLock.lockForRead(); + while (!_signalProxySet.empty()) { + _signalProxySetLock.unlock(); + delete *_signalProxySet.begin(); + _signalProxySetLock.lockForRead(); + } + _signalProxySetLock.unlock(); +} + + void ScriptEngineV8::deleteUnusedValueWrappers() { while (!_scriptValueWrappersToDelete.empty()) { auto wrapper = _scriptValueWrappersToDelete.dequeue(); @@ -485,6 +500,9 @@ void ScriptEngineV8::registerGetterSetter(const QString& name, ScriptEngine::Fun } v8::Local ScriptEngineV8::getContext() { +#ifdef OVERTE_SCRIPT_USE_AFTER_DELETE_GUARD + Q_ASSERT(!_wasDestroyed); +#endif v8::EscapableHandleScope handleScope(_v8Isolate); Q_ASSERT(!_contexts.isEmpty()); return handleScope.Escape(_contexts.last().get()->toV8Value()); diff --git a/libraries/script-engine/src/v8/ScriptEngineV8.h b/libraries/script-engine/src/v8/ScriptEngineV8.h index 9e87edfb62..5badba271e 100644 --- a/libraries/script-engine/src/v8/ScriptEngineV8.h +++ b/libraries/script-engine/src/v8/ScriptEngineV8.h @@ -48,6 +48,7 @@ class ScriptManager; class ScriptObjectV8Proxy; class ScriptMethodV8Proxy; class ScriptValueV8Wrapper; +class ScriptSignalV8Proxy; template class V8ScriptValueTemplate; typedef V8ScriptValueTemplate V8ScriptValue; @@ -57,6 +58,8 @@ using ScriptContextV8Pointer = std::shared_ptr; const double GARBAGE_COLLECTION_TIME_LIMIT_S = 1.0; +#define OVERTE_SCRIPT_USE_AFTER_DELETE_GUARD + Q_DECLARE_METATYPE(ScriptEngine::FunctionSignature) /// [V8] Implements ScriptEngine for V8 and translates calls for QScriptEngine @@ -144,6 +147,7 @@ public: // ScriptEngine implementation void scheduleValueWrapperForDeletion(ScriptValueV8Wrapper* wrapper) {_scriptValueWrappersToDelete.enqueue(wrapper);} void deleteUnusedValueWrappers(); virtual void perManagerLoopIterationCleanup() override; + virtual void disconnectSignalProxies() override; // helper to detect and log warnings when other code invokes QScriptEngine/BaseScriptEngine in thread-unsafe ways inline bool IS_THREADSAFE_INVOCATION(const QString& method) { return ScriptEngine::IS_THREADSAFE_INVOCATION(method); } @@ -276,12 +280,21 @@ private: std::atomic scriptValueProxyCount{0}; #endif +#ifdef OVERTE_SCRIPT_USE_AFTER_DELETE_GUARD + bool _wasDestroyed{false}; +#endif // Pointers to profiling classes. These are valid only when profiler is running, otherwise null // Smart pointer cannot be used here since profiler has private destructor v8::CpuProfiler *_profiler{nullptr}; v8::ProfilerId _profilerId{0}; + // Set of script signal proxy pointers. Used for disconnecting signals on cleanup. + // V8TODO: later it would be also worth to make sure that script proxies themselves get deleted together with script engine + QReadWriteLock _signalProxySetLock; + QSet _signalProxySet; + friend ScriptValueV8Wrapper; + friend ScriptSignalV8Proxy; }; // This class is used to automatically add context to script engine's context list that is used by C++ calls diff --git a/libraries/script-engine/src/v8/ScriptObjectV8Proxy.cpp b/libraries/script-engine/src/v8/ScriptObjectV8Proxy.cpp index 616e252d7c..177eeb259f 100644 --- a/libraries/script-engine/src/v8/ScriptObjectV8Proxy.cpp +++ b/libraries/script-engine/src/v8/ScriptObjectV8Proxy.cpp @@ -1157,6 +1157,9 @@ ScriptSignalV8Proxy::ScriptSignalV8Proxy(ScriptEngineV8* engine, QObject* object _objectLifetime.Reset(isolate, lifetime.get()); _objectLifetime.SetWeak(this, weakHandleCallback, v8::WeakCallbackType::kParameter); _v8Context.Reset(isolate, _engine->getContext()); + _engine->_signalProxySetLock.lockForWrite(); + _engine->_signalProxySet.insert(this); + _engine->_signalProxySetLock.unlock(); } ScriptSignalV8Proxy::~ScriptSignalV8Proxy() { @@ -1166,6 +1169,12 @@ ScriptSignalV8Proxy::~ScriptSignalV8Proxy() { v8::HandleScope handleScope(isolate); _objectLifetime.Reset(); _v8Context.Reset(); +#ifdef OVERTE_SCRIPT_USE_AFTER_DELETE_GUARD + Q_ASSERT(!_engine->_wasDestroyed); +#endif + _engine->_signalProxySetLock.lockForWrite(); + _engine->_signalProxySet.remove(this); + _engine->_signalProxySetLock.unlock(); } void ScriptSignalV8Proxy::weakHandleCallback(const v8::WeakCallbackInfo& info) { diff --git a/libraries/script-engine/src/v8/ScriptObjectV8Proxy.h b/libraries/script-engine/src/v8/ScriptObjectV8Proxy.h index eb1e51ccdc..8c7011daf1 100644 --- a/libraries/script-engine/src/v8/ScriptObjectV8Proxy.h +++ b/libraries/script-engine/src/v8/ScriptObjectV8Proxy.h @@ -270,6 +270,9 @@ public: // API //Moved to public temporarily for debugging: QString fullName() const; + // Disconnects all signals from the proxy + void disconnectAll() { QObject::disconnect(this, nullptr, nullptr, nullptr); }; + private: // storage ScriptEngineV8* _engine; diff --git a/libraries/script-engine/src/v8/ScriptValueV8Wrapper.cpp b/libraries/script-engine/src/v8/ScriptValueV8Wrapper.cpp index fb2cf85a9f..fc329c9e19 100644 --- a/libraries/script-engine/src/v8/ScriptValueV8Wrapper.cpp +++ b/libraries/script-engine/src/v8/ScriptValueV8Wrapper.cpp @@ -248,6 +248,9 @@ ScriptEnginePointer ScriptValueV8Wrapper::engine() const { if (!_engine) { return ScriptEnginePointer(); } +#ifdef OVERTE_SCRIPT_USE_AFTER_DELETE_GUARD + Q_ASSERT(!_engine->_wasDestroyed); +#endif return _engine->shared_from_this(); }