From a54dfe7deb2b23303a4f1f3a0559a7a8c36bca56 Mon Sep 17 00:00:00 2001 From: ObsidianSnoo <66971815+ObsidianSnoo@users.noreply.github.com> Date: Tue, 23 Feb 2021 12:35:19 -0800 Subject: [PATCH] 26.0.2.1-rpan --- CI/full-build-macos.sh | 85 -- CI/install-dependencies-linux.sh | 50 - CI/install-dependencies-osx.sh | 81 -- UI/auth-mixer.cpp | 313 ------- UI/auth-mixer.hpp | 30 - UI/window-reddit-login-dialog2.cpp | 21 +- cmake/Modules/FindLibUDev.cmake | 33 - plugins/enc-amf | 2 +- plugins/obs-ffmpeg/ffmpeg-encoded-output.c | 873 ------------------ plugins/obs-ffmpeg/ffmpeg-encoded-output.h | 108 --- plugins/obs-vst | 2 +- .../graphics-hook/d3d1x_shaders.hpp | 31 - plugins/win-dshow/libdshowcapture | 2 +- 13 files changed, 14 insertions(+), 1617 deletions(-) delete mode 100755 CI/install-dependencies-linux.sh delete mode 100755 CI/install-dependencies-osx.sh delete mode 100644 UI/auth-mixer.cpp delete mode 100644 UI/auth-mixer.hpp delete mode 100644 cmake/Modules/FindLibUDev.cmake delete mode 100644 plugins/obs-ffmpeg/ffmpeg-encoded-output.c delete mode 100644 plugins/obs-ffmpeg/ffmpeg-encoded-output.h delete mode 100644 plugins/win-capture/graphics-hook/d3d1x_shaders.hpp diff --git a/CI/full-build-macos.sh b/CI/full-build-macos.sh index 514c7fe..45743c8 100755 --- a/CI/full-build-macos.sh +++ b/CI/full-build-macos.sh @@ -143,90 +143,6 @@ check_ccache() { info "${CCACHE_STATUS}" } -install_obs-deps() { - hr "Setting up pre-built macOS dependencies v${1}" - ensure_dir ${DEPS_BUILD_DIR} - step "Download..." - ${CURLCMD} --progress-bar -L -C - -O https://github.com/obsproject/obs-deps/releases/download/${1}/macos-deps-${1}.tar.gz - step "Unpack..." - tar -xf ./macos-deps-${1}.tar.gz -C /tmp -} - -install_qt-deps() { - hr "Setting up pre-built dependency QT v${1}" - ensure_dir ${DEPS_BUILD_DIR} - step "Download..." - ${CURLCMD} --progress-bar -L -C - -O https://github.com/obsproject/obs-deps/releases/download/${2}/macos-qt-${1}-${2}.tar.gz - step "Unpack..." - tar -xf ./macos-qt-${1}-${2}.tar.gz -C /tmp - xattr -r -d com.apple.quarantine /tmp/obsdeps -} - -install_vlc() { - hr "Setting up dependency VLC v${1}" - ensure_dir ${DEPS_BUILD_DIR} - step "Download..." - ${CURLCMD} --progress-bar -L -C - -O https://downloads.videolan.org/vlc/${1}/vlc-${1}.tar.xz - step "Unpack ..." - tar -xf vlc-${1}.tar.xz -} - -install_sparkle() { - hr "Setting up dependency Sparkle v${1} (might prompt for password)" - ensure_dir ${DEPS_BUILD_DIR}/sparkle - step "Download..." - ${CURLCMD} --progress-bar -L -C - -o sparkle.tar.bz2 https://github.com/sparkle-project/Sparkle/releases/download/${1}/Sparkle-${1}.tar.bz2 - step "Unpack..." - tar -xf ./sparkle.tar.bz2 - step "Copy to destination..." - if [ -d /Library/Frameworks/Sparkle.framework/ ]; then - info "Warning - Sparkle framework already found in /Library/Frameworks" - else - sudo cp -R ./Sparkle.framework/ /Library/Frameworks/Sparkle.framework/ - fi -} - -install_cef() { - hr "Building dependency CEF v${1}" - ensure_dir ${DEPS_BUILD_DIR} - step "Download..." - ${CURLCMD} --progress-bar -L -C - -O https://obs-nightly.s3-us-west-2.amazonaws.com/cef_binary_${1}_macosx64.tar.bz2 - step "Unpack..." - tar -xf ./cef_binary_${1}_macosx64.tar.bz2 - cd ./cef_binary_${1}_macosx64 - step "Fix tests..." - # remove a broken test - sed -i '.orig' '/add_subdirectory(tests\/ceftests)/d' ./CMakeLists.txt - # target 10.11 - sed -i '.orig' s/\"10.9\"/\"10.11\"/ ./cmake/cef_variables.cmake - ensure_dir ./build - step "Run CMAKE..." - cmake \ - -DCMAKE_CXX_FLAGS="-std=c++11 -stdlib=libc++"\ - -DCMAKE_EXE_LINKER_FLAGS="-std=c++11 -stdlib=libc++"\ - -DCMAKE_OSX_DEPLOYMENT_TARGET=10.11 \ - .. - step "Build..." - make -j4 - if [ ! -d libcef_dll ]; then mkdir libcef_dll; fi -} - -## CHECK AND INSTALL PACKAGING DEPENDENCIES ## -install_dmgbuild() { - if ! exists dmgbuild; then - if exists "pip3"; then - PIPCMD="pip3" - elif exists "pip"; then - PIPCMD="pip" - else - error "Pip not found - please install pip via 'python -m ensurepip'" - exit 1 - fi - - ${PIPCMD} install dmgbuild - fi -} - ## OBS BUILD FROM SOURCE ## configure_obs_build() { ensure_dir "${CHECKOUT_DIR}/${BUILD_DIR}" @@ -564,7 +480,6 @@ package_macos() { hr "Creating macOS .dmg image" trap "caught_error 'package app'" ERR - install_dmgbuild prepare_macos_image } diff --git a/CI/install-dependencies-linux.sh b/CI/install-dependencies-linux.sh deleted file mode 100755 index 0d92fbf..0000000 --- a/CI/install-dependencies-linux.sh +++ /dev/null @@ -1,50 +0,0 @@ -#!/bin/sh -set -ex - -curl -L https://packagecloud.io/github/git-lfs/gpgkey | sudo apt-key add - - -sudo apt-get -qq update -sudo apt-get install -y \ - build-essential \ - checkinstall \ - cmake \ - libasound2-dev \ - libavcodec-dev \ - libavdevice-dev \ - libavfilter-dev \ - libavformat-dev \ - libavutil-dev \ - libcurl4-openssl-dev \ - libfdk-aac-dev \ - libfontconfig-dev \ - libfreetype6-dev \ - libgl1-mesa-dev \ - libjack-jackd2-dev \ - libjansson-dev \ - libluajit-5.1-dev \ - libpulse-dev \ - libqt5x11extras5-dev \ - libspeexdsp-dev \ - libswresample-dev \ - libswscale-dev \ - libudev-dev \ - libv4l-dev \ - libva-dev \ - libvlc-dev \ - libx11-dev \ - libx264-dev \ - libxcb-randr0-dev \ - libxcb-shm0-dev \ - libxcb-xinerama0-dev \ - libxcomposite-dev \ - libxinerama-dev \ - libmbedtls-dev \ - pkg-config \ - python3-dev \ - qtbase5-dev \ - libqt5svg5-dev \ - swig - -# build cef -wget --quiet --retry-connrefused --waitretry=1 https://cdn-fastly.obsproject.com/downloads/cef_binary_${CEF_BUILD_VERSION}_linux64.tar.bz2 -tar -xjf ./cef_binary_${CEF_BUILD_VERSION}_linux64.tar.bz2 diff --git a/CI/install-dependencies-osx.sh b/CI/install-dependencies-osx.sh deleted file mode 100755 index 35e4fb1..0000000 --- a/CI/install-dependencies-osx.sh +++ /dev/null @@ -1,81 +0,0 @@ -hr() { - echo "───────────────────────────────────────────────────" - echo $1 - echo "───────────────────────────────────────────────────" -} - -# Exit if something fails -set -e - -# Echo all commands before executing -set -v - -if [[ $TRAVIS ]]; then - git fetch --unshallow -fi - -git fetch origin --tags - -# Leave obs-studio folder -cd ../ - -# Install Packages app so we can build a package later -# http://s.sudre.free.fr/Software/Packages/about.html -hr "Downloading Packages app" -wget --quiet --retry-connrefused --waitretry=1 https://s3-us-west-2.amazonaws.com/obs-nightly/Packages.pkg -sudo installer -pkg ./Packages.pkg -target / - -brew update - -#Base OBS Deps and ccache -for DEPENDENCY in jack speexdsp ccache mbedtls freetype fdk-aac cmocka; do - if [ ! -d "$(brew --cellar)/${DEPENDENCY}" ]; then - brew install $DEPENDENCY - else - brew upgrade $DEPENDENCY - fi -done - -brew install https://gist.githubusercontent.com/DDRBoxman/9c7a2b08933166f4b61ed9a44b242609/raw/ef4de6c587c6bd7f50210eccd5bd51ff08e6de13/qt.rb -if [ -d "$(brew --cellar)/swig" ]; then - brew unlink swig -fi -brew install https://gist.githubusercontent.com/DDRBoxman/4cada55c51803a2f963fa40ce55c9d3e/raw/572c67e908bfbc1bcb8c476ea77ea3935133f5b5/swig.rb - -pip install dmgbuild - -export PATH=/usr/local/opt/ccache/libexec:$PATH -ccache -s || echo "CCache is not available." - -# Fetch and untar prebuilt OBS deps that are compatible with older versions of OSX -hr "Downloading OBS deps" -wget --quiet --retry-connrefused --waitretry=1 https://github.com/obsproject/obs-deps/releases/download/2020-04-24/osx-deps-2020-04-24.tar.gz -tar -xf ./osx-deps-2020-04-24.tar.gz -C /tmp - -# Fetch vlc codebase -hr "Downloading VLC repo" -wget --quiet --retry-connrefused --waitretry=1 https://downloads.videolan.org/vlc/3.0.8/vlc-3.0.8.tar.xz -tar -xf vlc-3.0.8.tar.xz - -# Get sparkle -hr "Downloading Sparkle framework" -wget --quiet --retry-connrefused --waitretry=1 -O sparkle.tar.bz2 https://github.com/sparkle-project/Sparkle/releases/download/1.23.0/Sparkle-1.23.0.tar.bz2 -mkdir ./sparkle -tar -xf ./sparkle.tar.bz2 -C ./sparkle -sudo cp -R ./sparkle/Sparkle.framework /Library/Frameworks/Sparkle.framework - -# CEF Stuff -hr "Downloading CEF" -wget --quiet --retry-connrefused --waitretry=1 https://obs-nightly.s3-us-west-2.amazonaws.com/cef_binary_${CEF_BUILD_VERSION}_macosx64.tar.bz2 -tar -xf ./cef_binary_${CEF_BUILD_VERSION}_macosx64.tar.bz2 -cd ./cef_binary_${CEF_BUILD_VERSION}_macosx64 -# remove a broken test -sed -i '.orig' '/add_subdirectory(tests\/ceftests)/d' ./CMakeLists.txt -# target 10.11 -sed -i '.orig' s/\"10.9\"/\"10.11\"/ ./cmake/cef_variables.cmake -mkdir build -cd ./build -cmake -DCMAKE_CXX_FLAGS="-std=c++11 -stdlib=libc++" -DCMAKE_EXE_LINKER_FLAGS="-std=c++11 -stdlib=libc++" -DCMAKE_OSX_DEPLOYMENT_TARGET=10.11 .. -make -j4 -mkdir libcef_dll -cd ../../ diff --git a/UI/auth-mixer.cpp b/UI/auth-mixer.cpp deleted file mode 100644 index b379e14..0000000 --- a/UI/auth-mixer.cpp +++ /dev/null @@ -1,313 +0,0 @@ -#include "auth-mixer.hpp" - -#include -#include -#include - -#include -#include - -#include "window-dock-browser.hpp" -#include "window-basic-main.hpp" -#include "remote-text.hpp" - -#include - -#include - -#include "ui-config.h" -#include "obf.h" - -using namespace json11; - -/* ------------------------------------------------------------------------- */ - -#define MIXER_AUTH_URL "https://obsproject.com/app-auth/mixer?action=redirect" -#define MIXER_TOKEN_URL "https://obsproject.com/app-auth/mixer-token" - -#define MIXER_SCOPE_VERSION 1 - -static Auth::Def mixerDef = {"Mixer", Auth::Type::OAuth_StreamKey}; - -/* ------------------------------------------------------------------------- */ - -MixerAuth::MixerAuth(const Def &d) : OAuthStreamKey(d) {} - -bool MixerAuth::GetChannelInfo(bool allow_retry) -try { - std::string client_id = MIXER_CLIENTID; - deobfuscate_str(&client_id[0], MIXER_HASH); - - if (!GetToken(MIXER_TOKEN_URL, client_id, MIXER_SCOPE_VERSION)) - return false; - if (token.empty()) - return false; - if (!key_.empty()) - return true; - - std::string auth; - auth += "Authorization: Bearer "; - auth += token; - - std::vector headers; - headers.push_back(std::string("Client-ID: ") + client_id); - headers.push_back(std::move(auth)); - - std::string output; - std::string error; - Json json; - bool success; - - if (id.empty()) { - auto func = [&]() { - success = GetRemoteFile( - "https://mixer.com/api/v1/users/current", - output, error, nullptr, "application/json", - nullptr, headers, nullptr, 5); - }; - - ExecThreadedWithoutBlocking( - func, QTStr("Auth.LoadingChannel.Title"), - QTStr("Auth.LoadingChannel.Text").arg(service())); - if (!success || output.empty()) - throw ErrorInfo("Failed to get user info from remote", - error); - - Json json = Json::parse(output, error); - if (!error.empty()) - throw ErrorInfo("Failed to parse json", error); - - error = json["error"].string_value(); - if (!error.empty()) - throw ErrorInfo( - error, - json["error_description"].string_value()); - - id = std::to_string(json["channel"]["id"].int_value()); - name = json["channel"]["token"].string_value(); - } - - /* ------------------ */ - - std::string url; - url += "https://mixer.com/api/v1/channels/"; - url += id; - url += "/details"; - - output.clear(); - - auto func = [&]() { - success = GetRemoteFile(url.c_str(), output, error, nullptr, - "application/json", nullptr, headers, - nullptr, 5); - }; - - ExecThreadedWithoutBlocking( - func, QTStr("Auth.LoadingChannel.Title"), - QTStr("Auth.LoadingChannel.Text").arg(service())); - if (!success || output.empty()) - throw ErrorInfo("Failed to get stream key from remote", error); - - json = Json::parse(output, error); - if (!error.empty()) - throw ErrorInfo("Failed to parse json", error); - - error = json["error"].string_value(); - if (!error.empty()) - throw ErrorInfo(error, - json["error_description"].string_value()); - - std::string key_suffix = json["streamKey"].string_value(); - - /* Mixer does not throw an error; instead it gives you the channel data - * json without the data you normally have privileges for, which means - * it'll be an empty stream key usually. So treat empty stream key as - * an error. */ - if (key_suffix.empty()) { - if (allow_retry && RetryLogin()) { - return GetChannelInfo(false); - } - throw ErrorInfo("Auth Failure", "Could not get channel data"); - } - - key_ = id + "-" + key_suffix; - - return true; -} catch (ErrorInfo info) { - QString title = QTStr("Auth.ChannelFailure.Title"); - QString text = QTStr("Auth.ChannelFailure.Text") - .arg(service(), info.message.c_str(), - info.error.c_str()); - - QMessageBox::warning(OBSBasic::Get(), title, text); - - blog(LOG_WARNING, "%s: %s: %s", __FUNCTION__, info.message.c_str(), - info.error.c_str()); - return false; -} - -void MixerAuth::SaveInternal() -{ - OBSBasic *main = OBSBasic::Get(); - config_set_string(main->Config(), service(), "Name", name.c_str()); - config_set_string(main->Config(), service(), "Id", id.c_str()); - if (uiLoaded) { - config_set_string(main->Config(), service(), "DockState", - main->saveState().toBase64().constData()); - } - OAuthStreamKey::SaveInternal(); -} - -static inline std::string get_config_str(OBSBasic *main, const char *section, - const char *name) -{ - const char *val = config_get_string(main->Config(), section, name); - return val ? val : ""; -} - -bool MixerAuth::LoadInternal() -{ - if (!cef) - return false; - - OBSBasic *main = OBSBasic::Get(); - name = get_config_str(main, service(), "Name"); - id = get_config_str(main, service(), "Id"); - firstLoad = false; - return OAuthStreamKey::LoadInternal(); -} - -static const char *elixr_script = "\ -var elixr = document.createElement('script');\ -elixr.setAttribute('src','https://api.mixrelixr.com/scripts/elixr-emotes-embedded-chat.bundle.js');\ -document.head.appendChild(elixr);"; - -void MixerAuth::LoadUI() -{ - if (!cef) - return; - if (uiLoaded) - return; - if (!GetChannelInfo()) - return; - - OBSBasic::InitBrowserPanelSafeBlock(); - OBSBasic *main = OBSBasic::Get(); - - std::string script = ""; - std::string url; - url += "https://mixer.com/embed/chat/"; - url += id; - - QSize size = main->frameSize(); - QPoint pos = main->pos(); - - chat.reset(new BrowserDock()); - chat->setObjectName("mixerChat"); - chat->resize(300, 600); - chat->setMinimumSize(200, 300); - chat->setWindowTitle(QTStr("Auth.Chat")); - chat->setAllowedAreas(Qt::AllDockWidgetAreas); - - QCefWidget *browser = cef->create_widget(nullptr, url, panel_cookies); - chat->SetWidget(browser); - - const int mxAddonChoice = - config_get_int(main->Config(), service(), "AddonChoice"); - if (mxAddonChoice) { - if (mxAddonChoice & 0x1) - script += elixr_script; - } - - browser->setStartupScript(script); - - main->addDockWidget(Qt::RightDockWidgetArea, chat.data()); - chatMenu.reset(main->AddDockWidget(chat.data())); - - /* ----------------------------------- */ - - chat->setFloating(true); - chat->move(pos.x() + size.width() - chat->width() - 50, pos.y() + 50); - - if (firstLoad) { - chat->setVisible(true); - } else { - const char *dockStateStr = config_get_string( - main->Config(), service(), "DockState"); - QByteArray dockState = - QByteArray::fromBase64(QByteArray(dockStateStr)); - main->restoreState(dockState); - } - - uiLoaded = true; -} - -bool MixerAuth::RetryLogin() -{ - if (!cef) - return false; - - OAuthLogin login(OBSBasic::Get(), MIXER_AUTH_URL, false); - cef->add_popup_whitelist_url("about:blank", &login); - - if (login.exec() == QDialog::Rejected) { - return false; - } - - std::shared_ptr auth = std::make_shared(mixerDef); - std::string client_id = MIXER_CLIENTID; - deobfuscate_str(&client_id[0], MIXER_HASH); - - return GetToken(MIXER_TOKEN_URL, client_id, MIXER_SCOPE_VERSION, - QT_TO_UTF8(login.GetCode()), true); -} - -std::shared_ptr MixerAuth::Login(QWidget *parent) -{ - if (!cef) { - return nullptr; - } - - OAuthLogin login(parent, MIXER_AUTH_URL, false); - cef->add_popup_whitelist_url("about:blank", &login); - - if (login.exec() == QDialog::Rejected) { - return nullptr; - } - - std::shared_ptr auth = std::make_shared(mixerDef); - - std::string client_id = MIXER_CLIENTID; - deobfuscate_str(&client_id[0], MIXER_HASH); - - if (!auth->GetToken(MIXER_TOKEN_URL, client_id, MIXER_SCOPE_VERSION, - QT_TO_UTF8(login.GetCode()))) { - return nullptr; - } - - std::string error; - if (auth->GetChannelInfo(false)) { - return auth; - } - - return nullptr; -} - -static std::shared_ptr CreateMixerAuth() -{ - return std::make_shared(mixerDef); -} - -static void DeleteCookies() -{ - if (panel_cookies) { - panel_cookies->DeleteCookies("mixer.com", std::string()); - panel_cookies->DeleteCookies("microsoft.com", std::string()); - } -} - -void RegisterMixerAuth() -{ - OAuth::RegisterOAuth(mixerDef, CreateMixerAuth, MixerAuth::Login, - DeleteCookies); -} diff --git a/UI/auth-mixer.hpp b/UI/auth-mixer.hpp deleted file mode 100644 index 19f3037..0000000 --- a/UI/auth-mixer.hpp +++ /dev/null @@ -1,30 +0,0 @@ -#pragma once - -#include "auth-oauth.hpp" - -class BrowserDock; - -class MixerAuth : public OAuthStreamKey { - Q_OBJECT - - QSharedPointer chat; - QSharedPointer chatMenu; - bool uiLoaded = false; - - std::string name; - std::string id; - - virtual bool RetryLogin() override; - - virtual void SaveInternal() override; - virtual bool LoadInternal() override; - - bool GetChannelInfo(bool allow_retry = true); - - virtual void LoadUI() override; - -public: - MixerAuth(const Def &d); - - static std::shared_ptr Login(QWidget *parent); -}; diff --git a/UI/window-reddit-login-dialog2.cpp b/UI/window-reddit-login-dialog2.cpp index 71101e2..9eaa1a9 100644 --- a/UI/window-reddit-login-dialog2.cpp +++ b/UI/window-reddit-login-dialog2.cpp @@ -3,6 +3,7 @@ #include "window-reddit-login-dialog2.hpp" #include +#include #include #include "api-reddit.hpp" @@ -226,18 +227,18 @@ void RedditLoginDialog2::AuthorizeResult(const QString &text, for (const QString &header : responseHeaders) { if (header.left(10) == "location: ") { QString val = header.mid(10); - int codeIdx = val.indexOf("code="); - if (codeIdx < 0) { - SetPage(PAGE_SIGNIN); + QUrl location(val); + QUrlQuery query(location); + + if(!query.hasQueryItem("code")) { + SetPage(PAGE_SIGNIN); + serverTextResponse.reset(new QString(text)); + serverErrorResponse.reset(new QString(errorText)); + errorStep = QTStr("Reddit.ErrorLog.Step.Authorization"); return; } - int codeEndIdx = val.indexOf('&', codeIdx); - if (codeEndIdx < 0) { - newCode = val.mid(codeIdx + 5); - } else { - newCode = val.mid(codeIdx + 5, - codeIdx + 5 - codeEndIdx); - } + + newCode = query.queryItemValue("code"); break; } } diff --git a/cmake/Modules/FindLibUDev.cmake b/cmake/Modules/FindLibUDev.cmake deleted file mode 100644 index 7731ec0..0000000 --- a/cmake/Modules/FindLibUDev.cmake +++ /dev/null @@ -1,33 +0,0 @@ -# Once done these will be defined: -# -# UDEV_FOUND -# UDEV_INCLUDE_DIRS -# UDEV_LIBRARIES - -find_package(PkgConfig QUIET) -if (PKG_CONFIG_FOUND) - pkg_check_modules(_UDEV QUIET udev) -endif() - -find_path(UDEV_INCLUDE_DIR - NAMES libudev.h - HINTS - ${_UDEV_INCLUDE_DIRS} - PATHS - /usr/include /usr/local/include /opt/local/include) - -find_library(UDEV_LIB - NAMES udev libudev - HINTS - ${_UDEV_LIBRARY_DIRS} - PATHS - /usr/lib /usr/local/lib /opt/local/lib) - -include(FindPackageHandleStandardArgs) -find_package_handle_standard_args(UDev DEFAULT_MSG UDEV_LIB UDEV_INCLUDE_DIR) -mark_as_advanced(UDEV_INCLUDE_DIR UDEV_LIB) - -if(UDEV_FOUND) - set(UDEV_INCLUDE_DIRS ${UDEV_INCLUDE_DIR}) - set(UDEV_LIBRARIES ${UDEV_LIB}) -endif() diff --git a/plugins/enc-amf b/plugins/enc-amf index 6e934b6..aa50203 160000 --- a/plugins/enc-amf +++ b/plugins/enc-amf @@ -1 +1 @@ -Subproject commit 6e934b6b281cf675522ea43391376bd6bb089256 +Subproject commit aa502039e3ab9a1ec6d13b42c491aaebf06b57ad diff --git a/plugins/obs-ffmpeg/ffmpeg-encoded-output.c b/plugins/obs-ffmpeg/ffmpeg-encoded-output.c deleted file mode 100644 index 4693932..0000000 --- a/plugins/obs-ffmpeg/ffmpeg-encoded-output.c +++ /dev/null @@ -1,873 +0,0 @@ -/****************************************************************************** - Copyright (C) 2019 Haivision Systems Inc. - Copyright (C) 2014 by Hugh Bailey - - This program is free software: you can redistribute it and/or modify - it under the terms of the GNU General Public License as published by - the Free Software Foundation, either version 2 of the License, or - (at your option) any later version. - - This program is distributed in the hope that it will be useful, - but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - GNU General Public License for more details. - - You should have received a copy of the GNU General Public License - along with this program. If not, see . -******************************************************************************/ - -#include "ffmpeg-encoded-output.h" - -static int proto_init(struct ffmpeg_encoded_output *stream); -static int proto_try_connect(struct ffmpeg_encoded_output *stream); -static int proto_connect_time(struct ffmpeg_encoded_output *stream); -static void proto_close(struct ffmpeg_encoded_output *stream); -static void proto_set_output_error(struct ffmpeg_encoded_output *stream); -static int proto_send_packet(struct ffmpeg_encoded_output *stream, - struct encoder_packet *packet, bool is_header); - -static inline size_t num_buffered_packets(struct ffmpeg_encoded_output *stream); -static inline void free_packets(struct ffmpeg_encoded_output *stream); -static inline bool stopping(struct ffmpeg_encoded_output *stream); -static inline bool connecting(struct ffmpeg_encoded_output *stream); -static inline bool active(struct ffmpeg_encoded_output *stream); -static inline bool disconnected(struct ffmpeg_encoded_output *stream); -static inline bool get_next_packet(struct ffmpeg_encoded_output *stream, - struct encoder_packet *packet); -#ifdef TEST_FRAMEDROPS -static void droptest_cap_data_rate(struct ffmpeg_encoded_output *stream, - size_t size); -#endif -static inline bool can_shutdown_stream(struct ffmpeg_encoded_output *stream, - struct encoder_packet *packet); -static void *send_thread(void *data); -static bool send_sps_pps(struct ffmpeg_encoded_output *stream); -static inline bool reset_semaphore(struct ffmpeg_encoded_output *stream); -static bool init_connect(struct ffmpeg_encoded_output *stream); -static int init_send(struct ffmpeg_encoded_output *stream); -static void *connect_thread(void *data); -static inline bool add_packet(struct ffmpeg_encoded_output *stream, - struct encoder_packet *packet); -static inline size_t num_buffered_packets(struct ffmpeg_encoded_output *stream); -static void drop_frames(struct ffmpeg_encoded_output *stream, const char *name, - int highest_priority, bool pframes); -static bool find_first_video_packet(struct ffmpeg_encoded_output *stream, - struct encoder_packet *first); -static void check_to_drop_frames(struct ffmpeg_encoded_output *stream, - bool pframes); -static bool add_video_packet(struct ffmpeg_encoded_output *stream, - struct encoder_packet *packet); - -static const char *ffmpeg_encoded_output_getname(void *unused) -{ - UNUSED_PARAMETER(unused); - - return obs_module_text("FFmpegEncodedOutput"); -} - -static inline bool add_packet(struct ffmpeg_encoded_output *stream, - struct encoder_packet *packet) -{ - circlebuf_push_back(&stream->packets, packet, - sizeof(struct encoder_packet)); - - return true; -} - -static bool add_video_packet(struct ffmpeg_encoded_output *stream, - struct encoder_packet *packet) -{ - check_to_drop_frames(stream, false); - check_to_drop_frames(stream, true); - - /* if currently dropping frames, drop packets until it reaches the - * desired priority */ - if (packet->drop_priority < stream->min_priority) { - stream->dropped_frames++; - return false; - } else { - stream->min_priority = 0; - } - - stream->last_dts_usec = packet->dts_usec; - - return add_packet(stream, packet); -} - -static inline bool get_next_packet(struct ffmpeg_encoded_output *stream, - struct encoder_packet *packet) -{ - bool new_packet = false; - - pthread_mutex_lock(&stream->packets_mutex); - if (stream->packets.size) { - circlebuf_pop_front(&stream->packets, packet, - sizeof(struct encoder_packet)); - new_packet = true; - } - pthread_mutex_unlock(&stream->packets_mutex); - - return new_packet; -} - -static inline void free_packets(struct ffmpeg_encoded_output *stream) -{ - size_t num_packets; - - pthread_mutex_lock(&stream->packets_mutex); - - num_packets = num_buffered_packets(stream); - if (num_packets) - info("Freeing %d remaining packets", (int)num_packets); - - while (stream->packets.size) { - struct encoder_packet packet; - circlebuf_pop_front(&stream->packets, &packet, sizeof(packet)); - obs_encoder_packet_release(&packet); - } - pthread_mutex_unlock(&stream->packets_mutex); -} - -static inline bool stopping(struct ffmpeg_encoded_output *stream) -{ - return os_event_try(stream->stop_event) != EAGAIN; -} - -static inline bool connecting(struct ffmpeg_encoded_output *stream) -{ - return os_atomic_load_bool(&stream->connecting); -} - -static inline bool active(struct ffmpeg_encoded_output *stream) -{ - return os_atomic_load_bool(&stream->active); -} - -static inline bool disconnected(struct ffmpeg_encoded_output *stream) -{ - return os_atomic_load_bool(&stream->disconnected); -} - -#ifdef TEST_FRAMEDROPS -static void droptest_cap_data_rate(struct ffmpeg_encoded_output *stream, - size_t size) -{ - uint64_t ts = os_gettime_ns(); - struct droptest_info info; - - info.ts = ts; - info.size = size; - - circlebuf_push_back(&stream->droptest_info, &info, sizeof(info)); - stream->droptest_size += size; - - if (stream->droptest_info.size) { - circlebuf_peek_front(&stream->droptest_info, &info, - sizeof(info)); - - if (stream->droptest_size > DROPTEST_MAX_BYTES) { - uint64_t elapsed = ts - info.ts; - - if (elapsed < 1000000000ULL) { - elapsed = 1000000000ULL - elapsed; - os_sleepto_ns(ts + elapsed); - } - - while (stream->droptest_size > DROPTEST_MAX_BYTES) { - circlebuf_pop_front(&stream->droptest_info, - &info, sizeof(info)); - stream->droptest_size -= info.size; - } - } - } -} -#endif - -static void *send_thread(void *data) -{ - struct ffmpeg_encoded_output *stream = data; - - os_set_thread_name("ffmpeg-stream: send_thread"); - - while (os_sem_wait(stream->send_sem) == 0) { - struct encoder_packet packet; - - if (stopping(stream) && stream->stop_ts == 0) - break; - - if (!get_next_packet(stream, &packet)) - continue; - - if (stopping(stream)) { - if (can_shutdown_stream(stream, &packet)) { - obs_encoder_packet_release(&packet); - break; - } - } - if (!stream->sent_sps_pps) { - if (!send_sps_pps(stream)) { - os_atomic_set_bool(&stream->disconnected, true); - break; - } - } - if (proto_send_packet(stream, &packet, false) < 0) { - os_atomic_set_bool(&stream->disconnected, true); - break; - } - } - - if (disconnected(stream)) - info("Disconnected from %s", stream->path.array); - else - info("User stopped the stream"); - - proto_set_output_error(stream); - proto_close(stream); - - if (!stopping(stream)) { - pthread_detach(stream->send_thread); - obs_output_signal_stop(stream->output, OBS_OUTPUT_DISCONNECTED); - } else { - obs_output_end_data_capture(stream->output); - } - - free_packets(stream); - os_event_reset(stream->stop_event); - os_atomic_set_bool(&stream->active, false); - stream->sent_sps_pps = false; - - return NULL; -} - -static bool send_sps_pps(struct ffmpeg_encoded_output *stream) -{ - obs_output_t *context = stream->output; - obs_encoder_t *vencoder = obs_output_get_video_encoder(context); - uint8_t *header; - struct encoder_packet packet = { - .type = OBS_ENCODER_VIDEO, .timebase_den = 1, .keyframe = true}; - - if (obs_encoder_get_extra_data(vencoder, &header, &packet.size)) { - packet.data = bmemdup(header, packet.size); - stream->sent_sps_pps = - proto_send_packet(stream, &packet, true) >= 0; - } - return stream->sent_sps_pps; -} - -static inline bool reset_semaphore(struct ffmpeg_encoded_output *stream) -{ - os_sem_destroy(stream->send_sem); - - return os_sem_init(&stream->send_sem, 0) == 0; -} - -static bool init_connect(struct ffmpeg_encoded_output *stream) -{ - obs_service_t *service; - obs_data_t *settings; - int64_t drop_p; - int64_t drop_b; - - if (stopping(stream)) - pthread_join(stream->send_thread, NULL); - - free_packets(stream); - - service = obs_output_get_service(stream->output); - if (!service) - return false; - - os_atomic_set_bool(&stream->disconnected, false); - stream->total_bytes_sent = 0; - stream->dropped_frames = 0; - stream->min_priority = 0; - stream->got_first_video = false; - - settings = obs_output_get_settings(stream->output); - dstr_copy(&stream->path, obs_service_get_url(service)); - dstr_copy(&stream->key, obs_service_get_key(service)); - dstr_copy(&stream->username, obs_service_get_username(service)); - dstr_copy(&stream->password, obs_service_get_password(service)); - dstr_depad(&stream->path); - dstr_depad(&stream->key); - drop_b = (int64_t)obs_data_get_int(settings, OPT_DROP_THRESHOLD); - drop_p = (int64_t)obs_data_get_int(settings, OPT_PFRAME_DROP_THRESHOLD); - stream->max_shutdown_time_sec = - (int)obs_data_get_int(settings, OPT_MAX_SHUTDOWN_TIME_SEC); - - if (drop_p < (drop_b + 200)) - drop_p = drop_b + 200; - - stream->drop_threshold_usec = 1000 * drop_b; - stream->pframe_drop_threshold_usec = 1000 * drop_p; - - obs_data_release(settings); - - return true; -} - -static void *connect_thread(void *data) -{ - struct ffmpeg_encoded_output *stream = data; - int ret; - - os_set_thread_name("ffmpeg-stream: connect_thread"); - - if (!init_connect(stream)) { - obs_output_signal_stop(stream->output, OBS_OUTPUT_BAD_PATH); - return NULL; - } - - ret = proto_try_connect(stream); - if (ret == OBS_OUTPUT_SUCCESS) - ret = init_send(stream); - - if (ret != OBS_OUTPUT_SUCCESS) { - obs_output_signal_stop(stream->output, ret); - info("Connection to %s failed: %d", stream->path.array, ret); - } - - if (!stopping(stream)) - pthread_detach(stream->connect_thread); - - os_atomic_set_bool(&stream->connecting, false); - - return NULL; -} - -static inline size_t num_buffered_packets(struct ffmpeg_encoded_output *stream) -{ - return stream->packets.size / sizeof(struct encoder_packet); -} - -static void drop_frames(struct ffmpeg_encoded_output *stream, const char *name, - int highest_priority, bool pframes) -{ - UNUSED_PARAMETER(pframes); - struct circlebuf new_buf = {0}; - int num_frames_dropped = 0; - -#ifdef _DEBUG - int start_packets = (int)num_buffered_packets(stream); -#else - UNUSED_PARAMETER(name); -#endif - - circlebuf_reserve(&new_buf, sizeof(struct encoder_packet) * 8); - - while (stream->packets.size) { - struct encoder_packet packet; - circlebuf_pop_front(&stream->packets, &packet, sizeof(packet)); - - /* do not drop audio data or video keyframes */ - if (packet.type == OBS_ENCODER_AUDIO || - packet.drop_priority >= highest_priority) { - circlebuf_push_back(&new_buf, &packet, sizeof(packet)); - - } else { - num_frames_dropped++; - obs_encoder_packet_release(&packet); - } - } - - circlebuf_free(&stream->packets); - stream->packets = new_buf; - - if (stream->min_priority < highest_priority) - stream->min_priority = highest_priority; - if (!num_frames_dropped) - return; - - stream->dropped_frames += num_frames_dropped; -#ifdef _DEBUG - debug("Dropped %s, prev packet count: %d, new packet count: %d", name, - start_packets, (int)num_buffered_packets(stream)); -#endif -} - -static bool find_first_video_packet(struct ffmpeg_encoded_output *stream, - struct encoder_packet *first) -{ - size_t count = stream->packets.size / sizeof(*first); - - for (size_t i = 0; i < count; i++) { - struct encoder_packet *cur = - circlebuf_data(&stream->packets, i * sizeof(*first)); - if (cur->type == OBS_ENCODER_VIDEO && !cur->keyframe) { - *first = *cur; - return true; - } - } - - return false; -} - -static void check_to_drop_frames(struct ffmpeg_encoded_output *stream, - bool pframes) -{ - struct encoder_packet first; - int64_t buffer_duration_usec; - size_t num_packets = num_buffered_packets(stream); - const char *name = pframes ? "p-frames" : "b-frames"; - int priority = pframes ? OBS_NAL_PRIORITY_HIGHEST - : OBS_NAL_PRIORITY_HIGH; - int64_t drop_threshold = pframes ? stream->pframe_drop_threshold_usec - : stream->drop_threshold_usec; - - if (num_packets < 5) { - if (!pframes) - stream->congestion = 0.0f; - return; - } - - if (!find_first_video_packet(stream, &first)) - return; - - /* if the amount of time stored in the buffered packets waiting to be - * sent is higher than threshold, drop frames */ - buffer_duration_usec = stream->last_dts_usec - first.dts_usec; - - if (!pframes) { - stream->congestion = - (float)buffer_duration_usec / (float)drop_threshold; - } - - if (buffer_duration_usec > drop_threshold) { - debug("buffer_duration_usec: %" PRId64, buffer_duration_usec); - drop_frames(stream, name, priority, pframes); - } -} - -static int init_send(struct ffmpeg_encoded_output *stream) -{ - int ret; - reset_semaphore(stream); - - ret = pthread_create(&stream->send_thread, NULL, send_thread, stream); - if (ret != 0) { - proto_close(stream); - warn("Failed to create send thread"); - return OBS_OUTPUT_ERROR; - } - - os_atomic_set_bool(&stream->active, true); - obs_output_begin_data_capture(stream->output, 0); - - return OBS_OUTPUT_SUCCESS; -} - -static inline bool can_shutdown_stream(struct ffmpeg_encoded_output *stream, - struct encoder_packet *packet) -{ - uint64_t cur_time = os_gettime_ns(); - bool timeout = cur_time >= stream->shutdown_timeout_ts; - - if (timeout) - info("Stream shutdown timeout reached (%d second(s))", - stream->max_shutdown_time_sec); - - return timeout || packet->sys_dts_usec >= (int64_t)stream->stop_ts; -} - -static int proto_init(struct ffmpeg_encoded_output *stream) -{ - AVOutputFormat *outfmt = NULL; - int ret = 0; - - //1. set up output format - outfmt = av_guess_format("mpegts", NULL, "video/MP2T"); - if (outfmt == NULL) { - ret = -1; - } else { - stream->ff_data.output = avformat_alloc_context(); - if (stream->ff_data.output) - stream->ff_data.output->oformat = outfmt; - else - ret = -1; - } - - return ret; -} - -static int get_audio_mix_count(int audio_mix_mask) -{ - int mix_count = 0; - - for (int i = 0; i < MAX_AUDIO_MIXES; i++) { - if ((audio_mix_mask & (1 << i)) != 0) - mix_count++; - } - - return mix_count; -} - -static inline int encoder_bitrate(obs_encoder_t *encoder) -{ - obs_data_t *settings = obs_encoder_get_settings(encoder); - const int bitrate = (int)obs_data_get_int(settings, "bitrate"); - - obs_data_release(settings); - - return bitrate; -} - -static int proto_try_connect(struct ffmpeg_encoded_output *stream) -{ - int ret = 0; - const char *url = stream->path.array; - obs_encoder_t *vencoder = obs_output_get_video_encoder(stream->output); - obs_encoder_t *aencoder = - obs_output_get_audio_encoder(stream->output, 0); - video_t *video = obs_encoder_video(vencoder); - audio_t *audio = obs_encoder_audio(aencoder); - const struct video_output_info *voi = video_output_get_info(video); - struct ffmpeg_cfg config; - bool success; - - config.url = url; - config.format_name = "mpegts"; - config.format_mime_type = "video/MP2T"; - config.muxer_settings = ""; - config.video_bitrate = encoder_bitrate(vencoder); - config.audio_bitrate = encoder_bitrate(aencoder); - config.gop_size = 250; - config.video_encoder = ""; - config.video_encoder_id = AV_CODEC_ID_H264; - config.audio_encoder = ""; - config.audio_encoder_id = AV_CODEC_ID_AAC; - config.video_settings = ""; - config.audio_settings = ""; - config.scale_width = 0; - config.scale_height = 0; - config.width = obs_encoder_get_width(vencoder); - config.height = obs_encoder_get_height(vencoder); - config.audio_tracks = (int)obs_output_get_mixer(stream->output); - config.audio_mix_count = 1; - config.format = - obs_to_ffmpeg_video_format(video_output_get_format(video)); - - if (format_is_yuv(voi->format)) { - config.color_range = voi->range == VIDEO_RANGE_FULL - ? AVCOL_RANGE_JPEG - : AVCOL_RANGE_MPEG; - config.color_space = voi->colorspace == VIDEO_CS_709 - ? AVCOL_SPC_BT709 - : AVCOL_SPC_BT470BG; - } else { - config.color_range = AVCOL_RANGE_UNSPECIFIED; - config.color_space = AVCOL_SPC_RGB; - } - - if (config.format == AV_PIX_FMT_NONE) { - blog(LOG_DEBUG, "invalid pixel format used for FFmpeg output"); - return false; - } - - if (!config.scale_width) - config.scale_width = config.width; - if (!config.scale_height) - config.scale_height = config.height; - - success = ffmpeg_data_init(&stream->ff_data, &config); - if (!success) - return -1; - - return ret; -} - -static int proto_connect_time(struct ffmpeg_encoded_output *stream) -{ - UNUSED_PARAMETER(stream); - - return 0; -} - -static void proto_close(struct ffmpeg_encoded_output *stream) -{ - ffmpeg_data_free(&stream->ff_data); -} - -static inline int64_t _rescale_ts(struct ffmpeg_encoded_output *stream, - int64_t val, int idx) -{ - AVStream *avstream = stream->ff_data.output->streams[idx]; - - return av_rescale_q_rnd(val / avstream->codec->time_base.num, - avstream->codec->time_base, avstream->time_base, - AV_ROUND_NEAR_INF | AV_ROUND_PASS_MINMAX); -} - -static int proto_send_packet(struct ffmpeg_encoded_output *stream, - struct encoder_packet *obs_packet, bool is_header) -{ - AVPacket av_packet = {0}; - int ret = 0; - int streamIdx = (obs_packet->type == OBS_ENCODER_VIDEO) ? 0 : 1; - - //2. send - av_init_packet(&av_packet); - - av_packet.data = obs_packet->data; - av_packet.size = (int)obs_packet->size; - av_packet.stream_index = streamIdx; - av_packet.pts = _rescale_ts(stream, obs_packet->pts, streamIdx); - av_packet.dts = _rescale_ts(stream, obs_packet->dts, streamIdx); - - if (obs_packet->keyframe) - av_packet.flags = AV_PKT_FLAG_KEY; - - ret = av_interleaved_write_frame(stream->ff_data.output, &av_packet); - stream->total_bytes_sent += obs_packet->size; - if (is_header) - bfree(obs_packet->data); - else - obs_encoder_packet_release(obs_packet); - - return ret; -} - -static void proto_set_output_error(struct ffmpeg_encoded_output *stream) -{ - UNUSED_PARAMETER(stream); -} - -static void ffmpeg_encoded_output_destroy(void *data) -{ - struct ffmpeg_encoded_output *stream = data; - - if (stopping(stream) && !connecting(stream)) { - pthread_join(stream->send_thread, NULL); - - } else if (connecting(stream) || active(stream)) { - if (stream->connecting) - pthread_join(stream->connect_thread, NULL); - - stream->stop_ts = 0; - os_event_signal(stream->stop_event); - - if (active(stream)) { - os_sem_post(stream->send_sem); - obs_output_end_data_capture(stream->output); - pthread_join(stream->send_thread, NULL); - } - } - - free_packets(stream); - dstr_free(&stream->path); - dstr_free(&stream->key); - dstr_free(&stream->username); - dstr_free(&stream->password); - dstr_free(&stream->encoder_name); - os_event_destroy(stream->stop_event); - os_sem_destroy(stream->send_sem); - pthread_mutex_destroy(&stream->packets_mutex); - circlebuf_free(&stream->packets); -#ifdef TEST_FRAMEDROPS - circlebuf_free(&stream->droptest_info); -#endif - - os_event_destroy(stream->buffer_space_available_event); - os_event_destroy(stream->buffer_has_data_event); - os_event_destroy(stream->send_thread_signaled_exit); - pthread_mutex_destroy(&stream->write_buf_mutex); - ffmpeg_data_free(&stream->ff_data); - if (stream->write_buf) - bfree(stream->write_buf); - bfree(stream); -} - -static void *ffmpeg_encoded_output_create(obs_data_t *settings, - obs_output_t *output) -{ - struct ffmpeg_encoded_output *stream = - bzalloc(sizeof(struct ffmpeg_encoded_output)); - - stream->output = output; - pthread_mutex_init_value(&stream->packets_mutex); - - proto_init(stream); - - if (pthread_mutex_init(&stream->packets_mutex, NULL) != 0) - goto fail; - if (os_event_init(&stream->stop_event, OS_EVENT_TYPE_MANUAL) != 0) - goto fail; - - if (pthread_mutex_init(&stream->write_buf_mutex, NULL) != 0) { - warn("Failed to initialize write buffer mutex"); - goto fail; - } - - if (os_event_init(&stream->buffer_space_available_event, - OS_EVENT_TYPE_AUTO) != 0) { - warn("Failed to initialize write buffer event"); - goto fail; - } - if (os_event_init(&stream->buffer_has_data_event, OS_EVENT_TYPE_AUTO) != - 0) { - warn("Failed to initialize data buffer event"); - goto fail; - } - if (os_event_init(&stream->send_thread_signaled_exit, - OS_EVENT_TYPE_MANUAL) != 0) { - warn("Failed to initialize socket exit event"); - goto fail; - } - - UNUSED_PARAMETER(settings); - - return stream; - -fail: - ffmpeg_encoded_output_destroy(stream); - return NULL; -} - -static bool ffmpeg_encoded_output_start(void *data) -{ - struct ffmpeg_encoded_output *stream = data; - - if (!obs_output_can_begin_data_capture(stream->output, 0)) - return false; - if (!obs_output_initialize_encoders(stream->output, 0)) - return false; - - os_atomic_set_bool(&stream->connecting, true); - - return pthread_create(&stream->connect_thread, NULL, connect_thread, - stream) == 0; -} - -#define MILLISECOND_DEN 1000 - -static int32_t get_ms_time(struct encoder_packet *packet, int64_t val) -{ - return (int32_t)(val * MILLISECOND_DEN / packet->timebase_den); -} - -static void ffmpeg_encoded_output_data(void *data, - struct encoder_packet *packet) -{ - struct ffmpeg_encoded_output *stream = data; - struct encoder_packet new_packet; - bool added_packet = false; - - if (disconnected(stream) || !active(stream)) - return; - - if (packet->type == OBS_ENCODER_VIDEO) { - if (!stream->got_first_video) { - stream->start_dts_offset = - get_ms_time(packet, packet->dts); - stream->got_first_video = true; - } - } - obs_encoder_packet_ref(&new_packet, packet); - - pthread_mutex_lock(&stream->packets_mutex); - - if (!disconnected(stream)) - added_packet = (packet->type == OBS_ENCODER_VIDEO) - ? add_video_packet(stream, &new_packet) - : add_packet(stream, &new_packet); - - pthread_mutex_unlock(&stream->packets_mutex); - - if (added_packet) - os_sem_post(stream->send_sem); - else - obs_encoder_packet_release(&new_packet); -} - -static void ffmpeg_encoded_output_stop(void *data, uint64_t ts) -{ - struct ffmpeg_encoded_output *stream = data; - - if (stopping(stream) && ts != 0) - return; - - if (connecting(stream)) - pthread_join(stream->connect_thread, NULL); - - stream->stop_ts = ts / 1000ULL; - - if (ts) - stream->shutdown_timeout_ts = - ts + - (uint64_t)stream->max_shutdown_time_sec * 1000000000ULL; - - if (active(stream)) { - os_event_signal(stream->stop_event); - if (stream->stop_ts == 0) - os_sem_post(stream->send_sem); - } else { - obs_output_signal_stop(stream->output, OBS_OUTPUT_SUCCESS); - } -} - -static void ffmpeg_encoded_output_defaults(obs_data_t *defaults) -{ - obs_data_set_default_int(defaults, OPT_DROP_THRESHOLD, 700); - obs_data_set_default_int(defaults, OPT_PFRAME_DROP_THRESHOLD, 900); - obs_data_set_default_int(defaults, OPT_MAX_SHUTDOWN_TIME_SEC, 30); -} - -static obs_properties_t *ffmpeg_encoded_output_properties(void *unused) -{ - UNUSED_PARAMETER(unused); - obs_properties_t *props = obs_properties_create(); - - obs_properties_add_int( - props, OPT_DROP_THRESHOLD, - obs_module_text("FFmpegEncodedOutput.DropThreshold"), 200, - 10000, 100); - - return props; -} - -static uint64_t ffmpeg_encoded_output_total_bytes_sent(void *data) -{ - struct ffmpeg_encoded_output *stream = data; - - return stream->total_bytes_sent; -} - -static int ffmpeg_encoded_output_dropped_frames(void *data) -{ - struct ffmpeg_encoded_output *stream = data; - - return stream->dropped_frames; -} - -static float ffmpeg_encoded_output_congestion(void *data) -{ - struct ffmpeg_encoded_output *stream = data; - - return stream->min_priority > 0 ? 1.0f : stream->congestion; -} - -static int ffmpeg_encoded_output_connect_time(void *data) -{ - struct ffmpeg_encoded_output *stream = data; - - return proto_connect_time(stream); -} - -struct obs_output_info ffmpeg_encoded_output_info = { - .id = "ffmpeg_encoded_output", - .flags = OBS_OUTPUT_AV | OBS_OUTPUT_ENCODED | OBS_OUTPUT_SERVICE | - OBS_OUTPUT_MULTI_TRACK, - .encoded_video_codecs = "h264", - .encoded_audio_codecs = "aac", - .get_name = ffmpeg_encoded_output_getname, - .create = ffmpeg_encoded_output_create, - .destroy = ffmpeg_encoded_output_destroy, - .start = ffmpeg_encoded_output_start, - .stop = ffmpeg_encoded_output_stop, - .encoded_packet = ffmpeg_encoded_output_data, - .get_defaults = ffmpeg_encoded_output_defaults, - .get_properties = ffmpeg_encoded_output_properties, - .get_total_bytes = ffmpeg_encoded_output_total_bytes_sent, - .get_congestion = ffmpeg_encoded_output_congestion, - .get_connect_time_ms = ffmpeg_encoded_output_connect_time, - .get_dropped_frames = ffmpeg_encoded_output_dropped_frames}; diff --git a/plugins/obs-ffmpeg/ffmpeg-encoded-output.h b/plugins/obs-ffmpeg/ffmpeg-encoded-output.h deleted file mode 100644 index 6012754..0000000 --- a/plugins/obs-ffmpeg/ffmpeg-encoded-output.h +++ /dev/null @@ -1,108 +0,0 @@ -/****************************************************************************** - Copyright (C) 2019 Haivision Systems Inc. - Copyright (C) 2014 by Hugh Bailey - - This program is free software: you can redistribute it and/or modify - it under the terms of the GNU General Public License as published by - the Free Software Foundation, either version 2 of the License, or - (at your option) any later version. - - This program is distributed in the hope that it will be useful, - but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - GNU General Public License for more details. - - You should have received a copy of the GNU General Public License - along with this program. If not, see . -******************************************************************************/ - -#include -#include -#include -#include -#include -#include -#include -#include "obs-ffmpeg-output.h" -#include "obs-ffmpeg-formats.h" - -#define do_log(level, format, ...) \ - blog(level, "[ffmpeg-encoded-output: '%s'] " format, \ - obs_output_get_name(stream->output), ##__VA_ARGS__) - -#define warn(format, ...) do_log(LOG_WARNING, format, ##__VA_ARGS__) -#define info(format, ...) do_log(LOG_INFO, format, ##__VA_ARGS__) -#define debug(format, ...) do_log(LOG_DEBUG, format, ##__VA_ARGS__) - -#define OPT_DROP_THRESHOLD "drop_threshold_ms" -#define OPT_PFRAME_DROP_THRESHOLD "pframe_drop_threshold_ms" -#define OPT_MAX_SHUTDOWN_TIME_SEC "max_shutdown_time_sec" -#define OPT_LOWLATENCY_ENABLED "low_latency_mode_enabled" - -//#define TEST_FRAMEDROPS - -#ifdef TEST_FRAMEDROPS - -#define DROPTEST_MAX_KBPS 3000 -#define DROPTEST_MAX_BYTES (DROPTEST_MAX_KBPS * 1000 / 8) - -struct droptest_info { - uint64_t ts; - size_t size; -}; -#endif - -struct ffmpeg_encoded_output { - obs_output_t *output; - - pthread_mutex_t packets_mutex; - struct circlebuf packets; - bool sent_sps_pps; - - bool got_first_video; - int64_t start_dts_offset; - - volatile bool connecting; - pthread_t connect_thread; - - volatile bool active; - volatile bool disconnected; - pthread_t send_thread; - - int max_shutdown_time_sec; - - os_sem_t *send_sem; - os_event_t *stop_event; - uint64_t stop_ts; - uint64_t shutdown_timeout_ts; - - struct dstr path, key; - struct dstr username, password; - struct dstr encoder_name; - - /* frame drop variables */ - int64_t drop_threshold_usec; - int64_t pframe_drop_threshold_usec; - int min_priority; - float congestion; - - int64_t last_dts_usec; - - uint64_t total_bytes_sent; - int dropped_frames; - -#ifdef TEST_FRAMEDROPS - struct circlebuf droptest_info; - size_t droptest_size; -#endif - - uint8_t *write_buf; - size_t write_buf_len; - size_t write_buf_size; - pthread_mutex_t write_buf_mutex; - os_event_t *buffer_space_available_event; - os_event_t *buffer_has_data_event; - os_event_t *send_thread_signaled_exit; - - struct ffmpeg_data ff_data; -}; diff --git a/plugins/obs-vst b/plugins/obs-vst index 83e1bf2..cca219f 160000 --- a/plugins/obs-vst +++ b/plugins/obs-vst @@ -1 +1 @@ -Subproject commit 83e1bf241007480a796c0c3f65018d1f0e482447 +Subproject commit cca219fa3613dbc65de676ab7ba29e76865fa6f8 diff --git a/plugins/win-capture/graphics-hook/d3d1x_shaders.hpp b/plugins/win-capture/graphics-hook/d3d1x_shaders.hpp deleted file mode 100644 index a12b351..0000000 --- a/plugins/win-capture/graphics-hook/d3d1x_shaders.hpp +++ /dev/null @@ -1,31 +0,0 @@ -#pragma once - -static const char vertex_shader_string[] = "struct VertData \ -{ \ - float4 pos : SV_Position; \ - float2 texCoord : TexCoord0; \ -}; \ -VertData main(VertData input) \ -{ \ - VertData output; \ - output.pos = input.pos; \ - output.texCoord = input.texCoord; \ - return output; \ -}"; - -static const char pixel_shader_string[] = "uniform Texture2D diffuseTexture; \ -SamplerState textureSampler \ -{ \ - AddressU = Clamp; \ - AddressV = Clamp; \ - Filter = Linear; \ -}; \ -struct VertData \ -{ \ - float4 pos : SV_Position; \ - float2 texCoord : TexCoord0; \ -}; \ -float4 main(VertData input) : SV_Target \ -{ \ - return diffuseTexture.Sample(textureSampler, input.texCoord); \ -}"; diff --git a/plugins/win-dshow/libdshowcapture b/plugins/win-dshow/libdshowcapture index 03fbb38..09f5a36 160000 --- a/plugins/win-dshow/libdshowcapture +++ b/plugins/win-dshow/libdshowcapture @@ -1 +1 @@ -Subproject commit 03fbb3814b2ad678f5be4b23b8ca05fb4172e12e +Subproject commit 09f5a36788d2cada8643605775fb8fdbd8e5e2c7