Fixed deadlocks in Recording API

This commit is contained in:
ksuprynowicz 2024-05-16 01:09:57 +02:00
parent f9ffccfe00
commit fe6d76b935
3 changed files with 25 additions and 16 deletions

View file

@ -40,7 +40,7 @@
*/ */
class AgentScriptingInterface : public QObject { class AgentScriptingInterface : public QObject {
Q_OBJECT Q_OBJECT
Q_PROPERTY(bool isAvatar READ isAvatar WRITE setIsAvatar) Q_PROPERTY(bool isAvatar READ getIsAvatar WRITE setIsAvatar)
Q_PROPERTY(bool isPlayingAvatarSound READ isPlayingAvatarSound) Q_PROPERTY(bool isPlayingAvatarSound READ isPlayingAvatarSound)
Q_PROPERTY(bool isListeningToAudioStream READ isListeningToAudioStream WRITE setIsListeningToAudioStream) Q_PROPERTY(bool isListeningToAudioStream READ isListeningToAudioStream WRITE setIsListeningToAudioStream)
Q_PROPERTY(bool isNoiseGateEnabled READ isNoiseGateEnabled WRITE setIsNoiseGateEnabled) Q_PROPERTY(bool isNoiseGateEnabled READ isNoiseGateEnabled WRITE setIsNoiseGateEnabled)
@ -77,15 +77,15 @@ public slots:
/*@jsdoc /*@jsdoc
* Checks whether the script is emulating an avatar. * Checks whether the script is emulating an avatar.
* @function Agent.isAvatar * @function Agent.getIsAvatar
* @returns {boolean} <code>true</code> if the script is emulating an avatar, otherwise <code>false</code>. * @returns {boolean} <code>true</code> if the script is emulating an avatar, otherwise <code>false</code>.
* @example <caption>Check whether the agent is emulating an avatar.</caption> * @example <caption>Check whether the agent is emulating an avatar.</caption>
* (function () { * (function () {
* print("Agent is avatar: " + Agent.isAvatar()); * print("Agent is avatar: " + Agent.getIsAvatar());
* print("Agent is avatar: " + Agent.isAvatar); // Same result. * print("Agent is avatar: " + Agent.isAvatar); // Same result.
* }()); * }());
*/ */
bool isAvatar() const { return _agent->isAvatar(); } bool getIsAvatar() const { return _agent->isAvatar(); }
/*@jsdoc /*@jsdoc
* Plays a sound from the position and with the orientation of the emulated avatar's head. No sound is played unless * Plays a sound from the position and with the orientation of the emulated avatar's head. No sound is played unless

View file

@ -38,27 +38,33 @@ using namespace recording;
static const QString HFR_EXTENSION = "hfr"; static const QString HFR_EXTENSION = "hfr";
RecordingScriptingInterface::RecordingScriptingInterface() { RecordingScriptingInterface::RecordingScriptingInterface() {
Locker(_mutex);
_player = DependencyManager::get<Deck>(); _player = DependencyManager::get<Deck>();
_recorder = DependencyManager::get<Recorder>(); _recorder = DependencyManager::get<Recorder>();
} }
bool RecordingScriptingInterface::isPlaying() const { bool RecordingScriptingInterface::isPlaying() const {
Locker(_mutex);
return _player->isPlaying(); return _player->isPlaying();
} }
bool RecordingScriptingInterface::isPaused() const { bool RecordingScriptingInterface::isPaused() const {
Locker(_mutex);
return _player->isPaused(); return _player->isPaused();
} }
float RecordingScriptingInterface::playerElapsed() const { float RecordingScriptingInterface::playerElapsed() const {
Locker(_mutex);
return _player->position(); return _player->position();
} }
float RecordingScriptingInterface::playerLength() const { float RecordingScriptingInterface::playerLength() const {
Locker(_mutex);
return _player->length(); return _player->length();
} }
void RecordingScriptingInterface::playClip(NetworkClipLoaderPointer clipLoader, const QString& url, const ScriptValue& callback) { void RecordingScriptingInterface::playClip(NetworkClipLoaderPointer clipLoader, const QString& url, const ScriptValue& callback) {
Locker(_mutex);
_player->queueClip(clipLoader->getClip()); _player->queueClip(clipLoader->getClip());
if (callback.isFunction()) { if (callback.isFunction()) {
@ -69,10 +75,7 @@ void RecordingScriptingInterface::playClip(NetworkClipLoaderPointer clipLoader,
} }
void RecordingScriptingInterface::loadRecording(const QString& url, const ScriptValue& callback) { void RecordingScriptingInterface::loadRecording(const QString& url, const ScriptValue& callback) {
if (QThread::currentThread() != thread()) { Locker(_mutex);
BLOCKING_INVOKE_METHOD(this, "loadRecording", Q_ARG(const QString&, url), Q_ARG(const ScriptValue&, callback));
return;
}
auto clipLoader = DependencyManager::get<recording::ClipCache>()->getClipLoader(url); auto clipLoader = DependencyManager::get<recording::ClipCache>()->getClipLoader(url);
@ -131,6 +134,7 @@ void RecordingScriptingInterface::startPlaying() {
return; return;
} }
Locker(_mutex);
_player->play(); _player->play();
} }
@ -140,6 +144,7 @@ void RecordingScriptingInterface::setPlayerVolume(float volume) {
return; return;
} }
Locker(_mutex);
_player->setVolume(std::min(std::max(volume, 0.0f), 1.0f)); _player->setVolume(std::min(std::max(volume, 0.0f), 1.0f));
} }
@ -152,6 +157,7 @@ void RecordingScriptingInterface::setPlayerTime(float time) {
BLOCKING_INVOKE_METHOD(this, "setPlayerTime", Q_ARG(float, time)); BLOCKING_INVOKE_METHOD(this, "setPlayerTime", Q_ARG(float, time));
return; return;
} }
Locker(_mutex);
_player->seek(time); _player->seek(time);
} }
@ -165,6 +171,7 @@ void RecordingScriptingInterface::setPlayerLoop(bool loop) {
return; return;
} }
Locker(_mutex);
_player->loop(loop); _player->loop(loop);
} }
@ -189,6 +196,7 @@ void RecordingScriptingInterface::pausePlayer() {
BLOCKING_INVOKE_METHOD(this, "pausePlayer"); BLOCKING_INVOKE_METHOD(this, "pausePlayer");
return; return;
} }
Locker(_mutex);
_player->pause(); _player->pause();
} }
@ -197,6 +205,7 @@ void RecordingScriptingInterface::stopPlaying() {
BLOCKING_INVOKE_METHOD(this, "stopPlaying"); BLOCKING_INVOKE_METHOD(this, "stopPlaying");
return; return;
} }
Locker(_mutex);
_player->stop(); _player->stop();
} }
@ -219,6 +228,7 @@ void RecordingScriptingInterface::startRecording() {
return; return;
} }
Locker(_mutex);
_recorder->start(); _recorder->start();
} }
@ -233,6 +243,7 @@ void RecordingScriptingInterface::stopRecording() {
return; return;
} }
Locker(_mutex);
_recorder->stop(); _recorder->stop();
_lastClip = _recorder->getClip(); _lastClip = _recorder->getClip();
_lastClip->seek(0); _lastClip->seek(0);
@ -253,6 +264,7 @@ void RecordingScriptingInterface::saveRecording(const QString& filename) {
return; return;
} }
Locker(_mutex);
if (!_lastClip) { if (!_lastClip) {
qWarning() << "There is no recording to save"; qWarning() << "There is no recording to save";
return; return;
@ -267,14 +279,7 @@ bool RecordingScriptingInterface::saveRecordingToAsset(const ScriptValue& getCli
return false; return false;
} }
if (QThread::currentThread() != thread()) { Locker(_mutex);
bool result;
BLOCKING_INVOKE_METHOD(this, "saveRecordingToAsset",
Q_RETURN_ARG(bool, result),
Q_ARG(const ScriptValue&, getClipAtpUrl));
return result;
}
if (!_lastClip) { if (!_lastClip) {
qWarning() << "There is no recording to save"; qWarning() << "There is no recording to save";
return false; return false;
@ -316,6 +321,8 @@ void RecordingScriptingInterface::loadLastRecording() {
return; return;
} }
Locker(_mutex);
if (!_lastClip) { if (!_lastClip) {
qCDebug(scriptengine) << "There is no recording to load"; qCDebug(scriptengine) << "There is no recording to load";
return; return;

View file

@ -357,6 +357,8 @@ protected:
using Locker = std::unique_lock<Mutex>; using Locker = std::unique_lock<Mutex>;
using Flag = std::atomic<bool>; using Flag = std::atomic<bool>;
Mutex _mutex;
QSharedPointer<recording::Deck> _player; QSharedPointer<recording::Deck> _player;
QSharedPointer<recording::Recorder> _recorder; QSharedPointer<recording::Recorder> _recorder;