From 6e19ac230f2ea3dd6897283d3b0e5c94672a0670 Mon Sep 17 00:00:00 2001 From: iphydf Date: Fri, 27 Dec 2024 17:27:14 +0000 Subject: [PATCH] cleanup: Add openal logging and audio log category. Also link static builds as PIE for security reasons (ASLR). --- .ci-scripts/build-qtox-linux-static.sh | 10 +-- CMakeLists.txt | 2 +- audio/src/backend/openal.cpp | 103 ++++++++++++++++++------- cspell.config.yaml | 2 + 4 files changed, 85 insertions(+), 32 deletions(-) diff --git a/.ci-scripts/build-qtox-linux-static.sh b/.ci-scripts/build-qtox-linux-static.sh index 953a6ccb45..8b7d92e3d9 100755 --- a/.ci-scripts/build-qtox-linux-static.sh +++ b/.ci-scripts/build-qtox-linux-static.sh @@ -12,11 +12,11 @@ cmake \ -DBUILD_TESTING=OFF \ -DFULLY_STATIC=ON \ -GNinja \ - -B_build \ + -B_build-static \ -H. -cmake --build _build +cmake --build _build-static -ls -lh _build/qtox -file _build/qtox -QT_QPA_PLATFORM=offscreen _build/qtox --help +ls -lh _build-static/qtox +file _build-static/qtox +QT_QPA_PLATFORM=offscreen _build-static/qtox --help diff --git a/CMakeLists.txt b/CMakeLists.txt index 6b5083ab18..c83d789bac 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -690,7 +690,7 @@ if(QT_FEATURE_static) endif() endif() if(FULLY_STATIC) - target_link_options(${PROJECT_NAME} PRIVATE -static) + target_link_options(${PROJECT_NAME} PRIVATE -static-pie) endif() if(BUILD_TESTING) diff --git a/audio/src/backend/openal.cpp b/audio/src/backend/openal.cpp index 9443976204..b2f38e3bdc 100644 --- a/audio/src/backend/openal.cpp +++ b/audio/src/backend/openal.cpp @@ -9,6 +9,7 @@ #include #include +#include #include #include #include @@ -17,6 +18,20 @@ #include +#if defined(QT_STATIC) +extern "C" +{ + typedef void alsoftLogCallback(void* userptr, char level, const char* message, int length) noexcept; + /** + * @brief Set a callback for OpenAL logging. + * + * @note This function is only available in statically linked builds where we know for sure we + * link against openal-soft. + */ + void alsoft_set_log_callback(alsoftLogCallback* callback, void* userptr) noexcept; +} +#endif + namespace { void applyGain(int16_t* buffer, uint32_t bufferSize, qreal gainFactor) { @@ -41,12 +56,45 @@ void applyGain(int16_t* buffer, uint32_t bufferSize, qreal gainFactor) constexpr unsigned int BUFFER_COUNT = 16; constexpr uint32_t AUDIO_CHANNELS = 2; + +namespace logcat { +#if defined(QT_STATIC) +Q_LOGGING_CATEGORY(openal, "openal") +#endif +Q_LOGGING_CATEGORY(audio, "qtox.audio") +} // namespace logcat } // namespace OpenAL::OpenAL(IAudioSettings& _settings) : settings{_settings} , audioThread{new QThread} { +#if defined(QT_STATIC) + alsoft_set_log_callback( + [](void* userptr, char level, const char* message, int length) noexcept { + std::ignore = userptr; + // OpenAL passes .data() and .size() to the callback, + // so we can't rely on null-termination. + const std::string_view msg{message, static_cast(length)}; + switch (level) { + case 'E': + qCCritical(logcat::openal).noquote() << msg; + break; + case 'W': + qCWarning(logcat::openal).noquote() << msg; + break; + case 'I': + qCDebug(logcat::openal).noquote() << msg; + break; + default: + // Shouldn't happen, but if it does, we'll log it as a warning. + qCWarning(logcat::openal).noquote() << level << msg; + break; + } + }, + nullptr); +#endif + // initialize OpenAL error stack alGetError(); alcGetError(nullptr); @@ -92,15 +140,17 @@ OpenAL::~OpenAL() void OpenAL::checkAlError() noexcept { const ALenum al_err = alGetError(); - if (al_err != AL_NO_ERROR) - qWarning("OpenAL error: %d", al_err); + if (al_err != AL_NO_ERROR) { + qWarning("OpenAL error: %s", alGetString(al_err)); + } } void OpenAL::checkAlcError(ALCdevice* device) noexcept { const ALCenum alc_err = alcGetError(device); - if (alc_err) - qWarning("OpenAL error: %d", alc_err); + if (alc_err) { + qWarning("OpenAL error: %s", alcGetString(device, alc_err)); + } } /** @@ -112,9 +162,7 @@ void OpenAL::setOutputVolume(qreal volume) { QMutexLocker locker(&audioLock); - volume = std::max(0.0, std::min(volume, 1.0)); - - alListenerf(AL_GAIN, static_cast(volume)); + alListenerf(AL_GAIN, static_cast(std::max(0.0, std::min(volume, 1.0)))); checkAlError(); } @@ -223,7 +271,7 @@ std::unique_ptr OpenAL::makeSink() } sinks.insert(sink); - qDebug() << "Audio source" << sid << "created. Sources active:" << sinks.size(); + qCDebug(logcat::audio) << "Audio source" << sid << "created. Sources active:" << sinks.size(); return std::unique_ptr{sink}; } @@ -241,7 +289,7 @@ void OpenAL::destroySink(AlSink& sink) const auto soundSinksErased = soundSinks.erase(&sink); if (sinksErased == 0 && soundSinksErased == 0) { - qWarning() << "Destroying non-existent sink"; + qCWarning(logcat::audio) << "Destroying non-existent sink"; return; } @@ -251,9 +299,9 @@ void OpenAL::destroySink(AlSink& sink) // stop playing, marks all buffers as processed alSourceStop(sid); cleanupBuffers(sid); - qDebug() << "Audio source" << sid << "deleted. Sources active:" << sinks.size(); + qCDebug(logcat::audio) << "Audio source" << sid << "deleted. Sources active:" << sinks.size(); } else { - qWarning() << "Trying to delete invalid audio source" << sid; + qCWarning(logcat::audio) << "Trying to delete invalid audio source" << sid; } if (sinks.empty() && soundSinks.empty()) { @@ -278,7 +326,8 @@ std::unique_ptr OpenAL::makeSource() auto* const source = new AlSource(*this); sources.insert(source); - qDebug() << "Subscribed to audio input device [" << sources.size() << "subscriptions ]"; + qCDebug(logcat::audio) << "Subscribed to audio input device [" << sources.size() + << "subscriptions ]"; return std::unique_ptr{source}; } @@ -294,13 +343,14 @@ void OpenAL::destroySource(AlSource& source) const auto s = sources.find(&source); if (s == sources.end()) { - qWarning() << "Destroyed non-existent source"; + qCWarning(logcat::audio) << "Destroyed non-existent source"; return; } sources.erase(s); - qDebug() << "Unsubscribed from audio input device [" << sources.size() << "subscriptions left ]"; + qCDebug(logcat::audio) << "Unsubscribed from audio input device [" << sources.size() + << "subscriptions left ]"; if (sources.empty()) { cleanupInput(); @@ -338,7 +388,7 @@ bool OpenAL::initInput(const QString& deviceName, uint32_t channels) return false; } - qDebug() << "Opening audio input" << deviceName; + qCDebug(logcat::audio) << "Opening audio input" << deviceName; assert(!alInDev); // TODO: Try to actually detect if our audio source is stereo @@ -356,7 +406,7 @@ bool OpenAL::initInput(const QString& deviceName, uint32_t channels) // Restart the capture if necessary if (!alInDev) { - qWarning() << "Failed to initialize audio input device:" << deviceName; + qCWarning(logcat::audio) << "Failed to initialize audio input device:" << deviceName; return false; } @@ -364,7 +414,7 @@ bool OpenAL::initInput(const QString& deviceName, uint32_t channels) setInputGain(settings.getAudioInGainDecibel()); setInputThreshold(settings.getAudioThreshold()); - qDebug() << "Opened audio input" << deviceName; + qCDebug(logcat::audio) << "Opened audio input" << deviceName; alcCaptureStart(alInDev); return true; @@ -382,7 +432,7 @@ bool OpenAL::initOutput(const QString& deviceName) if (!settings.getAudioOutDevEnabled()) return false; - qDebug() << "Opening audio output" << deviceName; + qCDebug(logcat::audio) << "Opening audio output" << deviceName; assert(!alOutDev); const QByteArray qDevName = deviceName.toUtf8(); @@ -390,16 +440,16 @@ bool OpenAL::initOutput(const QString& deviceName) alOutDev = alcOpenDevice(tmpDevName); if (!alOutDev) { - qWarning() << "Cannot open audio output device" << deviceName; + qCWarning(logcat::audio) << "Cannot open audio output device" << deviceName; return false; } - qDebug() << "Opened audio output" << deviceName; + qCDebug(logcat::audio) << "Opened audio output" << deviceName; alOutContext = alcCreateContext(alOutDev, nullptr); checkAlcError(alOutDev); if (!alcMakeContextCurrent(alOutContext)) { - qWarning() << "Cannot create audio output context"; + qCWarning(logcat::audio) << "Cannot create audio output context"; return false; } @@ -419,14 +469,14 @@ void OpenAL::playMono16Sound(AlSink& sink, const IAudioSink::Sound& sound) const uint sourceId = sink.getSourceId(); QFile sndFile(IAudioSink::getSound(sound)); if (!sndFile.exists()) { - qDebug() << "Trying to open non-existent sound file"; + qCDebug(logcat::audio) << "Trying to open non-existent sound file"; return; } sndFile.open(QIODevice::ReadOnly); const QByteArray data{sndFile.readAll()}; if (data.isEmpty()) { - qDebug() << "Sound file contained no data"; + qCDebug(logcat::audio) << "Sound file contained no data"; return; } @@ -524,12 +574,12 @@ void OpenAL::cleanupInput() if (!alInDev) return; - qDebug() << "Closing audio input"; + qCDebug(logcat::audio) << "Closing audio input"; alcCaptureStop(alInDev); if (alcCaptureCloseDevice(alInDev) == ALC_TRUE) { alInDev = nullptr; } else { - qWarning() << "Failed to close input"; + qCWarning(logcat::audio) << "Failed to close input"; } delete[] inputBuffer; @@ -550,7 +600,7 @@ void OpenAL::cleanupOutput() alcDestroyContext(alOutContext); alOutContext = nullptr; - qDebug() << "Closing audio output"; + qCDebug(logcat::audio) << "Closing audio output"; if (alcCloseDevice(alOutDev)) { alOutDev = nullptr; } else { @@ -686,6 +736,7 @@ QStringList OpenAL::inDeviceNames() const ALchar* pDeviceList = alcGetString(nullptr, ALC_CAPTURE_DEVICE_SPECIFIER); if (pDeviceList) { + qCDebug(logcat::audio) << "Available input devices:" << pDeviceList; while (*pDeviceList) { auto len = strlen(pDeviceList); list << QString::fromUtf8(pDeviceList, len); diff --git a/cspell.config.yaml b/cspell.config.yaml index f0981599f5..5c70b1fd8d 100644 --- a/cspell.config.yaml +++ b/cspell.config.yaml @@ -45,6 +45,7 @@ words: - "afrm" - "agilob" - "alices" + - "alsoft" - "amproperty" - "ampropsetid" - "anthonybilinski" @@ -140,6 +141,7 @@ words: - "nurupo" - "olestr" - "oming" + - "openal" - "opengl" - "ostype" - "pacman"