Changes of Revision 103

obs-studio.changes Changed
x
 
1
@@ -1,4 +1,272 @@
2
 -------------------------------------------------------------------
3
+Sun Jan 08 14:16:44 UTC 2023 - hpj@urpla.net
4
+
5
+- Update to version 29.0.0:
6
+  * rtmp-services: Specify RTMP_SERVICES_FORMAT_VERSION in package.json
7
+  * CI: Revise repository conditions to validate JSON schema of services
8
+  * libobs: Update version to 29.0.0
9
+  * win-capture: Always reset timeout when searching for target display
10
+  * UI: Lock volume meter sliders to LTR
11
+  * UI: Use stream encoder when resetting encoders
12
+  * obs-filters: Log NVIDIA Effects version only if lib is found
13
+  * CI: Use Flatpak build-bundle option
14
+  * CI: Update Flatpak Actions
15
+  * Revert "libobs-d3d11: Default to Intel IGPU on IGPU+DGPU systems"
16
+  * Revert "libobs-d3d11: Make sure libobs knows the new adapter index"
17
+  * Revert "libobs: Fix adapter index not getting applied to resets"
18
+  * UI: Restrict GNOME wayland override
19
+  * obs-ffmpeg, obs-qsv11: Ensure adapter order in encoder tests
20
+  * obs-ffmpeg: Remove EnumOutputs from encoder tests
21
+  * libobs: Add funcs to get windows video adapter LUIDs
22
+  * rtmp-services: Remove defunct servers/services
23
+  * obs-ffmpeg: Set chroma location for VA-API
24
+  * obs-qsv11: Put mastering primaries in GBR order
25
+  * mac-videotoolbox: Add HDR metadata
26
+  * obs-ffmpeg: Fix AMF default CQP value
27
+  * libobs: Duplicate URL string for OBS_BUTTON_URL
28
+  * UI: Fix Qt AutoUic warning
29
+  * obs-ffmpeg: Add new rate control method mappings for AVC/HEVC
30
+  * obs-filter: Fix upward compressor
31
+  * obs-ffmpeg: Use enum for av1 encoders
32
+  * cmake: Fix FindGio.cmake to find libgio
33
+  * docs: Add missing source functions
34
+  * obs-ffmpeg: Bump AMF version to v1.4.29
35
+  * obs-ffmpeg: Add new rate control methods for AMD AVC/HEVC
36
+  * docs: Fix reference count info of obs_frontend_get_streaming_service
37
+  * libcaption: Fix invalid data at utf8_load_text_file
38
+  * docs: Add versionadded for 29.0.0 functions
39
+  * docs: Add info on property modified callback
40
+  * mac-videotoolbox: Don't parse HEVC as AVC
41
+  * UI: Guard GetMonitorName behind Qt < 6.4
42
+  * UI: Fix slide counter with no slides
43
+  * obs-ffmpeg: Improve chroma location decision
44
+  * docs: Clarify signal_handler_connect()
45
+  * obs-filters: Use correct signal to reset greenscreen filter
46
+  * image-source: Remove cleared missing files from slideshow
47
+  * libobs: Override fps ovi for aux views
48
+  * updater: Bump to version 2.2
49
+  * obs-ffmpeg: Allow srt stream to disconnect after timeout
50
+  * rtmp-services: Add Bitmovin
51
+  * obs-qsv11: Set subprocess timeout to 10 sec
52
+  * win-dshow: Ignore FFmpeg colorspace if overridden
53
+  * obs-ffmpeg: Disable VBAQ for H264 CQP rate control
54
+  * obs-qsv11: Fix QSV detection
55
+  * obs-qsv11: Fix encoder capping resolution on dgpus
56
+  * Revert "obs-qsv11: Don't set to low power mode if AV1"
57
+  * obs-qsv11: Fix HDR not working with AV1
58
+  * obs-qsv11: Don't set to low power mode if AV1
59
+  * obs-qsv11: Keep ExtParam value around
60
+  * UI: Hide "Update Channel" label on macOS
61
+  * UI: Guard AutoBetaOptIn as Windows-only
62
+  * UI: Delay timed update check until branch migration
63
+  * UI: Fix monitor name for projectors on Windows
64
+  * obs-qsv11: Remove statics, fix buffer misuse
65
+  * obs-qsv11: Add HEVC
66
+  * obs-qsv11: Don't declare vars in switch w/o braces
67
+  * obs-qsv11: Remove unused function
68
+  * obs-qsv11: Fix profile default for AV1
69
+  * obs-qsv11: Remove unused function declaration
70
+  * obs-ffmpeg: Disable VBAQ for HEVC CQP rate control
71
+  * libobs: Suppress LNK4098
72
+  * rtmp-services: Suppress LNK4098
73
+  * UI: Add QSV AV1 to simple output mode
74
+  * obs-qsv11: Simplify CQP
75
+  * obs-qsv11: Add QSV AV1 encoder
76
+  * obs-qsv11: Add codec enum
77
+  * obs-qsv11: Only reinitialize bitrate
78
+  * obs-qsv11: Add startup process to test QSV support
79
+  * libobs: Fix adapter index not getting applied to resets
80
+  * UI: Switch to beta branch when running beta/rc for the first time
81
+  * UI: Add update channels (Windows)
82
+  * updater: Add --branch/--portable command line arguments
83
+  * cmake/libobs: Set OBS_COMMIT based on git describe
84
+  * flatpak: Update deps based on obs-deps 2022-11-21
85
+  * CI: Update deps to obs-deps 2022-11-21 release
86
+  * libobs-d3d11: Make sure libobs knows the new adapter index
87
+  * obs-ffmpeg: Fix building without HEVC on Windows
88
+  * cmake: Specify utf-8 for MSVC builds
89
+  * CI,obs-vst: Update Flatpak KDE Runtime to version 6.4
90
+  * UI, image-source: Add slide counter to slideshow toolbar
91
+  * obs-vst: Improve some string handling
92
+  * obs-vst: Use libobs memory allocation functions
93
+  * UI: Disable toolbar buttons when no source is selected
94
+  * obs-ffmpeg: Improve RIST/SRT log messages
95
+  * UI: Use blog for "Attempted path" log messages
96
+  * ffmpeg: fix cqp rate control on svtav1
97
+  * ffmpeg: fix "cqp" mode for libaom
98
+  * libobs: Deprecate obs_get/set_master_volume
99
+  * obs-browser: Don't use QPointF for pointer position
100
+  * obs-browser: Update version to 2.19.0
101
+  * libdshowcapture: Support more capture cards with uncoupled audio
102
+  * obs-ffmpeg: Use Libva in FFmpeg VA-API
103
+  * UI: Add filters button to scenes toolbar
104
+  * UI: Remove Qt taskbar overlay
105
+  * obs-ffmpeg: Fix SVT-AV1 rate control mode selection
106
+  * libobs: Allow overriding video resolution per view
107
+  * decklink: Always output BGRA
108
+  * UI: Clarify that RGB output format is BGRA
109
+  * flatpak: Add Jansson to modules
110
+  * mac-videotoolbox: Default to High profile
111
+  * UI: Add Apple Hardware Encoder to AutoConfig
112
+  * win-dshow: Recognise higher FPS values from devices
113
+  * CI: Name Docs zip based on commit/tag
114
+  * obs-ffmpeg: Fix encoder preset quality fallbacks for AVC/HEVC/AV1
115
+  * obs-ffmpeg: Suggest docs to reference for AMF/FFmpeg options
116
+  * UI: Add AMD AV1 to simple output mode
117
+  * obs-ffmpeg: Add AMF AV1 encoder
118
+  * obs-ffmpeg: Use codec enum for AMF texture encode check
119
+  * obs-ffmpeg: Make AMF AVC encoder name consistent w/ others
120
+  * obs-ffmpeg: Only show b-frames AMF property for AVC
121
+  * obs-ffmpeg: Only allow AMF high/baseline profiles for AVC
122
+  * obs-ffmpeg: Allow 0-51 for CQP property
123
+  * obs-ffmpeg: Use codec enum for amf_properties_internal
124
+  * obs-ffmpeg: Fix transcoding API typo
125
+  * obs-ffmpeg: Update AMF SDK for AV1 support
126
+  * UI: Change Simple Output NVENC default preset to P5
127
+  * CI: Re-enable scripting in Windows builds
128
+  * obs-ffmpeg: Change default nvenc preset to P5
129
+  * win-capture,UI: Look up display by id, not index
130
+  * Revert "virtualcam-module: Don't send frames on initial pause"
131
+  * obs-websocket: Update version to 5.1.0
132
+  * obs-ffmpeg: Fix SRT error type comparison (#7802)
133
+  * win-capture: Invert output when drawing monochrome cursors
134
+  * rtmp-services: Add ffmpeg-mpegts-muxer in schema v4
135
+  * obs-ffmpeg: Direct setting of  encryption & auth for SRT & RIST
136
+  * UI: Use weak source for projectors
137
+  * obs-ffmpeg: Use compatibility options on nvnenc init fail
138
+  * libobs: Fix SRGB to SCRGB async video rendering
139
+  * CI: Fix building in PowerShell 7.3.x
140
+  * CI: Fix services checkers using wrong port for RTMPS
141
+  * UI: Add separator in source toolbar
142
+  * obs-outputs: Shorten dynamic bitrate increment timeout
143
+  * rtmp-services: Add IRLToolkit
144
+  * UI: Remove number from multiview labels
145
+  * CI: Add debian debug symbols to CI artifacts
146
+  * cmake: Fix debian packages loosing all debug symbols
147
+  * mac-capture: Disable all SCK modes besides WindowCapture on macOS 12
148
+  * mac-videotoolbox: Support P010 and HDR color spaces
149
+  * obs-filters: NVIDIA Background Removal variable mask refresh
150
+  * obs-filters: Add temporal processing to Background Removal
151
+  * obs-filters: Warn if NVIDIA Audio FX is outdated
152
+  * obs-filters: Warn if NVIDIA Video FX is oudated
153
+  * obs-outputs: Increase librtmp send timeout to 15 seconds
154
+  * UI: Fix snprintf calls with literals as buffer sizes
155
+  * obs-outputs: Fix snprintf calls with literals as buffer sizes
156
+  * obs-filters: Fix snprintf calls with literals as buffer sizes
157
+  * image-source: Fix snprintf calls with literals as buffer sizes
158
+  * coreaudio-encoder: Fix snprintf calls with literals as buffer sizes
159
+  * obs-x264: Fix snprintf calls with literals as buffer sizes
160
+  * win-capture: Replace invocations of sprintf with snprintf
161
+  * obs-ffmpeg: Replace invocations of sprintf with snprintf
162
+  * libobs-d3d11: Replace invocations of sprintf with snprintf
163
+  * linux-v4l2: Replace invocations of sprintf with snprintf
164
+  * linux-capture: Replace invocations of sprintf with snprintf
165
+  * UI: Replace invocations of sprintf with snprintf
166
+  * obs-outputs: Replace invocations of sprintf with snprintf
167
+  * mac-capture: Replace invocations of sprintf with snprintf
168
+  * libobs: Replace invocations of sprintf with snprintf
169
+  * deps: Replace invocations of sprintf with snprintf
170
+  * obs-ffmpeg: Fix deprecation of channels member of several structs
171
+  * libobs: Change audio resampler to new channel API
172
+  * obs-ffmpeg: Update mpegts to channel API change
173
+  * docs: Add info on funcs to use for properties
174
+  * aja: Fix capturing UHD/4K YUV on Kona HDMI.
175
+  * UI: Fix QStyle memory leak
176
+  * libobs-d3d11: Support advanced SDR window preview
177
+  * mac-capture: Support P3 for HDR recordings
178
+  * libobs: Add P3 shaders for Mac
179
+  * libobs-opengl: Support l10r IOSurface
180
+  * decklink-output-ui: Pipeline GPU data for preview
181
+  * libobs: Log Windows emulation status
182
+  * libobs: Log macOS Rosetta status
183
+  * UI: Remove Rosetta detection log
184
+  * libobs/util: Add function to get Windows x64 emulation status
185
+  * UI: Use on_foo_bar properly for docks context menu
186
+  * UI: Replace manual usage of on_foo_bar for show/hide transition
187
+  * UI: Remove support for toggling Aero
188
+  * libobs: Remove Aero logging
189
+  * mac-avcapture: Add DeskCam support
190
+  * rtmp-services: Add Whowatch
191
+  * libobs: Fix reading Windows release name
192
+  * UI: Set Replay Buffer Memory limit dynamically
193
+  * libobs: Add utility function to get total RAM
194
+  * libobs: Move async filtering from render to tick
195
+  * libobs: Add "source_update" signal
196
+  * docs: Add clarifications
197
+  * UI: Fix always on top not being saved on exit
198
+  * libobs: Update to 28.1.2
199
+  * CI: Upload beta builds as Steam Playtest
200
+  * obs-filters: Add a simple 3-band equalizer
201
obs-studio.spec Changed
26
 
1
@@ -8,7 +8,7 @@
2
 %endif
3
 
4
 Name:           obs-studio
5
-Version:        28.1.2
6
+Version:        29.0.0
7
 Release:        0
8
 Summary:        A recording/broadcasting program
9
 Group:          Productivity/Multimedia/Video/Editors and Convertors
10
@@ -55,12 +55,13 @@
11
 BuildRequires:  pkgconfig(libavfilter)
12
 BuildRequires:  pkgconfig(libavformat)
13
 BuildRequires:  pkgconfig(libavutil)
14
+BuildRequires:  pkgconfig(librist)
15
 BuildRequires:  pkgconfig(libswresample)
16
 BuildRequires:  pkgconfig(libswscale)
17
-BuildRequires:  pkgconfig(librist)
18
+BuildRequires:  pkgconfig(libva)
19
 BuildRequires:  pkgconfig(lua)
20
 BuildRequires:  pkgconfig(luajit)
21
-BuildRequires:  pkgconfig(srt)
22
+BuildRequires:  pkgconfig(srt) >= 1.4
23
 BuildRequires:  python3-devel
24
 %if 0%{?suse_version} != 1315 || 0%{?is_opensuse}
25
 BuildRequires:  speexdsp-devel
26
_service Changed
10
 
1
@@ -1,7 +1,7 @@
2
 <services>
3
   <service name="tar_scm" mode="disabled">
4
     <param name="versionformat">@PARENT_TAG@</param>
5
-    <param name="revision">refs/tags/28.1.2</param>
6
+    <param name="revision">refs/tags/29.0.0</param>
7
     <param name="url">https://github.com/obsproject/obs-studio.git</param>
8
     <param name="scm">git</param>
9
     <param name="changesgenerate">enable</param>
10
_servicedata Changed
10
 
1
@@ -1,6 +1,6 @@
2
 <servicedata>
3
   <service name="tar_scm">
4
     <param name="url">https://github.com/obsproject/obs-studio.git</param>
5
-    <param name="changesrevision">c1841e43b04e662ae3be146ce10596eb7be866fa</param>
6
+    <param name="changesrevision">b51773b97b551cb409603146a97b2827d2a66ec3</param>
7
   </service>
8
 </servicedata>
9
\ No newline at end of file
10
obs-studio-28.1.2.tar.xz/plugins/rtmp-services/data/schema/service-schema-v3.json Deleted
201
 
1
@@ -1,238 +0,0 @@
2
-{
3
-    "$schema": "http://json-schema.org/draft-07/schema#",
4
-    "type": "object",
5
-    "properties": {
6
-        "format_version": {
7
-            "type": "integer",
8
-            "description": "Identifier for parsing this file.\n- v3 introduced 'ffmpeg_hls_muxer' to services/recommended/output\n - v2 introduced 'alt_names' to services"
9
-        },
10
-        "services": {
11
-            "type": "array",
12
-            "items": {
13
-                "type": "object",
14
-                "properties": {
15
-                    "name": {
16
-                        "type": "string",
17
-                        "description": "Name of the streaming service. Will be displayed in the Service dropdown.",
18
-                        "minLength": 1
19
-                    },
20
-                    "common": {
21
-                        "type": "boolean",
22
-                        "description": "Whether or not the service is shown in the list before it is expanded to all services by the user.",
23
-                        "default": false
24
-                    },
25
-                    "stream_key_link": {
26
-                        "$ref": "#/definitions/saneUrl",
27
-                        "description": "Link where a logged-in user can find the 'stream key', presented as a button alongside the stream key field."
28
-                    },
29
-                    "supported video codecs": {
30
-                        "type": "array",
31
-                        "description": "Video codecs that are supported by the service.",
32
-                        "items": {
33
-                            "type": "string",
34
-                            "description": "Short-form codec names.",
35
-                            "minLength": 1,
36
-                            "enum": 
37
-                                "h264",
38
-                                "hevc"
39
-                            
40
-                        }
41
-                    },
42
-                    "supported audio codecs": {
43
-                        "type": "array",
44
-                        "description": "Audio codecs that are supported by the service.",
45
-                        "items": {
46
-                            "type": "string",
47
-                            "description": "Short-form codec names.",
48
-                            "minLength": 1,
49
-                            "enum": 
50
-                                "aac",
51
-                                "opus"
52
-                            
53
-                        }
54
-                    },
55
-                    "servers": {
56
-                        "type": "array",
57
-                        "description": "List of servers.",
58
-                        "items": {
59
-                            "type": "object",
60
-                            "properties": {
61
-                                "name": {
62
-                                    "type": "string",
63
-                                    "description": "Name of the server (e.g. location, primary/backup), displayed in the Server dropdown.",
64
-                                    "minLength": 1
65
-                                },
66
-                                "url": {
67
-                                    "$ref": "#/definitions/serviceUri",
68
-                                    "description": "RTMP(S) or HLS URL of the ingest server.",
69
-                                    "minLength": 1
70
-                                }
71
-                            },
72
-                            "additionalProperties": false,
73
-                            "required": 
74
-                                "name",
75
-                                "url"
76
-                            
77
-                        },
78
-                        "default": 
79
-                            {
80
-                                "name": "",
81
-                                "url": ""
82
-                            }
83
-                        ,
84
-                        "minItems": 1,
85
-                        "additionalItems": true
86
-                    },
87
-                    "recommended": {
88
-                        "type": "object",
89
-                        "description": "Recommended service settings. Users will be unable to choose values outside of these by default, so choose recommended values carefully.",
90
-                        "properties": {
91
-                            "keyint": {
92
-                                "type": "integer",
93
-                                "description": "Keyframe interval (seconds)."
94
-                            },
95
-                            "max video bitrate": {
96
-                                "type": "integer",
97
-                                "description": "Highest supported video bitrate (kbps)."
98
-                            },
99
-                            "max audio bitrate": {
100
-                                "type": "integer",
101
-                                "description": "Highest supported audio bitrate (kbps)."
102
-                            },
103
-                            "x264opts": {
104
-                                "type": "string",
105
-                                "description": "Additional x264 encoder options. Space-separated.",
106
-                                "pattern": "^(\\S+=\\S+\\s*)+$"
107
-                            },
108
-                            "output": {
109
-                                "type": "string",
110
-                                "description": "OBS output module used.",
111
-                                "enum": 
112
-                                    "rtmp_output",
113
-                                    "ffmpeg_hls_muxer",
114
-                                    "ftl_output"
115
-                                
116
-                            },
117
-                            "profile": {
118
-                                "type": "string",
119
-                                "description": "H.264 Profile.",
120
-                                "minLength": 1,
121
-                                "enum": 
122
-                                    "high",
123
-                                    "main",
124
-                                    "baseline"
125
-                                
126
-                            },
127
-                            "bframes": {
128
-                                "type": "integer",
129
-                                "description": "Maximum allowed number of B-Frames."
130
-                            },
131
-                            "supported resolutions": {
132
-                                "type": "array",
133
-                                "description": "List of supported resolutions in format {width}x{height}",
134
-                                "items": {
135
-                                    "$ref": "#/definitions/resolution"
136
-                                },
137
-                                "minItems": 1,
138
-                                "additionalItems": true
139
-                            },
140
-                            "max fps": {
141
-                                "type": "integer",
142
-                                "description": "Maximum supported framerate."
143
-                            },
144
-                            "bitrate matrix": {
145
-                                "type": "array",
146
-                                "description": "List of resolutions and frame rate combinations with their recommended maximum bitrate.",
147
-                                "items": {
148
-                                    "type": "object",
149
-                                    "properties": {
150
-                                        "res": {
151
-                                            "$ref": "#/definitions/resolution",
152
-                                            "description": "Resolution in format {width}x{height}"
153
-                                        },
154
-                                        "fps": {
155
-                                            "type": "integer",
156
-                                            "description": "Frame rate"
157
-                                        },
158
-                                        "max bitrate": {
159
-                                            "type": "integer",
160
-                                            "description": "Maximum bitrate in kbps."
161
-                                        }
162
-                                    },
163
-                                    "minItems": 1,
164
-                                    "additionalProperties": false,
165
-                                    "required": 
166
-                                        "res",
167
-                                        "fps",
168
-                                        "max bitrate"
169
-                                    
170
-                                },
171
-                                "default": 
172
-                                    {
173
-                                        "res": "",
174
-                                        "fps": "",
175
-                                        "max bitrate": ""
176
-                                    }
177
-                                ,
178
-                                "additionalItems": true
179
-                            }
180
-                        },
181
-                        "additionalProperties": false
182
-                    },
183
-                    "more_info_link": {
184
-                        "$ref": "#/definitions/saneUrl",
185
-                        "description": "Link that provides additional info about the service, presented in the UI as a button next to the services dropdown."
186
-                    },
187
-                    "alt_names": {
188
-                        "type": "array",
189
-                        "description": "Previous names of the service used for migrating existing users to the updated entry.",
190
-                        "items": {
191
-                            "type": "string",
192
-                            "minLength": 1
193
-                        },
194
-                        "default": 
195
-                            ""
196
-                        
197
-                    }
198
-                },
199
-                "additionalProperties": false,
200
-                "required": 
201
obs-studio-28.1.2.tar.xz/.github/workflows/docs.yml -> obs-studio-29.0.0.tar.xz/.github/workflows/docs.yml Changed
21
 
1
@@ -14,6 +14,10 @@
2
     steps:
3
       - name: Checkout
4
         uses: actions/checkout@v3
5
+      - name: Setup Environment
6
+        id: setup
7
+        run: |
8
+          echo "commitHash=$(git describe --exact-match --tags 2> /dev/null || git rev-parse --short HEAD)" >> $GITHUB_OUTPUT
9
       - name: Bump Version Number
10
         shell: bash
11
         if: github.event_name == 'push'
12
@@ -45,7 +49,7 @@
13
           pre_build_commands: 'pip install -Iv sphinx==5.1.1'
14
       - uses: actions/upload-artifact@v3
15
         with:
16
-          name: OBS Studio Documentation (HTML)
17
+          name: 'OBS Studio Docs ${{ steps.setup.outputs.commitHash }}'
18
           path: |
19
             ${{ runner.temp }}/_github_home/_build
20
             !${{ runner.temp }}/_github_home/_build/.doctrees
21
obs-studio-28.1.2.tar.xz/.github/workflows/flatpak.yml -> obs-studio-29.0.0.tar.xz/.github/workflows/flatpak.yml Changed
37
 
1
@@ -50,7 +50,7 @@
2
     env:
3
       FLATPAK_BUILD_PATH: flatpak_app/files/share
4
     container:
5
-      image: bilelmoussaoui/flatpak-github-actions:kde-6.3
6
+      image: bilelmoussaoui/flatpak-github-actions:kde-6.4
7
       options: --privileged
8
     strategy:
9
       matrix:
10
@@ -68,7 +68,7 @@
11
           echo "commitHash=$(git rev-parse --short=9 HEAD)" >> $GITHUB_OUTPUT
12
 
13
       - name: Build Flatpak Manifest
14
-        uses: bilelmoussaoui/flatpak-github-actions/flatpak-builder@master
15
+        uses: flatpak/flatpak-github-actions/flatpak-builder@v5
16
         with:
17
           bundle: obs-studio-${{ steps.setup.outputs.commitHash }}.flatpak
18
           manifest-path: CI/flatpak/com.obsproject.Studio.json
19
@@ -94,7 +94,7 @@
20
           ostree commit --repo=repo --canonical-permissions --branch=screenshots/x86_64 flatpak_app/screenshots
21
 
22
       - name: Publish to Flathub Beta
23
-        uses: bilelmoussaoui/flatpak-github-actions/flat-manager@v4
24
+        uses: flatpak/flatpak-github-actions/flat-manager@v5
25
         if: matrix.branch == 'beta'
26
         with:
27
           flat-manager-url: https://hub.flathub.org/
28
@@ -102,7 +102,7 @@
29
           token: ${{ secrets.FLATHUB_BETA_TOKEN }}
30
 
31
       - name: Publish to Flathub
32
-        uses: bilelmoussaoui/flatpak-github-actions/flat-manager@v4
33
+        uses: flatpak/flatpak-github-actions/flat-manager@v5
34
         if: matrix.branch == 'stable'
35
         with:
36
           flat-manager-url: https://hub.flathub.org/
37
obs-studio-28.1.2.tar.xz/.github/workflows/main.yml -> obs-studio-29.0.0.tar.xz/.github/workflows/main.yml Changed
92
 
1
@@ -18,15 +18,15 @@
2
   CEF_HASH_MAC_ARM64: '98679b92eea6ea9959ac5aa54f46ca60681d8a86c768c35f496dbdd409bf0642'
3
   CEF_BUILD_VERSION_LINUX: '5060'
4
   CEF_BUILD_VERSION_WIN: '5060'
5
-  QT_VERSION_MAC: '6.3.1'
6
-  QT_HASH_MAC_X86_64: 'a83f72a11023b03b6cb2dc365f0a66ad9df31163bbb4fe2df32d601856a9fad3'
7
-  QT_HASH_MAC_ARM64: '2f30af90c049670a5660656adbb440668aa1b0567f75a5f29e1def9108928403'
8
-  QT_HASH_MAC_UNIVERSAL: '252e6684f43ab9c6f262c73af739e2296ce391b998da2c4ee04c254aaa07db18'
9
-  QT_VERSION_WIN: '6.3.1'
10
-  DEPS_VERSION_MAC: '2022-08-02'
11
-  DEPS_HASH_MAC_X86_64: '7637e52305e6fc53014b5aabd583f1a4490b1d97450420e977cae9a336a29525'
12
-  DEPS_HASH_MAC_ARM64: '755e0fa69b17a3ae444e1befa9d91d77e3cafe628fbd1c6333686091826595cd'
13
-  DEPS_VERSION_WIN: '2022-08-02'
14
+  QT_VERSION_MAC: '6.4.1'
15
+  QT_HASH_MAC_X86_64: 'c5ed7bc9f6e802910ec539066bcf0a8d64100fafce568071f264c88c22c5859b'
16
+  QT_HASH_MAC_ARM64: '1ce472fd1e28f947456b72b1d7ab929d6e93cb774c2928e22eca9bb751b12ccf'
17
+  QT_HASH_MAC_UNIVERSAL: '873f7c9c9f7fcee740a79c075b32a505c932c816d928807fa16f3439c610fbfd'
18
+  QT_VERSION_WIN: '6.4.1'
19
+  DEPS_VERSION_MAC: '2022-11-21'
20
+  DEPS_HASH_MAC_X86_64: 'ed0a145e88496f8975da14a07939dbe5633e60510aada34509a4aef64a66e438'
21
+  DEPS_HASH_MAC_ARM64: 'f397dc524e5ee7f85684f0b9661c45957446e28d166fcd6dfacf895c9d4d2521'
22
+  DEPS_VERSION_WIN: '2022-11-21'
23
   VLC_VERSION_MAC: '3.0.8'
24
   VLC_HASH_MAC: 'e0149ef4a20a19b9ecd87309c2d27787ee3f47dfd47c6639644bc1f6fd95bdf6'
25
   VLC_VERSION_WIN: '3.0.0-git'
26
@@ -273,6 +273,7 @@
27
           CI/linux/03_package_obs.sh
28
           ARTIFACT_NAME=$(basename $(/usr/bin/find build  -maxdepth 1 -type f -name "obs-studio-*.deb" | sort -rn | head -1))
29
           echo "FILE_NAME=${ARTIFACT_NAME}" >> $GITHUB_ENV
30
+          echo "DEBUG_FILE_NAME=${ARTIFACT_NAME//.deb/-dbgsym.ddeb}" >> $GITHUB_ENV
31
 
32
       - name: 'Upload build Artifact'
33
         if: ${{ success() && (github.event_name != 'pull_request' || env.SEEKING_TESTERS == '1') }}
34
@@ -281,6 +282,13 @@
35
           name: 'obs-studio-${{ matrix.ubuntu }}-${{ steps.setup.outputs.commitHash }}'
36
           path: '${{ github.workspace }}/obs-studio/build/${{ env.FILE_NAME }}'
37
 
38
+      - name: 'Upload debug symbol Artifact'
39
+        if: ${{ success() && (github.event_name != 'pull_request' || env.SEEKING_TESTERS == '1') }}
40
+        uses: actions/upload-artifact@v3
41
+        with:
42
+          name: 'obs-studio-${{ matrix.ubuntu }}-${{ steps.setup.outputs.commitHash }}-dbgsym'
43
+          path: '${{ github.workspace }}/obs-studio/build/${{ env.DEBUG_FILE_NAME }}'
44
+
45
   windows_build:
46
     name: '02 - Windows'
47
     runs-on: windows-2022
48
@@ -362,7 +370,7 @@
49
           path: '${{ env.FILE_NAME }}'
50
 
51
   linux_package:
52
-    name: '02 - Flatpak Bundle'
53
+    name: '02 - Flatpak'
54
     runs-on: ubuntu-latest
55
     needs: clang_check
56
     if: always()
57
@@ -370,7 +378,7 @@
58
       run:
59
         shell: bash
60
     container:
61
-      image: bilelmoussaoui/flatpak-github-actions:kde-6.3
62
+      image: bilelmoussaoui/flatpak-github-actions:kde-6.4
63
       options: --privileged
64
     steps:
65
       - name: 'Check for Github Labels'
66
@@ -385,13 +393,11 @@
67
 
68
       - name: 'Checkout'
69
         uses: actions/checkout@v3
70
-        if: ${{ success() && (github.event_name != 'pull_request' || env.SEEKING_TESTERS == '1') }}
71
         with:
72
           submodules: 'recursive'
73
           fetch-depth: 0
74
 
75
       - name: 'Setup build environment'
76
-        if: ${{ success() && (github.event_name != 'pull_request' || env.SEEKING_TESTERS == '1') }}
77
         run: |
78
           git config --global --add safe.directory $GITHUB_WORKSPACE
79
           echo "OBS_GIT_BRANCH=$(git rev-parse --abbrev-ref HEAD)" >> $GITHUB_ENV
80
@@ -399,9 +405,9 @@
81
           echo "OBS_GIT_TAG=$(git describe --tags --abbrev=0)" >> $GITHUB_ENV
82
 
83
       - name: Build Flatpak Manifest
84
-        uses: bilelmoussaoui/flatpak-github-actions/flatpak-builder@v4
85
-        if: ${{ success() && (github.event_name != 'pull_request' || env.SEEKING_TESTERS == '1') }}
86
+        uses: flatpak/flatpak-github-actions/flatpak-builder@v5
87
         with:
88
+          build-bundle: ${{ github.event_name != 'pull_request' || env.SEEKING_TESTERS == '1' }}
89
           bundle: obs-studio-flatpak-${{ env.OBS_GIT_HASH }}.flatpak
90
           manifest-path: CI/flatpak/com.obsproject.Studio.json
91
           cache-key: flatpak-builder-${{ hashFiles('CI/flatpak/com.obsproject.Studio.json') }}
92
obs-studio-28.1.2.tar.xz/.github/workflows/services-json.yml -> obs-studio-29.0.0.tar.xz/.github/workflows/services-json.yml Changed
18
 
1
@@ -17,6 +17,7 @@
2
   schema:
3
     name: Schema
4
     runs-on: ubuntu-20.04
5
+    if: ${{ github.repository_owner == 'obsproject' || github.event_name != 'schedule' }}
6
     steps:
7
       - name: Checkout
8
         uses: actions/checkout@v3
9
@@ -46,7 +47,7 @@
10
     name: Service Check
11
     runs-on: ubuntu-20.04
12
     needs: schema
13
-    if: ${{ github.repository_owner == 'obsproject' && (github.event_name == 'schedule' || github.event_name == 'workflow_dispatch') }}
14
+    if: ${{ github.repository_owner == 'obsproject' && github.event_name == 'schedule' || github.event_name == 'workflow_dispatch' }}
15
 
16
     steps:
17
       - name: Checkout
18
obs-studio-28.1.2.tar.xz/.github/workflows/steam.yml -> obs-studio-29.0.0.tar.xz/.github/workflows/steam.yml Changed
51
 
1
@@ -27,6 +27,7 @@
2
   STEAM_NIGHTLY_BRANCH: nightly
3
   STEAM_STABLE_BRANCH: staging
4
   STEAM_BETA_BRANCH: beta_staging
5
+  STEAM_PLAYTEST_BRANCH: staging
6
   SEVENZIP_HASH: 5290409e7bbe2f133d0bd7e7482548678157ea2be276b0f9cb440600f4be9a2d
7
 
8
 jobs:
9
@@ -145,8 +146,14 @@
10
       run: |
11
         if  '${{ steps.build-info.outputs.type }}' == 'release' || '${{ steps.cache.outputs.cache-hit }}' != 'true' ; then
12
             echo "result=true" >> $GITHUB_OUTPUT
13
+            if  '${{ steps.build-info.outputs.branch }}' == '${{ env.STEAM_BETA_BRANCH }}' ; then
14
+                echo "result_playtest=true" >> $GITHUB_OUTPUT
15
+            else
16
+                echo "result_playtest=false" >> $GITHUB_OUTPUT
17
+            fi
18
         else
19
             echo "result=false" >> $GITHUB_OUTPUT
20
+            echo "result_playtest=false" >> $GITHUB_OUTPUT
21
         fi
22
 
23
     - name: Download and prepare builds
24
@@ -256,6 +263,26 @@
25
         steamcmd +login '${{ secrets.STEAM_USER }}' '${{ secrets.STEAM_PASSWORD }}' '${{ steps.steam-totp.outputs.code }}' +run_app_build "$(pwd)/build.vdf" +quit
26
         echo "::endgroup::"
27
 
28
+    - name: Generate Steam auth code (Playtest)
29
+      if: steps.should-run.outputs.result_playtest == 'true'
30
+      id: steam-totp-playtest
31
+      uses: CyberAndrii/steam-totp@0fc9e59dc5bbf4368d23d5a33956f104248da31a
32
+      with:
33
+        shared_secret: ${{ secrets.STEAM_SHARED_SECRET }}
34
+
35
+    - name: Upload to Steam (Playtest)
36
+      if: steps.should-run.outputs.result_playtest == 'true'
37
+      run: |
38
+        cd steam
39
+        echo "::group::Prepare Steam build script"
40
+        sed 's/@@DESC@@/${{ steps.build-info.outputs.branch }}-${{ steps.build-info.outputs.desc }}/;s/@@BRANCH@@/${{ env.STEAM_PLAYTEST_BRANCH }}/' ../source/CI/steam/obs_playtest_build.vdf > build_playtest.vdf
41
+        echo "Generated file:"
42
+        cat build_playtest.vdf
43
+        echo "::endgroup::"
44
+        echo "::group::Upload to Steam"
45
+        steamcmd +login '${{ secrets.STEAM_USER }}' '${{ secrets.STEAM_PASSWORD }}' '${{ steps.steam-totp-playtest.outputs.code }}' +run_app_build "$(pwd)/build_playtest.vdf" +quit
46
+        echo "::endgroup::"
47
+
48
     - name: Upload Steam build logs
49
       if: steps.should-run.outputs.result == 'true'
50
       uses: actions/upload-artifact@v3
51
obs-studio-28.1.2.tar.xz/CI/check-services.py -> obs-studio-29.0.0.tar.xz/CI/check-services.py Changed
16
 
1
@@ -87,7 +87,13 @@
2
     """Try connecting and sending a RTMP handshake (with SSL if necessary)"""
3
     parsed = urlparse(uri)
4
     hostname, port = parsed.netloc.partition(':')::2
5
-    port = int(port) if port else 1935
6
+
7
+    if port:
8
+        port = int(port)
9
+    elif parsed.scheme == 'rtmps':
10
+        port = 443
11
+    else:
12
+        port = 1935
13
 
14
     try:
15
         recv = b''
16
obs-studio-28.1.2.tar.xz/CI/flatpak/com.obsproject.Studio.json -> obs-studio-29.0.0.tar.xz/CI/flatpak/com.obsproject.Studio.json Changed
187
 
1
@@ -1,7 +1,7 @@
2
 {
3
   "app-id": "com.obsproject.Studio",
4
   "runtime": "org.kde.Platform",
5
-  "runtime-version": "6.3",
6
+  "runtime-version": "6.4",
7
   "sdk": "org.kde.Sdk",
8
   "command": "obs",
9
   "finish-args": 
10
@@ -19,7 +19,7 @@
11
     "--talk-name=org.a11y.Bus",
12
     "--own-name=org.kde.StatusNotifierItem-2-2",
13
     "--system-talk-name=org.freedesktop.Avahi",
14
-    "--env=VST_PATH=/app/extensions/Plugins/lxvst"
15
+    "--env=VST_PATH=/app/extensions/Plugins/vst"
16
   ,
17
   "add-extensions": {
18
     "com.obsproject.Studio.Plugin": {
19
@@ -32,9 +32,9 @@
20
     },
21
     "org.freedesktop.LinuxAudio.Plugins": {
22
       "directory": "extensions/Plugins",
23
-      "version": "21.08",
24
+      "version": "22.08",
25
       "add-ld-path": "lib",
26
-      "merge-dirs": "lxvst",
27
+      "merge-dirs": "vst",
28
       "subdirectories": true,
29
       "no-autodownload": true
30
     }
31
@@ -58,7 +58,7 @@
32
         {
33
           "type": "git",
34
           "url": "https://code.videolan.org/videolan/x264.git",
35
-          "commit": "baee400fa9ced6f5481a728138fed6e867b0ff7f"
36
+          "commit": "b093bbe7d9bc642c8f24067cbdcc73bb43562eab"
37
         }
38
       
39
     },
40
@@ -119,8 +119,8 @@
41
         {
42
           "type": "git",
43
           "url": "https://github.com/Haivision/srt.git",
44
-          "tag": "v1.5.0",
45
-          "commit": "060c0d1a2af8082c053018fa6a6fe2dba67d71a8"
46
+          "tag": "v1.5.1",
47
+          "commit": "0bc3b03202b3159fc9b085b3ae6d66ec071c25d6"
48
         }
49
       
50
     },
51
@@ -191,8 +191,8 @@
52
         {
53
           "type": "git",
54
           "url": "https://aomedia.googlesource.com/aom.git",
55
-          "commit": "fc430c57c7b0307b4c5ffb686cd90b3c010d08d2",
56
-          "tag": "v3.4.0"
57
+          "commit": "bcfe6fbfed315f83ee8a95465c654ee8078dbff9",
58
+          "tag": "v3.5.0"
59
         }
60
       
61
     },
62
@@ -216,8 +216,8 @@
63
         {
64
           "type": "git",
65
           "url": "https://gitlab.com/AOMediaCodec/SVT-AV1.git",
66
-          "tag": "v1.1.0",
67
-          "commit": "6e87a1de98281840abebc030781780edd822bae5"
68
+          "tag": "v1.3.0",
69
+          "commit": "91b94efb2809e83d9bf041d8575b32f234dfef27"
70
         }
71
       
72
     },
73
@@ -248,22 +248,20 @@
74
         {
75
           "type": "git",
76
           "url": "https://github.com/FFmpeg/FFmpeg.git",
77
-          "commit": "9687cae2b468e09e35df4cea92cc2e6a0e6c93b3",
78
+          "commit": "eacfcbae690f914a4b1b4ad06999f138540cc3d8",
79
           "disable-shallow-clone": true
80
         },
81
         {
82
           "type": "git",
83
           "dest": "obs-deps",
84
           "url": "https://github.com/obsproject/obs-deps.git",
85
-          "tag": "2022-07-29",
86
-          "commit": "71df2dddb0eb732496448f626553b61bcda8f993"
87
+          "tag": "2022-11-21",
88
+          "commit": "ba56d64e4cbcc0aec4fa0efcef5f5a1e6a1ba91d"
89
         },
90
         {
91
           "type": "shell",
92
           "commands": 
93
-            "patch -Np1 -i obs-deps/deps.ffmpeg/patches/FFmpeg/0001-FFmpeg-9010.patch",
94
-            "patch -Np1 -i obs-deps/deps.ffmpeg/patches/FFmpeg/0002-FFmpeg-5.0.1-OBS.patch",
95
-            "patch -Np1 -i obs-deps/deps.ffmpeg/patches/FFmpeg/0003-FFmpeg-5.0.1-librist-7f3f3539e8.patch"
96
+            "patch -Np1 -i obs-deps/deps.ffmpeg/patches/FFmpeg/0001-FFmpeg-5.1.2-OBS.patch"
97
           
98
         }
99
       
100
@@ -281,7 +279,7 @@
101
         {
102
           "type": "git",
103
           "url": "https://luajit.org/git/luajit-2.0.git",
104
-          "commit": "3065c910ad6027031aabe2dfd3c26a3d0f014b4f",
105
+          "commit": "7a0cf5fd4c6c841d0455a51271af4fd4390c7884",
106
           "disable-shallow-clone": true
107
         },
108
         {
109
@@ -333,56 +331,47 @@
110
       
111
     },
112
     {
113
-      "name": "pipewire",
114
-      "buildsystem": "meson",
115
+      "name": "ntv2",
116
+      "buildsystem": "cmake-ninja",
117
+      "builddir": true,
118
       "config-opts": 
119
-        "-Daudiotestsrc=disabled",
120
-        "-Droc=disabled",
121
-        "-Dvideotestsrc=disabled",
122
-        "-Dvolume=disabled",
123
-        "-Dvulkan=disabled",
124
-        "-Ddocs=disabled",
125
-        "-Dman=disabled",
126
-        "-Dbluez5-codec-ldac=disabled",
127
-        "-Dbluez5-codec-aptx=disabled",
128
-        "-Dlibcamera=disabled",
129
-        "-Dudevrulesdir=/app/lib/udev/rules.d/",
130
-        "-Dsession-managers=",
131
-        "-Dtests=disabled",
132
-        "-Dexamples=disabled",
133
-        "-Dpw-cat=disabled"
134
+        "-DCMAKE_BUILD_TYPE=Release",
135
+        "-DAJA_BUILD_OPENSOURCE=ON",
136
+        "-DAJA_BUILD_APPS=OFF",
137
+        "-DAJA_INSTALL_HEADERS=ON"
138
       ,
139
       "cleanup": 
140
-        "/bin"
141
+        "/include"
142
       ,
143
       "sources": 
144
         {
145
           "type": "git",
146
-          "url": "https://github.com/pipewire/pipewire.git",
147
-          "tag": "0.3.40",
148
-          "commit": "7afd80052b7c49754a13c9ab49c368f95b60e0a7"
149
+          "url": "https://github.com/aja-video/ntv2.git",
150
+          "tag": "v16.2-bugfix5",
151
+          "commit": "0acbac70a0b5e6509cca78cfbf69974c73c10db9"
152
         }
153
       
154
     },
155
     {
156
-      "name": "ntv2",
157
+      "name": "jansson",
158
       "buildsystem": "cmake-ninja",
159
       "builddir": true,
160
       "config-opts": 
161
-        "-DCMAKE_BUILD_TYPE=Release",
162
-        "-DAJA_BUILD_OPENSOURCE=ON",
163
-        "-DAJA_BUILD_APPS=OFF",
164
-        "-DAJA_INSTALL_HEADERS=ON"
165
+        "-DJANSSON_EXAMPLES=OFF",
166
+        "-DJANSSON_BUILD_DOCS=OFF",
167
+        "-DJANSSON_BUILD_SHARED_LIBS=ON",
168
+        "-DJANSSON_WITHOUT_TESTS=ON"
169
       ,
170
       "cleanup": 
171
-        "/include"
172
+        "/include",
173
+        "/lib/pkgconfig"
174
       ,
175
       "sources": 
176
         {
177
           "type": "git",
178
-          "url": "https://github.com/aja-video/ntv2.git",
179
-          "tag": "v16.2-bugfix5",
180
-          "commit": "0acbac70a0b5e6509cca78cfbf69974c73c10db9"
181
+          "url": "https://github.com/akheron/jansson.git",
182
+          "tag": "v2.14",
183
+          "commit": "684e18c927e89615c2d501737e90018f4930d6c5"
184
         }
185
       
186
     },
187
obs-studio-28.1.2.tar.xz/CI/linux/03_package_obs.sh -> obs-studio-29.0.0.tar.xz/CI/linux/03_package_obs.sh Changed
17
 
1
@@ -21,9 +21,14 @@
2
     cmake --build ${BUILD_DIR} -t package
3
 
4
     DEB_NAME=$(find ${BUILD_DIR} -maxdepth 1 -type f -name "obs*.deb" | sort -rn | head -1)
5
+    DEBUG_NAME="${DEB_NAME//.deb/-dbgsym.ddeb}"
6
 
7
     if  "${DEB_NAME}" ; then
8
-        mv ${DEB_NAME} ${BUILD_DIR}/${FILE_NAME}
9
+        mv "${DEB_NAME}" "${BUILD_DIR}/${FILE_NAME}"
10
+
11
+        if  "${DEBUG_NAME}" ; then
12
+            mv "${DEBUG_NAME}" "${BUILD_DIR}/${FILE_NAME//.deb/-dbgsym.ddeb}"
13
+        fi
14
     else
15
         error "ERROR No suitable OBS debian package generated"
16
     fi
17
obs-studio-29.0.0.tar.xz/CI/steam/obs_playtest_build.vdf Added
37
 
1
@@ -0,0 +1,35 @@
2
+"AppBuild"
3
+{
4
+   "AppID" "1905640"
5
+   "Desc" "github_@@DESC@@"
6
+
7
+   "ContentRoot" "./"
8
+   "BuildOutput" "build/"
9
+   
10
+   "SetLive" "@@BRANCH@@"
11
+
12
+   "Depots"
13
+   {
14
+       "1905642" // Windows
15
+       {
16
+           "ContentRoot" "./steam-windows"
17
+           "FileMapping"
18
+           {
19
+               "LocalPath" "*"
20
+               "DepotPath" "."
21
+               "recursive" "1"
22
+           }
23
+       }
24
+       
25
+       "1905641" // Mac
26
+       {
27
+           "ContentRoot" "./steam-macos"
28
+           "FileMapping"
29
+           {
30
+               "LocalPath" "*"
31
+               "DepotPath" "."
32
+               "recursive" "1"
33
+           }
34
+       }
35
+   }
36
+}
37
obs-studio-28.1.2.tar.xz/CI/windows/02_build_obs.ps1 -> obs-studio-29.0.0.tar.xz/CI/windows/02_build_obs.ps1 Changed
86
 
1
@@ -63,31 +63,59 @@
2
     $BuildDirectoryActual = "${BuildDirectory}$(if (${BuildArch} -eq "x64") { "64" } else { "32" })"
3
     $GeneratorPlatform = "$(if (${BuildArch} -eq "x64") { "x64" } else { "Win32" })"
4
 
5
-    $CmakeCommand = @(
6
-        "-G", ${CmakeGenerator}
7
-        "-DCMAKE_GENERATOR_PLATFORM=`"${GeneratorPlatform}`"",
8
-        "-DCMAKE_SYSTEM_VERSION=`"${CmakeSystemVersion}`"",
9
-        "-DCMAKE_PREFIX_PATH:PATH=`"${CmakePrefixPath}`"",
10
-        "-DCEF_ROOT_DIR:PATH=`"${CefDirectory}`"",
11
-        "-DENABLE_BROWSER=ON",
12
-        "-DVLC_PATH:PATH=`"${CheckoutDir}/../obs-build-dependencies/vlc-${WindowsVlcVersion}`"",
13
-        "-DENABLE_VLC=ON",
14
-        "-DCMAKE_INSTALL_PREFIX=`"${BuildDirectoryActual}/install`"",
15
-        "-DVIRTUALCAM_GUID=`"${Env:VIRTUALCAM-GUID}`"",
16
-        "-DTWITCH_CLIENTID=`"${Env:TWITCH_CLIENTID}`"",
17
-        "-DTWITCH_HASH=`"${Env:TWITCH_HASH}`"",
18
-        "-DRESTREAM_CLIENTID=`"${Env:RESTREAM_CLIENTID}`"",
19
-        "-DRESTREAM_HASH=`"${Env:RESTREAM_HASH}`"",
20
-        "-DYOUTUBE_CLIENTID=`"${Env:YOUTUBE_CLIENTID}`"",
21
-        "-DYOUTUBE_CLIENTID_HASH=`"${Env:YOUTUBE_CLIENTID_HASH}`"",
22
-        "-DYOUTUBE_SECRET=`"${Env:YOUTUBE_SECRET}`"",
23
-        "-DYOUTUBE_SECRET_HASH=`"${Env:YOUTUBE_SECRET_HASH}`"",
24
-        "-DCOPIED_DEPENDENCIES=OFF",
25
-        "-DCOPY_DEPENDENCIES=ON",
26
-        "-DBUILD_FOR_DISTRIBUTION=`"$(if (Test-Path Env:BUILD_FOR_DISTRIBUTION) { "ON" } else { "OFF" })`"",
27
-        "$(if (Test-Path Env:CI) { "-DOBS_BUILD_NUMBER=${Env:GITHUB_RUN_ID}" })",
28
-        "$(if (Test-Path Variable:$Quiet) { "-Wno-deprecated -Wno-dev --log-level=ERROR" })"
29
-    )
30
+    if ( $PSVersionTable.PSVersion -ge '7.3.0' ) {
31
+        $CmakeCommand = @(
32
+            "-G", ${CmakeGenerator}
33
+            "-DCMAKE_GENERATOR_PLATFORM=${GeneratorPlatform}",
34
+            "-DCMAKE_SYSTEM_VERSION=${CmakeSystemVersion}",
35
+            "-DCMAKE_PREFIX_PATH:PATH=${CmakePrefixPath}",
36
+            "-DCEF_ROOT_DIR:PATH=${CefDirectory}",
37
+            "-DENABLE_BROWSER=ON",
38
+            "-DVLC_PATH:PATH=${CheckoutDir}/../obs-build-dependencies/vlc-${WindowsVlcVersion}",
39
+            "-DENABLE_VLC=ON",
40
+            "-DCMAKE_INSTALL_PREFIX=${BuildDirectoryActual}/install",
41
+            "-DVIRTUALCAM_GUID=${Env:VIRTUALCAM-GUID}",
42
+            "-DTWITCH_CLIENTID=${Env:TWITCH_CLIENTID}",
43
+            "-DTWITCH_HASH=${Env:TWITCH_HASH}",
44
+            "-DRESTREAM_CLIENTID=${Env:RESTREAM_CLIENTID}",
45
+            "-DRESTREAM_HASH=${Env:RESTREAM_HASH}",
46
+            "-DYOUTUBE_CLIENTID=${Env:YOUTUBE_CLIENTID}",
47
+            "-DYOUTUBE_CLIENTID_HASH=${Env:YOUTUBE_CLIENTID_HASH}",
48
+            "-DYOUTUBE_SECRET=${Env:YOUTUBE_SECRET}",
49
+            "-DYOUTUBE_SECRET_HASH=${Env:YOUTUBE_SECRET_HASH}",
50
+            "-DCOPIED_DEPENDENCIES=OFF",
51
+            "-DCOPY_DEPENDENCIES=ON",
52
+            "-DBUILD_FOR_DISTRIBUTION=$(if (Test-Path Env:BUILD_FOR_DISTRIBUTION) { "ON" } else { "OFF" })",
53
+            "$(if (Test-Path Env:CI) { "-DOBS_BUILD_NUMBER=${Env:GITHUB_RUN_ID}" })",
54
+            "$(if (Test-Path Variable:$Quiet) { "-Wno-deprecated -Wno-dev --log-level=ERROR" })"
55
+        )
56
+    } else {
57
+        $CmakeCommand = @(
58
+            "-G", ${CmakeGenerator}
59
+            "-DCMAKE_GENERATOR_PLATFORM=`"${GeneratorPlatform}`"",
60
+            "-DCMAKE_SYSTEM_VERSION=`"${CmakeSystemVersion}`"",
61
+            "-DCMAKE_PREFIX_PATH:PATH=`"${CmakePrefixPath}`"",
62
+            "-DCEF_ROOT_DIR:PATH=`"${CefDirectory}`"",
63
+            "-DENABLE_BROWSER=ON",
64
+            "-DVLC_PATH:PATH=`"${CheckoutDir}/../obs-build-dependencies/vlc-${WindowsVlcVersion}`"",
65
+            "-DENABLE_VLC=ON",
66
+            "-DCMAKE_INSTALL_PREFIX=`"${BuildDirectoryActual}/install`"",
67
+            "-DVIRTUALCAM_GUID=`"${Env:VIRTUALCAM-GUID}`"",
68
+            "-DTWITCH_CLIENTID=`"${Env:TWITCH_CLIENTID}`"",
69
+            "-DTWITCH_HASH=`"${Env:TWITCH_HASH}`"",
70
+            "-DRESTREAM_CLIENTID=`"${Env:RESTREAM_CLIENTID}`"",
71
+            "-DRESTREAM_HASH=`"${Env:RESTREAM_HASH}`"",
72
+            "-DYOUTUBE_CLIENTID=`"${Env:YOUTUBE_CLIENTID}`"",
73
+            "-DYOUTUBE_CLIENTID_HASH=`"${Env:YOUTUBE_CLIENTID_HASH}`"",
74
+            "-DYOUTUBE_SECRET=`"${Env:YOUTUBE_SECRET}`"",
75
+            "-DYOUTUBE_SECRET_HASH=`"${Env:YOUTUBE_SECRET_HASH}`"",
76
+            "-DCOPIED_DEPENDENCIES=OFF",
77
+            "-DCOPY_DEPENDENCIES=ON",
78
+            "-DBUILD_FOR_DISTRIBUTION=`"$(if (Test-Path Env:BUILD_FOR_DISTRIBUTION) { "ON" } else { "OFF" })`"",
79
+            "$(if (Test-Path Env:CI) { "-DOBS_BUILD_NUMBER=${Env:GITHUB_RUN_ID}" })",
80
+            "$(if (Test-Path Variable:$Quiet) { "-Wno-deprecated -Wno-dev --log-level=ERROR" })"
81
+        )
82
+    }
83
 
84
     Invoke-External cmake -S . -B  "${BuildDirectoryActual}" @CmakeCommand
85
 
86
obs-studio-28.1.2.tar.xz/UI/api-interface.cpp -> obs-studio-29.0.0.tar.xz/UI/api-interface.cpp Changed
28
 
1
@@ -642,6 +642,26 @@
2
        return Str(string);
3
    }
4
 
5
+   bool obs_frontend_is_theme_dark(void) override
6
+   {
7
+       return App()->IsThemeDark();
8
+   }
9
+
10
+   char *obs_frontend_get_last_recording(void) override
11
+   {
12
+       return bstrdup(main->outputHandler->lastRecordingPath.c_str());
13
+   }
14
+
15
+   char *obs_frontend_get_last_screenshot(void) override
16
+   {
17
+       return bstrdup(main->lastScreenshot.c_str());
18
+   }
19
+
20
+   char *obs_frontend_get_last_replay(void) override
21
+   {
22
+       return bstrdup(main->lastReplay.c_str());
23
+   }
24
+
25
    void on_load(obs_data_t *settings) override
26
    {
27
        for (size_t i = saveCallbacks.size(); i > 0; i--) {
28
obs-studio-28.1.2.tar.xz/UI/context-bar-controls.cpp -> obs-studio-29.0.0.tar.xz/UI/context-bar-controls.cpp Changed
15
 
1
@@ -311,10 +311,12 @@
2
    const char *device_str =
3
        get_os_text(mod, "Monitor", "DisplayCapture.Display", "Screen");
4
    ui->deviceLabel->setText(device_str);
5
+#ifndef _WIN32
6
    is_int = true;
7
+#endif
8
 
9
 #ifdef _WIN32
10
-   prop_name = "monitor";
11
+   prop_name = "monitor_id";
12
 #elif __APPLE__
13
    prop_name = "display";
14
 #else
15
obs-studio-28.1.2.tar.xz/UI/data/locale/en-US.ini -> obs-studio-29.0.0.tar.xz/UI/data/locale/en-US.ini Changed
124
 
1
@@ -113,6 +113,7 @@
2
 SourceProperties="Open Source Properties"
3
 SourceFilters="Open Source Filters"
4
 MixerToolbarMenu="Audio Mixer Menu"
5
+SceneFilters="Open Scene Filters"
6
 
7
 # warning for plugin load failures
8
 PluginsFailedToLoad.Title="Plugin Load Error"
9
@@ -204,6 +205,7 @@
10
 Basic.AutoConfig.StreamPage.Service.Custom="Custom..."
11
 Basic.AutoConfig.StreamPage.Server="Server"
12
 Basic.AutoConfig.StreamPage.StreamKey="Stream Key"
13
+Basic.AutoConfig.StreamPage.StreamKey.ToolTip="RIST: enter the encryption passphrase.\nRTMP: enter the key provided by the service.\nSRT: enter the streamid if the service uses one."
14
 Basic.AutoConfig.StreamPage.StreamKey.LinkToSite="(Link)"
15
 Basic.AutoConfig.StreamPage.EncoderKey="Encoder Key"
16
 Basic.AutoConfig.StreamPage.ConnectedAccount="Connected account"
17
@@ -266,6 +268,8 @@
18
 Updater.Running.Text="Outputs are currently active, please shut down any active outputs before attempting to update"
19
 Updater.NoUpdatesAvailable.Title="No updates available"
20
 Updater.NoUpdatesAvailable.Text="No updates are currently available"
21
+Updater.BranchNotFound.Title="Update Channel Removed"
22
+Updater.BranchNotFound.Text="Your selected update channel is no longer available, OBS has been reset to the default."
23
 Updater.RepairButUpdatesAvailable.Title="Integrity Check Unavailable"
24
 Updater.RepairButUpdatesAvailable.Text="Checking file integrity is only possible for the latest version available. Use Help → Check For Updates to verify and update your OBS installation."
25
 Updater.RepairConfirm.Title="Confirm Integrity Check"
26
@@ -833,6 +837,10 @@
27
 Basic.Settings.General="General"
28
 Basic.Settings.General.Theme="Theme"
29
 Basic.Settings.General.Language="Language"
30
+Basic.Settings.General.Updater="Updates"
31
+Basic.Settings.General.UpdateChannel="Update Channel"
32
+Basic.Settings.General.UpdateChannelDisabled="(Disabled)"
33
+Basic.Settings.General.UpdateChannelDefault="(Default)"
34
 Basic.Settings.General.EnableAutoUpdates="Automatically check for updates on startup"
35
 Basic.Settings.General.OpenStatsOnStartup="Open stats dialog on startup"
36
 Basic.Settings.General.HideOBSWindowsFromCapture="Hide OBS windows from screen capture"
37
@@ -884,12 +892,20 @@
38
 Basic.Settings.General.MultiviewLayout.16Scene="Scenes only (16 Scenes)"
39
 Basic.Settings.General.MultiviewLayout.25Scene="Scenes only (25 Scenes)"
40
 
41
+# default channel name translations
42
+Basic.Settings.General.ChannelName.stable="Stable"
43
+Basic.Settings.General.ChannelDescription.stable="Latest stable release"
44
+Basic.Settings.General.ChannelName.beta="Betas / Release Candidates"
45
+Basic.Settings.General.ChannelDescription.beta="Potentially unstable pre-release versions"
46
+
47
 # basic mode 'stream' settings
48
 Basic.Settings.Stream="Stream"
49
 Basic.Settings.Stream.StreamType="Stream Type"
50
 Basic.Settings.Stream.Custom.UseAuthentication="Use authentication"
51
 Basic.Settings.Stream.Custom.Username="Username"
52
 Basic.Settings.Stream.Custom.Password="Password"
53
+Basic.Settings.Stream.Custom.Username.ToolTip="RIST: enter the srp_username.\nRTMP: enter the username.\nSRT: not used."
54
+Basic.Settings.Stream.Custom.Password.ToolTip="RIST: enter the srp_password.\nRTMP: enter the password.\nSRT: enter the encryption passphrase."
55
 Basic.Settings.Stream.BandwidthTestMode="Enable Bandwidth Test Mode"
56
 Basic.Settings.Stream.TTVAddon="Twitch Chat Add-Ons"
57
 Basic.Settings.Stream.TTVAddon.None="None"
58
@@ -926,6 +942,7 @@
59
 Basic.Settings.Output.ReplayBuffer.SecondsMax="Maximum Replay Time"
60
 Basic.Settings.Output.ReplayBuffer.MegabytesMax="Maximum Memory (Megabytes)"
61
 Basic.Settings.Output.ReplayBuffer.Estimate="Estimated memory usage: %1 MB"
62
+Basic.Settings.Output.ReplayBuffer.EstimateTooLarge="Warning: Estimated memory usage of %1 MiB is larger than recommended maximum of %2 MiB"
63
 Basic.Settings.Output.ReplayBuffer.EstimateUnknown="Cannot estimate memory usage. Please set maximum memory limit."
64
 Basic.Settings.Output.ReplayBuffer.Prefix="Replay Buffer Filename Prefix"
65
 Basic.Settings.Output.ReplayBuffer.Suffix="Suffix"
66
@@ -944,12 +961,15 @@
67
 Basic.Settings.Output.Simple.Warn.Lossless.Title="Lossless quality warning!"
68
 Basic.Settings.Output.Simple.Encoder.Software="Software (x264)"
69
 Basic.Settings.Output.Simple.Encoder.Hardware.QSV.H264="Hardware (QSV, H.264)"
70
+Basic.Settings.Output.Simple.Encoder.Hardware.QSV.AV1="Hardware (QSV, AV1)"
71
 Basic.Settings.Output.Simple.Encoder.Hardware.AMD.H264="Hardware (AMD, H.264)"
72
 Basic.Settings.Output.Simple.Encoder.Hardware.AMD.HEVC="Hardware (AMD, HEVC)"
73
+Basic.Settings.Output.Simple.Encoder.Hardware.AMD.AV1="Hardware (AMD, AV1)"
74
 Basic.Settings.Output.Simple.Encoder.Hardware.NVENC.H264="Hardware (NVENC, H.264)"
75
 Basic.Settings.Output.Simple.Encoder.Hardware.NVENC.AV1="Hardware (NVENC, AV1)"
76
 Basic.Settings.Output.Simple.Encoder.Hardware.NVENC.HEVC="Hardware (NVENC, HEVC)"
77
 Basic.Settings.Output.Simple.Encoder.Hardware.Apple.H264="Hardware (Apple, H.264)"
78
+Basic.Settings.Output.Simple.Encoder.Hardware.Apple.HEVC="Hardware (Apple, HEVC)"
79
 Basic.Settings.Output.Simple.Encoder.SoftwareLowCPU="Software (x264 low CPU usage preset, increases file size)"
80
 Basic.Settings.Output.Simple.TwitchVodTrack="Twitch VOD Track (Uses Track 2)"
81
 Basic.Settings.Output.Warn.EnforceResolutionFPS.Title="Incompatible Resolution/Framerate"
82
@@ -1046,7 +1066,6 @@
83
 Basic.Settings.Video.ScaledResolution="Output (Scaled) Resolution"
84
 Basic.Settings.Video.DownscaleFilter="Downscale Filter"
85
 Basic.Settings.Video.DownscaleFilter.Unavailable="Resolutions match, no downscaling required"
86
-Basic.Settings.Video.DisableAeroWindows="Disable Aero (Windows only)"
87
 Basic.Settings.Video.FPS="FPS"
88
 Basic.Settings.Video.FPSCommon="Common FPS Values"
89
 Basic.Settings.Video.FPSInteger="Integer FPS Value"
90
@@ -1056,7 +1075,6 @@
91
 Basic.Settings.Video.Renderer="Renderer"
92
 Basic.Settings.Video.InvalidResolution="Invalid resolution value. Must be widthxheight (i.e. 1920x1080)"
93
 Basic.Settings.Video.CurrentlyActive="Video output is currently active. Please turn off any outputs to change video settings."
94
-Basic.Settings.Video.DisableAero="Disable Aero"
95
 
96
 # scale filters
97
 Basic.Settings.Video.DownscaleFilter.Bilinear="Bilinear (Fastest, but blurry if scaling)"
98
@@ -1135,7 +1153,7 @@
99
 Basic.Settings.Advanced.Video.ColorFormat.I444="I444 (8-bit, 4:4:4, 3 planes)"
100
 Basic.Settings.Advanced.Video.ColorFormat.P010="P010 (10-bit, 4:2:0, 2 planes)"
101
 Basic.Settings.Advanced.Video.ColorFormat.I010="I010 (10-bit, 4:2:0, 3 planes)"
102
-Basic.Settings.Advanced.Video.ColorFormat.RGB="RGB (8-bit)"
103
+Basic.Settings.Advanced.Video.ColorFormat.BGRA="BGRA (8-bit)"
104
 Basic.Settings.Advanced.Video.ColorSpace="Color Space"
105
 Basic.Settings.Advanced.Video.ColorSpace.sRGB="sRGB"
106
 Basic.Settings.Advanced.Video.ColorSpace.601="Rec. 601"
107
@@ -1164,7 +1182,7 @@
108
 Basic.Settings.Advanced.Hotkeys.NeverDisableHotkeys="Never disable hotkeys"
109
 Basic.Settings.Advanced.Hotkeys.DisableHotkeysInFocus="Disable hotkeys when main window is in focus"
110
 Basic.Settings.Advanced.Hotkeys.DisableHotkeysOutOfFocus="Disable hotkeys when main window is not in focus"
111
-Basic.Settings.Advanced.AutoRemux="Automatically remux to mp4"
112
+Basic.Settings.Advanced.AutoRemux="Automatically remux to %1"
113
 Basic.Settings.Advanced.AutoRemux.MP4="(record as mkv)"
114
 
115
 # advanced audio properties
116
@@ -1254,6 +1272,7 @@
117
 # Output warnings
118
 OutputWarnings.NoTracksSelected="You must select at least one track"
119
 OutputWarnings.MP4Recording="Warning: Recordings saved to MP4/MOV will be unrecoverable if the file cannot be finalized (e.g. as a result of BSODs, power losses, etc.). If you want to record multiple audio tracks consider using MKV and remux the recording to MP4/MOV after it is finished (File → Remux Recordings)"
120
+OutputWarnings.ProResRecording="Apple ProRes is not supported by the %1 container format - supported container formats are mov (preferred) and mkv."
121
 OutputWarnings.CannotPause="Warning: Recordings cannot be paused if the recording encoder is set to \"(Use stream encoder)\""
122
 
123
 # deleting final scene
124
obs-studio-28.1.2.tar.xz/UI/data/themes/Acri.qss -> obs-studio-29.0.0.tar.xz/UI/data/themes/Acri.qss Changed
30
 
1
@@ -524,6 +524,10 @@
2
     qproperty-icon: url(./Dark/media-pause.svg);
3
 }
4
 
5
+* themeID="filtersIcon" {
6
+    qproperty-icon: url(./Dark/filter.svg);
7
+}
8
+
9
 QToolBarExtension {
10
     background: palette(button);
11
     min-width: 12px;
12
@@ -819,7 +823,7 @@
13
     background-color: rgb(22,31,65);
14
 }
15
 
16
-QPushButton:disabled {
17
+QPushButton:disabled, QToolButton:disabled {
18
     background-color: rgb(22,31,65);
19
 }
20
 
21
@@ -1519,3 +1523,8 @@
22
 OBSBasicStats {
23
     background: palette(dark);
24
 }
25
+
26
+/* Advanced audio dialog */
27
+OBSBasicAdvAudio #scrollAreaWidgetContents {
28
+    background: palette(dark);
29
+}
30
obs-studio-28.1.2.tar.xz/UI/data/themes/Dark.qss -> obs-studio-29.0.0.tar.xz/UI/data/themes/Dark.qss Changed
21
 
1
@@ -328,6 +328,10 @@
2
     qproperty-icon: url(./Dark/cogs.svg);
3
 }
4
 
5
+* themeID="filtersIcon" {
6
+    qproperty-icon: url(./Dark/filter.svg);
7
+}
8
+
9
 /* Tab Widget */
10
 
11
 QTabWidget::pane { /* The tab widget frame */
12
@@ -530,7 +534,7 @@
13
     background-color: palette(base);
14
 }
15
 
16
-QPushButton:disabled {
17
+QPushButton:disabled, QToolButton:disabled {
18
     background-color: rgb(46,45,46);
19
 }
20
 
21
obs-studio-28.1.2.tar.xz/UI/data/themes/Grey.qss -> obs-studio-29.0.0.tar.xz/UI/data/themes/Grey.qss Changed
30
 
1
@@ -522,6 +522,10 @@
2
     qproperty-icon: url(./Dark/media-pause.svg);
3
 }
4
 
5
+* themeID="filtersIcon" {
6
+    qproperty-icon: url(./Dark/filter.svg);
7
+}
8
+
9
 QToolBarExtension {
10
     background: palette(button);
11
     min-width: 12px;
12
@@ -808,7 +812,7 @@
13
     background-color: rgb(28,28,28);
14
 }
15
 
16
-QPushButton:disabled {
17
+QPushButton:disabled, QToolButton:disabled {
18
     background-color: rgb(28,28,28);
19
 }
20
 
21
@@ -1507,3 +1511,8 @@
22
 OBSBasicStats {
23
     background: palette(dark);
24
 }
25
+
26
+/* Advanced audio dialog */
27
+OBSBasicAdvAudio #scrollAreaWidgetContents {
28
+    background: palette(dark);
29
+}
30
obs-studio-28.1.2.tar.xz/UI/data/themes/Light.qss -> obs-studio-29.0.0.tar.xz/UI/data/themes/Light.qss Changed
30
 
1
@@ -522,6 +522,10 @@
2
     qproperty-icon: url(./Light/media-pause.svg);
3
 }
4
 
5
+* themeID="filtersIcon" {
6
+    qproperty-icon: url(./Light/filter.svg);
7
+}
8
+
9
 QToolBarExtension {
10
     background: palette(button);
11
     min-width: 12px;
12
@@ -808,7 +812,7 @@
13
     background-color: rgb(193,193,193);
14
 }
15
 
16
-QPushButton:disabled {
17
+QPushButton:disabled, QToolButton:disabled {
18
     background-color: rgb(193,193,193);
19
 }
20
 
21
@@ -1513,3 +1517,8 @@
22
 OBSBasicStats {
23
     background: palette(dark);
24
 }
25
+
26
+/* Advanced audio dialog */
27
+OBSBasicAdvAudio #scrollAreaWidgetContents {
28
+    background: palette(dark);
29
+}
30
obs-studio-28.1.2.tar.xz/UI/data/themes/Rachni.qss -> obs-studio-29.0.0.tar.xz/UI/data/themes/Rachni.qss Changed
30
 
1
@@ -530,6 +530,10 @@
2
     qproperty-icon: url(./Dark/media-pause.svg);
3
 }
4
 
5
+* themeID="filtersIcon" {
6
+    qproperty-icon: url(./Dark/filter.svg);
7
+}
8
+
9
 QToolBarExtension {
10
     background: palette(button);
11
     min-width: 12px;
12
@@ -812,7 +816,7 @@
13
     background-color: rgb(240,98,146);
14
 }
15
 
16
-QPushButton:disabled {
17
+QPushButton:disabled, QToolButton:disabled {
18
     background-color: rgb(0,139,163);
19
 }
20
 
21
@@ -1511,3 +1515,8 @@
22
 OBSBasicStats {
23
     background: palette(dark);
24
 }
25
+
26
+/* Advanced audio dialog */
27
+OBSBasicAdvAudio #scrollAreaWidgetContents {
28
+    background: palette(dark);
29
+}
30
obs-studio-28.1.2.tar.xz/UI/data/themes/System.qss -> obs-studio-29.0.0.tar.xz/UI/data/themes/System.qss Changed
12
 
1
@@ -58,6 +58,10 @@
2
     qproperty-icon: url(:/res/images/cogs.svg);
3
 }
4
 
5
+* themeID="filtersIcon" {
6
+    qproperty-icon: url(:/res/images/filter.svg);
7
+}
8
+
9
 MuteCheckBox {
10
     outline: none;
11
 }
12
obs-studio-28.1.2.tar.xz/UI/data/themes/Yami.qss -> obs-studio-29.0.0.tar.xz/UI/data/themes/Yami.qss Changed
30
 
1
@@ -526,6 +526,10 @@
2
     qproperty-icon: url(./Dark/media-pause.svg);
3
 }
4
 
5
+* themeID="filtersIcon" {
6
+    qproperty-icon: url(./Dark/filter.svg);
7
+}
8
+
9
 QToolBarExtension {
10
     background: palette(button);
11
     min-width: 12px;
12
@@ -812,7 +816,7 @@
13
     background-color: rgb(25,27,38);
14
 }
15
 
16
-QPushButton:disabled {
17
+QPushButton:disabled, QToolButton:disabled {
18
     background-color: rgb(25,27,38);
19
 }
20
 
21
@@ -1511,3 +1515,8 @@
22
 OBSBasicStats {
23
     background: palette(dark);
24
 }
25
+
26
+/* Advanced audio dialog */
27
+OBSBasicAdvAudio #scrollAreaWidgetContents {
28
+    background: palette(dark);
29
+}
30
obs-studio-28.1.2.tar.xz/UI/forms/OBSBasic.ui -> obs-studio-29.0.0.tar.xz/UI/forms/OBSBasic.ui Changed
40
 
1
@@ -842,6 +842,8 @@
2
           <addaction name="actionAddScene"/>
3
           <addaction name="actionRemoveScene"/>
4
           <addaction name="separator"/>
5
+          <addaction name="actionSceneFilters"/>
6
+          <addaction name="separator"/>
7
           <addaction name="actionSceneUp"/>
8
           <addaction name="actionSceneDown"/>
9
          </widget>
10
@@ -979,6 +981,7 @@
11
           </property>
12
           <addaction name="actionAddSource"/>
13
           <addaction name="actionRemoveSource"/>
14
+          <addaction name="separator"/>
15
           <addaction name="actionSourceProperties"/>
16
           <addaction name="separator"/>
17
           <addaction name="actionSourceUp"/>
18
@@ -2348,6 +2351,21 @@
19
     <string>menuIconSmall</string>
20
    </property>
21
   </action>
22
+  <action name="actionSceneFilters">
23
+   <property name="icon">
24
+    <iconset resource="obs.qrc">
25
+     <normaloff>:/res/images/filter.svg</normaloff>:/res/images/filter.svg</iconset>
26
+   </property>
27
+   <property name="text">
28
+    <string>SceneFilters</string>
29
+   </property>
30
+   <property name="toolTip">
31
+    <string>SceneFilters</string>
32
+   </property>
33
+   <property name="themeID" stdset="0">
34
+    <string>filtersIcon</string>
35
+   </property>
36
+  </action>
37
  </widget>
38
  <customwidgets>
39
   <customwidget>
40
obs-studio-28.1.2.tar.xz/UI/forms/OBSBasicSettings.ui -> obs-studio-29.0.0.tar.xz/UI/forms/OBSBasicSettings.ui Changed
107
 
1
@@ -256,23 +256,13 @@
2
                     </spacer>
3
                    </item>
4
                    <item row="2" column="1">
5
-                    <widget class="QCheckBox" name="enableAutoUpdates">
6
-                     <property name="text">
7
-                      <string>Basic.Settings.General.EnableAutoUpdates</string>
8
-                     </property>
9
-                     <property name="checked">
10
-                      <bool>true</bool>
11
-                     </property>
12
-                    </widget>
13
-                   </item>
14
-                   <item row="3" column="1">
15
                     <widget class="QCheckBox" name="openStatsOnStartup">
16
                      <property name="text">
17
                       <string>Basic.Settings.General.OpenStatsOnStartup</string>
18
                      </property>
19
                     </widget>
20
                    </item>
21
-                   <item row="4" column="1">
22
+                   <item row="3" column="1">
23
                     <widget class="QCheckBox" name="hideOBSFromCapture">
24
                      <property name="toolTip">
25
                       <string>Basic.Settings.General.HideOBSWindowsFromCapture.Tooltip</string>
26
@@ -286,6 +276,64 @@
27
                  </widget>
28
                 </item>
29
                 <item>
30
+                 <widget class="QGroupBox" name="updateSettingsGroupBox">
31
+                  <property name="title">
32
+                   <string>Basic.Settings.General.Updater</string>
33
+                  </property>
34
+                  <layout class="QFormLayout" name="formLayout_20">
35
+                   <property name="labelAlignment">
36
+                    <set>Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter</set>
37
+                   </property>
38
+                   <property name="fieldGrowthPolicy">
39
+                    <enum>QFormLayout::AllNonFixedFieldsGrow</enum>
40
+                   </property>
41
+                   <item row="0" column="0">
42
+                    <widget class="QLabel" name="updateChannelLabel">
43
+                     <property name="text">
44
+                      <string>Basic.Settings.General.UpdateChannel</string>
45
+                     </property>
46
+                    </widget>
47
+                   </item>
48
+                   <item row="0" column="1">
49
+                    <widget class="QComboBox" name="updateChannelBox">
50
+                     <property name="editable">
51
+                      <bool>false</bool>
52
+                     </property>
53
+                     <property name="currentText">
54
+                      <string/>
55
+                     </property>
56
+                    </widget>
57
+                   </item>
58
+                   <item row="1" column="1">
59
+                    <widget class="QCheckBox" name="enableAutoUpdates">
60
+                     <property name="text">
61
+                      <string>Basic.Settings.General.EnableAutoUpdates</string>
62
+                     </property>
63
+                     <property name="checked">
64
+                      <bool>true</bool>
65
+                     </property>
66
+                    </widget>
67
+                   </item>
68
+                   <item row="1" column="0">
69
+                    <spacer name="horizontalSpacer_29">
70
+                     <property name="orientation">
71
+                      <enum>Qt::Horizontal</enum>
72
+                     </property>
73
+                     <property name="sizeType">
74
+                      <enum>QSizePolicy::Fixed</enum>
75
+                     </property>
76
+                     <property name="sizeHint" stdset="0">
77
+                      <size>
78
+                       <width>170</width>
79
+                       <height>20</height>
80
+                      </size>
81
+                     </property>
82
+                    </spacer>
83
+                   </item>
84
+                  </layout>
85
+                 </widget>
86
+                </item>
87
+                <item>
88
                  <widget class="QGroupBox" name="groupBox_16">
89
                   <property name="title">
90
                    <string>Basic.Settings.Output</string>
91
@@ -7473,6 +7521,7 @@
92
   <tabstop>scrollArea_2</tabstop>
93
   <tabstop>language</tabstop>
94
   <tabstop>theme</tabstop>
95
+  <tabstop>updateChannelBox</tabstop>
96
   <tabstop>enableAutoUpdates</tabstop>
97
   <tabstop>openStatsOnStartup</tabstop>
98
   <tabstop>warnBeforeStreamStart</tabstop>
99
@@ -7615,7 +7664,6 @@
100
   <tabstop>colorSpace</tabstop>
101
   <tabstop>colorRange</tabstop>
102
   <tabstop>sdrWhiteLevel</tabstop>
103
-  <tabstop>horizontalLayout_sdrPaperWhite</tabstop>
104
   <tabstop>hdrNominalPeakLevel</tabstop>
105
   <tabstop>disableOSXVSync</tabstop>
106
   <tabstop>resetOSXVSync</tabstop>
107
obs-studio-28.1.2.tar.xz/UI/frontend-plugins/decklink-output-ui/decklink-ui-main.cpp -> obs-studio-29.0.0.tar.xz/UI/frontend-plugins/decklink-output-ui/decklink-ui-main.cpp Changed
121
 
1
@@ -21,6 +21,8 @@
2
 
3
 obs_output_t *output;
4
 
5
+constexpr size_t STAGE_BUFFER_COUNT = 3;
6
+
7
 struct preview_output {
8
    bool enabled;
9
    obs_source_t *current_source;
10
@@ -28,7 +30,9 @@
11
 
12
    video_t *video_queue;
13
    gs_texrender_t *texrender;
14
-   gs_stagesurf_t *stagesurface;
15
+   gs_stagesurf_t *stagesurfacesSTAGE_BUFFER_COUNT;
16
+   bool surf_writtenSTAGE_BUFFER_COUNT;
17
+   size_t stage_index;
18
    uint8_t *video_data;
19
    uint32_t video_linesize;
20
 
21
@@ -131,7 +135,8 @@
22
    obs_source_release(context.current_source);
23
 
24
    obs_enter_graphics();
25
-   gs_stagesurface_destroy(context.stagesurface);
26
+   for (gs_stagesurf_t *surf : context.stagesurfaces)
27
+       gs_stagesurface_destroy(surf);
28
    gs_texrender_destroy(context.texrender);
29
    obs_leave_graphics();
30
 
31
@@ -161,10 +166,15 @@
32
 
33
        obs_enter_graphics();
34
        context.texrender = gs_texrender_create(GS_BGRA, GS_ZS_NONE);
35
-       context.stagesurface =
36
-           gs_stagesurface_create(width, height, GS_BGRA);
37
+       for (gs_stagesurf_t *&surf : context.stagesurfaces)
38
+           surf = gs_stagesurface_create(width, height, GS_BGRA);
39
        obs_leave_graphics();
40
 
41
+       for (bool &written : context.surf_written)
42
+           written = false;
43
+
44
+       context.stage_index = 0;
45
+
46
        const video_output_info *mainVOI =
47
            video_output_get_info(obs_get_video());
48
 
49
@@ -266,34 +276,47 @@
50
        gs_blend_state_pop();
51
        gs_texrender_end(ctx->texrender);
52
 
53
-       struct video_frame output_frame;
54
-       if (video_output_lock_frame(ctx->video_queue, &output_frame, 1,
55
-                       os_gettime_ns())) {
56
-           gs_stage_texture(
57
-               ctx->stagesurface,
58
-               gs_texrender_get_texture(ctx->texrender));
59
-
60
-           if (gs_stagesurface_map(ctx->stagesurface,
61
-                       &ctx->video_data,
62
-                       &ctx->video_linesize)) {
63
-               uint32_t linesize = output_frame.linesize0;
64
-               for (uint32_t i = 0; i < ctx->ovi.base_height;
65
-                    i++) {
66
-                   uint32_t dst_offset = linesize * i;
67
-                   uint32_t src_offset =
68
-                       ctx->video_linesize * i;
69
-                   memcpy(output_frame.data0 +
70
-                              dst_offset,
71
-                          ctx->video_data + src_offset,
72
-                          linesize);
73
+       const size_t write_stage_index = ctx->stage_index;
74
+       gs_stage_texture(ctx->stagesurfaceswrite_stage_index,
75
+                gs_texrender_get_texture(ctx->texrender));
76
+       ctx->surf_writtenwrite_stage_index = true;
77
+
78
+       const size_t read_stage_index =
79
+           (write_stage_index + 1) % STAGE_BUFFER_COUNT;
80
+       if (ctx->surf_writtenread_stage_index) {
81
+           struct video_frame output_frame;
82
+           if (video_output_lock_frame(ctx->video_queue,
83
+                           &output_frame, 1,
84
+                           os_gettime_ns())) {
85
+               gs_stagesurf_t *const read_surf =
86
+                   ctx->stagesurfacesread_stage_index;
87
+               if (gs_stagesurface_map(read_surf,
88
+                           &ctx->video_data,
89
+                           &ctx->video_linesize)) {
90
+                   uint32_t linesize =
91
+                       output_frame.linesize0;
92
+                   for (uint32_t i = 0;
93
+                        i < ctx->ovi.base_height; i++) {
94
+                       uint32_t dst_offset =
95
+                           linesize * i;
96
+                       uint32_t src_offset =
97
+                           ctx->video_linesize * i;
98
+                       memcpy(output_frame.data0 +
99
+                                  dst_offset,
100
+                              ctx->video_data +
101
+                                  src_offset,
102
+                              linesize);
103
+                   }
104
+
105
+                   gs_stagesurface_unmap(read_surf);
106
+                   ctx->video_data = nullptr;
107
                }
108
 
109
-               gs_stagesurface_unmap(ctx->stagesurface);
110
-               ctx->video_data = nullptr;
111
+               video_output_unlock_frame(ctx->video_queue);
112
            }
113
-
114
-           video_output_unlock_frame(ctx->video_queue);
115
        }
116
+
117
+       ctx->stage_index = read_stage_index;
118
    }
119
 }
120
 
121
obs-studio-28.1.2.tar.xz/UI/frontend-plugins/decklink-output-ui/forms/output.ui -> obs-studio-29.0.0.tar.xz/UI/frontend-plugins/decklink-output-ui/forms/output.ui Changed
10
 
1
@@ -122,7 +122,7 @@
2
    <item>
3
     <widget class="QLabel" name="keyerLabel">
4
      <property name="text">
5
-      <string>Keyer output requires RGB mode in advanced settings.</string>
6
+      <string>Keyer output requires BGRA mode in advanced settings.</string>
7
      </property>
8
     </widget>
9
    </item>
10
obs-studio-28.1.2.tar.xz/UI/media-controls.cpp -> obs-studio-29.0.0.tar.xz/UI/media-controls.cpp Changed
110
 
1
@@ -31,6 +31,18 @@
2
    QMetaObject::invokeMethod(media, "SetPlayingState");
3
 }
4
 
5
+void MediaControls::OBSMediaNext(void *data, calldata_t *)
6
+{
7
+   MediaControls *media = static_cast<MediaControls *>(data);
8
+   QMetaObject::invokeMethod(media, "UpdateSlideCounter");
9
+}
10
+
11
+void MediaControls::OBSMediaPrevious(void *data, calldata_t *)
12
+{
13
+   MediaControls *media = static_cast<MediaControls *>(data);
14
+   QMetaObject::invokeMethod(media, "UpdateSlideCounter");
15
+}
16
+
17
 MediaControls::MediaControls(QWidget *parent)
18
    : QWidget(parent), ui(new Ui::MediaControls)
19
 {
20
@@ -196,6 +208,7 @@
21
 
22
    prevPaused = false;
23
 
24
+   UpdateSlideCounter();
25
    StartMediaTimer();
26
 }
27
 
28
@@ -219,8 +232,15 @@
29
        QTStr("ContextBar.MediaControls.RestartMedia"));
30
 
31
    ui->slider->setValue(0);
32
-   ui->timerLabel->setText("--:--:--");
33
-   ui->durationLabel->setText("--:--:--");
34
+
35
+   if (!isSlideshow) {
36
+       ui->timerLabel->setText("--:--:--");
37
+       ui->durationLabel->setText("--:--:--");
38
+   } else {
39
+       ui->timerLabel->setText("-");
40
+       ui->durationLabel->setText("-");
41
+   }
42
+
43
    ui->slider->setEnabled(false);
44
 
45
    StopMediaTimer();
46
@@ -255,9 +275,6 @@
47
 
48
    isSlideshow = strcmp(id, "slideshow") == 0;
49
    ui->slider->setVisible(!isSlideshow);
50
-   ui->timerLabel->setVisible(!isSlideshow);
51
-   ui->label->setVisible(!isSlideshow);
52
-   ui->durationLabel->setVisible(!isSlideshow);
53
    ui->emptySpaceAgain->setVisible(isSlideshow);
54
 
55
    obs_media_state state = obs_source_media_get_state(source);
56
@@ -278,7 +295,10 @@
57
        break;
58
    }
59
 
60
-   SetSliderPosition();
61
+   if (isSlideshow)
62
+       UpdateSlideCounter();
63
+   else
64
+       SetSliderPosition();
65
 }
66
 
67
 OBSSource MediaControls::GetSource()
68
@@ -299,6 +319,8 @@
69
        sigs.emplace_back(sh, "media_stopped", OBSMediaStopped, this);
70
        sigs.emplace_back(sh, "media_started", OBSMediaStarted, this);
71
        sigs.emplace_back(sh, "media_ended", OBSMediaStopped, this);
72
+       sigs.emplace_back(sh, "media_next", OBSMediaNext, this);
73
+       sigs.emplace_back(sh, "media_previous", OBSMediaPrevious, this);
74
    } else {
75
        weakSource = nullptr;
76
    }
77
@@ -467,3 +489,32 @@
78
    obs_source_media_set_time(source, ms);
79
    SetSliderPosition();
80
 }
81
+
82
+void MediaControls::UpdateSlideCounter()
83
+{
84
+   if (!isSlideshow)
85
+       return;
86
+
87
+   OBSSource source = OBSGetStrongRef(weakSource);
88
+
89
+   if (!source)
90
+       return;
91
+
92
+   proc_handler_t *ph = obs_source_get_proc_handler(source);
93
+   calldata_t cd = {};
94
+
95
+   proc_handler_call(ph, "current_index", &cd);
96
+   int slide = calldata_int(&cd, "current_index");
97
+
98
+   proc_handler_call(ph, "total_files", &cd);
99
+   int total = calldata_int(&cd, "total_files");
100
+   calldata_free(&cd);
101
+
102
+   if (total > 0) {
103
+       ui->timerLabel->setText(QString::number(slide + 1));
104
+       ui->durationLabel->setText(QString::number(total));
105
+   } else {
106
+       ui->timerLabel->setText("-");
107
+       ui->durationLabel->setText("-");
108
+   }
109
+}
110
obs-studio-28.1.2.tar.xz/UI/media-controls.hpp -> obs-studio-29.0.0.tar.xz/UI/media-controls.hpp Changed
19
 
1
@@ -33,6 +33,8 @@
2
    static void OBSMediaPlay(void *data, calldata_t *calldata);
3
    static void OBSMediaPause(void *data, calldata_t *calldata);
4
    static void OBSMediaStarted(void *data, calldata_t *calldata);
5
+   static void OBSMediaNext(void *data, calldata_t *calldata);
6
+   static void OBSMediaPrevious(void *data, calldata_t *calldata);
7
 
8
    std::unique_ptr<Ui_MediaControls> ui;
9
 
10
@@ -61,6 +63,8 @@
11
    void MoveSliderFoward(int seconds = 5);
12
    void MoveSliderBackwards(int seconds = 5);
13
 
14
+   void UpdateSlideCounter();
15
+
16
 public slots:
17
    void PlayMedia();
18
    void PauseMedia();
19
obs-studio-28.1.2.tar.xz/UI/multiview.cpp -> obs-studio-29.0.0.tar.xz/UI/multiview.cpp Changed
13
 
1
@@ -174,9 +174,8 @@
2
        multiviewScenes.emplace_back(OBSGetWeakRef(src));
3
        obs_source_inc_showing(src);
4
 
5
-       std::string name = std::to_string(numSrcs) + " - " +
6
-                  obs_source_get_name(src);
7
-       multiviewLabels.emplace_back(CreateLabel(name.c_str(), h / 3));
8
+       multiviewLabels.emplace_back(
9
+           CreateLabel(obs_source_get_name(src), h / 3));
10
    }
11
 
12
    obs_frontend_source_list_free(&scenes);
13
obs-studio-28.1.2.tar.xz/UI/obs-app.cpp -> obs-studio-29.0.0.tar.xz/UI/obs-app.cpp Changed
177
 
1
@@ -55,6 +55,7 @@
2
 #include <curl/curl.h>
3
 
4
 #ifdef _WIN32
5
+#include <json11.hpp>
6
 #include <windows.h>
7
 #include <filesystem>
8
 #else
9
@@ -376,7 +377,7 @@
10
    va_copy(args2, args);
11
 #endif
12
 
13
-   vsnprintf(str, 4095, msg, args);
14
+   vsnprintf(str, sizeof(str), msg, args);
15
 
16
 #ifdef _WIN32
17
    if (IsDebuggerPresent()) {
18
@@ -1264,6 +1265,112 @@
19
    return SetTheme("System");
20
 }
21
 
22
+#ifdef _WIN32
23
+void ParseBranchesJson(const std::string &jsonString, vector<UpdateBranch> &out,
24
+              std::string &error)
25
+{
26
+   json11::Json root;
27
+   root = json11::Json::parse(jsonString, error);
28
+   if (!error.empty() || !root.is_array())
29
+       return;
30
+
31
+   for (const json11::Json &item : root.array_items()) {
32
+#ifdef _WIN32
33
+       if (!item"windows".bool_value())
34
+           continue;
35
+#endif
36
+
37
+       UpdateBranch branch = {
38
+           QString::fromStdString(item"name".string_value()),
39
+           QString::fromStdString(
40
+               item"display_name".string_value()),
41
+           QString::fromStdString(
42
+               item"description".string_value()),
43
+           item"enabled".bool_value(),
44
+           item"visible".bool_value(),
45
+       };
46
+       out.push_back(branch);
47
+   }
48
+}
49
+
50
+bool LoadBranchesFile(vector<UpdateBranch> &out)
51
+{
52
+   string error;
53
+   string branchesText;
54
+
55
+   BPtr<char> branchesFilePath =
56
+       GetConfigPathPtr("obs-studio/updates/branches.json");
57
+
58
+   QFile branchesFile(branchesFilePath.Get());
59
+   if (!branchesFile.open(QIODevice::ReadOnly)) {
60
+       error = "Opening file failed.";
61
+       goto fail;
62
+   }
63
+
64
+   branchesText = branchesFile.readAll();
65
+   if (branchesText.empty()) {
66
+       error = "File empty.";
67
+       goto fail;
68
+   }
69
+
70
+   ParseBranchesJson(branchesText, out, error);
71
+   if (error.empty())
72
+       return !out.empty();
73
+
74
+fail:
75
+   blog(LOG_WARNING, "Loading branches from file failed: %s",
76
+        error.c_str());
77
+   return false;
78
+}
79
+#endif
80
+
81
+void OBSApp::SetBranchData(const string &data)
82
+{
83
+#ifdef _WIN32
84
+   string error;
85
+   vector<UpdateBranch> result;
86
+
87
+   ParseBranchesJson(data, result, error);
88
+
89
+   if (!error.empty()) {
90
+       blog(LOG_WARNING, "Reading branches JSON response failed: %s",
91
+            error.c_str());
92
+       return;
93
+   }
94
+
95
+   if (!result.empty())
96
+       updateBranches = result;
97
+
98
+   branches_loaded = true;
99
+#else
100
+   UNUSED_PARAMETER(data);
101
+#endif
102
+}
103
+
104
+std::vector<UpdateBranch> OBSApp::GetBranches()
105
+{
106
+   vector<UpdateBranch> out;
107
+   /* Always ensure the default branch exists */
108
+   out.push_back(UpdateBranch{"stable", "", "", true, true});
109
+
110
+#ifdef _WIN32
111
+   if (!branches_loaded) {
112
+       vector<UpdateBranch> result;
113
+       if (LoadBranchesFile(result))
114
+           updateBranches = result;
115
+
116
+       branches_loaded = true;
117
+   }
118
+#endif
119
+
120
+   /* Copy additional branches to result (if any) */
121
+   if (!updateBranches.empty())
122
+       out.insert(out.end(), updateBranches.begin(),
123
+              updateBranches.end());
124
+
125
+   return out;
126
+}
127
+
128
 OBSApp::OBSApp(int &argc, char **argv, profiler_name_store_t *store)
129
    : QApplication(argc, argv), profilerNameStore(store)
130
 {
131
@@ -2199,10 +2306,12 @@
132
    /* NOTE: Qt doesn't use the Wayland platform on GNOME, so we have to
133
     * force it using the QT_QPA_PLATFORM env var. It's still possible to
134
     * use other QPA platforms using this env var, or the -platform command
135
-    * line option. */
136
+    * line option. Remove after Qt 6.3 is everywhere. */
137
 
138
+   const char *desktop = getenv("XDG_CURRENT_DESKTOP");
139
    const char *session_type = getenv("XDG_SESSION_TYPE");
140
-   if (session_type && strcmp(session_type, "wayland") == 0)
141
+   if (session_type && desktop && strcmp(desktop, "GNOME") == 0 &&
142
+       strcmp(session_type, "wayland") == 0)
143
        setenv("QT_QPA_PLATFORM", "wayland", false);
144
 #endif
145
 #endif
146
@@ -2316,10 +2425,6 @@
147
                audio_permission, accessibility_permission);
148
            check->exec();
149
        }
150
-
151
-       bool rosettaTranslated = os_get_emulation_status();
152
-       blog(LOG_INFO, "Rosetta translation used: %s",
153
-            rosettaTranslated ? "true" : "false");
154
 #endif
155
 
156
 #ifdef _WIN32
157
@@ -2662,16 +2767,16 @@
158
    if (!home)
159
        return;
160
 
161
-   if (snprintf(old_path, 512, "%s/.obs-studio", home) <= 0)
162
+   if (snprintf(old_path, sizeof(old_path), "%s/.obs-studio", home) <= 0)
163
        return;
164
 
165
    /* make base xdg path if it doesn't already exist */
166
-   if (GetConfigPath(new_path, 512, "") <= 0)
167
+   if (GetConfigPath(new_path, sizeof(new_path), "") <= 0)
168
        return;
169
    if (os_mkdirs(new_path) == MKDIR_ERROR)
170
        return;
171
 
172
-   if (GetConfigPath(new_path, 512, "obs-studio") <= 0)
173
+   if (GetConfigPath(new_path, sizeof(new_path), "obs-studio") <= 0)
174
        return;
175
 
176
    if (os_file_exists(old_path) && !os_file_exists(new_path)) {
177
obs-studio-28.1.2.tar.xz/UI/obs-app.hpp -> obs-studio-29.0.0.tar.xz/UI/obs-app.hpp Changed
35
 
1
@@ -74,6 +74,14 @@
2
    std::string author;
3
 };
4
 
5
+struct UpdateBranch {
6
+   QString name;
7
+   QString display_name;
8
+   QString description;
9
+   bool is_enabled;
10
+   bool is_visible;
11
+};
12
+
13
 class OBSApp : public QApplication {
14
    Q_OBJECT
15
 
16
@@ -86,6 +94,8 @@
17
    TextLookup textLookup;
18
    QPointer<OBSMainWindow> mainWindow;
19
    profiler_name_store_t *profilerNameStore = nullptr;
20
+   std::vector<UpdateBranch> updateBranches;
21
+   bool branches_loaded = false;
22
 
23
    bool libobs_initialized = false;
24
 
25
@@ -142,6 +152,9 @@
26
    bool SetTheme(std::string name, std::string path = "");
27
    inline bool IsThemeDark() const { return themeDarkMode; };
28
 
29
+   void SetBranchData(const std::string &data);
30
+   std::vector<UpdateBranch> GetBranches();
31
+
32
    inline lookup_t *GetTextLookup() const { return textLookup; }
33
 
34
    inline const char *GetString(const char *lookupVal) const
35
obs-studio-28.1.2.tar.xz/UI/obs-frontend-api/obs-frontend-api.cpp -> obs-studio-29.0.0.tar.xz/UI/obs-frontend-api/obs-frontend-api.cpp Changed
28
 
1
@@ -571,3 +571,26 @@
2
    return !!callbacks_valid() ? c->obs_frontend_get_locale_string(string)
3
                   : nullptr;
4
 }
5
+
6
+bool obs_frontend_is_theme_dark(void)
7
+{
8
+   return !!callbacks_valid() ? c->obs_frontend_is_theme_dark() : false;
9
+}
10
+
11
+char *obs_frontend_get_last_recording(void)
12
+{
13
+   return !!callbacks_valid() ? c->obs_frontend_get_last_recording()
14
+                  : nullptr;
15
+}
16
+
17
+char *obs_frontend_get_last_screenshot(void)
18
+{
19
+   return !!callbacks_valid() ? c->obs_frontend_get_last_screenshot()
20
+                  : nullptr;
21
+}
22
+
23
+char *obs_frontend_get_last_replay(void)
24
+{
25
+   return !!callbacks_valid() ? c->obs_frontend_get_last_replay()
26
+                  : nullptr;
27
+}
28
obs-studio-28.1.2.tar.xz/UI/obs-frontend-api/obs-frontend-api.h -> obs-studio-29.0.0.tar.xz/UI/obs-frontend-api/obs-frontend-api.h Changed
23
 
1
@@ -60,6 +60,8 @@
2
    OBS_FRONTEND_EVENT_SCRIPTING_SHUTDOWN,
3
    OBS_FRONTEND_EVENT_PROFILE_RENAMED,
4
    OBS_FRONTEND_EVENT_SCENE_COLLECTION_RENAMED,
5
+   OBS_FRONTEND_EVENT_THEME_CHANGED,
6
+   OBS_FRONTEND_EVENT_SCREENSHOT_TAKEN,
7
 };
8
 
9
 /* ------------------------------------------------------------------------- */
10
@@ -228,6 +230,12 @@
11
 EXPORT char *obs_frontend_get_current_record_output_path(void);
12
 EXPORT const char *obs_frontend_get_locale_string(const char *string);
13
 
14
+EXPORT bool obs_frontend_is_theme_dark(void);
15
+
16
+EXPORT char *obs_frontend_get_last_recording(void);
17
+EXPORT char *obs_frontend_get_last_screenshot(void);
18
+EXPORT char *obs_frontend_get_last_replay(void);
19
+
20
 /* ------------------------------------------------------------------------- */
21
 
22
 #ifdef __cplusplus
23
obs-studio-28.1.2.tar.xz/UI/obs-frontend-api/obs-frontend-internal.hpp -> obs-studio-29.0.0.tar.xz/UI/obs-frontend-api/obs-frontend-internal.hpp Changed
14
 
1
@@ -145,6 +145,12 @@
2
    virtual char *obs_frontend_get_current_record_output_path(void) = 0;
3
    virtual const char *
4
    obs_frontend_get_locale_string(const char *string) = 0;
5
+
6
+   virtual bool obs_frontend_is_theme_dark(void) = 0;
7
+
8
+   virtual char *obs_frontend_get_last_recording(void) = 0;
9
+   virtual char *obs_frontend_get_last_screenshot(void) = 0;
10
+   virtual char *obs_frontend_get_last_replay(void) = 0;
11
 };
12
 
13
 EXPORT void
14
obs-studio-28.1.2.tar.xz/UI/platform-windows.cpp -> obs-studio-29.0.0.tar.xz/UI/platform-windows.cpp Changed
113
 
1
@@ -37,11 +37,6 @@
2
 #include <util/windows/HRError.hpp>
3
 #include <util/windows/ComPtr.hpp>
4
 
5
-#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
6
-#include <QWinTaskbarButton>
7
-#include <QMainWindow>
8
-#endif
9
-
10
 using namespace std;
11
 
12
 static inline bool check_path(const char *data, const char *path,
13
@@ -51,7 +46,7 @@
14
    str << path << data;
15
    output = str.str();
16
 
17
-   printf("Attempted path: %s\n", output.c_str());
18
+   blog(LOG_DEBUG, "Attempted path: %s", output.c_str());
19
 
20
    return os_file_exists(output.c_str());
21
 }
22
@@ -176,33 +171,6 @@
23
    return build;
24
 }
25
 
26
-void SetAeroEnabled(bool enable)
27
-{
28
-   static HRESULT(WINAPI * func)(UINT) = nullptr;
29
-   static bool failed = false;
30
-
31
-   if (!func) {
32
-       if (failed) {
33
-           return;
34
-       }
35
-
36
-       HMODULE dwm = LoadLibraryW(L"dwmapi");
37
-       if (!dwm) {
38
-           failed = true;
39
-           return;
40
-       }
41
-
42
-       func = reinterpret_cast<decltype(func)>(
43
-           GetProcAddress(dwm, "DwmEnableComposition"));
44
-       if (!func) {
45
-           failed = true;
46
-           return;
47
-       }
48
-   }
49
-
50
-   func(enable ? DWM_EC_ENABLECOMPOSITION : DWM_EC_DISABLECOMPOSITION);
51
-}
52
-
53
 bool IsAlwaysOnTop(QWidget *window)
54
 {
55
    DWORD exStyle = GetWindowLong((HWND)window->winId(), GWL_EXSTYLE);
56
@@ -388,6 +356,7 @@
57
    return true;
58
 }
59
 
60
+#if QT_VERSION < QT_VERSION_CHECK(6, 4, 0)
61
 #define GENERIC_MONITOR_NAME QStringLiteral("Generic PnP Monitor")
62
 
63
 QString GetMonitorName(const QString &id)
64
@@ -449,6 +418,7 @@
65
 
66
    return QString::fromWCharArray(target.monitorFriendlyDeviceName);
67
 }
68
+#endif
69
 
70
 /* Based on https://www.winehq.org/pipermail/wine-devel/2008-September/069387.html */
71
 typedef const char *(CDECL *WINEGETVERSION)(void);
72
@@ -470,35 +440,6 @@
73
    return false;
74
 }
75
 
76
-#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
77
-QWinTaskbarButton *taskBtn;
78
-
79
-void TaskbarOverlayInit()
80
-{
81
-   QMainWindow *main = App()->GetMainWindow();
82
-   taskBtn = new QWinTaskbarButton(main);
83
-   taskBtn->setWindow(main->windowHandle());
84
-}
85
-
86
-void TaskbarOverlaySetStatus(TaskbarOverlayStatus status)
87
-{
88
-   if (status == TaskbarOverlayStatusInactive) {
89
-       taskBtn->clearOverlayIcon();
90
-       return;
91
-   }
92
-
93
-   QIcon icon;
94
-   if (status == TaskbarOverlayStatusActive) {
95
-       icon = QIcon::fromTheme("obs-active",
96
-                   QIcon(":/res/images/active.png"));
97
-   } else {
98
-       icon = QIcon::fromTheme("obs-paused",
99
-                   QIcon(":/res/images/paused.png"));
100
-   }
101
-   taskBtn->setOverlayIcon(icon);
102
-}
103
-#else
104
-
105
 HWND hwnd;
106
 void TaskbarOverlayInit()
107
 {
108
@@ -554,4 +495,3 @@
109
    }
110
    taskbarIcon->Release();
111
 }
112
-#endif
113
obs-studio-28.1.2.tar.xz/UI/platform-x11.cpp -> obs-studio-29.0.0.tar.xz/UI/platform-x11.cpp Changed
10
 
1
@@ -178,7 +178,7 @@
2
    str << path << data;
3
    output = str.str();
4
 
5
-   printf("Attempted path: %s\n", output.c_str());
6
+   blog(LOG_DEBUG, "Attempted path: %s", output.c_str());
7
 
8
    return (access(output.c_str(), R_OK) == 0);
9
 }
10
obs-studio-28.1.2.tar.xz/UI/platform.hpp -> obs-studio-29.0.0.tar.xz/UI/platform.hpp Changed
19
 
1
@@ -55,7 +55,6 @@
2
 #ifdef _WIN32
3
 uint32_t GetWindowsVersion();
4
 uint32_t GetWindowsBuild();
5
-void SetAeroEnabled(bool enable);
6
 void SetProcessPriority(const char *priority);
7
 void SetWin32DropStyle(QWidget *window);
8
 bool DisableAudioDucking(bool disable);
9
@@ -75,7 +74,9 @@
10
    RunOnceMutex &operator=(RunOnceMutex &&rom);
11
 };
12
 
13
+#if QT_VERSION < QT_VERSION_CHECK(6, 4, 0)
14
 QString GetMonitorName(const QString &id);
15
+#endif
16
 bool IsRunningOnWine();
17
 #endif
18
 
19
obs-studio-28.1.2.tar.xz/UI/qt-wrappers.cpp -> obs-studio-29.0.0.tar.xz/UI/qt-wrappers.cpp Changed
31
 
1
@@ -30,6 +30,7 @@
2
 #include <QStandardItemModel>
3
 #include <QLabel>
4
 #include <QPushButton>
5
+#include <QToolBar>
6
 
7
 #if !defined(_WIN32) && !defined(__APPLE__)
8
 #include <obs-nix-platform.h>
9
@@ -42,7 +43,7 @@
10
 static inline void OBSErrorBoxva(QWidget *parent, const char *msg, va_list args)
11
 {
12
    char full_message4096;
13
-   vsnprintf(full_message, 4095, msg, args);
14
+   vsnprintf(full_message, sizeof(full_message), msg, args);
15
 
16
    QMessageBox::critical(parent, "Error", full_message);
17
 }
18
@@ -406,3 +407,12 @@
19
 
20
    SetLabelText(label, newText);
21
 }
22
+
23
+void RefreshToolBarStyling(QToolBar *toolBar)
24
+{
25
+   for (QAction *action : toolBar->actions()) {
26
+       QWidget *widget = toolBar->widgetForAction(action);
27
+       widget->style()->unpolish(widget);
28
+       widget->style()->polish(widget);
29
+   }
30
+}
31
obs-studio-28.1.2.tar.xz/UI/qt-wrappers.hpp -> obs-studio-29.0.0.tar.xz/UI/qt-wrappers.hpp Changed
15
 
1
@@ -39,6 +39,7 @@
2
 class QString;
3
 struct gs_window;
4
 class QLabel;
5
+class QToolBar;
6
 
7
 class OBSMessageBox {
8
 public:
9
@@ -122,3 +123,5 @@
10
 
11
 void TruncateLabel(QLabel *label, QString newText,
12
           int length = MAX_LABEL_LENGTH);
13
+
14
+void RefreshToolBarStyling(QToolBar *toolBar);
15
obs-studio-28.1.2.tar.xz/UI/volume-control.cpp -> obs-studio-29.0.0.tar.xz/UI/volume-control.cpp Changed
17
 
1
@@ -204,6 +204,7 @@
2
 
3
        volMeter = new VolumeMeter(nullptr, obs_volmeter, true);
4
        slider = new VolumeSlider(obs_fader, Qt::Vertical);
5
+       slider->setLayoutDirection(Qt::LeftToRight);
6
 
7
        nameLayout->setAlignment(Qt::AlignCenter);
8
        meterLayout->setAlignment(Qt::AlignCenter);
9
@@ -253,6 +254,7 @@
10
 
11
        volMeter = new VolumeMeter(nullptr, obs_volmeter, false);
12
        slider = new VolumeSlider(obs_fader, Qt::Horizontal);
13
+       slider->setLayoutDirection(Qt::LeftToRight);
14
 
15
        textLayout->setContentsMargins(0, 0, 0, 0);
16
        textLayout->addWidget(nameLabel);
17
obs-studio-28.1.2.tar.xz/UI/win-update/updater/http.cpp -> obs-studio-29.0.0.tar.xz/UI/win-update/updater/http.cpp Changed
10
 
1
@@ -125,7 +125,7 @@
2
    /* -------------------------------------- *
3
     * connect to server                      */
4
 
5
-   hSession = WinHttpOpen(L"OBS Studio Updater/2.1",
6
+   hSession = WinHttpOpen(L"OBS Studio Updater/2.2",
7
                   WINHTTP_ACCESS_TYPE_DEFAULT_PROXY,
8
                   WINHTTP_NO_PROXY_NAME, WINHTTP_NO_PROXY_BYPASS,
9
                   0);
10
obs-studio-28.1.2.tar.xz/UI/win-update/updater/updater.cpp -> obs-studio-29.0.0.tar.xz/UI/win-update/updater/updater.cpp Changed
91
 
1
@@ -376,7 +376,7 @@
2
 
3
    const DWORD enableHTTP2Flag = WINHTTP_PROTOCOL_FLAG_HTTP2;
4
 
5
-   HttpHandle hSession = WinHttpOpen(L"OBS Studio Updater/2.1",
6
+   HttpHandle hSession = WinHttpOpen(L"OBS Studio Updater/2.2",
7
                      WINHTTP_ACCESS_TYPE_DEFAULT_PROXY,
8
                      WINHTTP_NO_PROXY_NAME,
9
                      WINHTTP_NO_PROXY_BYPASS, 0);
10
@@ -652,7 +652,8 @@
11
 #define UPDATE_URL L"https://cdn-fastly.obsproject.com/update_studio"
12
 
13
 static bool AddPackageUpdateFiles(const Json &root, size_t idx,
14
-                 const wchar_t *tempPath)
15
+                 const wchar_t *tempPath,
16
+                 const wchar_t *branch)
17
 {
18
    const Json &package = rootidx;
19
    const Json &name = package"name";
20
@@ -726,8 +727,9 @@
21
            return false;
22
        }
23
 
24
-       StringCbPrintf(sourceURL, sizeof(sourceURL), L"%s/%s/%s",
25
-                  UPDATE_URL, wPackageName, updateFileName);
26
+       StringCbPrintf(sourceURL, sizeof(sourceURL), L"%s/%s/%s/%s",
27
+                  UPDATE_URL, branch, wPackageName,
28
+                  updateFileName);
29
        StringCbPrintf(tempFilePath, sizeof(tempFilePath), L"%s\\%s",
30
                   tempPath, updateHashStr);
31
 
32
@@ -1092,7 +1094,7 @@
33
 
34
    const DWORD tlsProtocols = WINHTTP_FLAG_SECURE_PROTOCOL_TLS1_2;
35
 
36
-   HttpHandle hSession = WinHttpOpen(L"OBS Studio Updater/2.1",
37
+   HttpHandle hSession = WinHttpOpen(L"OBS Studio Updater/2.2",
38
                      WINHTTP_ACCESS_TYPE_DEFAULT_PROXY,
39
                      WINHTTP_NO_PROXY_NAME,
40
                      WINHTTP_NO_PROXY_BYPASS, 0);
41
@@ -1284,6 +1286,7 @@
42
     * Check if updating portable build      */
43
 
44
    bool bIsPortable = false;
45
+   wstring branch = L"stable";
46
 
47
    if (cmdLine0) {
48
        int argc;
49
@@ -1292,10 +1295,17 @@
50
        if (argv) {
51
            for (int i = 0; i < argc; i++) {
52
                if (wcscmp(argvi, L"Portable") == 0) {
53
+                   // Legacy OBS
54
+                   bIsPortable = true;
55
+                   break;
56
+               } else if (wcsncmp(argvi, L"--branch=", 9) ==
57
+                      0) {
58
+                   branch = argvi + 9;
59
+               } else if (wcscmp(argvi, L"--portable") ==
60
+                      0) {
61
                    bIsPortable = true;
62
                }
63
            }
64
-
65
            LocalFree((HLOCAL)argv);
66
        }
67
    }
68
@@ -1396,7 +1406,8 @@
69
 
70
    const Json::array &packages = root"packages".array_items();
71
    for (size_t i = 0; i < packages.size(); i++) {
72
-       if (!AddPackageUpdateFiles(packages, i, tempPath)) {
73
+       if (!AddPackageUpdateFiles(packages, i, tempPath,
74
+                      branch.c_str())) {
75
            Status(L"Update failed: Failed to process update packages");
76
            return false;
77
        }
78
@@ -1481,7 +1492,11 @@
79
              Z_BEST_COMPRESSION);
80
        compressedJson.resize(compressSize);
81
 
82
-       bool success = !!HTTPPostData(PATCH_MANIFEST_URL,
83
+       wstring manifestUrl(PATCH_MANIFEST_URL);
84
+       if (branch != L"stable")
85
+           manifestUrl += L"?branch=" + branch;
86
+
87
+       bool success = !!HTTPPostData(manifestUrl.c_str(),
88
                          (BYTE *)&compressedJson0,
89
                          (int)compressedJson.size(),
90
                          L"Accept-Encoding: gzip",
91
obs-studio-28.1.2.tar.xz/UI/win-update/win-update.cpp -> obs-studio-29.0.0.tar.xz/UI/win-update/win-update.cpp Changed
201
 
1
@@ -36,6 +36,18 @@
2
 #define WIN_MANIFEST_URL "https://obsproject.com/update_studio/manifest.json"
3
 #endif
4
 
5
+#ifndef WIN_MANIFEST_BASE_URL
6
+#define WIN_MANIFEST_BASE_URL "https://obsproject.com/update_studio/"
7
+#endif
8
+
9
+#ifndef WIN_BRANCHES_URL
10
+#define WIN_BRANCHES_URL "https://obsproject.com/update_studio/branches.json"
11
+#endif
12
+
13
+#ifndef WIN_DEFAULT_BRANCH
14
+#define WIN_DEFAULT_BRANCH "stable"
15
+#endif
16
+
17
 #ifndef WIN_WHATSNEW_URL
18
 #define WIN_WHATSNEW_URL "https://obsproject.com/update_studio/whatsnew.json"
19
 #endif
20
@@ -399,8 +411,30 @@
21
 
22
 /* ------------------------------------------------------------------------ */
23
 
24
+#if defined(OBS_RELEASE_CANDIDATE) && OBS_RELEASE_CANDIDATE > 0
25
+#define CUR_VER                                                               \
26
+   ((uint64_t)OBS_RELEASE_CANDIDATE_VER << 16ULL | OBS_RELEASE_CANDIDATE \
27
+                               << 8ULL)
28
+#define PRE_RELEASE true
29
+#elif OBS_BETA > 0
30
+#define CUR_VER ((uint64_t)OBS_BETA_VER << 16ULL | OBS_BETA)
31
+#define PRE_RELEASE true
32
+#elif defined(OBS_COMMIT)
33
+#define CUR_VER 1 << 16ULL
34
+#define CUR_COMMIT OBS_COMMIT
35
+#define PRE_RELEASE true
36
+#else
37
+#define CUR_VER ((uint64_t)LIBOBS_API_VER << 16ULL)
38
+#define PRE_RELEASE false
39
+#endif
40
+
41
+#ifndef CUR_COMMIT
42
+#define CUR_COMMIT "00000000"
43
+#endif
44
+
45
 static bool ParseUpdateManifest(const char *manifest, bool *updatesAvailable,
46
-               string &notes_str, int &updateVer)
47
+               string &notes_str, uint64_t &updateVer,
48
+               string &branch)
49
 try {
50
 
51
    string error;
52
@@ -415,8 +449,11 @@
53
    int major = root"version_major".int_value();
54
    int minor = root"version_minor".int_value();
55
    int patch = root"version_patch".int_value();
56
+   int rc = root"rc".int_value();
57
+   int beta = root"beta".int_value();
58
+   string commit_hash = root"commit".string_value();
59
 
60
-   if (major == 0)
61
+   if (major == 0 && commit_hash.empty())
62
        throw strprintf("Invalid version number: %d.%d.%d", major,
63
                minor, patch);
64
 
65
@@ -430,11 +467,35 @@
66
    if (!packages.is_array())
67
        throw string("'packages' value invalid");
68
 
69
-   int cur_ver = LIBOBS_API_VER;
70
-   int new_ver = MAKE_SEMANTIC_VERSION(major, minor, patch);
71
+   uint64_t cur_ver;
72
+   uint64_t new_ver;
73
+
74
+   if (commit_hash.empty()) {
75
+       cur_ver = CUR_VER;
76
+       new_ver = MAKE_SEMANTIC_VERSION(
77
+           (uint64_t)major, (uint64_t)minor, (uint64_t)patch);
78
+       new_ver <<= 16;
79
+       /* RC builds are shifted so that rc1 and beta1 versions do not result
80
+        * in the same new_ver. */
81
+       if (rc > 0)
82
+           new_ver |= (uint64_t)rc << 8;
83
+       else if (beta > 0)
84
+           new_ver |= (uint64_t)beta;
85
+   } else {
86
+       /* Test or nightly builds may not have a (valid) version number,
87
+        * so compare commit hashes instead. */
88
+       cur_ver = stoul(CUR_COMMIT, nullptr, 16);
89
+       new_ver = stoul(commit_hash.substr(0, 8), nullptr, 16);
90
+   }
91
 
92
    updateVer = new_ver;
93
-   *updatesAvailable = new_ver > cur_ver;
94
+
95
+   /* When using a pre-release build or non-default branch we only check if
96
+    * the manifest version is different, so that it can be rolled-back. */
97
+   if (branch != WIN_DEFAULT_BRANCH || PRE_RELEASE)
98
+       *updatesAvailable = new_ver != cur_ver;
99
+   else
100
+       *updatesAvailable = new_ver > cur_ver;
101
 
102
    return true;
103
 
104
@@ -443,6 +504,10 @@
105
    return false;
106
 }
107
 
108
+#undef CUR_COMMIT
109
+#undef CUR_VER
110
+#undef PRE_RELEASE
111
+
112
 /* ------------------------------------------------------------------------ */
113
 
114
 void GenerateGUID(string &guid)
115
@@ -481,6 +546,127 @@
116
    return guid;
117
 }
118
 
119
+/* ------------------------------------------------------------------------ */
120
+
121
+bool GetBranchAndUrl(string &selectedBranch, string &manifestUrl)
122
+{
123
+   const char *config_branch =
124
+       config_get_string(GetGlobalConfig(), "General", "UpdateBranch");
125
+   if (!config_branch)
126
+       return true;
127
+
128
+   bool found = false;
129
+   for (const UpdateBranch &branch : App()->GetBranches()) {
130
+       if (branch.name != config_branch)
131
+           continue;
132
+       /* A branch that is found but disabled will just silently fall back to
133
+        * the default. But if the branch was removed entirely, the user should
134
+        * be warned, so leave this false *only* if the branch was removed. */
135
+       found = true;
136
+
137
+       if (branch.is_enabled) {
138
+           selectedBranch = branch.name.toStdString();
139
+           if (branch.name != WIN_DEFAULT_BRANCH) {
140
+               manifestUrl = WIN_MANIFEST_BASE_URL;
141
+               manifestUrl += "manifest_" +
142
+                          branch.name.toStdString() +
143
+                          ".json";
144
+           }
145
+       }
146
+       break;
147
+   }
148
+
149
+   return found;
150
+}
151
+
152
+/* ------------------------------------------------------------------------ */
153
+
154
+static bool
155
+FetchAndVerifyFile(const char *name, const char *file, const char *url,
156
+          string &text,
157
+          const vector<string> &extraHeaders = vector<string>())
158
+{
159
+   long responseCode;
160
+   vector<string> headers;
161
+   string error;
162
+   string signature;
163
+   BYTE fileHashBLAKE2_HASH_LENGTH;
164
+   bool success;
165
+
166
+   BPtr<char> filePath = GetConfigPathPtr(file);
167
+
168
+   if (!extraHeaders.empty()) {
169
+       headers.insert(headers.end(), extraHeaders.begin(),
170
+                  extraHeaders.end());
171
+   }
172
+
173
+   /* ----------------------------------- *
174
+    * avoid downloading json again        */
175
+
176
+   if (CalculateFileHash(filePath, fileHash)) {
177
+       char hashStringBLAKE2_HASH_STR_LENGTH;
178
+       HashToString(fileHash, hashString);
179
+
180
+       string header = "If-None-Match: ";
181
+       header += hashString;
182
+       headers.push_back(move(header));
183
+   }
184
+
185
+   /* ----------------------------------- *
186
+    * get current install GUID            */
187
+
188
+   string guid = GetProgramGUID();
189
+
190
+   if (!guid.empty()) {
191
+       string header = "X-OBS2-GUID: ";
192
+       header += guid;
193
+       headers.push_back(move(header));
194
+   }
195
+
196
+   /* ----------------------------------- *
197
+    * get json from server                */
198
+
199
+   success = GetRemoteFile(url, text, error, &responseCode, nullptr, "",
200
+               nullptr, headers, &signature);
201
obs-studio-28.1.2.tar.xz/UI/window-basic-auto-config-test.cpp -> obs-studio-29.0.0.tar.xz/UI/window-basic-auto-config-test.cpp Changed
36
 
1
@@ -894,6 +894,8 @@
2
            wiz->streamingEncoder = AutoConfig::Encoder::NVENC;
3
        else if (wiz->qsvAvailable)
4
            wiz->streamingEncoder = AutoConfig::Encoder::QSV;
5
+       else if (wiz->appleAvailable)
6
+           wiz->streamingEncoder = AutoConfig::Encoder::Apple;
7
        else
8
            wiz->streamingEncoder = AutoConfig::Encoder::AMD;
9
    } else {
10
@@ -927,6 +929,8 @@
11
            wiz->recordingEncoder = AutoConfig::Encoder::NVENC;
12
        else if (wiz->qsvAvailable)
13
            wiz->recordingEncoder = AutoConfig::Encoder::QSV;
14
+       else if (wiz->appleAvailable)
15
+           wiz->recordingEncoder = AutoConfig::Encoder::Apple;
16
        else
17
            wiz->recordingEncoder = AutoConfig::Encoder::AMD;
18
    } else {
19
@@ -948,6 +952,7 @@
20
 #define ENCODER_NVENC ENCODER_TEXT("Hardware.NVENC.H264")
21
 #define ENCODER_QSV ENCODER_TEXT("Hardware.QSV.H264")
22
 #define ENCODER_AMD ENCODER_TEXT("Hardware.AMD.H264")
23
+#define ENCODER_APPLE ENCODER_TEXT("Hardware.Apple.H264")
24
 
25
 #define QUALITY_SAME "Basic.Settings.Output.Simple.RecordingQuality.Stream"
26
 #define QUALITY_HIGH "Basic.Settings.Output.Simple.RecordingQuality.Small"
27
@@ -990,6 +995,8 @@
28
            return QTStr(ENCODER_QSV);
29
        case AutoConfig::Encoder::AMD:
30
            return QTStr(ENCODER_AMD);
31
+       case AutoConfig::Encoder::Apple:
32
+           return QTStr(ENCODER_APPLE);
33
        case AutoConfig::Encoder::Stream:
34
            return QTStr(QUALITY_SAME);
35
        }
36
obs-studio-28.1.2.tar.xz/UI/window-basic-auto-config.cpp -> obs-studio-29.0.0.tar.xz/UI/window-basic-auto-config.cpp Changed
38
 
1
@@ -969,7 +969,7 @@
2
        /* Newer generations of NVENC have a high enough quality to
3
         * bitrate ratio that if NVENC is available, it makes sense to
4
         * just always prefer hardware encoding by default */
5
-       bool preferHardware = nvencAvailable ||
6
+       bool preferHardware = nvencAvailable || appleAvailable ||
7
                      os_get_physical_cores() <= 4;
8
        streamPage->ui->preferHardware->setChecked(preferHardware);
9
    }
10
@@ -1000,6 +1000,18 @@
11
            hardwareEncodingAvailable = qsvAvailable = true;
12
        else if (strcmp(id, "h264_texture_amf") == 0)
13
            hardwareEncodingAvailable = vceAvailable = true;
14
+#ifdef __APPLE__
15
+       else if (strcmp(id,
16
+               "com.apple.videotoolbox.videoencoder.ave.avc") ==
17
+                0
18
+#ifndef __aarch64__
19
+            && os_get_emulation_status() == true
20
+#endif
21
+       )
22
+           if (__builtin_available(macOS 13.0, *))
23
+               hardwareEncodingAvailable = appleAvailable =
24
+                   true;
25
+#endif
26
    }
27
 }
28
 
29
@@ -1047,6 +1059,8 @@
30
        return SIMPLE_ENCODER_QSV;
31
    case Encoder::AMD:
32
        return SIMPLE_ENCODER_AMD;
33
+   case Encoder::Apple:
34
+       return SIMPLE_ENCODER_APPLE_H264;
35
    default:
36
        return SIMPLE_ENCODER_X264;
37
    }
38
obs-studio-28.1.2.tar.xz/UI/window-basic-auto-config.hpp -> obs-studio-29.0.0.tar.xz/UI/window-basic-auto-config.hpp Changed
17
 
1
@@ -47,6 +47,7 @@
2
        NVENC,
3
        QSV,
4
        AMD,
5
+       Apple,
6
        Stream,
7
    };
8
 
9
@@ -89,6 +90,7 @@
10
    bool nvencAvailable = false;
11
    bool qsvAvailable = false;
12
    bool vceAvailable = false;
13
+   bool appleAvailable = false;
14
 
15
    int startingBitrate = 2500;
16
    bool customServer = false;
17
obs-studio-28.1.2.tar.xz/UI/window-basic-main-outputs.cpp -> obs-studio-29.0.0.tar.xz/UI/window-basic-main-outputs.cpp Changed
143
 
1
@@ -288,11 +288,14 @@
2
    int CalcCRF(int crf);
3
 
4
    void UpdateRecordingSettings_x264_crf(int crf);
5
-   void UpdateRecordingSettings_qsv11(int crf);
6
+   void UpdateRecordingSettings_qsv11(int crf, bool av1);
7
    void UpdateRecordingSettings_nvenc(int cqp);
8
    void UpdateRecordingSettings_nvenc_hevc_av1(int cqp);
9
    void UpdateRecordingSettings_amd_cqp(int cqp);
10
    void UpdateRecordingSettings_apple(int quality);
11
+#ifdef ENABLE_HEVC
12
+   void UpdateRecordingSettings_apple_hevc(int quality);
13
+#endif
14
    void UpdateRecordingSettings();
15
    void UpdateRecordingAudioSettings();
16
    virtual void Update() override;
17
@@ -367,13 +370,17 @@
18
    } else if (strcmp(encoder, SIMPLE_ENCODER_X264_LOWCPU) == 0) {
19
        return "obs_x264";
20
    } else if (strcmp(encoder, SIMPLE_ENCODER_QSV) == 0) {
21
-       return "obs_qsv11";
22
+       return "obs_qsv11_v2";
23
+   } else if (strcmp(encoder, SIMPLE_ENCODER_QSV_AV1) == 0) {
24
+       return "obs_qsv11_av1";
25
    } else if (strcmp(encoder, SIMPLE_ENCODER_AMD) == 0) {
26
        return "h264_texture_amf";
27
 #ifdef ENABLE_HEVC
28
    } else if (strcmp(encoder, SIMPLE_ENCODER_AMD_HEVC) == 0) {
29
        return "h265_texture_amf";
30
 #endif
31
+   } else if (strcmp(encoder, SIMPLE_ENCODER_AMD_AV1) == 0) {
32
+       return "av1_texture_amf";
33
    } else if (strcmp(encoder, SIMPLE_ENCODER_NVENC) == 0) {
34
        return EncoderAvailable("jim_nvenc") ? "jim_nvenc"
35
                             : "ffmpeg_nvenc";
36
@@ -386,6 +393,10 @@
37
        return "jim_av1_nvenc";
38
    } else if (strcmp(encoder, SIMPLE_ENCODER_APPLE_H264) == 0) {
39
        return "com.apple.videotoolbox.videoencoder.ave.avc";
40
+#ifdef ENABLE_HEVC
41
+   } else if (strcmp(encoder, SIMPLE_ENCODER_APPLE_HEVC) == 0) {
42
+       return "com.apple.videotoolbox.videoencoder.ave.hevc";
43
+#endif
44
    }
45
 
46
    return "obs_x264";
47
@@ -526,6 +537,9 @@
48
    if (strcmp(encoder, SIMPLE_ENCODER_QSV) == 0) {
49
        presetType = "QSVPreset";
50
 
51
+   } else if (strcmp(encoder, SIMPLE_ENCODER_QSV_AV1) == 0) {
52
+       presetType = "QSVPreset";
53
+
54
    } else if (strcmp(encoder, SIMPLE_ENCODER_AMD) == 0) {
55
        presetType = "AMDPreset";
56
 
57
@@ -542,6 +556,9 @@
58
        presetType = "NVENCPreset2";
59
 #endif
60
 
61
+   } else if (strcmp(encoder, SIMPLE_ENCODER_AMD_AV1) == 0) {
62
+       presetType = "AMDAV1Preset";
63
+
64
    } else if (strcmp(encoder, SIMPLE_ENCODER_NVENC_AV1) == 0) {
65
        presetType = "NVENCPreset2";
66
 
67
@@ -653,21 +670,19 @@
68
    return icq_found;
69
 }
70
 
71
-void SimpleOutput::UpdateRecordingSettings_qsv11(int crf)
72
+void SimpleOutput::UpdateRecordingSettings_qsv11(int crf, bool av1)
73
 {
74
    bool icq = icq_available(videoRecording);
75
 
76
    OBSDataAutoRelease settings = obs_data_create();
77
    obs_data_set_string(settings, "profile", "high");
78
 
79
-   if (icq) {
80
+   if (icq && !av1) {
81
        obs_data_set_string(settings, "rate_control", "ICQ");
82
        obs_data_set_int(settings, "icq_quality", crf);
83
    } else {
84
        obs_data_set_string(settings, "rate_control", "CQP");
85
-       obs_data_set_int(settings, "qpi", crf);
86
-       obs_data_set_int(settings, "qpp", crf);
87
-       obs_data_set_int(settings, "qpb", crf);
88
+       obs_data_set_int(settings, "cqp", crf);
89
    }
90
 
91
    obs_encoder_update(videoRecording, settings);
92
@@ -703,6 +718,18 @@
93
    obs_encoder_update(videoRecording, settings);
94
 }
95
 
96
+#ifdef ENABLE_HEVC
97
+void SimpleOutput::UpdateRecordingSettings_apple_hevc(int quality)
98
+{
99
+   OBSDataAutoRelease settings = obs_data_create();
100
+   obs_data_set_string(settings, "rate_control", "CRF");
101
+   obs_data_set_string(settings, "profile", "main");
102
+   obs_data_set_int(settings, "quality", quality);
103
+
104
+   obs_encoder_update(videoRecording, settings);
105
+}
106
+#endif
107
+
108
 void SimpleOutput::UpdateRecordingSettings_amd_cqp(int cqp)
109
 {
110
    OBSDataAutoRelease settings = obs_data_create();
111
@@ -722,7 +749,10 @@
112
        UpdateRecordingSettings_x264_crf(crf);
113
 
114
    } else if (videoEncoder == SIMPLE_ENCODER_QSV) {
115
-       UpdateRecordingSettings_qsv11(crf);
116
+       UpdateRecordingSettings_qsv11(crf, false);
117
+
118
+   } else if (videoEncoder == SIMPLE_ENCODER_QSV_AV1) {
119
+       UpdateRecordingSettings_qsv11(crf, true);
120
 
121
    } else if (videoEncoder == SIMPLE_ENCODER_AMD) {
122
        UpdateRecordingSettings_amd_cqp(crf);
123
@@ -745,6 +775,10 @@
124
    } else if (videoEncoder == SIMPLE_ENCODER_APPLE_H264) {
125
        /* These are magic numbers. 0 - 100, more is better. */
126
        UpdateRecordingSettings_apple(ultra_hq ? 70 : 50);
127
+#ifdef ENABLE_HEVC
128
+   } else if (videoEncoder == SIMPLE_ENCODER_APPLE_HEVC) {
129
+       UpdateRecordingSettings_apple_hevc(ultra_hq ? 70 : 50);
130
+#endif
131
    }
132
    UpdateRecordingAudioSettings();
133
 }
134
@@ -1323,7 +1357,7 @@
135
 
136
    for (int i = 0; i < MAX_AUDIO_MIXES; i++) {
137
        char name9;
138
-       sprintf(name, "adv_aac%d", i);
139
+       snprintf(name, sizeof(name), "adv_aac%d", i);
140
 
141
        if (!CreateAACEncoder(aacTracki, aacEncoderIDi,
142
                      GetAudioBitrate(i), name, i))
143
obs-studio-28.1.2.tar.xz/UI/window-basic-main-profiles.cpp -> obs-studio-29.0.0.tar.xz/UI/window-basic-main-profiles.cpp Changed
113
 
1
@@ -373,13 +373,15 @@
2
    char profilePath512;
3
    char basePath512;
4
 
5
-   int ret = GetConfigPath(basePath, 512, "obs-studio/basic/profiles");
6
+   int ret = GetConfigPath(basePath, sizeof(basePath),
7
+               "obs-studio/basic/profiles");
8
    if (ret <= 0) {
9
        blog(LOG_WARNING, "Failed to get profiles config path");
10
        return;
11
    }
12
 
13
-   ret = snprintf(profilePath, 512, "%s/%s/*", basePath, profileDir);
14
+   ret = snprintf(profilePath, sizeof(profilePath), "%s/%s/*", basePath,
15
+              profileDir);
16
    if (ret <= 0) {
17
        blog(LOG_WARNING, "Failed to get path for profile dir '%s'",
18
             profileDir);
19
@@ -404,7 +406,8 @@
20
 
21
    os_globfree(glob);
22
 
23
-   ret = snprintf(profilePath, 512, "%s/%s", basePath, profileDir);
24
+   ret = snprintf(profilePath, sizeof(profilePath), "%s/%s", basePath,
25
+              profileDir);
26
    if (ret <= 0) {
27
        blog(LOG_WARNING, "Failed to get path for profile dir '%s'",
28
             profileDir);
29
@@ -819,12 +822,15 @@
30
    const char *curRecEncoder =
31
        config_get_string(basicConfig, "SimpleOutput", "RecEncoder");
32
    bool qsv_supported = false;
33
+   bool qsv_av1_supported = false;
34
    bool amd_supported = false;
35
    bool nve_supported = false;
36
 #ifdef ENABLE_HEVC
37
    bool amd_hevc_supported = false;
38
    bool nve_hevc_supported = false;
39
+   bool apple_hevc_supported = false;
40
 #endif
41
+   bool amd_av1_supported = false;
42
    bool apple_supported = false;
43
    bool changed = false;
44
    size_t idx = 0;
45
@@ -835,6 +841,8 @@
46
            amd_supported = true;
47
        else if (strcmp(id, "obs_qsv11") == 0)
48
            qsv_supported = true;
49
+       else if (strcmp(id, "obs_qsv11_av1") == 0)
50
+           qsv_av1_supported = true;
51
        else if (strcmp(id, "ffmpeg_nvenc") == 0)
52
            nve_supported = true;
53
 #ifdef ENABLE_HEVC
54
@@ -843,10 +851,18 @@
55
        else if (strcmp(id, "ffmpeg_hevc_nvenc") == 0)
56
            nve_hevc_supported = true;
57
 #endif
58
+       else if (strcmp(id, "av1_texture_amf") == 0)
59
+           amd_av1_supported = true;
60
        else if (strcmp(id,
61
                "com.apple.videotoolbox.videoencoder.ave.avc") ==
62
             0)
63
            apple_supported = true;
64
+#ifdef ENABLE_HEVC
65
+       else if (strcmp(id,
66
+               "com.apple.videotoolbox.videoencoder.ave.hevc") ==
67
+            0)
68
+           apple_hevc_supported = true;
69
+#endif
70
    }
71
 
72
    auto CheckEncoder = &(const char *&name) {
73
@@ -856,6 +872,12 @@
74
                name = SIMPLE_ENCODER_X264;
75
                return false;
76
            }
77
+       } else if (strcmp(name, SIMPLE_ENCODER_QSV_AV1) == 0) {
78
+           if (!qsv_av1_supported) {
79
+               changed = true;
80
+               name = SIMPLE_ENCODER_X264;
81
+               return false;
82
+           }
83
        } else if (strcmp(name, SIMPLE_ENCODER_NVENC) == 0) {
84
            if (!nve_supported) {
85
                changed = true;
86
@@ -888,12 +910,26 @@
87
                name = SIMPLE_ENCODER_X264;
88
                return false;
89
            }
90
+       } else if (strcmp(name, SIMPLE_ENCODER_AMD_AV1) == 0) {
91
+           if (!amd_av1_supported) {
92
+               changed = true;
93
+               name = SIMPLE_ENCODER_X264;
94
+               return false;
95
+           }
96
        } else if (strcmp(name, SIMPLE_ENCODER_APPLE_H264) == 0) {
97
            if (!apple_supported) {
98
                changed = true;
99
                name = SIMPLE_ENCODER_X264;
100
                return false;
101
            }
102
+#ifdef ENABLE_HEVC
103
+       } else if (strcmp(name, SIMPLE_ENCODER_APPLE_HEVC) == 0) {
104
+           if (!apple_hevc_supported) {
105
+               changed = true;
106
+               name = SIMPLE_ENCODER_X264;
107
+               return false;
108
+           }
109
+#endif
110
        }
111
 
112
        return true;
113
obs-studio-28.1.2.tar.xz/UI/window-basic-main-screenshot.cpp -> obs-studio-29.0.0.tar.xz/UI/window-basic-main-screenshot.cpp Changed
14
 
1
@@ -53,6 +53,12 @@
2
            main->ShowStatusBarMessage(
3
                QTStr("Basic.StatusBar.ScreenshotSavedTo")
4
                    .arg(QT_UTF8(path.c_str())));
5
+
6
+           main->lastScreenshot = path;
7
+
8
+           if (main->api)
9
+               main->api->on_event(
10
+                   OBS_FRONTEND_EVENT_SCREENSHOT_TAKEN);
11
        }
12
    }
13
 }
14
obs-studio-28.1.2.tar.xz/UI/window-basic-main-transitions.cpp -> obs-studio-29.0.0.tar.xz/UI/window-basic-main-transitions.cpp Changed
33
 
1
@@ -1053,7 +1053,7 @@
2
    return menu;
3
 }
4
 
5
-void OBSBasic::on_actionShowTransitionProperties_triggered()
6
+void OBSBasic::ShowTransitionProperties()
7
 {
8
    OBSSceneItem item = GetCurrentSceneItem();
9
    OBSSource source = obs_sceneitem_get_transition(item, true);
10
@@ -1062,7 +1062,7 @@
11
        CreatePropertiesWindow(source);
12
 }
13
 
14
-void OBSBasic::on_actionHideTransitionProperties_triggered()
15
+void OBSBasic::HideTransitionProperties()
16
 {
17
    OBSSceneItem item = GetCurrentSceneItem();
18
    OBSSource source = obs_sceneitem_get_transition(item, false);
19
@@ -1247,10 +1247,9 @@
20
    menu->addAction(durationAction);
21
    if (curId && obs_is_source_configurable(curId)) {
22
        menu->addSeparator();
23
-       menu->addAction(
24
-           QTStr("Properties"), this,
25
-           visible ? SLOT(on_actionShowTransitionProperties_triggered())
26
-               : SLOT(on_actionHideTransitionProperties_triggered()));
27
+       menu->addAction(QTStr("Properties"), this,
28
+               visible ? SLOT(ShowTransitionProperties())
29
+                   : SLOT(HideTransitionProperties()));
30
    }
31
 
32
    auto copyTransition = this(QAction *, bool visible) {
33
obs-studio-28.1.2.tar.xz/UI/window-basic-main.cpp -> obs-studio-29.0.0.tar.xz/UI/window-basic-main.cpp Changed
201
 
1
@@ -272,14 +272,14 @@
2
    setAcceptDrops(true);
3
 
4
    setContextMenuPolicy(Qt::CustomContextMenu);
5
-   connect(this, SIGNAL(customContextMenuRequested(const QPoint &)), this,
6
-       SLOT(on_customContextMenuRequested(const QPoint &)));
7
 
8
    api = InitializeAPIInterface(this);
9
 
10
    ui->setupUi(this);
11
    ui->previewDisabledWidget->setVisible(false);
12
-   ui->contextContainer->setStyle(new OBSContextBarProxyStyle);
13
+   QStyle *contextBarStyle = new OBSContextBarProxyStyle();
14
+   contextBarStyle->setParent(ui->contextContainer);
15
+   ui->contextContainer->setStyle(contextBarStyle);
16
    ui->broadcastButton->setVisible(false);
17
 
18
    startingDockLayout = saveState();
19
@@ -1399,7 +1399,7 @@
20
    config_set_default_string(basicConfig, "SimpleOutput", "Preset",
21
                  "veryfast");
22
    config_set_default_string(basicConfig, "SimpleOutput", "NVENCPreset2",
23
-                 "p6");
24
+                 "p5");
25
    config_set_default_string(basicConfig, "SimpleOutput", "RecQuality",
26
                  "Stream");
27
    config_set_default_bool(basicConfig, "SimpleOutput", "RecRB", false);
28
@@ -1882,7 +1882,6 @@
29
        disableSaving++;
30
    }
31
 
32
-   TimedCheckForUpdates();
33
    loaded = true;
34
 
35
    previewEnabled = config_get_bool(App()->GlobalConfig(), "BasicWindow",
36
@@ -1893,15 +1892,6 @@
37
                      Qt::QueuedConnection,
38
                      Q_ARG(bool, previewEnabled));
39
 
40
-#ifdef _WIN32
41
-   uint32_t winVer = GetWindowsVersion();
42
-   if (winVer > 0 && winVer < 0x602) {
43
-       bool disableAero =
44
-           config_get_bool(basicConfig, "Video", "DisableAero");
45
-       SetAeroEnabled(!disableAero);
46
-   }
47
-#endif
48
-
49
    RefreshSceneCollections();
50
    RefreshProfiles();
51
    disableSaving--;
52
@@ -2036,6 +2026,19 @@
53
        QMetaObject::invokeMethod(this, "on_autoConfigure_triggered",
54
                      Qt::QueuedConnection);
55
 
56
+#if defined(_WIN32) && (OBS_RELEASE_CANDIDATE > 0 || OBS_BETA > 0)
57
+   /* Automatically set branch to "beta" the first time a pre-release build is run. */
58
+   if (!config_get_bool(App()->GlobalConfig(), "General",
59
+                "AutoBetaOptIn")) {
60
+       config_set_string(App()->GlobalConfig(), "General",
61
+                 "UpdateBranch", "beta");
62
+       config_set_bool(App()->GlobalConfig(), "General",
63
+               "AutoBetaOptIn", true);
64
+       config_save_safe(App()->GlobalConfig(), "tmp", nullptr);
65
+   }
66
+#endif
67
+   TimedCheckForUpdates();
68
+
69
    ToggleMixerLayout(config_get_bool(App()->GlobalConfig(), "BasicWindow",
70
                      "VerticalVolControl"));
71
 
72
@@ -2717,17 +2720,6 @@
73
            ui->lockDocks->isChecked());
74
    config_save_safe(App()->GlobalConfig(), "tmp", nullptr);
75
 
76
-#ifdef _WIN32
77
-   uint32_t winVer = GetWindowsVersion();
78
-   if (winVer > 0 && winVer < 0x602) {
79
-       bool disableAero =
80
-           config_get_bool(basicConfig, "Video", "DisableAero");
81
-       if (disableAero) {
82
-           SetAeroEnabled(true);
83
-       }
84
-   }
85
-#endif
86
-
87
 #ifdef BROWSER_AVAILABLE
88
    DestroyPanelCookieManager();
89
    delete cef;
90
@@ -3097,13 +3089,26 @@
91
                  Qt::QueuedConnection, Q_ARG(bool, force));
92
 }
93
 
94
+void OBSBasic::SourceToolBarActionsSetEnabled(bool enable)
95
+{
96
+   ui->actionRemoveSource->setEnabled(enable);
97
+   ui->actionSourceProperties->setEnabled(enable);
98
+   ui->actionSourceUp->setEnabled(enable);
99
+   ui->actionSourceDown->setEnabled(enable);
100
+
101
+   RefreshToolBarStyling(ui->sourcesToolbar);
102
+}
103
+
104
 void OBSBasic::UpdateContextBar(bool force)
105
 {
106
+   OBSSceneItem item = GetCurrentSceneItem();
107
+   bool enable = item != nullptr;
108
+
109
+   SourceToolBarActionsSetEnabled(enable);
110
+
111
    if (!ui->contextContainer->isVisible() && !force)
112
        return;
113
 
114
-   OBSSceneItem item = GetCurrentSceneItem();
115
-
116
    if (item) {
117
        obs_source_t *source = obs_sceneitem_get_source(item);
118
 
119
@@ -5056,13 +5061,13 @@
120
        QRect screenGeometry = screen->geometry();
121
        qreal ratio = screen->devicePixelRatio();
122
        QString name = "";
123
-#ifdef _WIN32
124
+#if defined(_WIN32) && QT_VERSION < QT_VERSION_CHECK(6, 4, 0)
125
        QTextStream fullname(&name);
126
        fullname << GetMonitorName(screen->name());
127
        fullname << " (";
128
        fullname << (i + 1);
129
        fullname << ")";
130
-#elif defined(__APPLE__)
131
+#elif defined(__APPLE__) || defined(_WIN32)
132
        name = screen->name();
133
 #else
134
        name = screen->model().simplified();
135
@@ -7166,7 +7171,16 @@
136
 
137
    QString output = input;
138
    output.resize(output.size() - suffix.size());
139
-   output += "mp4";
140
+
141
+   const obs_encoder_t *videoEncoder =
142
+       obs_output_get_video_encoder(outputHandler->fileOutput);
143
+   const char *codecName = obs_encoder_get_codec(videoEncoder);
144
+
145
+   if (strcmp(codecName, "prores") == 0) {
146
+       output += "mov";
147
+   } else {
148
+       output += "mp4";
149
+   }
150
 
151
    OBSRemux *remux = new OBSRemux(QT_TO_UTF8(path), this, true);
152
    if (!no_show)
153
@@ -7478,15 +7492,17 @@
154
    proc_handler_t *ph =
155
        obs_output_get_proc_handler(outputHandler->replayBuffer);
156
    proc_handler_call(ph, "get_last_replay", &cd);
157
-   QString path = QT_UTF8(calldata_string(&cd, "path"));
158
-   QString msg = QTStr("Basic.StatusBar.ReplayBufferSavedTo").arg(path);
159
+   std::string path = calldata_string(&cd, "path");
160
+   QString msg = QTStr("Basic.StatusBar.ReplayBufferSavedTo")
161
+                 .arg(QT_UTF8(path.c_str()));
162
    ShowStatusBarMessage(msg);
163
+   lastReplay = path;
164
    calldata_free(&cd);
165
 
166
    if (api)
167
        api->on_event(OBS_FRONTEND_EVENT_REPLAY_BUFFER_SAVED);
168
 
169
-   AutoRemux(path);
170
+   AutoRemux(QT_UTF8(path.c_str()));
171
 }
172
 
173
 void OBSBasic::ReplayBufferStop(int code)
174
@@ -10151,7 +10167,7 @@
175
    foreach(OBSBasicStats * s, list) s->Reset();
176
 }
177
 
178
-void OBSBasic::on_customContextMenuRequested(const QPoint &pos)
179
+void OBSBasic::on_OBSBasic_customContextMenuRequested(const QPoint &pos)
180
 {
181
    QWidget *widget = childAt(pos);
182
    const char *className = nullptr;
183
@@ -10206,6 +10222,14 @@
184
    OpenFilters();
185
 }
186
 
187
+void OBSBasic::on_actionSceneFilters_triggered()
188
+{
189
+   OBSSource sceneSource = GetCurrentSceneSource();
190
+
191
+   if (sceneSource)
192
+       OpenFilters(sceneSource);
193
+}
194
+
195
 void OBSBasic::on_sourceInteractButton_clicked()
196
 {
197
    on_actionInteract_triggered();
198
@@ -10318,4 +10342,7 @@
199
        ActivateAudioSource(source);
200
 
201
obs-studio-28.1.2.tar.xz/UI/window-basic-main.hpp -> obs-studio-29.0.0.tar.xz/UI/window-basic-main.hpp Changed
58
 
1
@@ -65,12 +65,15 @@
2
 #define SIMPLE_ENCODER_X264 "x264"
3
 #define SIMPLE_ENCODER_X264_LOWCPU "x264_lowcpu"
4
 #define SIMPLE_ENCODER_QSV "qsv"
5
+#define SIMPLE_ENCODER_QSV_AV1 "qsv_av1"
6
 #define SIMPLE_ENCODER_NVENC "nvenc"
7
 #define SIMPLE_ENCODER_NVENC_AV1 "nvenc_av1"
8
 #define SIMPLE_ENCODER_NVENC_HEVC "nvenc_hevc"
9
 #define SIMPLE_ENCODER_AMD "amd"
10
 #define SIMPLE_ENCODER_AMD_HEVC "amd_hevc"
11
+#define SIMPLE_ENCODER_AMD_AV1 "amd_av1"
12
 #define SIMPLE_ENCODER_APPLE_H264 "apple_h264"
13
+#define SIMPLE_ENCODER_APPLE_HEVC "apple_hevc"
14
 
15
 #define PREVIEW_EDGE_SIZE 10
16
 
17
@@ -189,6 +192,7 @@
18
    friend class OBSPermissions;
19
    friend struct BasicOutputHandler;
20
    friend struct OBSStudioAPI;
21
+   friend class ScreenshotObj;
22
 
23
    enum class MoveDir { Up, Down, Left, Right };
24
 
25
@@ -634,6 +638,10 @@
26
    bool drawSpacingHelpers = true;
27
 
28
    float GetDevicePixelRatio();
29
+   void SourceToolBarActionsSetEnabled(bool enable);
30
+
31
+   std::string lastScreenshot;
32
+   std::string lastReplay;
33
 
34
 public slots:
35
    void DeferSaveBegin();
36
@@ -1026,8 +1034,9 @@
37
    void on_actionCenterToScreen_triggered();
38
    void on_actionVerticalCenter_triggered();
39
    void on_actionHorizontalCenter_triggered();
40
+   void on_actionSceneFilters_triggered();
41
 
42
-   void on_customContextMenuRequested(const QPoint &pos);
43
+   void on_OBSBasic_customContextMenuRequested(const QPoint &pos);
44
 
45
    void on_scenes_currentItemChanged(QListWidgetItem *current,
46
                      QListWidgetItem *prev);
47
@@ -1107,8 +1116,8 @@
48
    void on_transitionDuration_valueChanged(int value);
49
    void on_tbar_position_valueChanged(int value);
50
 
51
-   void on_actionShowTransitionProperties_triggered();
52
-   void on_actionHideTransitionProperties_triggered();
53
+   void ShowTransitionProperties();
54
+   void HideTransitionProperties();
55
 
56
    void on_modeSwitch_clicked();
57
 
58
obs-studio-28.1.2.tar.xz/UI/window-basic-settings-stream.cpp -> obs-studio-29.0.0.tar.xz/UI/window-basic-settings-stream.cpp Changed
99
 
1
@@ -132,6 +132,29 @@
2
        ui->authUsername->setText(QT_UTF8(username));
3
        ui->authPw->setText(QT_UTF8(password));
4
        ui->useAuth->setChecked(use_auth);
5
+
6
+       /* add tooltips for stream key, user, password fields */
7
+       QString file = !App()->IsThemeDark()
8
+                      ? ":/res/images/help.svg"
9
+                      : ":/res/images/help_light.svg";
10
+       QString lStr = "<html>%1 <img src='%2' style=' \
11
+               vertical-align: bottom;  \
12
+               ' /></html>";
13
+
14
+       ui->streamKeyLabel->setText(
15
+           lStr.arg(ui->streamKeyLabel->text(), file));
16
+       ui->streamKeyLabel->setToolTip(
17
+           QTStr("Basic.AutoConfig.StreamPage.StreamKey.ToolTip"));
18
+
19
+       ui->authUsernameLabel->setText(
20
+           lStr.arg(ui->authUsernameLabel->text(), file));
21
+       ui->authUsernameLabel->setToolTip(
22
+           QTStr("Basic.Settings.Stream.Custom.Username.ToolTip"));
23
+
24
+       ui->authPwLabel->setText(
25
+           lStr.arg(ui->authPwLabel->text(), file));
26
+       ui->authPwLabel->setToolTip(
27
+           QTStr("Basic.Settings.Stream.Custom.Password.ToolTip"));
28
    } else {
29
        int idx = ui->service->findText(service);
30
        if (idx == -1) {
31
@@ -305,7 +328,7 @@
32
    if (serviceName == "Dacast") {
33
        ui->streamKeyLabel->setText(
34
            QTStr("Basic.AutoConfig.StreamPage.EncoderKey"));
35
-   } else {
36
+   } else if (!IsCustomService()) {
37
        ui->streamKeyLabel->setText(
38
            QTStr("Basic.AutoConfig.StreamPage.StreamKey"));
39
    }
40
@@ -1244,6 +1267,8 @@
41
        return "jim_nvenc";
42
    if (enc == "h265_texture_amf")
43
        return "h264_texture_amf";
44
+   if (enc == "com.apple.videotoolbox.videoencoder.ave.hevc")
45
+       return "com.apple.videotoolbox.videoencoder.ave.avc";
46
    return "obs_x264";
47
 }
48
 
49
@@ -1255,6 +1280,8 @@
50
        return SIMPLE_ENCODER_NVENC;
51
    if (enc == SIMPLE_ENCODER_AMD_HEVC)
52
        return SIMPLE_ENCODER_AMD;
53
+   if (enc == SIMPLE_ENCODER_APPLE_HEVC)
54
+       return SIMPLE_ENCODER_APPLE_H264;
55
    return SIMPLE_ENCODER_X264;
56
 }
57
 
58
@@ -1315,7 +1342,7 @@
59
 
60
 void OBSBasicSettings::ResetEncoders(bool streamOnly)
61
 {
62
-   QString lastAdvEnc = ui->advOutRecEncoder->currentData().toString();
63
+   QString lastAdvEnc = ui->advOutEncoder->currentData().toString();
64
    QString lastEnc = ui->simpleOutStrEncoder->currentData().toString();
65
    OBSService service = SpawnTempService();
66
    const char **codecs = obs_service_get_supported_video_codecs(service);
67
@@ -1384,6 +1411,10 @@
68
        ui->simpleOutStrEncoder->addItem(
69
            ENCODER_STR("Hardware.QSV.H264"),
70
            QString(SIMPLE_ENCODER_QSV));
71
+   if (service_supports_encoder(codecs, "obs_qsv11_av1"))
72
+       ui->simpleOutStrEncoder->addItem(
73
+           ENCODER_STR("Hardware.QSV.AV1"),
74
+           QString(SIMPLE_ENCODER_QSV_AV1));
75
    if (service_supports_encoder(codecs, "ffmpeg_nvenc"))
76
        ui->simpleOutStrEncoder->addItem(
77
            ENCODER_STR("Hardware.NVENC.H264"),
78
@@ -1420,6 +1451,20 @@
79
                QString(SIMPLE_ENCODER_APPLE_H264));
80
        }
81
    }
82
+#ifdef ENABLE_HEVC
83
+   if (service_supports_encoder(
84
+           codecs, "com.apple.videotoolbox.videoencoder.ave.hevc")
85
+#ifndef __aarch64__
86
+       && os_get_emulation_status() == true
87
+#endif
88
+   ) {
89
+       if (__builtin_available(macOS 13.0, *)) {
90
+           ui->simpleOutStrEncoder->addItem(
91
+               ENCODER_STR("Hardware.Apple.HEVC"),
92
+               QString(SIMPLE_ENCODER_APPLE_HEVC));
93
+       }
94
+   }
95
+#endif
96
 #endif
97
 #undef ENCODER_STR
98
 
99
obs-studio-28.1.2.tar.xz/UI/window-basic-settings.cpp -> obs-studio-29.0.0.tar.xz/UI/window-basic-settings.cpp Changed
201
 
1
@@ -259,13 +259,6 @@
2
             id);
3
 }
4
 
5
-#ifdef _WIN32
6
-void OBSBasicSettings::ToggleDisableAero(bool checked)
7
-{
8
-   SetAeroEnabled(!checked);
9
-}
10
-#endif
11
-
12
 static void PopulateAACBitrates(initializer_list<QComboBox *> boxes)
13
 {
14
    auto &bitrateMap = GetAACEncoderBitrateMap();
15
@@ -380,6 +373,7 @@
16
    /* clang-format off */
17
    HookWidget(ui->language,             COMBO_CHANGED,  GENERAL_CHANGED);
18
    HookWidget(ui->theme,            COMBO_CHANGED,  GENERAL_CHANGED);
19
+   HookWidget(ui->updateChannelBox,     COMBO_CHANGED,  GENERAL_CHANGED);
20
    HookWidget(ui->enableAutoUpdates,    CHECK_CHANGED,  GENERAL_CHANGED);
21
    HookWidget(ui->openStatsOnStartup,   CHECK_CHANGED,  GENERAL_CHANGED);
22
    HookWidget(ui->hideOBSFromCapture,   CHECK_CHANGED,  GENERAL_CHANGED);
23
@@ -587,6 +581,19 @@
24
 #if !defined(_WIN32) && !defined(__APPLE__)
25
    delete ui->enableAutoUpdates;
26
    ui->enableAutoUpdates = nullptr;
27
+   delete ui->updateChannelBox;
28
+   ui->updateChannelBox = nullptr;
29
+   delete ui->updateSettingsGroupBox;
30
+   ui->updateSettingsGroupBox = nullptr;
31
+#elif defined(__APPLE__)
32
+   delete ui->updateChannelBox;
33
+   ui->updateChannelBox = nullptr;
34
+   delete ui->updateChannelLabel;
35
+   ui->updateChannelLabel = nullptr;
36
+#else
37
+   // Hide update section if disabled
38
+   if (App()->IsUpdaterDisabled())
39
+       ui->updateSettingsGroupBox->hide();
40
 #endif
41
 
42
    // Remove the Advanced Audio section if monitoring is not supported, as the monitoring device selection is the only item in the group box.
43
@@ -598,20 +605,6 @@
44
    }
45
 
46
 #ifdef _WIN32
47
-   uint32_t winVer = GetWindowsVersion();
48
-   if (winVer > 0 && winVer < 0x602) {
49
-       // Older than Windows 8
50
-       toggleAero = new QCheckBox(
51
-           QTStr("Basic.Settings.Video.DisableAero"), this);
52
-       QFormLayout *videoLayout = reinterpret_cast<QFormLayout *>(
53
-           ui->videoPage->layout());
54
-       videoLayout->addRow(nullptr, toggleAero);
55
-
56
-       HookWidget(toggleAero, CHECK_CHANGED, VIDEO_CHANGED);
57
-       connect(toggleAero, &QAbstractButton::toggled, this,
58
-           &OBSBasicSettings::ToggleDisableAero);
59
-   }
60
-
61
    if (!SetDisplayAffinitySupported()) {
62
        delete ui->hideOBSFromCapture;
63
        ui->hideOBSFromCapture = nullptr;
64
@@ -823,8 +816,6 @@
65
        SLOT(AdvReplayBufferChanged()));
66
    connect(ui->advRBSecMax, SIGNAL(valueChanged(int)), this,
67
        SLOT(AdvReplayBufferChanged()));
68
-   connect(ui->listWidget, SIGNAL(currentRowChanged(int)), this,
69
-       SLOT(SimpleRecordingEncoderChanged()));
70
 
71
    // Get Bind to IP Addresses
72
    obs_properties_t *ppts = obs_get_output_properties("rtmp_output");
73
@@ -910,10 +901,10 @@
74
        SLOT(AdvOutRecCheckWarnings()));
75
    connect(ui->advOutRecEncoder, SIGNAL(currentIndexChanged(int)), this,
76
        SLOT(AdvOutRecCheckWarnings()));
77
-   AdvOutRecCheckWarnings();
78
 
79
    SimpleRecordingQualityChanged();
80
    AdvOutSplitFileChanged();
81
+   AdvOutRecCheckWarnings();
82
 
83
    UpdateAutomaticReplayBufferCheckboxes();
84
 
85
@@ -1020,7 +1011,7 @@
86
 #define CF_I444_STR QTStr("Basic.Settings.Advanced.Video.ColorFormat.I444")
87
 #define CF_P010_STR QTStr("Basic.Settings.Advanced.Video.ColorFormat.P010")
88
 #define CF_I010_STR QTStr("Basic.Settings.Advanced.Video.ColorFormat.I010")
89
-#define CF_RGB_STR QTStr("Basic.Settings.Advanced.Video.ColorFormat.RGB")
90
+#define CF_BGRA_STR QTStr("Basic.Settings.Advanced.Video.ColorFormat.BGRA")
91
 
92
 void OBSBasicSettings::LoadColorFormats()
93
 {
94
@@ -1029,7 +1020,7 @@
95
    ui->colorFormat->addItem(CF_I444_STR, "I444");
96
    ui->colorFormat->addItem(CF_P010_STR, "P010");
97
    ui->colorFormat->addItem(CF_I010_STR, "I010");
98
-   ui->colorFormat->addItem(CF_RGB_STR, "RGB");
99
+   ui->colorFormat->addItem(CF_BGRA_STR, "RGB"); // Avoid config break
100
 }
101
 
102
 #define AV_FORMAT_DEFAULT_STR \
103
@@ -1215,6 +1206,73 @@
104
        ui->theme->setCurrentIndex(idx);
105
 }
106
 
107
+#ifdef _WIN32
108
+void TranslateBranchInfo(const QString &name, QString &displayName,
109
+            QString &description)
110
+{
111
+   QString translatedName =
112
+       QTStr("Basic.Settings.General.ChannelName." + name.toUtf8());
113
+   QString translatedDesc = QTStr(
114
+       "Basic.Settings.General.ChannelDescription." + name.toUtf8());
115
+
116
+   if (!translatedName.startsWith("Basic.Settings."))
117
+       displayName = translatedName;
118
+   if (!translatedDesc.startsWith("Basic.Settings."))
119
+       description = translatedDesc;
120
+}
121
+#endif
122
+
123
+void OBSBasicSettings::LoadBranchesList()
124
+{
125
+#ifdef _WIN32
126
+   bool configBranchRemoved = true;
127
+   QString configBranch =
128
+       config_get_string(GetGlobalConfig(), "General", "UpdateBranch");
129
+
130
+   for (const UpdateBranch &branch : App()->GetBranches()) {
131
+       if (branch.name == configBranch)
132
+           configBranchRemoved = false;
133
+       if (!branch.is_visible && branch.name != configBranch)
134
+           continue;
135
+
136
+       QString displayName = branch.display_name;
137
+       QString description = branch.description;
138
+
139
+       TranslateBranchInfo(branch.name, displayName, description);
140
+       QString itemDesc = displayName + " - " + description;
141
+
142
+       if (!branch.is_enabled) {
143
+           itemDesc.prepend(" ");
144
+           itemDesc.prepend(QTStr(
145
+               "Basic.Settings.General.UpdateChannelDisabled"));
146
+       } else if (branch.name == "stable") {
147
+           itemDesc.append(" ");
148
+           itemDesc.append(QTStr(
149
+               "Basic.Settings.General.UpdateChannelDefault"));
150
+       }
151
+
152
+       ui->updateChannelBox->addItem(itemDesc, branch.name);
153
+
154
+       // Disable item if branch is disabled
155
+       if (!branch.is_enabled) {
156
+           QStandardItemModel *model =
157
+               dynamic_cast<QStandardItemModel *>(
158
+                   ui->updateChannelBox->model());
159
+           QStandardItem *item =
160
+               model->item(ui->updateChannelBox->count() - 1);
161
+           item->setFlags(Qt::NoItemFlags);
162
+       }
163
+   }
164
+
165
+   // Fall back to default if not yet set or user-selected branch has been removed
166
+   if (configBranch.isEmpty() || configBranchRemoved)
167
+       configBranch = "stable";
168
+
169
+   int idx = ui->updateChannelBox->findData(configBranch);
170
+   ui->updateChannelBox->setCurrentIndex(idx);
171
+#endif
172
+}
173
+
174
 void OBSBasicSettings::LoadGeneralSettings()
175
 {
176
    loading = true;
177
@@ -1226,6 +1284,10 @@
178
    bool enableAutoUpdates = config_get_bool(GetGlobalConfig(), "General",
179
                         "EnableAutoUpdates");
180
    ui->enableAutoUpdates->setChecked(enableAutoUpdates);
181
+
182
+#ifdef _WIN32
183
+   LoadBranchesList();
184
+#endif
185
 #endif
186
    bool openStatsOnStartup = config_get_bool(main->Config(), "General",
187
                          "OpenStatsOnStartup");
188
@@ -1707,16 +1769,6 @@
189
    LoadFPSData();
190
    LoadDownscaleFilters();
191
 
192
-#ifdef _WIN32
193
-   if (toggleAero) {
194
-       bool disableAero =
195
-           config_get_bool(main->Config(), "Video", "DisableAero");
196
-       toggleAero->setChecked(disableAero);
197
-
198
-       aeroWasDisabled = disableAero;
199
-   }
200
-#endif
201
obs-studio-28.1.2.tar.xz/UI/window-basic-settings.hpp -> obs-studio-29.0.0.tar.xz/UI/window-basic-settings.hpp Changed
38
 
1
@@ -120,6 +120,7 @@
2
    int pageIndex = 0;
3
    bool loading = true;
4
    bool forceAuthReload = false;
5
+   bool forceUpdateCheck = false;
6
    std::string savedTheme;
7
    int sampleRateIndex = 0;
8
    int channelIndex = 0;
9
@@ -146,6 +147,7 @@
10
    QString curQSVPreset;
11
    QString curNVENCPreset;
12
    QString curAMDPreset;
13
+   QString curAMDAV1Preset;
14
 
15
    QString curAdvStreamEncoder;
16
    QString curAdvRecordEncoder;
17
@@ -218,12 +220,6 @@
18
        EnableApplyButton(false);
19
    }
20
 
21
-#ifdef _WIN32
22
-   bool aeroWasDisabled = false;
23
-   QCheckBox *toggleAero = nullptr;
24
-   void ToggleDisableAero(bool checked);
25
-#endif
26
-
27
    void HookWidget(QWidget *widget, const char *signal, const char *slot);
28
 
29
    bool QueryChanges();
30
@@ -255,6 +251,7 @@
31
    /* general */
32
    void LoadLanguageList();
33
    void LoadThemeList();
34
+   void LoadBranchesList();
35
 
36
    /* stream */
37
    void InitStreamPage();
38
obs-studio-28.1.2.tar.xz/UI/window-projector.cpp -> obs-studio-29.0.0.tar.xz/UI/window-projector.cpp Changed
83
 
1
@@ -16,11 +16,12 @@
2
 
3
 OBSProjector::OBSProjector(QWidget *widget, obs_source_t *source_, int monitor,
4
               ProjectorType type_)
5
-   : OBSQTDisplay(widget, Qt::Window),
6
-     source(source_),
7
-     removedSignal(obs_source_get_signal_handler(source), "remove",
8
-           OBSSourceRemoved, this)
9
+   : OBSQTDisplay(widget, Qt::Window), weakSource(OBSGetWeakRef(source_))
10
 {
11
+   OBSSource source = GetSource();
12
+   destroyedSignal.Connect(obs_source_get_signal_handler(source),
13
+               "destroy", OBSSourceDestroyed, this);
14
+
15
    isAlwaysOnTop = config_get_bool(GetGlobalConfig(), "BasicWindow",
16
                    "ProjectorAlwaysOnTop");
17
 
18
@@ -103,6 +104,7 @@
19
        GetDisplay(), isMultiview ? OBSRenderMultiview : OBSRender,
20
        this);
21
 
22
+   OBSSource source = GetSource();
23
    if (source)
24
        obs_source_dec_showing(source);
25
 
26
@@ -157,7 +159,7 @@
27
        return;
28
 
29
    OBSBasic *main = reinterpret_cast<OBSBasic *>(App()->GetMainWindow());
30
-   OBSSource source = window->source;
31
+   OBSSource source = window->GetSource();
32
 
33
    uint32_t targetCX;
34
    uint32_t targetCY;
35
@@ -191,11 +193,11 @@
36
            obs_source_dec_showing(source);
37
            obs_source_inc_showing(curSource);
38
            source = curSource;
39
-           window->source = source;
40
+           window->weakSource = OBSGetWeakRef(source);
41
        }
42
    } else if (window->type == ProjectorType::Preview &&
43
           !main->IsPreviewProgramMode()) {
44
-       window->source = nullptr;
45
+       window->weakSource = nullptr;
46
    }
47
 
48
    if (source)
49
@@ -206,7 +208,7 @@
50
    endRegion();
51
 }
52
 
53
-void OBSProjector::OBSSourceRemoved(void *data, calldata_t *params)
54
+void OBSProjector::OBSSourceDestroyed(void *data, calldata_t *params)
55
 {
56
    OBSProjector *window = reinterpret_cast<OBSProjector *>(data);
57
    QMetaObject::invokeMethod(window, "EscapeTriggered");
58
@@ -367,7 +369,7 @@
59
 
60
 OBSSource OBSProjector::GetSource()
61
 {
62
-   return source;
63
+   return OBSGetStrongRef(weakSource);
64
 }
65
 
66
 ProjectorType OBSProjector::GetProjectorType()
67
@@ -410,6 +412,7 @@
68
    int monitor = sender()->property("monitor").toInt();
69
    SetMonitor(monitor);
70
 
71
+   OBSSource source = GetSource();
72
    UpdateProjectorTitle(QT_UTF8(obs_source_get_name(source)));
73
 }
74
 
75
@@ -426,6 +429,7 @@
76
 
77
    savedMonitor = -1;
78
 
79
+   OBSSource source = GetSource();
80
    UpdateProjectorTitle(QT_UTF8(obs_source_get_name(source)));
81
    screen = nullptr;
82
 }
83
obs-studio-28.1.2.tar.xz/UI/window-projector.hpp -> obs-studio-29.0.0.tar.xz/UI/window-projector.hpp Changed
17
 
1
@@ -18,12 +18,12 @@
2
    Q_OBJECT
3
 
4
 private:
5
-   OBSSource source;
6
-   OBSSignal removedSignal;
7
+   OBSWeakSourceAutoRelease weakSource;
8
+   OBSSignal destroyedSignal;
9
 
10
    static void OBSRenderMultiview(void *data, uint32_t cx, uint32_t cy);
11
    static void OBSRender(void *data, uint32_t cx, uint32_t cy);
12
-   static void OBSSourceRemoved(void *data, calldata_t *params);
13
+   static void OBSSourceDestroyed(void *data, calldata_t *params);
14
 
15
    void mousePressEvent(QMouseEvent *event) override;
16
    void mouseDoubleClickEvent(QMouseEvent *event) override;
17
obs-studio-28.1.2.tar.xz/cmake/Modules/CompilerConfig.cmake -> obs-studio-29.0.0.tar.xz/cmake/Modules/CompilerConfig.cmake Changed
11
 
1
@@ -67,7 +67,8 @@
2
     /DUNICODE
3
     /D_UNICODE
4
     /D_CRT_SECURE_NO_WARNINGS
5
-    /D_CRT_NONSTDC_NO_WARNINGS)
6
+    /D_CRT_NONSTDC_NO_WARNINGS
7
+    /utf-8)
8
 
9
   add_link_options(
10
     "LINKER:/OPT:REF"
11
obs-studio-28.1.2.tar.xz/cmake/Modules/FindGio.cmake -> obs-studio-29.0.0.tar.xz/cmake/Modules/FindGio.cmake Changed
10
 
1
@@ -21,7 +21,7 @@
2
 
3
 find_library(
4
   GIO_LIB
5
-  NAMES ${_GIO_LIBRARIES} gio-2.0 libgio-2.0 gio-unix-2.0
6
+  NAMES gio-2.0 libgio-2.0 gio-unix-2.0
7
   HINTS ${_GIO_LIBRARY_DIRS}
8
   PATHS /usr/lib /usr/local/lib /opt/local/lib /sw/lib)
9
 
10
obs-studio-29.0.0.tar.xz/cmake/Modules/FindLibva.cmake Added
51
 
1
@@ -0,0 +1,49 @@
2
+# * Try to find Libva, once done this will define
3
+#
4
+# * LIBVA_FOUND - system has Libva
5
+# * LIBVA_INCLUDE_DIRS - the Libva include directory
6
+# * LIBVA_LIBRARIES - the libraries needed to use Libva
7
+# * LIBVA_DEFINITIONS - Compiler switches required for using Libva
8
+
9
+# Use pkg-config to get the directories and then use these values in the
10
+# find_path() and find_library() calls
11
+
12
+find_package(PkgConfig QUIET)
13
+if(PKG_CONFIG_FOUND)
14
+  pkg_check_modules(_LIBVA libva)
15
+endif()
16
+
17
+find_path(
18
+  LIBVA_INCLUDE_DIR
19
+  NAMES va.h
20
+  HINTS ${_LIBVA_INCLUDE_DIRS}
21
+  PATHS /usr/include /usr/local/include /opt/local/include
22
+  PATH_SUFFIXES va/)
23
+
24
+find_library(
25
+  LIBVA_LIB
26
+  NAMES ${_LIBVA_LIBRARIES} libva
27
+  HINTS ${_LIBVA_LIBRARY_DIRS}
28
+  PATHS /usr/lib /usr/local/lib /opt/local/lib)
29
+
30
+include(FindPackageHandleStandardArgs)
31
+find_package_handle_standard_args(Libva REQUIRED_VARS LIBVA_LIB
32
+                                                      LIBVA_INCLUDE_DIR)
33
+mark_as_advanced(LIBVA_INCLUDE_DIR LIBVA_LIB)
34
+
35
+if(LIBVA_FOUND)
36
+  set(LIBVA_INCLUDE_DIRS ${LIBVA_INCLUDE_DIR})
37
+  set(LIBVA_LIBRARIES ${LIBVA_LIB})
38
+
39
+  if(NOT TARGET Libva::va)
40
+    if(IS_ABSOLUTE "${LIBVA_LIBRARIES}")
41
+      add_library(Libva::va UNKNOWN IMPORTED)
42
+      set_target_properties(Libva::va PROPERTIES IMPORTED_LOCATION
43
+                                                 "${LIBVA_LIBRARIES}")
44
+    else()
45
+      add_library(Libva::va INTERFACE IMPORTED)
46
+      set_target_properties(Libva::va PROPERTIES IMPORTED_LIBNAME
47
+                                                 "${LIBVA_LIBRARIES}")
48
+    endif()
49
+  endif()
50
+endif()
51
obs-studio-28.1.2.tar.xz/cmake/Modules/ObsDefaults_Linux.cmake -> obs-studio-29.0.0.tar.xz/cmake/Modules/ObsDefaults_Linux.cmake Changed
26
 
1
@@ -49,7 +49,7 @@
2
                └ libobs
3
                └ obs-plugins
4
                └ obs-studio
5
-   
6
+   #
7
 
8
   if(CMAKE_SIZEOF_VOID_P EQUAL 8)
9
     set(_ARCH_SUFFIX 64)
10
@@ -114,14 +114,13 @@
11
   set(CPACK_RESOURCE_FILE_LICENSE
12
       "${CMAKE_SOURCE_DIR}/UI/data/license/gplv2.txt")
13
   set(CPACK_PACKAGE_VERSION "${OBS_VERSION_CANONICAL}-${OBS_BUILD_NUMBER}")
14
-  set(CPACK_STRIP_FILES "bin/obs" "bin/obs-ffmpeg-mux")
15
-  set(CPACK_SOURCE_STRIP_FILES "")
16
   set(CPACK_PACKAGE_EXECUTABLES "obs")
17
 
18
   if(OS_LINUX AND NOT LINUX_PORTABLE)
19
     set(CPACK_GENERATOR "DEB")
20
     set(CPACK_DEBIAN_PACKAGE_SHLIBDEPS ON)
21
     set(CPACK_SET_DESTDIR ON)
22
+    set(CPACK_DEBIAN_DEBUGINFO_PACKAGE ON)
23
   elseif(OS_FREEBSD)
24
     option(ENABLE_CPACK_GENERATOR
25
            "Enable FreeBSD CPack generator (experimental)" OFF)
26
obs-studio-28.1.2.tar.xz/cmake/Modules/VersionConfig.cmake -> obs-studio-29.0.0.tar.xz/cmake/Modules/VersionConfig.cmake Changed
15
 
1
@@ -32,6 +32,13 @@
2
         string(REPLACE "." ";" _CANONICAL_SPLIT ${_CANONICAL_SPLIT})
3
         list(GET _CANONICAL_SPLIT 0 1 2 _OBS_VERSION_CANONICAL)
4
         string(REPLACE "." ";" _OBS_VERSION ${_OBS_VERSION})
5
+        # Get 8-character commit hash without "g" prefix
6
+        foreach(VERSION_PART ${_CANONICAL_SPLIT})
7
+          if(VERSION_PART MATCHES "^g")
8
+            string(SUBSTRING ${VERSION_PART}, 1, 8, OBS_COMMIT)
9
+            break()
10
+          endif()
11
+        endforeach()
12
       endif()
13
     endif()
14
   endif()
15
obs-studio-28.1.2.tar.xz/deps/libcaption/caption/caption.h -> obs-studio-29.0.0.tar.xz/deps/libcaption/caption/caption.h Changed
12
 
1
@@ -134,10 +134,6 @@
2
 /*! \brief
3
     \param
4
 */
5
-#define CAPTION_FRAME_DUMP_BUF_SIZE 8192
6
-size_t caption_frame_dump_buffer(caption_frame_t* frame, utf8_char_t* buf);
7
-void caption_frame_dump(caption_frame_t* frame);
8
-
9
 #ifdef __cplusplus
10
 }
11
 #endif
12
obs-studio-28.1.2.tar.xz/deps/libcaption/src/caption.c -> obs-studio-29.0.0.tar.xz/deps/libcaption/src/caption.c Changed
59
 
1
@@ -403,57 +403,3 @@
2
 
3
     return size;
4
 }
5
-////////////////////////////////////////////////////////////////////////////////
6
-size_t caption_frame_dump_buffer(caption_frame_t* frame, utf8_char_t* buf)
7
-{
8
-    int r, c;
9
-    size_t bytes, total = 0;
10
-    bytes = sprintf(buf, "   timestamp: %f\n   row: %02d    col: %02d    roll-up: %d\n",
11
-        frame->timestamp, frame->state.row, frame->state.col, caption_frame_rollup(frame));
12
-    total += bytes, buf += bytes;
13
-    bytes = sprintf(buf, "   00000000001111111111222222222233\t   00000000001111111111222222222233\n"
14
-                         "   01234567890123456789012345678901\t   01234567890123456789012345678901\n"
15
-                         "  %s--------------------------------%s\t  %s--------------------------------%s\n",
16
-        EIA608_CHAR_BOX_DRAWINGS_LIGHT_DOWN_AND_RIGHT, EIA608_CHAR_BOX_DRAWINGS_LIGHT_DOWN_AND_LEFT,
17
-        EIA608_CHAR_BOX_DRAWINGS_LIGHT_DOWN_AND_RIGHT, EIA608_CHAR_BOX_DRAWINGS_LIGHT_DOWN_AND_LEFT);
18
-    total += bytes;
19
-    buf += bytes;
20
-
21
-    for (r = 0; r < SCREEN_ROWS; ++r) {
22
-        bytes = sprintf(buf, "%02d%s", r, EIA608_CHAR_VERTICAL_LINE);
23
-        total += bytes, buf += bytes;
24
-
25
-        // front buffer
26
-        for (c = 0; c < SCREEN_COLS; ++c) {
27
-            caption_frame_cell_t* cell = frame_buffer_cell(&frame->front, r, c);
28
-            bytes = utf8_char_copy(buf, (!cell || 0 == cell->data0) ? EIA608_CHAR_SPACE : &cell->data0);
29
-            total += bytes, buf += bytes;
30
-        }
31
-
32
-        bytes = sprintf(buf, "%s\t%02d%s", EIA608_CHAR_VERTICAL_LINE, r, EIA608_CHAR_VERTICAL_LINE);
33
-        total += bytes, buf += bytes;
34
-
35
-        // back buffer
36
-        for (c = 0; c < SCREEN_COLS; ++c) {
37
-            caption_frame_cell_t* cell = frame_buffer_cell(&frame->back, r, c);
38
-            bytes = utf8_char_copy(buf, (!cell || 0 == cell->data0) ? EIA608_CHAR_SPACE : &cell->data0);
39
-            total += bytes, buf += bytes;
40
-        }
41
-
42
-        bytes = sprintf(buf, "%s\n", EIA608_CHAR_VERTICAL_LINE);
43
-        total += bytes, buf += bytes;
44
-    }
45
-
46
-    bytes = sprintf(buf, "  %s--------------------------------%s\t  %s--------------------------------%s\n",
47
-        EIA608_CHAR_BOX_DRAWINGS_LIGHT_UP_AND_RIGHT, EIA608_CHAR_BOX_DRAWINGS_LIGHT_UP_AND_LEFT,
48
-        EIA608_CHAR_BOX_DRAWINGS_LIGHT_UP_AND_RIGHT, EIA608_CHAR_BOX_DRAWINGS_LIGHT_UP_AND_LEFT);
49
-    total += bytes, buf += bytes;
50
-    return total;
51
-}
52
-
53
-void caption_frame_dump(caption_frame_t* frame)
54
-{
55
-    utf8_char_t buffCAPTION_FRAME_DUMP_BUF_SIZE;
56
-    caption_frame_dump_buffer(frame, buff);
57
-    fprintf(stderr, "%s\n", buff);
58
-}
59
obs-studio-28.1.2.tar.xz/deps/libcaption/src/utf8.c -> obs-studio-29.0.0.tar.xz/deps/libcaption/src/utf8.c Changed
13
 
1
@@ -220,7 +220,10 @@
2
         }
3
     }
4
 
5
-    data*size = 0;
6
+    if (data) {
7
+        data*size = 0;
8
+    }
9
+    
10
     return data;
11
 }
12
 
13
obs-studio-28.1.2.tar.xz/deps/media-playback/media-playback/media.c -> obs-studio-29.0.0.tar.xz/deps/media-playback/media-playback/media.c Changed
23
 
1
@@ -353,6 +353,12 @@
2
    struct mp_decode *d = &m->a;
3
    struct obs_source_audio audio = {0};
4
    AVFrame *f = d->frame;
5
+   int channels;
6
+#if LIBAVFORMAT_VERSION_INT < AV_VERSION_INT(59, 19, 100)
7
+   channels = f->channels;
8
+#else
9
+   channels = f->ch_layout.nb_channels;
10
+#endif
11
 
12
    if (!mp_media_can_play_frame(m, d))
13
        return;
14
@@ -365,7 +371,7 @@
15
        audio.datai = f->datai;
16
 
17
    audio.samples_per_sec = f->sample_rate * m->speed / 100;
18
-   audio.speakers = convert_speaker_layout(f->channels);
19
+   audio.speakers = convert_speaker_layout(channels);
20
    audio.format = convert_sample_format(f->format);
21
    audio.frames = f->nb_samples;
22
 
23
obs-studio-28.1.2.tar.xz/deps/obs-scripting/obs-scripting-python-import.c -> obs-studio-29.0.0.tar.xz/deps/obs-scripting/obs-scripting-python-import.c Changed
29
 
1
@@ -71,9 +71,9 @@
2
 
3
    char tempPATH_MAX;
4
 
5
-   sprintf(cur_version, VERSION_PATTERN, PY_MAJOR_VERSION_MAX,
6
-       PY_MINOR_VERSION_MAX);
7
-   sprintf(temp, FILE_PATTERN, cur_version);
8
+   snprintf(cur_version, sizeof(cur_version), VERSION_PATTERN,
9
+        PY_MAJOR_VERSION_MAX, PY_MINOR_VERSION_MAX);
10
+   snprintf(temp, sizeof(temp), FILE_PATTERN, cur_version);
11
 
12
    dstr_cat(&lib_candidate_path, temp);
13
 
14
@@ -87,10 +87,10 @@
15
            break;
16
        }
17
 
18
-       sprintf(cur_version, VERSION_PATTERN, PY_MAJOR_VERSION_MAX,
19
-           minor_version);
20
-       sprintf(next_version, VERSION_PATTERN, PY_MAJOR_VERSION_MAX,
21
-           --minor_version);
22
+       snprintf(cur_version, sizeof(cur_version), VERSION_PATTERN,
23
+            PY_MAJOR_VERSION_MAX, minor_version);
24
+       snprintf(next_version, sizeof(next_version), VERSION_PATTERN,
25
+            PY_MAJOR_VERSION_MAX, --minor_version);
26
        dstr_replace(&lib_candidate_path, cur_version, next_version);
27
    } while (minor_version > 5);
28
 
29
obs-studio-28.1.2.tar.xz/deps/obs-scripting/obs-scripting-python.c -> obs-studio-29.0.0.tar.xz/deps/obs-scripting/obs-scripting-python.c Changed
12
 
1
@@ -1666,8 +1666,8 @@
2
    if (python_path && *python_path) {
3
 #ifdef __APPLE__
4
        char tempPATH_MAX;
5
-       sprintf(temp, "%s/Python.framework/Versions/Current",
6
-           python_path);
7
+       snprintf(temp, sizeof(temp),
8
+            "%s/Python.framework/Versions/Current", python_path);
9
        os_utf8_to_wcs(temp, 0, home_path, PATH_MAX);
10
        Py_SetPythonHome(home_path);
11
 #else
12
obs-studio-28.1.2.tar.xz/docs/sphinx/reference-core.rst -> obs-studio-29.0.0.tar.xz/docs/sphinx/reference-core.rst Changed
75
 
1
@@ -363,7 +363,8 @@
2
 
3
 .. function:: obs_data_t *obs_save_source(obs_source_t *source)
4
 
5
-   :return: A new reference to a source's saved data
6
+   :return: A new reference to a source's saved data. Use
7
+            :c:func:`obs_data_release()` to release it when complete.
8
 
9
 ---------------------
10
 
11
@@ -478,13 +479,19 @@
12
 
13
 .. function:: void obs_set_master_volume(float volume)
14
 
15
-   Sets the master user volume.
16
+   No-op, only exists to keep ABI compatibility.
17
+
18
+   .. deprecated:: 29.0
19
 
20
 ---------------------
21
 
22
 .. function:: float obs_get_master_volume(void)
23
 
24
-   :return: The master user volume
25
+   No-op, only exists to keep ABI compatibility.
26
+
27
+   :return: Always returns 1
28
+
29
+   .. deprecated:: 29.0
30
 
31
 ---------------------
32
 
33
@@ -557,7 +564,8 @@
34
 
35
 .. function:: signal_handler_t *obs_get_signal_handler(void)
36
 
37
-   :return: The primary obs signal handler
38
+   :return: The primary obs signal handler. Should not be manually freed,
39
+            as its lifecycle is managed by libobs.
40
 
41
    See :ref:`core_signal_handler_reference` for more information on
42
    core signals.
43
@@ -566,7 +574,8 @@
44
 
45
 .. function:: proc_handler_t *obs_get_proc_handler(void)
46
 
47
-   :return: The primary obs procedure handler
48
+   :return: The primary obs procedure handler. Should not be manually freed,
49
+            as its lifecycle is managed by libobs.
50
 
51
 
52
 .. _core_signal_handler_reference:
53
@@ -587,6 +596,10 @@
54
    Called when a source has been removed (:c:func:`obs_source_remove()`
55
    has been called on the source).
56
 
57
+**source_update** (ptr source)
58
+
59
+   Called when a source's settings have been updated.
60
+
61
 **source_save** (ptr source)
62
 
63
    Called when a source is being saved.
64
@@ -639,10 +652,6 @@
65
 
66
    Called when :c:func:`obs_set_output_source()` has been called.
67
 
68
-**master_volume** (in out float volume)
69
-
70
-   Called when the master volume has changed.
71
-
72
 **hotkey_layout_change** ()
73
 
74
    Called when the hotkey layout has changed.
75
obs-studio-28.1.2.tar.xz/docs/sphinx/reference-encoders.rst -> obs-studio-29.0.0.tar.xz/docs/sphinx/reference-encoders.rst Changed
46
 
1
@@ -332,7 +332,7 @@
2
 .. function:: obs_encoder_t *obs_encoder_get_ref(obs_encoder_t *encoder)
3
 
4
    Returns an incremented reference if still valid, otherwise returns
5
-   *NULL*.
6
+   *NULL*. Release with :c:func:`obs_encoder_release()`.
7
 
8
 ---------------------
9
 
10
@@ -437,7 +437,8 @@
11
 .. function:: obs_data_t *obs_encoder_defaults(const char *id)
12
               obs_data_t *obs_encoder_get_defaults(const obs_encoder_t *encoder)
13
 
14
-   :return: An incremented reference to the encoder's default settings
15
+   :return: An incremented reference to the encoder's default settings.
16
+            Release with :c:func:`obs_data_release()`.
17
 
18
 ---------------------
19
 
20
@@ -461,19 +462,22 @@
21
 
22
 .. function:: obs_data_t *obs_encoder_get_settings(const obs_encoder_t *encoder)
23
 
24
-   :return: An incremented reference to the encoder's settings
25
+   :return: An incremented reference to the encoder's settings. Release with
26
+            :c:func:`obs_data_release()`.
27
 
28
 ---------------------
29
 
30
 .. function:: signal_handler_t *obs_encoder_get_signal_handler(const obs_encoder_t *encoder)
31
 
32
-   :return: The signal handler of the encoder
33
+   :return: The signal handler of the encoder. Should not be manually freed,
34
+            as its lifecycle is managed by libobs.
35
 
36
 ---------------------
37
 
38
 .. function:: proc_handler_t *obs_encoder_get_proc_handler(const obs_encoder_t *encoder)
39
 
40
-   :return: The procedure handler of the encoder
41
+   :return: The procedure handler of the encoder. Should not be manually freed,
42
+            as its lifecycle is managed by libobs.
43
 
44
 ---------------------
45
 
46
obs-studio-28.1.2.tar.xz/docs/sphinx/reference-frontend-api.rst -> obs-studio-29.0.0.tar.xz/docs/sphinx/reference-frontend-api.rst Changed
142
 
1
@@ -79,7 +79,7 @@
2
    - **OBS_FRONTEND_EVENT_TRANSITION_DURATION_CHANGED**
3
 
4
      Triggered when the transition duration has been changed by the
5
-    user.
6
+     user.
7
 
8
    - **OBS_FRONTEND_EVENT_TBAR_VALUE_CHANGED**
9
 
10
@@ -178,6 +178,17 @@
11
 
12
      Triggered when the virtual camera is stopped.
13
 
14
+   - **OBS_FRONTEND_EVENT_THEME_CHANGED**
15
+
16
+     Triggered when the theme is changed.
17
+
18
+     .. versionadded:: 29.0.0
19
+
20
+   - **OBS_FRONTEND_EVENT_SCREENSHOT_TAKEN**
21
+
22
+     Triggered when a screenshot is taken.
23
+
24
+     .. versionadded:: 29.0.0
25
 
26
 .. struct:: obs_frontend_source_list
27
 
28
@@ -259,7 +270,8 @@
29
 
30
 .. function:: obs_source_t *obs_frontend_get_current_scene(void)
31
 
32
-   :return: A new reference to the currently active scene
33
+   :return: A new reference to the currently active scene. Release with
34
+            :c:func:`obs_source_release()`.
35
 
36
 ---------------------------------------
37
 
38
@@ -280,7 +292,8 @@
39
 
40
 .. function:: obs_source_t *obs_frontend_get_current_transition(void)
41
 
42
-   :return: A new reference to the currently active transition
43
+   :return: A new reference to the currently active transition.
44
+            Release with :c:func:`obs_source_release()`.
45
 
46
 ---------------------------------------
47
 
48
@@ -605,19 +618,22 @@
49
 
50
 .. function:: obs_output_t *obs_frontend_get_streaming_output(void)
51
 
52
-   :return: A new reference to the current streaming output
53
+   :return: A new reference to the current streaming output.
54
+            Release with :c:func:`obs_output_release()`.
55
 
56
 ---------------------------------------
57
 
58
 .. function:: obs_output_t *obs_frontend_get_recording_output(void)
59
 
60
-   :return: A new reference to the current srecording output
61
+   :return: A new reference to the current recording output.
62
+            Release with :c:func:`obs_output_release()`.
63
 
64
 ---------------------------------------
65
 
66
 .. function:: obs_output_t *obs_frontend_get_replay_buffer_output(void)
67
 
68
-   :return: A new reference to the current replay buffer output
69
+   :return: A new reference to the current replay buffer output.
70
+            Release with :c:func:`obs_output_release()`.
71
 
72
 ---------------------------------------
73
 
74
@@ -643,7 +659,8 @@
75
 
76
 .. function:: obs_service_t *obs_frontend_get_streaming_service(void)
77
 
78
-   :return: A new reference to the current streaming service object
79
+   :return: The current streaming service object. Does not increment the
80
+            reference.
81
 
82
 ---------------------------------------
83
 
84
@@ -677,7 +694,8 @@
85
 .. function:: obs_source_t *obs_frontend_get_current_preview_scene(void)
86
 
87
    :return: A new reference to the current preview scene if studio mode
88
-            is active, or *NULL* if studio mode is not active.
89
+            is active, or *NULL* if studio mode is not active. Release
90
+            with :c:func:`obs_source_release()`.
91
 
92
 ---------------------------------------
93
 
94
@@ -721,7 +739,8 @@
95
 
96
 .. function:: obs_output_t *obs_frontend_get_virtualcam_output(void)
97
 
98
-   :return: A new reference to the current virtual camera output
99
+   :return: A new reference to the current virtual camera output.
100
+            Release with :c:func:`obs_output_release()`.
101
 
102
 ---------------------------------------
103
 
104
@@ -785,3 +804,37 @@
105
 .. function:: const char *obs_frontend_get_locale_string(const char *string)
106
 
107
    :return: Gets the frontend translation of a given string.
108
+
109
+---------------------------------------
110
+
111
+.. function:: bool obs_frontend_is_theme_dark(void)
112
+
113
+   :return: Checks if the current theme is dark or light.
114
+
115
+   .. versionadded:: 29.0.0
116
+
117
+---------------------------------------
118
+
119
+.. function:: char *obs_frontend_get_last_recording(void)
120
+
121
+   :return: The file path of the last recording. Free with :c:func:`bfree()`
122
+
123
+   .. versionadded:: 29.0.0
124
+
125
+---------------------------------------
126
+
127
+.. function:: char *obs_frontend_get_last_screenshot(void)
128
+
129
+   :return: The file path of the last screenshot taken. Free with
130
+            :c:func:`bfree()`
131
+
132
+   .. versionadded:: 29.0.0
133
+
134
+---------------------------------------
135
+
136
+.. function:: char *obs_frontend_get_last_replay(void)
137
+
138
+   :return: The file path of the last replay buffer saved. Free with
139
+            :c:func:`bfree()`
140
+
141
+   .. versionadded:: 29.0.0
142
obs-studio-28.1.2.tar.xz/docs/sphinx/reference-libobs-callback.rst -> obs-studio-29.0.0.tar.xz/docs/sphinx/reference-libobs-callback.rst Changed
75
 
1
@@ -22,7 +22,9 @@
2
 
3
 .. function:: void calldata_free(calldata_t *data)
4
 
5
-   Frees a calldata structure.
6
+   Frees a calldata structure. Should only be used if :c:func:`calldata_init()`
7
+   was used. If the object is received as a callback parameter, this function
8
+   should not be used.
9
 
10
    :param data: Calldata structure
11
 
12
@@ -110,7 +112,9 @@
13
 
14
 .. function:: void *calldata_ptr(const calldata_t *data, const char *name)
15
 
16
-   Gets a pointer parameter.
17
+   Gets a pointer parameter. For example, :ref:`core_signal_handler_reference`
18
+   that have ``ptr source`` as a parameter requires this function to get the
19
+   pointer, which can be casted to :c:type:`obs_source_t`. Does not have to be freed.
20
 
21
    :param data: Calldata structure
22
    :param name: Parameter name
23
@@ -188,11 +192,14 @@
24
 
25
 .. function:: void signal_handler_connect(signal_handler_t *handler, const char *signal, signal_callback_t callback, void *data)
26
 
27
-   Connect a callback to a signal on a signal handler.
28
+   Connects a callback to a signal on a signal handler. Does nothing
29
+   if the combination of ``signal``, ``callback``, and ``data``
30
+   is already connected to the handler.
31
 
32
    :param handler:  Signal handler object
33
+   :param signal:   Name of signal to handle
34
    :param callback: Signal callback
35
-   :param data:     Private data passed the callback
36
+   :param data:     Private data passed to the callback
37
 
38
    For scripting, use :py:func:`signal_handler_connect`.
39
 
40
@@ -200,23 +207,29 @@
41
 
42
 .. function:: void signal_handler_connect_ref(signal_handler_t *handler, const char *signal, signal_callback_t callback, void *data)
43
 
44
-   Connect a callback to a signal on a signal handler, and increments
45
+   Connects a callback to a signal on a signal handler, and increments
46
    the handler's internal reference counter, preventing it from being
47
-   destroyed until the signal has been disconnected.
48
+   destroyed until the signal has been disconnected. Even if the combination of
49
+   ``signal``, ``callback``, and ``data`` is already connected to the handler,
50
+   the reference counter is still incremented.
51
 
52
    :param handler:  Signal handler object
53
+   :param signal:   Name of signal to handle
54
    :param callback: Signal callback
55
-   :param data:     Private data passed the callback
56
+   :param data:     Private data passed to the callback
57
 
58
 ---------------------
59
 
60
 .. function:: void signal_handler_disconnect(signal_handler_t *handler, const char *signal, signal_callback_t callback, void *data)
61
 
62
-   Disconnects a callback from a signal on a signal handler.
63
+   Disconnects a callback from a signal on a signal handler. Does nothing
64
+   if the combination of ``signal``, ``callback``, and ``data``
65
+   is not yet connected to the handler.
66
 
67
    :param handler:  Signal handler object
68
+   :param signal:   Name of signal that was handled
69
    :param callback: Signal callback
70
-   :param data:     Private data passed the callback
71
+   :param data:     Private data passed to the callback
72
 
73
    For scripting, use :py:func:`signal_handler_disconnect`.
74
 
75
obs-studio-28.1.2.tar.xz/docs/sphinx/reference-libobs-util-platform.rst -> obs-studio-29.0.0.tar.xz/docs/sphinx/reference-libobs-util-platform.rst Changed
29
 
1
@@ -447,6 +447,14 @@
2
 
3
 ---------------------
4
 
5
+.. function:: uint64_t os_get_sys_total_size(void)
6
+
7
+   Returns the amount of memory installed.
8
+
9
+   .. versionadded:: 29.0.0
10
+
11
+---------------------
12
+
13
 .. struct:: os_proc_memory_usage
14
 
15
    Memory usage structure.
16
@@ -483,7 +491,8 @@
17
 
18
 .. function:: bool os_get_emulation_status(void)
19
 
20
-   Returns true if the current process is a x64 binary and is being emulated or translated
21
-   by the host operating system. On macOS, it returns true when a x64 binary is 
22
-   being translated by Rosetta and running on Apple Silicon Macs. This function is not yet
23
-   implemented on Windows and Linux and will always return false on those platforms.
24
+   Returns true if the current process is an x64 binary and is being emulated or translated
25
+   by the host operating system. On macOS, it returns true when an x64 binary is 
26
+   being translated by Rosetta and running on Apple Silicon Macs. On Windows, it 
27
+   returns true when an x64 binary is being emulated on Windows ARM64 PCs. On all other 
28
+   platforms, it will always returns false.
29
obs-studio-28.1.2.tar.xz/docs/sphinx/reference-outputs.rst -> obs-studio-29.0.0.tar.xz/docs/sphinx/reference-outputs.rst Changed
46
 
1
@@ -345,7 +345,7 @@
2
 .. function:: obs_output_t *obs_output_get_ref(obs_output_t *output)
3
 
4
    Returns an incremented reference if still valid, otherwise returns
5
-   *NULL*.
6
+   *NULL*. Release with :c:func:`obs_output_release()`.
7
 
8
 ---------------------
9
 
10
@@ -443,7 +443,8 @@
11
 
12
 .. function:: obs_data_t *obs_output_defaults(const char *id)
13
 
14
-   :return: An incremented reference to the output's default settings
15
+   :return: An incremented reference to the output's default settings.
16
+            Release with :c:func:`obs_data_release()`.
17
 
18
 ---------------------
19
 
20
@@ -488,19 +489,22 @@
21
 
22
 .. function:: obs_data_t *obs_output_get_settings(const obs_output_t *output)
23
 
24
-   :return: An incremented reference to the output's settings
25
+   :return: An incremented reference to the output's settings. Release with
26
+            :c:func:`obs_data_release()`.
27
 
28
 ---------------------
29
 
30
 .. function:: signal_handler_t *obs_output_get_signal_handler(const obs_output_t *output)
31
 
32
-   :return: The signal handler of the output
33
+   :return: The signal handler of the output. Should not be manually freed,
34
+            as its lifecycle is managed by libobs.
35
 
36
 ---------------------
37
 
38
 .. function:: proc_handler_t *obs_output_get_proc_handler(const obs_output_t *output)
39
 
40
-   :return: The procedure handler of the output
41
+   :return: The procedure handler of the output. Should not be manually freed,
42
+            as its lifecycle is managed by libobs.
43
 
44
 ---------------------
45
 
46
obs-studio-28.1.2.tar.xz/docs/sphinx/reference-properties.rst -> obs-studio-29.0.0.tar.xz/docs/sphinx/reference-properties.rst Changed
113
 
1
@@ -220,7 +220,8 @@
2
 
3
 .. function:: obs_property_t *obs_properties_add_color(obs_properties_t *props, const char *name, const char *description)
4
 
5
-   Adds a color property without alpha.
6
+   Adds a color property without alpha (stored internally with an alpha value of 255).
7
+   The color can be retrieved from an :c:type:`obs_data_t` object by using :c:func:`obs_data_get_int()`.
8
 
9
    :param    name:        Setting identifier string
10
    :param    description: Localized name shown to user
11
@@ -230,7 +231,8 @@
12
 
13
 .. function:: obs_property_t *obs_properties_add_color_alpha(obs_properties_t *props, const char *name, const char *description)
14
 
15
-   Adds a color property with alpha.
16
+   Adds a color property with alpha. The color can be retrieved from an
17
+   :c:type:`obs_data_t` object by using :c:func:`obs_data_get_int()`.
18
 
19
    :param    name:        Setting identifier string
20
    :param    description: Localized name shown to user
21
@@ -266,7 +268,8 @@
22
 
23
 .. function:: obs_property_t *obs_properties_add_font(obs_properties_t *props, const char *name, const char *description)
24
 
25
-   Adds a font property.
26
+   Adds a font property. The font can be retrieved from an :c:type:`obs_data_t`
27
+   object by using :c:func:`obs_data_get_obj()`.
28
 
29
    :param    name:        Setting identifier string
30
    :param    description: Localized name shown to user
31
@@ -276,7 +279,8 @@
32
 
33
 .. function:: obs_property_t *obs_properties_add_editable_list(obs_properties_t *props, const char *name, const char *description, enum obs_editable_list_type type, const char *filter, const char *default_path)
34
 
35
-   Adds a list in which the user can add/insert/remove items.
36
+   Adds a list in which the user can add/insert/remove items. The items can be
37
+   retrieved from an :c:type:`obs_data_t` object by using :c:func:`obs_data_get_array()`.
38
 
39
    :param    name:         Setting identifier string
40
    :param    description:  Localized name shown to user
41
@@ -576,7 +580,9 @@
42
               void obs_property_set_modified_callback2(obs_property_t *p, obs_property_modified2_t modified2, void *priv)
43
 
44
    Allows the ability to change the properties depending on what
45
-   settings are used by the user.
46
+   settings are used by the user. The callback should return ``true``
47
+   if the property widgets need to be refreshed due to changes to the
48
+   property layout.
49
 
50
    Relevant data types used with these functions:
51
 
52
@@ -649,36 +655,60 @@
53
 
54
    Adds a string to a string list.
55
 
56
+   :param    name: Localized name shown to user
57
+   :param    val:  The actual string value stored and will be returned by :c:func:`obs_data_get_string`
58
+   :returns: The index of the list item.
59
+
60
 ---------------------
61
 
62
 .. function:: size_t obs_property_list_add_int(obs_property_t *p, const char *name, long long val)
63
 
64
    Adds an integer to a integer list.
65
 
66
+   :param    name: Localized name shown to user
67
+   :param    val:  The actual int value stored and will be returned by :c:func:`obs_data_get_int`
68
+   :returns: The index of the list item.
69
+
70
 ---------------------
71
 
72
 .. function:: size_t obs_property_list_add_float(obs_property_t *p, const char *name, double val)
73
 
74
    Adds a floating point to a floating point list.
75
 
76
+   :param    name: Localized name shown to user
77
+   :param    val:  The actual float value stored and will be returned by :c:func:`obs_data_get_double`
78
+   :returns: The index of the list item.
79
+
80
 ---------------------
81
 
82
 .. function:: void obs_property_list_insert_string(obs_property_t *p, size_t idx, const char *name, const char *val)
83
 
84
    Inserts a string in to a string list.
85
 
86
+   :param    idx:  The index of the list item
87
+   :param    name: Localized name shown to user
88
+   :param    val:  The actual string value stored and will be returned by :c:func:`obs_data_get_string`
89
+
90
 ---------------------
91
 
92
 .. function:: void obs_property_list_insert_int(obs_property_t *p, size_t idx, const char *name, long long val)
93
 
94
    Inserts an integer in to an integer list.
95
 
96
+   :param    idx:  The index of the list item
97
+   :param    name: Localized name shown to user
98
+   :param    val:  The actual int value stored and will be returned by :c:func:`obs_data_get_int`
99
+
100
 ---------------------
101
 
102
 .. function:: void obs_property_list_insert_float(obs_property_t *p, size_t idx, const char *name, double val)
103
 
104
    Inserts a floating point in to a floating point list.
105
 
106
+   :param    idx:  The index of the list item.
107
+   :param    name: Localized name shown to user
108
+   :param    val:  The actual float value stored and will be returned by :c:func:`obs_data_get_double`
109
+
110
 ---------------------
111
 
112
 .. function:: void obs_property_list_item_disable(obs_property_t *p, size_t idx, bool disabled)
113
obs-studio-28.1.2.tar.xz/docs/sphinx/reference-scenes.rst -> obs-studio-29.0.0.tar.xz/docs/sphinx/reference-scenes.rst Changed
20
 
1
@@ -208,7 +208,7 @@
2
 .. function:: obs_scene_t *obs_scene_get_ref(obs_scene_t *scene)
3
 
4
    Returns an incremented reference if still valid, otherwise returns
5
-   *NULL*.
6
+   *NULL*. Release with :c:func:`obs_scene_release()`.
7
 
8
 ---------------------
9
 
10
@@ -549,7 +549,8 @@
11
 
12
    :return: An incremented reference to the private settings of the
13
             scene item.  Allows the front-end to set custom information
14
-            which is saved with the scene item
15
+            which is saved with the scene item. Release with
16
+            :c:func:`obs_data_release()`.
17
 
18
 ---------------------
19
 
20
obs-studio-28.1.2.tar.xz/docs/sphinx/reference-services.rst -> obs-studio-29.0.0.tar.xz/docs/sphinx/reference-services.rst Changed
30
 
1
@@ -200,7 +200,7 @@
2
 .. function:: obs_service_t *obs_service_get_ref(obs_service_t *service)
3
 
4
    Returns an incremented reference if still valid, otherwise returns
5
-   *NULL*.
6
+   *NULL*. Release with :c:func:`obs_service_release()`.
7
 
8
 ---------------------
9
 
10
@@ -236,7 +236,8 @@
11
 
12
 .. function:: obs_data_t *obs_service_defaults(const char *id)
13
 
14
-   :return: An incremented reference to the service's default settings
15
+   :return: An incremented reference to the service's default settings.
16
+            Release with :c:func:`obs_data_release()`.
17
 
18
 ---------------------
19
 
20
@@ -254,7 +255,8 @@
21
 
22
 .. function:: obs_data_t *obs_service_get_settings(const obs_service_t *service)
23
 
24
-   :return: An incremented reference to the service's settings
25
+   :return: An incremented reference to the service's settings. Release with
26
+            :c:func:`obs_data_release()`.
27
 
28
 ---------------------
29
 
30
obs-studio-28.1.2.tar.xz/docs/sphinx/reference-settings.rst -> obs-studio-29.0.0.tar.xz/docs/sphinx/reference-settings.rst Changed
98
 
1
@@ -25,7 +25,8 @@
2
 
3
 .. function:: obs_data_t *obs_data_create()
4
 
5
-   :return: A new reference to a data object.
6
+   :return: A new reference to a data object. Release with
7
+            :c:func:`obs_data_release()`.
8
 
9
 ---------------------
10
 
11
@@ -34,7 +35,8 @@
12
    Creates a data object from a Json string.
13
 
14
    :param json_string: Json string
15
-   :return:            A new reference to a data object
16
+   :return:            A new reference to a data object. Release with
17
+                       :c:func:`obs_data_release()`.
18
 
19
 ---------------------
20
 
21
@@ -43,7 +45,8 @@
22
    Creates a data object from a Json file.
23
 
24
    :param json_file: Json file path
25
-   :return:          A new reference to a data object
26
+   :return:          A new reference to a data object. Release with
27
+                     :c:func:`obs_data_release()`.
28
 
29
 ---------------------
30
 
31
@@ -54,7 +57,8 @@
32
 
33
    :param json_file:  Json file path
34
    :param backup_ext: Backup file extension
35
-   :return:           A new reference to a data object
36
+   :return:           A new reference to a data object. Release with
37
+                       :c:func:`obs_data_release()`.
38
 
39
 ---------------------
40
 
41
@@ -176,13 +180,15 @@
42
 
43
 .. function:: obs_data_t *obs_data_get_obj(obs_data_t *data, const char *name)
44
 
45
-   :return: An incremented reference to a data object.
46
+   :return: An incremented reference to a data object. Release with
47
+            :c:func:`obs_data_release()`.
48
 
49
 ---------------------
50
 
51
 .. function:: obs_data_array_t *obs_data_get_array(obs_data_t *data, const char *name)
52
 
53
-   :return: An incremented reference to a data array object.
54
+   :return: An incremented reference to a data array object. Release
55
+            with :c:func:`obs_data_array_release()`.
56
 
57
 ---------------------
58
 
59
@@ -224,7 +230,8 @@
60
 .. function:: void obs_data_set_default_obj(obs_data_t *data, const char *name, obs_data_t *obj)
61
               obs_data_t *obs_data_get_default_obj(obs_data_t *data, const char *name)
62
 
63
-   :return: An incremented reference to a data object.
64
+   :return: An incremented reference to a data object. Release with
65
+            :c:func:`obs_data_release()`.
66
 
67
 ----------------------
68
 
69
@@ -262,7 +269,8 @@
70
 .. function:: void obs_data_set_autoselect_obj(obs_data_t *data, const char *name, obs_data_t *obj)
71
               obs_data_t *obs_data_get_autoselect_obj(obs_data_t *data, const char *name)
72
 
73
-   :return: An incremented reference to a data object.
74
+   :return: An incremented reference to a data object. Release with
75
+            :c:func:`obs_data_release()`.
76
 
77
 ---------------------
78
 
79
@@ -272,7 +280,8 @@
80
 
81
 .. function:: obs_data_array_t *obs_data_array_create()
82
 
83
-   :return: A new reference to a data array object.
84
+   :return: A new reference to a data array object. Release
85
+            with :c:func:`obs_data_array_release()`.
86
 
87
 ---------------------
88
 
89
@@ -291,7 +300,7 @@
90
 .. function:: obs_data_t *obs_data_array_item(obs_data_array_t *array, size_t idx)
91
 
92
    :return: An incremented reference to the data object associated with
93
-            this array entry.
94
+            this array entry. Release with :c:func:`obs_data_release()`.
95
 
96
 ---------------------
97
 
98
obs-studio-28.1.2.tar.xz/docs/sphinx/reference-sources.rst -> obs-studio-29.0.0.tar.xz/docs/sphinx/reference-sources.rst Changed
201
 
1
@@ -31,6 +31,14 @@
2
 
3
    Unique string identifier for the source (required).
4
 
5
+.. member:: uint32_t version
6
+
7
+   Source version (optional).
8
+
9
+   This is used when a source's implementation is significantly
10
+   modified and the previous version is deprecated, but is kept to
11
+   prevent old sources from breaking.
12
+
13
 .. member:: enum obs_source_type obs_source_info.type
14
 
15
    Type of source.
16
@@ -502,6 +510,10 @@
17
    - **OBS_MEDIA_STATE_ENDED**     - Ended
18
    - **OBS_MEDIA_STATE_ERROR**     - Error
19
 
20
+.. member:: obs_missing_files_t *(*missing_files)(void *data)
21
+
22
+   Called to get the missing files of the source.
23
+
24
 .. member:: enum gs_color_space (*obs_source_info.video_get_color_space)(void *data, size_t count, const enum gs_color_space *preferred_spaces)
25
 
26
    Returns the color space of the source. Assume GS_CS_SRGB if not
27
@@ -531,6 +543,12 @@
28
    Called when the :c:func:`obs_source_remove()` function is called on
29
    the source.
30
 
31
+**update** (ptr source)
32
+
33
+   Called when the source's settings have been updated.
34
+
35
+   .. versionadded:: 29.0.0
36
+
37
 **save** (ptr source)
38
 
39
    Called when the source is being saved.
40
@@ -692,7 +710,7 @@
41
    Creates a source of the specified type with the specified settings.
42
 
43
    The "source" context is used for anything related to presenting
44
-   or modifying video/audio.  Use obs_source_release to release it.
45
+   or modifying video/audio.  Use :c:func:`obs_source_release` to release it.
46
 
47
    :param   id:             The source type string identifier
48
    :param   name:           The desired name of the source.  If this is
49
@@ -732,7 +750,8 @@
50
 
51
    Duplicates a source.  If the source has the
52
    OBS_SOURCE_DO_NOT_DUPLICATE output flag set, this only returns a
53
-   new reference to the same source.
54
+   new reference to the same source. Either way,
55
+   release with :c:func:`obs_source_release`.
56
 
57
    :param source:         The source to duplicate
58
    :param desired_name:   The desired name of the new source.  If this is
59
@@ -756,7 +775,7 @@
60
 .. function:: obs_source_t *obs_source_get_ref(obs_source_t *source)
61
 
62
    Returns an incremented reference if still valid, otherwise returns
63
-   *NULL*.
64
+   *NULL*. Use :c:func:`obs_source_release` to release it.
65
 
66
 ---------------------
67
 
68
@@ -937,15 +956,35 @@
69
 
70
 ---------------------
71
 
72
+.. function:: bool obs_source_is_scene(const obs_source_t *source)
73
+
74
+   :return: *true* if the source is a scene
75
+
76
+---------------------
77
+
78
+.. function:: bool obs_source_is_group(const obs_source_t *source)
79
+
80
+   :return: *true* if the source is a group
81
+
82
+---------------------
83
+
84
 .. function:: const char *obs_source_get_id(const obs_source_t *source)
85
 
86
-   :return: The source's type identifier string
87
+   :return: The source's type identifier string. If the source is versioned,
88
+            "_vN" is appended at the end, where "N" is the source's version.
89
+
90
+ ---------------------
91
+
92
+.. function:: const char *obs_source_get_unversioned_id(const obs_source_t *source)
93
+
94
+   :return: The source's unversioned type identifier string.
95
 
96
 ---------------------
97
 
98
 .. function:: signal_handler_t *obs_source_get_signal_handler(const obs_source_t *source)
99
 
100
-   :return: The source's signal handler
101
+   :return: The source's signal handler. Should not be manually freed,
102
+            as its lifecycle is managed by libobs.
103
 
104
    See the :ref:`source_signal_handler_reference` for more information
105
    on signals that are available for sources.
106
@@ -954,7 +993,8 @@
107
 
108
 .. function:: proc_handler_t *obs_source_get_proc_handler(const obs_source_t *source)
109
 
110
-   :return: The procedure handler for a source
111
+   :return: The procedure handler for a source. Should not be manually freed,
112
+            as its lifecycle is managed by libobs.
113
 
114
 ---------------------
115
 
116
@@ -995,7 +1035,7 @@
117
 .. function:: void obs_source_set_audio_mixers(obs_source_t *source, uint32_t mixers)
118
               uint32_t obs_source_get_audio_mixers(const obs_source_t *source)
119
 
120
-   Sets/gets the audio mixer channels that a source outputs to,
121
+   Sets/gets the audio mixer channels (i.e. audio tracks) that a source outputs to,
122
    depending on what bits are set.  Audio mixers allow filtering
123
    specific using multiple audio encoders to mix different sources
124
    together depending on what mixer channel they're set to.
125
@@ -1116,6 +1156,14 @@
126
 
127
 ---------------------
128
 
129
+.. function:: void obs_source_copy_single_filter(obs_source_t *dst, obs_source_t *filter)
130
+
131
+   Copies the filter from the source to the destination. If a filter by the
132
+   same name already exists in the destination source, the newer filter
133
+   will be given a unique name.
134
+
135
+---------------------
136
+
137
 .. function:: size_t obs_source_filter_count(const obs_source_t *source)
138
 
139
    Returns the number of filters the source has.
140
@@ -1182,7 +1230,8 @@
141
 .. function:: obs_data_t *obs_source_get_private_settings(obs_source_t *item)
142
 
143
    Gets private front-end settings data.  This data is saved/loaded
144
-   automatically.  Returns an incremented reference.
145
+   automatically.  Returns an incremented reference. Use :c:func:`obs_data_release()`
146
+   to release it.
147
 
148
 ---------------------
149
 
150
@@ -1221,6 +1270,77 @@
151
 
152
 ---------------------
153
 
154
+.. function:: enum obs_icon_type obs_source_get_icon_type(const char *id)
155
+
156
+   Calls the :c:member:`obs_source_info.icon_type` to get the icon type.
157
+
158
+---------------------
159
+
160
+.. function:: void obs_source_media_play_pause(obs_source_t *source, bool pause)
161
+
162
+   Calls the :c:member:`obs_source_info.media_play_pause` to pause or play media.
163
+
164
+---------------------
165
+
166
+.. function:: void obs_source_media_restart(obs_source_t *source)
167
+
168
+   Calls the :c:member:`obs_source_info.media_restart` to restart the media.
169
+
170
+---------------------
171
+
172
+.. function:: void obs_source_media_stop(obs_source_t *source)
173
+
174
+   Calls the :c:member:`obs_source_info.media_stop` to stop the media.
175
+
176
+---------------------
177
+
178
+.. function:: void obs_source_media_next(obs_source_t *source)
179
+
180
+   Calls the :c:member:`obs_source_info.media_next` to go to the next media.
181
+
182
+---------------------
183
+
184
+.. function:: void obs_source_media_previous(obs_source_t *source)
185
+
186
+   Calls the :c:member:`obs_source_info.media_previous` to go to the previous media.
187
+
188
+---------------------
189
+
190
+.. function:: int64_t obs_source_media_get_duration(obs_source_t *source)
191
+
192
+   Calls the :c:member:`obs_source_info.media_get_duration` to
193
+   get the media duration in milliseconds.
194
+
195
+---------------------
196
+
197
+.. function:: int64_t obs_source_media_get_time(obs_source_t *source)
198
+              void obs_source_media_set_time(obs_source_t *source, int64_t ms)
199
+
200
+   Calls the :c:member:`obs_source_info.media_get_time` or
201
obs-studio-28.1.2.tar.xz/libobs-d3d11/d3d11-subsystem.cpp -> obs-studio-29.0.0.tar.xz/libobs-d3d11/d3d11-subsystem.cpp Changed
148
 
1
@@ -66,7 +66,8 @@
2
        next->prev_next = prev_next;
3
 }
4
 
5
-static bool screen_supports_hdr(gs_device_t *device, HMONITOR hMonitor)
6
+static gs_monitor_color_info get_monitor_color_info(gs_device_t *device,
7
+                           HMONITOR hMonitor)
8
 {
9
    IDXGIFactory1 *factory1 = device->factory;
10
    if (!factory1->IsCurrent()) {
11
@@ -75,7 +76,8 @@
12
        device->monitor_to_hdr.clear();
13
    }
14
 
15
-   for (const std::pair<HMONITOR, bool> &pair : device->monitor_to_hdr) {
16
+   for (const std::pair<HMONITOR, gs_monitor_color_info> &pair :
17
+        device->monitor_to_hdr) {
18
        if (pair.first == hMonitor)
19
            return pair.second;
20
    }
21
@@ -96,15 +98,19 @@
22
                    const bool hdr =
23
                        desc1.ColorSpace ==
24
                        DXGI_COLOR_SPACE_RGB_FULL_G2084_NONE_P2020;
25
-                   device->monitor_to_hdr.emplace_back(
26
-                       hMonitor, hdr);
27
-                   return hdr;
28
+                   return device->monitor_to_hdr
29
+                       .emplace_back(
30
+                           hMonitor,
31
+                           gs_monitor_color_info(
32
+                               hdr,
33
+                               desc1.BitsPerColor))
34
+                       .second;
35
                }
36
            }
37
        }
38
    }
39
 
40
-   return false;
41
+   return gs_monitor_color_info(false, 8);
42
 }
43
 
44
 static enum gs_color_space get_next_space(gs_device_t *device, HWND hwnd,
45
@@ -115,8 +121,12 @@
46
        const HMONITOR hMonitor =
47
            MonitorFromWindow(hwnd, MONITOR_DEFAULTTONEAREST);
48
        if (hMonitor) {
49
-           if (screen_supports_hdr(device, hMonitor))
50
+           const gs_monitor_color_info info =
51
+               get_monitor_color_info(device, hMonitor);
52
+           if (info.hdr)
53
                next_space = GS_CS_709_SCRGB;
54
+           else if (info.bits_per_color > 8)
55
+               next_space = GS_CS_SRGB_16F;
56
        }
57
    }
58
 
59
@@ -126,7 +136,14 @@
60
 static enum gs_color_format
61
 get_swap_format_from_space(gs_color_space space, gs_color_format sdr_format)
62
 {
63
-   return (space == GS_CS_709_SCRGB) ? GS_RGBA16F : sdr_format;
64
+   gs_color_format format = sdr_format;
65
+   switch (space) {
66
+   case GS_CS_SRGB_16F:
67
+   case GS_CS_709_SCRGB:
68
+       format = GS_RGBA16F;
69
+   }
70
+
71
+   return format;
72
 }
73
 
74
 static inline enum gs_color_space
75
@@ -313,7 +330,8 @@
76
    int ver = 49;
77
 
78
    while (ver > 30) {
79
-       sprintf(d3dcompiler, "D3DCompiler_%02d.dll", ver);
80
+       snprintf(d3dcompiler, sizeof(d3dcompiler),
81
+            "D3DCompiler_%02d.dll", ver);
82
 
83
        HMODULE module = LoadLibraryA(d3dcompiler);
84
        if (module) {
85
@@ -346,45 +364,6 @@
86
        throw UnsupportedHWError("Failed to create DXGIFactory", hr);
87
 }
88
 
89
-#define VENDOR_ID_INTEL 0x8086
90
-#define IGPU_MEM (512 * 1024 * 1024)
91
-
92
-void gs_device::ReorderAdapters(uint32_t &adapterIdx)
93
-{
94
-   std::vector<uint32_t> adapterOrder;
95
-   ComPtr<IDXGIAdapter> adapter;
96
-   DXGI_ADAPTER_DESC desc;
97
-   uint32_t iGPUIndex = 0;
98
-   bool hasIGPU = false;
99
-   bool hasDGPU = false;
100
-   int idx = 0;
101
-
102
-   while (SUCCEEDED(factory->EnumAdapters(idx, &adapter))) {
103
-       if (SUCCEEDED(adapter->GetDesc(&desc))) {
104
-           if (desc.VendorId == VENDOR_ID_INTEL) {
105
-               if (desc.DedicatedVideoMemory <= IGPU_MEM) {
106
-                   hasIGPU = true;
107
-                   iGPUIndex = (uint32_t)idx;
108
-               } else {
109
-                   hasDGPU = true;
110
-               }
111
-           }
112
-       }
113
-
114
-       adapterOrder.push_back((uint32_t)idx++);
115
-   }
116
-
117
-   /* Intel specific adapter check for Intel integrated and Intel
118
-    * dedicated. If both exist, then change adapter priority so that the
119
-    * integrated comes first for the sake of improving overall
120
-    * performance */
121
-   if (hasIGPU && hasDGPU) {
122
-       adapterOrder.erase(adapterOrder.begin() + iGPUIndex);
123
-       adapterOrder.insert(adapterOrder.begin(), iGPUIndex);
124
-       adapterIdx = adapterOrderadapterIdx;
125
-   }
126
-}
127
-
128
 void gs_device::InitAdapter(uint32_t adapterIdx)
129
 {
130
    HRESULT hr = factory->EnumAdapters1(adapterIdx, &adapter);
131
@@ -923,7 +902,6 @@
132
 
133
    InitCompiler();
134
    InitFactory();
135
-   ReorderAdapters(adapterIdx);
136
    InitAdapter(adapterIdx);
137
    InitDevice(adapterIdx);
138
    device_set_render_target(this, NULL, NULL);
139
@@ -3068,7 +3046,7 @@
140
 extern "C" EXPORT bool device_is_monitor_hdr(gs_device_t *device, void *monitor)
141
 {
142
    const HMONITOR hMonitor = static_cast<HMONITOR>(monitor);
143
-   return screen_supports_hdr(device, hMonitor);
144
+   return get_monitor_color_info(device, hMonitor).hdr;
145
 }
146
 
147
 extern "C" EXPORT void device_debug_marker_begin(gs_device_t *,
148
obs-studio-28.1.2.tar.xz/libobs-d3d11/d3d11-subsystem.hpp -> obs-studio-29.0.0.tar.xz/libobs-d3d11/d3d11-subsystem.hpp Changed
31
 
1
@@ -979,6 +979,16 @@
2
    float mat16;
3
 };
4
 
5
+struct gs_monitor_color_info {
6
+   bool hdr;
7
+   UINT bits_per_color;
8
+
9
+   gs_monitor_color_info(bool hdr, int bits_per_color)
10
+       : hdr(hdr), bits_per_color(bits_per_color)
11
+   {
12
+   }
13
+};
14
+
15
 struct gs_device {
16
    ComPtr<IDXGIFactory1> factory;
17
    ComPtr<IDXGIAdapter1> adapter;
18
@@ -1035,11 +1045,10 @@
19
    vector<gs_device_loss> loss_callbacks;
20
    gs_obj *first_obj = nullptr;
21
 
22
-   vector<std::pair<HMONITOR, bool>> monitor_to_hdr;
23
+   vector<std::pair<HMONITOR, gs_monitor_color_info>> monitor_to_hdr;
24
 
25
    void InitCompiler();
26
    void InitFactory();
27
-   void ReorderAdapters(uint32_t &adapterIdx);
28
    void InitAdapter(uint32_t adapterIdx);
29
    void InitDevice(uint32_t adapterIdx);
30
 
31
obs-studio-28.1.2.tar.xz/libobs-opengl/gl-cocoa.m -> obs-studio-29.0.0.tar.xz/libobs-opengl/gl-cocoa.m Changed
46
 
1
@@ -342,21 +342,24 @@
2
    struct gs_texture_2d *tex = bzalloc(sizeof(struct gs_texture_2d));
3
 
4
    OSType pf = IOSurfaceGetPixelFormat(ref);
5
+   const bool l10r = pf == 'l10r';
6
    if (pf == 0)
7
        blog(LOG_ERROR, "Invalid IOSurface Buffer");
8
-   else if (pf != 'BGRA')
9
+   else if ((pf != 'BGRA') && !l10r)
10
        blog(LOG_ERROR, "Unexpected pixel format: %d (%c%c%c%c)", pf,
11
             pf >> 24, pf >> 16, pf >> 8, pf);
12
 
13
-   const enum gs_color_format color_format = GS_BGRA;
14
+   const enum gs_color_format color_format = l10r ? GS_R10G10B10A2
15
+                              : GS_BGRA;
16
 
17
    tex->base.device = device;
18
    tex->base.type = GS_TEXTURE_2D;
19
-   tex->base.format = GS_BGRA;
20
+   tex->base.format = color_format;
21
    tex->base.levels = 1;
22
-   tex->base.gl_format = convert_gs_format(color_format);
23
+   tex->base.gl_format = l10r ? GL_BGRA : convert_gs_format(color_format);
24
    tex->base.gl_internal_format = convert_gs_internal_format(color_format);
25
-   tex->base.gl_type = GL_UNSIGNED_INT_8_8_8_8_REV;
26
+   tex->base.gl_type = l10r ? GL_UNSIGNED_INT_2_10_10_10_REV
27
+                : GL_UNSIGNED_INT_8_8_8_8_REV;
28
    tex->base.gl_target = GL_TEXTURE_RECTANGLE_ARB;
29
    tex->base.is_dynamic = false;
30
    tex->base.is_render_target = false;
31
@@ -427,11 +430,9 @@
32
    OSType pf = IOSurfaceGetPixelFormat(ref);
33
    if (pf == 0) {
34
        blog(LOG_ERROR, "Invalid IOSurface buffer");
35
-   } else {
36
-       if (pf != 'BGRA')
37
-           blog(LOG_ERROR,
38
-                "Unexpected pixel format: %d (%c%c%c%c)", pf,
39
-                pf >> 24, pf >> 16, pf >> 8, pf);
40
+   } else if ((pf != 'BGRA') && (pf != 'l10r')) {
41
+       blog(LOG_ERROR, "Unexpected pixel format: %d (%c%c%c%c)", pf,
42
+            pf >> 24, pf >> 16, pf >> 8, pf);
43
    }
44
 
45
    tex->width = IOSurfaceGetWidth(ref);
46
obs-studio-28.1.2.tar.xz/libobs/CMakeLists.txt -> obs-studio-29.0.0.tar.xz/libobs/CMakeLists.txt Changed
29
 
1
@@ -290,6 +290,8 @@
2
             util/threading-windows.h
3
             util/pipe-windows.c
4
             util/platform-windows.c
5
+            util/windows/device-enum.c
6
+            util/windows/device-enum.h
7
             util/windows/obfuscate.c
8
             util/windows/obfuscate.h
9
             util/windows/win-registry.h
10
@@ -310,7 +312,7 @@
11
     libobs PRIVATE UNICODE _UNICODE _CRT_SECURE_NO_WARNINGS
12
                    _CRT_NONSTDC_NO_WARNINGS)
13
 
14
-  target_link_libraries(libobs PRIVATE Avrt Dwmapi winmm)
15
+  target_link_libraries(libobs PRIVATE dxgi Avrt Dwmapi winmm)
16
 
17
   if(MSVC)
18
     target_link_libraries(libobs PUBLIC OBS::w32-pthreads)
19
@@ -318,7 +320,8 @@
20
     target_compile_options(libobs PRIVATE "$<$<COMPILE_LANGUAGE:C>:/EHc->"
21
                                           "$<$<COMPILE_LANGUAGE:CXX>:/EHc->")
22
 
23
-    target_link_options(libobs PRIVATE "LINKER:/SAFESEH:NO")
24
+    target_link_options(libobs PRIVATE "LINKER:/IGNORE:4098"
25
+                        "LINKER:/SAFESEH:NO")
26
   endif()
27
 
28
 elseif(OS_MACOS)
29
obs-studio-28.1.2.tar.xz/libobs/data/color.effect -> obs-studio-29.0.0.tar.xz/libobs/data/color.effect Changed
16
 
1
@@ -26,6 +26,14 @@
2
    return float3(r, g, b);
3
 }
4
 
5
+float3 d65p3_to_rec709(float3 v)
6
+{
7
+   float r = dot(v, float3(1.2249401762805598, -0.22494017628055996, 0.));
8
+   float g = dot(v, float3(-0.042056954709688163, 1.0420569547096881, 0.));
9
+   float b = dot(v, float3(-0.019637554590334432, -0.078636045550631889, 1.0982736001409663));
10
+   return float3(r, g, b);
11
+}
12
+
13
 float3 rec2020_to_rec709(float3 v)
14
 {
15
    float r = dot(v, float3(1.6604910021084345, -0.58764113878854951, -0.072849863319884883));
16
obs-studio-28.1.2.tar.xz/libobs/data/default_rect.effect -> obs-studio-29.0.0.tar.xz/libobs/data/default_rect.effect Changed
32
 
1
@@ -27,6 +27,14 @@
2
    return image.Sample(def_sampler, vert_in.uv);
3
 }
4
 
5
+float4 PSDrawD65P3(VertInOut vert_in) : TARGET
6
+{
7
+   float4 rgba = image.Sample(def_sampler, vert_in.uv);
8
+   rgba.rgb = srgb_nonlinear_to_linear(rgba.rgb);
9
+   rgba.rgb = d65p3_to_rec709(rgba.rgb);
10
+   return rgba;
11
+}
12
+
13
 float4 PSDrawOpaque(VertInOut vert_in) : TARGET
14
 {
15
    return float4(image.Sample(def_sampler, vert_in.uv).rgb, 1.0);
16
@@ -48,6 +56,15 @@
17
    }
18
 }
19
 
20
+technique DrawD65P3
21
+{
22
+   pass
23
+   {
24
+       vertex_shader = VSDefault(vert_in);
25
+       pixel_shader  = PSDrawD65P3(vert_in);
26
+   }
27
+}
28
+
29
 technique DrawOpaque
30
 {
31
    pass
32
obs-studio-28.1.2.tar.xz/libobs/media-io/audio-resampler-ffmpeg.c -> obs-studio-29.0.0.tar.xz/libobs/media-io/audio-resampler-ffmpeg.c Changed
96
 
1
@@ -27,16 +27,20 @@
2
    bool opened;
3
 
4
    uint32_t input_freq;
5
-   uint64_t input_layout;
6
    enum AVSampleFormat input_format;
7
-
8
    uint8_t *output_bufferMAX_AV_PLANES;
9
-   uint64_t output_layout;
10
    enum AVSampleFormat output_format;
11
    int output_size;
12
    uint32_t output_ch;
13
    uint32_t output_freq;
14
    uint32_t output_planes;
15
+#if LIBSWRESAMPLE_VERSION_INT < AV_VERSION_INT(4, 5, 100)
16
+   uint64_t input_layout;
17
+   uint64_t output_layout;
18
+#else
19
+   AVChannelLayout input_ch_layout;
20
+   AVChannelLayout output_ch_layout;
21
+#endif
22
 };
23
 
24
 static inline enum AVSampleFormat convert_audio_format(enum audio_format format)
25
@@ -66,6 +70,7 @@
26
    return AV_SAMPLE_FMT_S16;
27
 }
28
 
29
+#if LIBSWRESAMPLE_VERSION_INT < AV_VERSION_INT(4, 5, 100)
30
 static inline uint64_t convert_speaker_layout(enum speaker_layout layout)
31
 {
32
    switch (layout) {
33
@@ -90,6 +95,7 @@
34
    /* shouldn't get here */
35
    return 0;
36
 }
37
+#endif
38
 
39
 audio_resampler_t *audio_resampler_create(const struct resample_info *dst,
40
                      const struct resample_info *src)
41
@@ -99,20 +105,38 @@
42
 
43
    rs->opened = false;
44
    rs->input_freq = src->samples_per_sec;
45
-   rs->input_layout = convert_speaker_layout(src->speakers);
46
    rs->input_format = convert_audio_format(src->format);
47
    rs->output_size = 0;
48
    rs->output_ch = get_audio_channels(dst->speakers);
49
    rs->output_freq = dst->samples_per_sec;
50
-   rs->output_layout = convert_speaker_layout(dst->speakers);
51
    rs->output_format = convert_audio_format(dst->format);
52
    rs->output_planes = is_audio_planar(dst->format) ? rs->output_ch : 1;
53
 
54
+#if (LIBSWRESAMPLE_VERSION_INT < AV_VERSION_INT(4, 5, 100))
55
+   rs->input_layout = convert_speaker_layout(src->speakers);
56
+   rs->output_layout = convert_speaker_layout(dst->speakers);
57
    rs->context = swr_alloc_set_opts(NULL, rs->output_layout,
58
                     rs->output_format,
59
                     dst->samples_per_sec, rs->input_layout,
60
                     rs->input_format, src->samples_per_sec,
61
                     0, NULL);
62
+#else
63
+   int nb_ch = get_audio_channels(src->speakers);
64
+   av_channel_layout_default(&rs->input_ch_layout, nb_ch);
65
+   av_channel_layout_default(&rs->output_ch_layout, rs->output_ch);
66
+   if (src->speakers == SPEAKERS_4POINT1)
67
+       rs->input_ch_layout =
68
+           (AVChannelLayout)AV_CHANNEL_LAYOUT_4POINT1;
69
+
70
+   if (dst->speakers == SPEAKERS_4POINT1)
71
+       rs->output_ch_layout =
72
+           (AVChannelLayout)AV_CHANNEL_LAYOUT_4POINT1;
73
+
74
+   swr_alloc_set_opts2(&rs->context, &rs->output_ch_layout,
75
+               rs->output_format, dst->samples_per_sec,
76
+               &rs->input_ch_layout, rs->input_format,
77
+               src->samples_per_sec, 0, NULL);
78
+#endif
79
 
80
    if (!rs->context) {
81
        blog(LOG_ERROR, "swr_alloc_set_opts failed");
82
@@ -120,7 +144,13 @@
83
        return NULL;
84
    }
85
 
86
+#if (LIBSWRESAMPLE_VERSION_INT < AV_VERSION_INT(4, 5, 100))
87
    if (rs->input_layout == AV_CH_LAYOUT_MONO && rs->output_ch > 1) {
88
+#else
89
+   AVChannelLayout test_ch = AV_CHANNEL_LAYOUT_MONO;
90
+   if (av_channel_layout_compare(&rs->input_ch_layout, &test_ch) == 0 &&
91
+       rs->output_ch > 1) {
92
+#endif
93
        const double matrixMAX_AUDIO_CHANNELSMAX_AUDIO_CHANNELS = {
94
            {1},
95
            {1, 1},
96
obs-studio-28.1.2.tar.xz/libobs/media-io/media-remux.c -> obs-studio-29.0.0.tar.xz/libobs/media-io/media-remux.c Changed
19
 
1
@@ -149,7 +149,16 @@
2
        av_dict_copy(&out_stream->metadata, in_stream->metadata, 0);
3
 
4
 #if LIBAVCODEC_VERSION_INT >= AV_VERSION_INT(57, 48, 101)
5
-       out_stream->codecpar->codec_tag = 0;
6
+       if (in_stream->codecpar->codec_tag != 0) {
7
+           out_stream->codecpar->codec_tag =
8
+               in_stream->codecpar->codec_tag;
9
+       } else if (in_stream->codecpar->codec_id == AV_CODEC_ID_HEVC) {
10
+           // Tag HEVC files with industry standard HVC1 tag for wider device compatibility
11
+           out_stream->codecpar->codec_tag =
12
+               MKTAG('h', 'v', 'c', '1');
13
+       } else {
14
+           out_stream->codecpar->codec_tag = 0;
15
+       }
16
 #else
17
        out_stream->codec->codec_tag = 0;
18
        out_stream->time_base = out_stream->codec->time_base;
19
obs-studio-28.1.2.tar.xz/libobs/obs-cocoa.m -> obs-studio-29.0.0.tar.xz/libobs/obs-cocoa.m Changed
22
 
1
@@ -123,6 +123,12 @@
2
         os_get_physical_cores(), os_get_logical_cores());
3
 }
4
 
5
+static void log_emulation_status(void)
6
+{
7
+   blog(LOG_INFO, "Rosetta translation used: %s",
8
+        os_get_emulation_status() ? "true" : "false");
9
+}
10
+
11
 static void log_available_memory(void)
12
 {
13
    size_t size;
14
@@ -162,6 +168,7 @@
15
    log_processor_cores();
16
    log_available_memory();
17
    log_os();
18
+   log_emulation_status();
19
    log_kernel_version();
20
 }
21
 
22
obs-studio-28.1.2.tar.xz/libobs/obs-config.h -> obs-studio-29.0.0.tar.xz/libobs/obs-config.h Changed
26
 
1
@@ -27,21 +27,21 @@
2
 /*
3
  * Increment if major breaking API changes
4
  */
5
-#define LIBOBS_API_MAJOR_VER 28
6
+#define LIBOBS_API_MAJOR_VER 29
7
 
8
 /*
9
  * Increment if backward-compatible additions
10
  *
11
  * Reset to zero each major version
12
  */
13
-#define LIBOBS_API_MINOR_VER 1
14
+#define LIBOBS_API_MINOR_VER 0
15
 
16
 /*
17
  * Increment if backward-compatible bug fix
18
  *
19
  * Reset to zero each major or minor version
20
  */
21
-#define LIBOBS_API_PATCH_VER 2
22
+#define LIBOBS_API_PATCH_VER 0
23
 
24
 #define MAKE_SEMANTIC_VERSION(major, minor, patch) \
25
    ((major << 24) | (minor << 16) | patch)
26
obs-studio-28.1.2.tar.xz/libobs/obs-encoder.c -> obs-studio-29.0.0.tar.xz/libobs/obs-encoder.c Changed
12
 
1
@@ -187,7 +187,9 @@
2
 
3
 static inline bool gpu_encode_available(const struct obs_encoder *encoder)
4
 {
5
-   struct obs_core_video_mix *video = obs->video.main_mix;
6
+   struct obs_core_video_mix *video = get_mix_for_video(encoder->media);
7
+   if (!video)
8
+       return false;
9
    return (encoder->info.caps & OBS_ENCODER_CAP_PASS_TEXTURE) != 0 &&
10
           (video->using_p010_tex || video->using_nv12_tex);
11
 }
12
obs-studio-28.1.2.tar.xz/libobs/obs-internal.h -> obs-studio-29.0.0.tar.xz/libobs/obs-internal.h Changed
53
 
1
@@ -284,6 +284,7 @@
2
    volatile bool gpu_encode_stop;
3
 
4
    video_t *video;
5
+   struct obs_video_info ovi;
6
 
7
    bool gpu_conversion;
8
    const char *conversion_techsNUM_CHANNELS;
9
@@ -292,7 +293,6 @@
10
    float conversion_height_i;
11
 
12
    float color_matrix16;
13
-   enum obs_scale_type scale_type;
14
 };
15
 
16
 extern struct obs_core_video_mix *
17
@@ -324,9 +324,6 @@
18
    uint32_t lagged_frames;
19
    bool thread_initialized;
20
 
21
-   uint32_t base_width;
22
-   uint32_t base_height;
23
-
24
    gs_texture_t *transparent_texture;
25
 
26
    gs_effect_t *deinterlace_discard_effect;
27
@@ -338,7 +335,6 @@
28
    gs_effect_t *deinterlace_yadif_effect;
29
    gs_effect_t *deinterlace_yadif_2x_effect;
30
 
31
-   struct obs_video_info ovi;
32
    float sdr_white_level;
33
    float hdr_nominal_peak_level;
34
 
35
@@ -365,8 +361,6 @@
36
    int max_buffering_ticks;
37
    bool fixed_buffer;
38
 
39
-   float user_volume;
40
-
41
    pthread_mutex_t monitoring_mutex;
42
    DARRAY(struct audio_monitor *) monitors;
43
    char *monitoring_device_name;
44
@@ -498,6 +492,8 @@
45
               uint64_t end_ts_in, uint64_t *out_ts,
46
               uint32_t mixers, struct audio_output_data *mixes);
47
 
48
+extern struct obs_core_video_mix *get_mix_for_video(video_t *video);
49
+
50
 extern void
51
 start_raw_video(video_t *video, const struct video_scale_info *conversion,
52
        void (*callback)(void *param, struct video_data *frame),
53
obs-studio-28.1.2.tar.xz/libobs/obs-nix-wayland.c -> obs-studio-29.0.0.tar.xz/libobs/obs-nix-wayland.c Changed
45
 
1
@@ -1061,6 +1061,21 @@
2
        return OBS_KEY_NUM8;
3
    case XKB_KEY_KP_9:
4
        return OBS_KEY_NUM9;
5
+
6
+   case XKB_KEY_XF86AudioPlay:
7
+       return OBS_KEY_VK_MEDIA_PLAY_PAUSE;
8
+   case XKB_KEY_XF86AudioStop:
9
+       return OBS_KEY_VK_MEDIA_STOP;
10
+   case XKB_KEY_XF86AudioPrev:
11
+       return OBS_KEY_VK_MEDIA_PREV_TRACK;
12
+   case XKB_KEY_XF86AudioNext:
13
+       return OBS_KEY_VK_MEDIA_NEXT_TRACK;
14
+   case XKB_KEY_XF86AudioMute:
15
+       return OBS_KEY_VK_VOLUME_MUTE;
16
+   case XKB_KEY_XF86AudioRaiseVolume:
17
+       return OBS_KEY_VK_VOLUME_DOWN;
18
+   case XKB_KEY_XF86AudioLowerVolume:
19
+       return OBS_KEY_VK_VOLUME_UP;
20
    }
21
    return OBS_KEY_NONE;
22
 }
23
@@ -1607,6 +1622,21 @@
24
        return XKB_KEY_KP_8;
25
    case OBS_KEY_NUM9:
26
        return XKB_KEY_KP_9;
27
+
28
+   case OBS_KEY_VK_MEDIA_PLAY_PAUSE:
29
+       return XKB_KEY_XF86AudioPlay;
30
+   case OBS_KEY_VK_MEDIA_STOP:
31
+       return XKB_KEY_XF86AudioStop;
32
+   case OBS_KEY_VK_MEDIA_PREV_TRACK:
33
+       return XKB_KEY_XF86AudioPrev;
34
+   case OBS_KEY_VK_MEDIA_NEXT_TRACK:
35
+       return XKB_KEY_XF86AudioNext;
36
+   case OBS_KEY_VK_VOLUME_MUTE:
37
+       return XKB_KEY_XF86AudioMute;
38
+   case OBS_KEY_VK_VOLUME_DOWN:
39
+       return XKB_KEY_XF86AudioRaiseVolume;
40
+   case OBS_KEY_VK_VOLUME_UP:
41
+       return XKB_KEY_XF86AudioLowerVolume;
42
    default:
43
        break;
44
    }
45
obs-studio-28.1.2.tar.xz/libobs/obs-nix-x11.c -> obs-studio-29.0.0.tar.xz/libobs/obs-nix-x11.c Changed
23
 
1
@@ -673,6 +673,21 @@
2
    case OBS_KEY_MOUSE5:
3
        return MOUSE_5;
4
 
5
+   case OBS_KEY_VK_MEDIA_PLAY_PAUSE:
6
+       return XF86XK_AudioPlay;
7
+   case OBS_KEY_VK_MEDIA_STOP:
8
+       return XF86XK_AudioStop;
9
+   case OBS_KEY_VK_MEDIA_PREV_TRACK:
10
+       return XF86XK_AudioPrev;
11
+   case OBS_KEY_VK_MEDIA_NEXT_TRACK:
12
+       return XF86XK_AudioNext;
13
+   case OBS_KEY_VK_VOLUME_MUTE:
14
+       return XF86XK_AudioMute;
15
+   case OBS_KEY_VK_VOLUME_DOWN:
16
+       return XF86XK_AudioRaiseVolume;
17
+   case OBS_KEY_VK_VOLUME_UP:
18
+       return XF86XK_AudioLowerVolume;
19
+
20
    /* TODO: Implement keys for non-US keyboards */
21
    default:;
22
    }
23
obs-studio-28.1.2.tar.xz/libobs/obs-properties.c -> obs-studio-29.0.0.tar.xz/libobs/obs-properties.c Changed
32
 
1
@@ -169,6 +169,12 @@
2
        bfree(data->suffix);
3
 }
4
 
5
+static inline void button_data_free(struct button_data *data)
6
+{
7
+   if (data->url)
8
+       bfree(data->url);
9
+}
10
+
11
 struct obs_properties;
12
 
13
 struct obs_property {
14
@@ -259,6 +265,8 @@
15
        int_data_free(get_property_data(property));
16
    else if (property->type == OBS_PROPERTY_FLOAT)
17
        float_data_free(get_property_data(property));
18
+   else if (property->type == OBS_PROPERTY_BUTTON)
19
+       button_data_free(get_property_data(property));
20
 
21
    bfree(property->name);
22
    bfree(property->desc);
23
@@ -1154,7 +1162,7 @@
24
    if (!data)
25
        return;
26
 
27
-   data->url = url;
28
+   data->url = bstrdup(url);
29
 }
30
 
31
 void obs_property_list_clear(obs_property_t *p)
32
obs-studio-28.1.2.tar.xz/libobs/obs-scene.c -> obs-studio-29.0.0.tar.xz/libobs/obs-scene.c Changed
25
 
1
@@ -1265,13 +1265,21 @@
2
 static uint32_t scene_getwidth(void *data)
3
 {
4
    obs_scene_t *scene = data;
5
-   return scene->custom_size ? scene->cx : obs->video.base_width;
6
+   if (scene->custom_size)
7
+       return scene->cx;
8
+   if (obs->video.main_mix)
9
+       return obs->video.main_mix->ovi.base_width;
10
+   return 0;
11
 }
12
 
13
 static uint32_t scene_getheight(void *data)
14
 {
15
    obs_scene_t *scene = data;
16
-   return scene->custom_size ? scene->cy : obs->video.base_height;
17
+   if (scene->custom_size)
18
+       return scene->cy;
19
+   if (obs->video.main_mix)
20
+       return obs->video.main_mix->ovi.base_height;
21
+   return 0;
22
 }
23
 
24
 static void apply_scene_item_audio_actions(struct obs_scene_item *item,
25
obs-studio-28.1.2.tar.xz/libobs/obs-source-deinterlace.c -> obs-studio-29.0.0.tar.xz/libobs/obs-source-deinterlace.c Changed
51
 
1
@@ -250,40 +250,24 @@
2
    }
3
 }
4
 
5
-static inline struct obs_source_frame *get_prev_frame(obs_source_t *source,
6
-                             bool *updated)
7
+void deinterlace_update_async_video(obs_source_t *source)
8
 {
9
-   struct obs_source_frame *frame = NULL;
10
+   if (source->deinterlace_rendered)
11
+       return;
12
+
13
+   source->deinterlace_rendered = true;
14
 
15
    pthread_mutex_lock(&source->async_mutex);
16
 
17
-   *updated = source->cur_async_frame != NULL;
18
-   frame = source->prev_async_frame;
19
+   const bool updated = source->cur_async_frame != NULL;
20
+   struct obs_source_frame *frame = source->prev_async_frame;
21
    source->prev_async_frame = NULL;
22
 
23
-   if (frame)
24
-       os_atomic_inc_long(&frame->refs);
25
-
26
    pthread_mutex_unlock(&source->async_mutex);
27
 
28
-   return frame;
29
-}
30
-
31
-void deinterlace_update_async_video(obs_source_t *source)
32
-{
33
-   struct obs_source_frame *frame;
34
-   bool updated;
35
-
36
-   if (source->deinterlace_rendered)
37
-       return;
38
-
39
-   frame = get_prev_frame(source, &updated);
40
-
41
-   source->deinterlace_rendered = true;
42
-   if (frame)
43
-       frame = filter_async_video(source, frame);
44
-
45
    if (frame) {
46
+       os_atomic_inc_long(&frame->refs);
47
+
48
        if (set_async_texture_size(source, frame)) {
49
            update_async_textures(source, frame,
50
                          source->async_prev_textures,
51
obs-studio-28.1.2.tar.xz/libobs/obs-source.c -> obs-studio-29.0.0.tar.xz/libobs/obs-source.c Changed
92
 
1
@@ -77,6 +77,7 @@
2
 static const char *source_signals = {
3
    "void destroy(ptr source)",
4
    "void remove(ptr source)",
5
+   "void update(ptr source)",
6
    "void save(ptr source)",
7
    "void load(ptr source)",
8
    "void activate(ptr source)",
9
@@ -203,7 +204,7 @@
10
        return false;
11
    if (pthread_mutex_init(&source->audio_mutex, NULL) != 0)
12
        return false;
13
-   if (pthread_mutex_init(&source->async_mutex, NULL) != 0)
14
+   if (pthread_mutex_init_recursive(&source->async_mutex) != 0)
15
        return false;
16
    if (pthread_mutex_init(&source->caption_cb_mutex, NULL) != 0)
17
        return false;
18
@@ -968,6 +969,7 @@
19
                    source->context.settings);
20
        os_atomic_compare_swap_long(&source->defer_update_count, count,
21
                        0);
22
+       obs_source_dosignal(source, "source_update", "update");
23
    }
24
 }
25
 
26
@@ -985,6 +987,7 @@
27
    } else if (source->context.data && source->info.update) {
28
        source->info.update(source->context.data,
29
                    source->context.settings);
30
+       obs_source_dosignal(source, "source_update", "update");
31
    }
32
 }
33
 
34
@@ -1184,8 +1187,20 @@
35
 
36
 static inline struct obs_source_frame *get_closest_frame(obs_source_t *source,
37
                             uint64_t sys_time);
38
-bool set_async_texture_size(struct obs_source *source,
39
-               const struct obs_source_frame *frame);
40
+
41
+static void filter_frame(obs_source_t *source,
42
+            struct obs_source_frame **ref_frame)
43
+{
44
+   struct obs_source_frame *frame = *ref_frame;
45
+   if (frame) {
46
+       os_atomic_inc_long(&frame->refs);
47
+       frame = filter_async_video(source, frame);
48
+       if (frame)
49
+           os_atomic_dec_long(&frame->refs);
50
+   }
51
+
52
+   *ref_frame = frame;
53
+}
54
 
55
 static void async_tick(obs_source_t *source)
56
 {
57
@@ -1205,6 +1220,11 @@
58
    }
59
 
60
    source->last_sys_timestamp = sys_time;
61
+
62
+   if (deinterlacing_enabled(source))
63
+       filter_frame(source, &source->prev_async_frame);
64
+   filter_frame(source, &source->cur_async_frame);
65
+
66
    pthread_mutex_unlock(&source->async_mutex);
67
 
68
    if (source->cur_async_frame)
69
@@ -2396,12 +2416,9 @@
70
 static void obs_source_update_async_video(obs_source_t *source)
71
 {
72
    if (!source->async_rendered) {
73
-       struct obs_source_frame *frame = obs_source_get_frame(source);
74
-
75
-       if (frame)
76
-           frame = filter_async_video(source, frame);
77
-
78
        source->async_rendered = true;
79
+
80
+       struct obs_source_frame *frame = obs_source_get_frame(source);
81
        if (frame) {
82
            check_to_swap_bgrx_bgra(source, frame);
83
 
84
@@ -2475,6 +2492,7 @@
85
                    nonlinear_alpha
86
                        ? "DrawNonlinearAlphaMultiply"
87
                        : "DrawMultiply";
88
+               linear_srgb = true;
89
                multiplier =
90
                    obs_get_video_sdr_white_level() / 80.0f;
91
            }
92
obs-studio-28.1.2.tar.xz/libobs/obs-video.c -> obs-studio-29.0.0.tar.xz/libobs/obs-video.c Changed
83
 
1
@@ -125,8 +125,8 @@
2
 static const char *render_main_texture_name = "render_main_texture";
3
 static inline void render_main_texture(struct obs_core_video_mix *video)
4
 {
5
-   uint32_t base_width = obs->video.base_width;
6
-   uint32_t base_height = obs->video.base_height;
7
+   uint32_t base_width = video->ovi.base_width;
8
+   uint32_t base_height = video->ovi.base_height;
9
 
10
    profile_start(render_main_texture_name);
11
    GS_DEBUG_MARKER_BEGIN(GS_DEBUG_COLOR_MAIN_TEXTURE,
12
@@ -170,12 +170,12 @@
13
    /* if the dimension is under half the size of the original image,
14
     * bicubic/lanczos can't sample enough pixels to create an accurate
15
     * image, so use the bilinear low resolution effect instead */
16
-   if (info->width < (video->base_width / 2) &&
17
-       info->height < (video->base_height / 2)) {
18
+   if (info->width < (mix->ovi.base_width / 2) &&
19
+       info->height < (mix->ovi.base_height / 2)) {
20
        return video->bilinear_lowres_effect;
21
    }
22
 
23
-   switch (mix->scale_type) {
24
+   switch (mix->ovi.scale_type) {
25
    case OBS_SCALE_BILINEAR:
26
        return video->default_effect;
27
    case OBS_SCALE_LANCZOS:
28
@@ -189,11 +189,11 @@
29
    return video->bicubic_effect;
30
 }
31
 
32
-static inline bool resolution_close(struct obs_core_video *video,
33
+static inline bool resolution_close(struct obs_core_video_mix *mix,
34
                    uint32_t width, uint32_t height)
35
 {
36
-   long width_cmp = (long)video->base_width - (long)width;
37
-   long height_cmp = (long)video->base_height - (long)height;
38
+   long width_cmp = (long)mix->ovi.base_width - (long)width;
39
+   long height_cmp = (long)mix->ovi.base_height - (long)height;
40
 
41
    return labs(width_cmp) <= 16 && labs(height_cmp) <= 16;
42
 }
43
@@ -203,7 +203,7 @@
44
 {
45
    struct obs_core_video *video = &obs->video;
46
 
47
-   if (resolution_close(video, width, height)) {
48
+   if (resolution_close(mix, width, height)) {
49
        return video->default_effect;
50
    } else {
51
        /* if the scale method couldn't be loaded, use either bicubic
52
@@ -233,8 +233,8 @@
53
    if (video_output_get_format(mix->video) == VIDEO_FORMAT_BGRA) {
54
        tech = gs_effect_get_technique(effect, "DrawAlphaDivide");
55
    } else {
56
-       if ((width == video->base_width) &&
57
-           (height == video->base_height))
58
+       if ((width == mix->ovi.base_width) &&
59
+           (height == mix->ovi.base_height))
60
            return texture;
61
 
62
        tech = gs_effect_get_technique(effect, "Draw");
63
@@ -254,15 +254,15 @@
64
 
65
    if (bres) {
66
        struct vec2 base;
67
-       vec2_set(&base, (float)video->base_width,
68
-            (float)video->base_height);
69
+       vec2_set(&base, (float)mix->ovi.base_width,
70
+            (float)mix->ovi.base_height);
71
        gs_effect_set_vec2(bres, &base);
72
    }
73
 
74
    if (bres_i) {
75
        struct vec2 base_i;
76
-       vec2_set(&base_i, 1.0f / (float)video->base_width,
77
-            1.0f / (float)video->base_height);
78
+       vec2_set(&base_i, 1.0f / (float)mix->ovi.base_width,
79
+            1.0f / (float)mix->ovi.base_height);
80
        gs_effect_set_vec2(bres_i, &base_i);
81
    }
82
 
83
obs-studio-28.1.2.tar.xz/libobs/obs-view.c -> obs-studio-29.0.0.tar.xz/libobs/obs-view.c Changed
21
 
1
@@ -162,10 +162,17 @@
2
 
3
 video_t *obs_view_add(obs_view_t *view)
4
 {
5
-   if (!view)
6
+   if (!obs->video.main_mix)
7
+       return NULL;
8
+   return obs_view_add2(view, &obs->video.main_mix->ovi);
9
+}
10
+
11
+video_t *obs_view_add2(obs_view_t *view, struct obs_video_info *ovi)
12
+{
13
+   if (!view || !ovi)
14
        return NULL;
15
 
16
-   struct obs_core_video_mix *mix = obs_create_video_mix(&obs->video.ovi);
17
+   struct obs_core_video_mix *mix = obs_create_video_mix(ovi);
18
    if (!mix) {
19
        return NULL;
20
    }
21
obs-studio-28.1.2.tar.xz/libobs/obs-windows.c -> obs-studio-29.0.0.tar.xz/libobs/obs-windows.c Changed
78
 
1
@@ -108,6 +108,13 @@
2
         os_get_physical_cores(), os_get_logical_cores());
3
 }
4
 
5
+static void log_emulation_status(void)
6
+{
7
+   if (os_get_emulation_status()) {
8
+       blog(LOG_WARNING, "Windows ARM64: Running with x64 emulation");
9
+   }
10
+}
11
+
12
 static void log_available_memory(void)
13
 {
14
    MEMORYSTATUSEX ms;
15
@@ -138,10 +145,13 @@
16
    bool b64 = is_64_bit_windows();
17
    const char *windows_bitness = b64 ? "64" : "32";
18
 
19
+   bool arm64 = is_arm64_windows();
20
+   const char *arm64_windows = arm64 ? "ARM " : "";
21
+
22
    blog(LOG_INFO,
23
-        "Windows Version: %d.%d Build %d (release: %s; revision: %d; %s-bit)",
24
+        "Windows Version: %d.%d Build %d (release: %s; revision: %d; %s%s-bit)",
25
         ver.major, ver.minor, ver.build, release_id, ver.revis,
26
-        windows_bitness);
27
+        arm64_windows, windows_bitness);
28
 }
29
 
30
 static void log_admin_status(void)
31
@@ -164,36 +174,6 @@
32
         success ? "true" : "false");
33
 }
34
 
35
-typedef HRESULT(WINAPI *dwm_is_composition_enabled_t)(BOOL *);
36
-
37
-static void log_aero(void)
38
-{
39
-   dwm_is_composition_enabled_t composition_enabled = NULL;
40
-
41
-   const char *aeroMessage =
42
-       win_ver >= 0x602
43
-           ? " (Aero is always on for windows 8 and above)"
44
-           : "";
45
-
46
-   HMODULE dwm = LoadLibraryW(L"dwmapi");
47
-   BOOL bComposition = true;
48
-
49
-   if (!dwm) {
50
-       return;
51
-   }
52
-
53
-   composition_enabled = (dwm_is_composition_enabled_t)GetProcAddress(
54
-       dwm, "DwmIsCompositionEnabled");
55
-   if (!composition_enabled) {
56
-       FreeLibrary(dwm);
57
-       return;
58
-   }
59
-
60
-   composition_enabled(&bComposition);
61
-   blog(LOG_INFO, "Aero is %s%s", bComposition ? "Enabled" : "Disabled",
62
-        aeroMessage);
63
-}
64
-
65
 #define WIN10_GAME_BAR_REG_KEY \
66
    L"Software\\Microsoft\\Windows\\CurrentVersion\\GameDVR"
67
 #define WIN10_GAME_DVR_POLICY_REG_KEY \
68
@@ -407,8 +387,8 @@
69
    log_processor_cores();
70
    log_available_memory();
71
    log_windows_version();
72
+   log_emulation_status();
73
    log_admin_status();
74
-   log_aero();
75
    log_gaming_features();
76
    log_security_products();
77
 }
78
obs-studio-28.1.2.tar.xz/libobs/obs.c -> obs-studio-29.0.0.tar.xz/libobs/obs.c Changed
130
 
1
@@ -398,7 +398,7 @@
2
    }
3
 
4
    video->render_texture =
5
-       gs_texture_create(obs->video.base_width, obs->video.base_height,
6
+       gs_texture_create(video->ovi.base_width, video->ovi.base_height,
7
                  format, 1, NULL, GS_RENDER_TARGET);
8
    if (!video->render_texture)
9
        success = false;
10
@@ -586,8 +586,20 @@
11
    pthread_mutex_init_value(&video->gpu_encoder_mutex);
12
 
13
    make_video_info(&vi, ovi);
14
+   video->ovi = *ovi;
15
+
16
+   /* main view graphics thread drives all frame output,
17
+    * so share FPS settings for aux views */
18
+   pthread_mutex_lock(&obs->video.mixes_mutex);
19
+   size_t num = obs->video.mixes.num;
20
+   if (num && obs->video.main_mix) {
21
+       struct obs_video_info main_ovi = obs->video.main_mix->ovi;
22
+       video->ovi.fps_num = main_ovi.fps_num;
23
+       video->ovi.fps_den = main_ovi.fps_den;
24
+   }
25
+   pthread_mutex_unlock(&obs->video.mixes_mutex);
26
+
27
    video->gpu_conversion = ovi->gpu_conversion;
28
-   video->scale_type = ovi->scale_type;
29
    video->gpu_was_active = false;
30
    video->raw_was_active = false;
31
    video->was_active = false;
32
@@ -634,8 +646,6 @@
33
 static int obs_init_video(struct obs_video_info *ovi)
34
 {
35
    struct obs_core_video *video = &obs->video;
36
-   video->base_width = ovi->base_width;
37
-   video->base_height = ovi->base_height;
38
    video->video_frame_interval_ns =
39
        util_mul_div64(1000000000ULL, ovi->fps_den, ovi->fps_num);
40
    video->video_half_frame_interval_ns =
41
@@ -646,9 +656,7 @@
42
    if (pthread_mutex_init(&video->mixes_mutex, NULL) < 0)
43
        return OBS_VIDEO_FAIL;
44
 
45
-   video->ovi = *ovi;
46
-
47
-   if (!obs_view_add(&obs->data.main_view))
48
+   if (!obs_view_add2(&obs->data.main_view, ovi))
49
        return OBS_VIDEO_FAIL;
50
 
51
    int errorcode;
52
@@ -831,8 +839,6 @@
53
    struct obs_task_info audio_init = {.task = set_audio_thread};
54
    circlebuf_push_back(&audio->tasks, &audio_init, sizeof(audio_init));
55
 
56
-   audio->user_volume = 1.0f;
57
-
58
    audio->monitoring_device_name = bstrdup("Default");
59
    audio->monitoring_device_id = bstrdup("default");
60
 
61
@@ -970,6 +976,7 @@
62
    "void source_create(ptr source)",
63
    "void source_destroy(ptr source)",
64
    "void source_remove(ptr source)",
65
+   "void source_update(ptr source)",
66
    "void source_save(ptr source)",
67
    "void source_load(ptr source)",
68
    "void source_activate(ptr source)",
69
@@ -987,7 +994,6 @@
70
    "void source_transition_stop(ptr source)",
71
 
72
    "void channel_change(int channel, in out ptr source, ptr prev_source)",
73
-   "void master_volume(in out float volume)",
74
 
75
    "void hotkey_layout_change()",
76
    "void hotkey_register(ptr hotkey)",
77
@@ -1509,10 +1515,10 @@
78
 
79
 bool obs_get_video_info(struct obs_video_info *ovi)
80
 {
81
-   if (!obs->video.graphics)
82
+   if (!obs->video.graphics || !obs->video.main_mix)
83
        return false;
84
 
85
-   *ovi = obs->video.ovi;
86
+   *ovi = obs->video.main_mix->ovi;
87
    return true;
88
 }
89
 
90
@@ -2090,19 +2096,12 @@
91
 
92
 void obs_set_master_volume(float volume)
93
 {
94
-   struct calldata data = {0};
95
-
96
-   calldata_set_float(&data, "volume", volume);
97
-   signal_handler_signal(obs->signals, "master_volume", &data);
98
-   volume = (float)calldata_float(&data, "volume");
99
-   calldata_free(&data);
100
-
101
-   obs->audio.user_volume = volume;
102
+   UNUSED_PARAMETER(volume);
103
 }
104
 
105
 float obs_get_master_volume(void)
106
 {
107
-   return obs->audio.user_volume;
108
+   return 1.f;
109
 }
110
 
111
 static obs_source_t *obs_load_source_type(obs_data_t *source_data,
112
@@ -2824,7 +2823,7 @@
113
 
114
 bool start_gpu_encode(obs_encoder_t *encoder)
115
 {
116
-   struct obs_core_video_mix *video = obs->video.main_mix;
117
+   struct obs_core_video_mix *video = get_mix_for_video(encoder->media);
118
    bool success = true;
119
 
120
    obs_enter_graphics();
121
@@ -2850,7 +2849,7 @@
122
 
123
 void stop_gpu_encode(obs_encoder_t *encoder)
124
 {
125
-   struct obs_core_video_mix *video = obs->video.main_mix;
126
+   struct obs_core_video_mix *video = get_mix_for_video(encoder->media);
127
    bool call_free = false;
128
 
129
    os_atomic_dec_long(&video->gpu_encoder_active);
130
obs-studio-28.1.2.tar.xz/libobs/obs.h -> obs-studio-29.0.0.tar.xz/libobs/obs.h Changed
28
 
1
@@ -754,10 +754,10 @@
2
 EXPORT gs_texture_t *obs_get_main_texture(void);
3
 
4
 /** Sets the master user volume */
5
-EXPORT void obs_set_master_volume(float volume);
6
+OBS_DEPRECATED EXPORT void obs_set_master_volume(float volume);
7
 
8
 /** Gets the master user volume */
9
-EXPORT float obs_get_master_volume(void);
10
+OBS_DEPRECATED EXPORT float obs_get_master_volume(void);
11
 
12
 /** Saves a source to settings data */
13
 EXPORT obs_data_t *obs_save_source(obs_source_t *source);
14
@@ -909,9 +909,12 @@
15
 /** Renders the sources of this view context */
16
 EXPORT void obs_view_render(obs_view_t *view);
17
 
18
-/** Adds a view to the main render loop */
19
+/** Adds a view to the main render loop, with current obs_get_video_info state */
20
 EXPORT video_t *obs_view_add(obs_view_t *view);
21
 
22
+/** Adds a view to the main render loop, with custom video settings */
23
+EXPORT video_t *obs_view_add2(obs_view_t *view, struct obs_video_info *ovi);
24
+
25
 /** Removes a view from the main render loop */
26
 EXPORT void obs_view_remove(obs_view_t *view);
27
 
28
obs-studio-28.1.2.tar.xz/libobs/obsconfig.h.in -> obs-studio-29.0.0.tar.xz/libobs/obsconfig.h.in Changed
9
 
1
@@ -16,6 +16,7 @@
2
 #define OBS_PLUGIN_DESTINATION "@OBS_PLUGIN_DESTINATION@"
3
 #define OBS_QT_VERSION @_QT_VERSION@
4
 
5
+#cmakedefine OBS_COMMIT "@OBS_COMMIT@"
6
 #cmakedefine LINUX_PORTABLE
7
 #cmakedefine GIO_FOUND
8
 #cmakedefine PULSEAUDIO_FOUND
9
obs-studio-28.1.2.tar.xz/libobs/util/platform-cocoa.m -> obs-studio-29.0.0.tar.xz/libobs/util/platform-cocoa.m Changed
30
 
1
@@ -386,6 +386,28 @@
2
    return vmstat.free_count * vm_page_size;
3
 }
4
 
5
+static uint64_t total_memory = 0;
6
+static bool total_memory_initialized = false;
7
+
8
+static void os_get_sys_total_size_internal()
9
+{
10
+   total_memory_initialized = true;
11
+
12
+   size_t size;
13
+   int ret;
14
+
15
+   size = sizeof(total_memory);
16
+   ret = sysctlbyname("hw.memsize", &total_memory, &size, NULL, 0);
17
+}
18
+
19
+uint64_t os_get_sys_total_size(void)
20
+{
21
+   if (!total_memory_initialized)
22
+       os_get_sys_total_size_internal();
23
+
24
+   return total_memory;
25
+}
26
+
27
 #ifndef MACH_TASK_BASIC_INFO
28
 typedef task_basic_info_data_t mach_task_basic_info_data_t;
29
 #endif
30
obs-studio-28.1.2.tar.xz/libobs/util/platform-nix.c -> obs-studio-29.0.0.tar.xz/libobs/util/platform-nix.c Changed
42
 
1
@@ -53,6 +53,9 @@
2
 #else
3
 #include <sys/resource.h>
4
 #endif
5
+#if !defined(__OpenBSD__)
6
+#include <sys/sysinfo.h>
7
+#endif
8
 #include <spawn.h>
9
 #endif
10
 
11
@@ -1100,6 +1103,30 @@
12
    return (uint64_t)statm.virtual_size;
13
 }
14
 #endif
15
+
16
+static uint64_t total_memory = 0;
17
+static bool total_memory_initialized = false;
18
+
19
+static void os_get_sys_total_size_internal()
20
+{
21
+   total_memory_initialized = true;
22
+
23
+#ifndef __OpenBSD__
24
+   struct sysinfo info;
25
+   if (sysinfo(&info) < 0)
26
+       return;
27
+
28
+   total_memory = (uint64_t)info.totalram * info.mem_unit;
29
+#endif
30
+}
31
+
32
+uint64_t os_get_sys_total_size(void)
33
+{
34
+   if (!total_memory_initialized)
35
+       os_get_sys_total_size_internal();
36
+
37
+   return total_memory;
38
+}
39
 #endif
40
 
41
 uint64_t os_get_free_disk_space(const char *dir)
42
obs-studio-28.1.2.tar.xz/libobs/util/platform-windows.c -> obs-studio-29.0.0.tar.xz/libobs/util/platform-windows.c Changed
68
 
1
@@ -1088,9 +1088,26 @@
2
 #endif
3
 }
4
 
5
+bool is_arm64_windows(void)
6
+{
7
+#if defined(_M_ARM64) || defined(_M_ARM64EC)
8
+   return true;
9
+#else
10
+   USHORT processMachine;
11
+   USHORT nativeMachine;
12
+   bool result = IsWow64Process2(GetCurrentProcess(), &processMachine,
13
+                     &nativeMachine);
14
+   return (result && (nativeMachine == IMAGE_FILE_MACHINE_ARM64));
15
+#endif
16
+}
17
+
18
 bool os_get_emulation_status(void)
19
 {
20
+#if defined(_M_ARM64) || defined(_M_ARM64EC)
21
    return false;
22
+#else
23
+   return is_arm64_windows();
24
+#endif
25
 }
26
 
27
 void get_reg_dword(HKEY hkey, LPCWSTR sub_key, LPCWSTR value_name,
28
@@ -1187,7 +1204,9 @@
29
        ver->build = wcstol(str, NULL, 10);
30
    }
31
 
32
-   if (get_reg_sz(key, L"ReleaseId", str, sizeof(str))) {
33
+   const wchar_t *release_key = ver->build > 19041 ? L"DisplayVersion"
34
+                           : L"ReleaseId";
35
+   if (get_reg_sz(key, release_key, str, sizeof(str))) {
36
        os_wcs_to_utf8(str, 0, win_release_id, MAX_SZ_LEN);
37
    }
38
 
39
@@ -1390,6 +1409,28 @@
40
    return msex.ullAvailPhys;
41
 }
42
 
43
+static uint64_t total_memory = 0;
44
+static bool total_memory_initialized = false;
45
+
46
+static void os_get_sys_total_size_internal()
47
+{
48
+   total_memory_initialized = true;
49
+
50
+   MEMORYSTATUSEX msex = {sizeof(MEMORYSTATUSEX)};
51
+   if (!os_get_sys_memory_usage_internal(&msex))
52
+       return;
53
+
54
+   total_memory = msex.ullTotalPhys;
55
+}
56
+
57
+uint64_t os_get_sys_total_size(void)
58
+{
59
+   if (!total_memory_initialized)
60
+       os_get_sys_total_size_internal();
61
+
62
+   return total_memory;
63
+}
64
+
65
 static inline bool
66
 os_get_proc_memory_usage_internal(PROCESS_MEMORY_COUNTERS *pmc)
67
 {
68
obs-studio-28.1.2.tar.xz/libobs/util/platform.c -> obs-studio-29.0.0.tar.xz/libobs/util/platform.c Changed
53
 
1
@@ -559,7 +559,8 @@
2
 double os_strtod(const char *str)
3
 {
4
    char buf64;
5
-   snprintf(buf, 64, "%s", str);
6
+   strncpy(buf, str, sizeof(buf) - 1);
7
+   bufsizeof(buf) - 1 = 0;
8
    to_locale(buf);
9
    return strtod(buf, NULL);
10
 }
11
@@ -760,23 +761,25 @@
12
        if (!convert0) {
13
            if (astrcmp_n(cmp, "%FPS", 4) == 0) {
14
                if (ovi.fps_den <= 1) {
15
-                   sprintf(convert, "%u", ovi.fps_num);
16
+                   snprintf(convert, sizeof(convert), "%u",
17
+                        ovi.fps_num);
18
                } else {
19
                    const double obsFPS =
20
                        (double)ovi.fps_num /
21
                        (double)ovi.fps_den;
22
-                   sprintf(convert, "%.2f", obsFPS);
23
+                   snprintf(convert, sizeof(convert),
24
+                        "%.2f", obsFPS);
25
                }
26
                replace_text(&sf, pos, 4, convert);
27
 
28
            } else if (astrcmp_n(cmp, "%CRES", 5) == 0) {
29
-               sprintf(convert, "%ux%u", ovi.base_width,
30
-                   ovi.base_height);
31
+               snprintf(convert, sizeof(convert), "%ux%u",
32
+                    ovi.base_width, ovi.base_height);
33
                replace_text(&sf, pos, 5, convert);
34
 
35
            } else if (astrcmp_n(cmp, "%ORES", 5) == 0) {
36
-               sprintf(convert, "%ux%u", ovi.output_width,
37
-                   ovi.output_height);
38
+               snprintf(convert, sizeof(convert), "%ux%u",
39
+                    ovi.output_width, ovi.output_height);
40
                replace_text(&sf, pos, 5, convert);
41
 
42
            } else if (astrcmp_n(cmp, "%VF", 3) == 0) {
43
@@ -785,7 +788,8 @@
44
                replace_text(&sf, pos, 3, convert);
45
 
46
            } else if (astrcmp_n(cmp, "%s", 2) == 0) {
47
-               sprintf(convert, "%" PRId64, (int64_t)now);
48
+               snprintf(convert, sizeof(convert), "%" PRId64,
49
+                    (int64_t)now);
50
                replace_text(&sf, pos, 2, convert);
51
            }
52
        }
53
obs-studio-28.1.2.tar.xz/libobs/util/platform.h -> obs-studio-29.0.0.tar.xz/libobs/util/platform.h Changed
9
 
1
@@ -189,6 +189,7 @@
2
 EXPORT int os_get_logical_cores(void);
3
 
4
 EXPORT uint64_t os_get_sys_free_size(void);
5
+EXPORT uint64_t os_get_sys_total_size(void);
6
 
7
 struct os_proc_memory_usage {
8
    uint64_t resident_size;
9
obs-studio-29.0.0.tar.xz/libobs/util/windows/device-enum.c Added
34
 
1
@@ -0,0 +1,32 @@
2
+#include "device-enum.h"
3
+#include "../dstr.h"
4
+
5
+#include <dxgi.h>
6
+
7
+void enum_graphics_device_luids(device_luid_cb device_luid, void *param)
8
+{
9
+   IDXGIFactory1 *factory;
10
+   IDXGIAdapter1 *adapter;
11
+   HRESULT hr;
12
+
13
+   hr = CreateDXGIFactory1(&IID_IDXGIFactory1, (void **)&factory);
14
+   if (FAILED(hr))
15
+       return;
16
+
17
+   for (UINT i = 0;
18
+        factory->lpVtbl->EnumAdapters1(factory, i, &adapter) == S_OK;
19
+        i++) {
20
+       DXGI_ADAPTER_DESC desc;
21
+
22
+       hr = adapter->lpVtbl->GetDesc(adapter, &desc);
23
+       adapter->lpVtbl->Release(adapter);
24
+       if (FAILED(hr))
25
+           continue;
26
+
27
+       uint64_t luid64 = *(uint64_t *)&desc.AdapterLuid;
28
+       if (!device_luid(param, i, luid64))
29
+           break;
30
+   }
31
+
32
+   factory->lpVtbl->Release(factory);
33
+}
34
obs-studio-29.0.0.tar.xz/libobs/util/windows/device-enum.h Added
16
 
1
@@ -0,0 +1,14 @@
2
+#pragma once
3
+
4
+#include "../c99defs.h"
5
+
6
+#ifdef __cplusplus
7
+extern "C" {
8
+#endif
9
+
10
+typedef bool (*device_luid_cb)(void *param, uint32_t idx, uint64_t luid);
11
+EXPORT void enum_graphics_device_luids(device_luid_cb device_luid, void *param);
12
+
13
+#ifdef __cplusplus
14
+}
15
+#endif
16
obs-studio-28.1.2.tar.xz/libobs/util/windows/win-version.h -> obs-studio-29.0.0.tar.xz/libobs/util/windows/win-version.h Changed
9
 
1
@@ -48,6 +48,7 @@
2
 }
3
 
4
 EXPORT bool is_64_bit_windows(void);
5
+EXPORT bool is_arm64_windows(void);
6
 EXPORT bool get_dll_ver(const wchar_t *lib, struct win_version_info *info);
7
 EXPORT void get_win_ver(struct win_version_info *info);
8
 EXPORT uint32_t get_win_ver_int(void);
9
obs-studio-28.1.2.tar.xz/plugins/aja/aja-presets.cpp -> obs-studio-29.0.0.tar.xz/plugins/aja/aja-presets.cpp Changed
29
 
1
@@ -187,6 +187,27 @@
2
          {DEVICE_ID_IO4KPLUS},
3
          false,
4
          false}},
5
+       {"HDMI_UHD_4K_YCbCr_Capture",
6
+        {"HDMI_UHD_4K_YCbCr_Capture",
7
+         ConnectionKind::HDMI,
8
+         NTV2_MODE_CAPTURE,
9
+         RasterDefinition::UHD_4K,
10
+         HDMIWireFormat::UHD_4K_YCBCR,
11
+         VPIDStandard_Unknown,
12
+         1,
13
+         2,
14
+         kEnable4KTSI,
15
+         "hdmi{ch1}0->tsi{ch1}0;"
16
+         "hdmi{ch1}1->tsi{ch1}1;"
17
+         "hdmi{ch1}2->tsi{ch2}0;"
18
+         "hdmi{ch1}3->tsi{ch2}1;"
19
+         "tsi{ch1}0->fb{ch1}0;"
20
+         "tsi{ch1}1->fb{ch1}1;"
21
+         "tsi{ch2}0->fb{ch2}0;"
22
+         "tsi{ch2}1->fb{ch2}1;",
23
+         {},
24
+         false,
25
+         false}},
26
        /*
27
         * HDMI YCbCr Display
28
         */
29
obs-studio-28.1.2.tar.xz/plugins/aja/aja-props.cpp -> obs-studio-29.0.0.tar.xz/plugins/aja/aja-props.cpp Changed
10
 
1
@@ -155,7 +155,7 @@
2
        NTV2_IS_4K_VIDEO_FORMAT(videoFormat)) {
3
        return NTV2_CHANNEL3;
4
    }
5
-   return NTV2InputSourceToChannel(InitialInputSource());
6
+   return Channel();
7
 }
8
 
9
 NTV2AudioSystem SourceProps::AudioSystem() const
10
obs-studio-28.1.2.tar.xz/plugins/aja/aja-source.cpp -> obs-studio-29.0.0.tar.xz/plugins/aja/aja-source.cpp Changed
33
 
1
@@ -269,6 +269,7 @@
2
                    sourceProps.pixelFormat);
3
    auto inputSource = sourceProps.InitialInputSource();
4
    auto channel = sourceProps.Channel();
5
+   auto framestore = sourceProps.Framestore();
6
    auto audioSystem = sourceProps.AudioSystem();
7
    // Current "on-air" frame on the card. The capture thread "Ping-pongs" between
8
    // two frames, starting at an index corresponding to the framestore channel.
9
@@ -278,12 +279,12 @@
10
    // Channel 3 (index 2) = frames 4/5
11
    // Channel 4 (index 3) = frames 6/7
12
    // etc...
13
-   ULWord currentCardFrame = (uint32_t)channel * 2;
14
+   ULWord currentCardFrame = GetIndexForNTV2Channel(framestore) * 2;
15
    card->WaitForInputFieldID(NTV2_FIELD0, channel);
16
 
17
    currentCardFrame ^= 1;
18
 
19
-   card->SetInputFrame(channel, currentCardFrame);
20
+   card->SetInputFrame(framestore, currentCardFrame);
21
 
22
    AudioOffsets offsets;
23
    ResetAudioBufferOffsets(card, audioSystem, offsets);
24
@@ -388,7 +389,7 @@
25
 
26
        obs_source_output_video2(ajaSource->mSource, &obsFrame);
27
 
28
-       card->SetInputFrame(channel, currentCardFrame);
29
+       card->SetInputFrame(framestore, currentCardFrame);
30
    }
31
 
32
    blog(LOG_INFO, "AJASource::Capturethread: Thread loop stopped");
33
obs-studio-28.1.2.tar.xz/plugins/coreaudio-encoder/encoder.cpp -> obs-studio-29.0.0.tar.xz/plugins/coreaudio-encoder/encoder.cpp Changed
10
 
1
@@ -175,7 +175,7 @@
2
 
3
    char array4096;
4
    va_start(args, fmt);
5
-   vsnprintf(array, 4096, fmt, args);
6
+   vsnprintf(array, sizeof(array), fmt, args);
7
    va_end(args);
8
 
9
    array4095 = 0;
10
obs-studio-28.1.2.tar.xz/plugins/decklink/decklink-device-instance.cpp -> obs-studio-29.0.0.tar.xz/plugins/decklink/decklink-device-instance.cpp Changed
65
 
1
@@ -521,10 +521,9 @@
2
 
3
    mode = mode_;
4
 
5
-   int keyerMode = device->GetKeyerMode();
6
-
7
    ComPtr<IDeckLinkKeyer> deckLinkKeyer;
8
    if (device->GetKeyer(&deckLinkKeyer)) {
9
+       const int keyerMode = device->GetKeyerMode();
10
        if (keyerMode) {
11
            deckLinkKeyer->Enable(keyerMode == 1);
12
            deckLinkKeyer->SetLevel(255);
13
@@ -537,21 +536,11 @@
14
    if (decklinkOutput == nullptr)
15
        return false;
16
 
17
-   int rowBytes = decklinkOutput->GetWidth() * 2;
18
-   if (decklinkOutput->keyerMode != 0) {
19
-       rowBytes = decklinkOutput->GetWidth() * 4;
20
-   }
21
-
22
-   BMDPixelFormat pixelFormat = bmdFormat8BitYUV;
23
-   if (keyerMode != 0) {
24
-       pixelFormat = bmdFormat8BitBGRA;
25
-   }
26
-
27
    HRESULT result;
28
-   result = output->CreateVideoFrame(decklinkOutput->GetWidth(),
29
-                     decklinkOutput->GetHeight(), rowBytes,
30
-                     pixelFormat, bmdFrameFlagDefault,
31
-                     &decklinkOutputFrame);
32
+   result = output->CreateVideoFrame(
33
+       decklinkOutput->GetWidth(), decklinkOutput->GetHeight(),
34
+       decklinkOutput->GetWidth() * 4, bmdFormat8BitBGRA,
35
+       bmdFrameFlagDefault, &decklinkOutputFrame);
36
    if (result != S_OK) {
37
        blog(LOG_ERROR, "failed to make frame 0x%X", result);
38
        return false;
39
@@ -571,8 +560,7 @@
40
    output->DisableVideoOutput();
41
    output->DisableAudioOutput();
42
 
43
-   if (decklinkOutputFrame != nullptr)
44
-       decklinkOutputFrame = nullptr;
45
+   decklinkOutputFrame.Clear();
46
 
47
    return true;
48
 }
49
@@ -588,12 +576,9 @@
50
 
51
    uint8_t *outData = frame->data0;
52
 
53
-   int rowBytes = decklinkOutput->GetWidth() * 2;
54
-   if (device->GetKeyerMode()) {
55
-       rowBytes = decklinkOutput->GetWidth() * 4;
56
-   }
57
-
58
-   std::copy(outData, outData + (decklinkOutput->GetHeight() * rowBytes),
59
+   std::copy(outData,
60
+         outData + (decklinkOutput->GetWidth() *
61
+                decklinkOutput->GetHeight() * 4),
62
          destData);
63
 
64
    output->DisplayVideoFrameSync(decklinkOutputFrame);
65
obs-studio-28.1.2.tar.xz/plugins/decklink/decklink-output.cpp -> obs-studio-29.0.0.tar.xz/plugins/decklink/decklink-output.cpp Changed
20
 
1
@@ -81,16 +81,10 @@
2
    decklink->SetSize(mode->GetWidth(), mode->GetHeight());
3
 
4
    struct video_scale_info to = {};
5
-
6
-   if (decklink->keyerMode != 0) {
7
-       to.format = VIDEO_FORMAT_BGRA;
8
-       to.range = VIDEO_RANGE_FULL;
9
-   } else {
10
-       to.format = VIDEO_FORMAT_UYVY;
11
-       to.range = VIDEO_RANGE_PARTIAL;
12
-   }
13
+   to.format = VIDEO_FORMAT_BGRA;
14
    to.width = mode->GetWidth();
15
    to.height = mode->GetHeight();
16
+   to.range = VIDEO_RANGE_FULL;
17
    to.colorspace = VIDEO_CS_709;
18
 
19
    obs_output_set_video_conversion(decklink->GetOutput(), &to);
20
obs-studio-28.1.2.tar.xz/plugins/image-source/obs-slideshow.c -> obs-studio-29.0.0.tar.xz/plugins/image-source/obs-slideshow.c Changed
131
 
1
@@ -186,7 +186,14 @@
2
 
3
 static inline size_t random_file(struct slideshow *ss)
4
 {
5
-   return (size_t)rand() % ss->files.num;
6
+   size_t next = ss->cur_item;
7
+
8
+   if (ss->files.num > 1) {
9
+       while (next == ss->cur_item)
10
+           next = (size_t)rand() % ss->files.num;
11
+   }
12
+
13
+   return next;
14
 }
15
 
16
 /* ------------------------------------------------------------------------- */
17
@@ -343,6 +350,11 @@
18
        const char *path = obs_data_get_string(item, "value");
19
        os_dir_t *dir = os_opendir(path);
20
 
21
+       if (!path || !*path) {
22
+           obs_data_release(item);
23
+           continue;
24
+       }
25
+
26
        if (dir) {
27
            struct dstr dir_path = {0};
28
            struct os_dirent *ent;
29
@@ -539,7 +551,9 @@
30
    if (!ss->files.num || obs_transition_get_time(ss->transition) < 1.0f)
31
        return;
32
 
33
-   if (++ss->cur_item >= ss->files.num)
34
+   if (ss->randomize)
35
+       ss->cur_item = random_file(ss);
36
+   else if (++ss->cur_item >= ss->files.num)
37
        ss->cur_item = 0;
38
 
39
    do_transition(ss, false);
40
@@ -552,7 +566,9 @@
41
    if (!ss->files.num || obs_transition_get_time(ss->transition) < 1.0f)
42
        return;
43
 
44
-   if (ss->cur_item == 0)
45
+   if (ss->randomize)
46
+       ss->cur_item = random_file(ss);
47
+   else if (ss->cur_item == 0)
48
        ss->cur_item = ss->files.num - 1;
49
    else
50
        --ss->cur_item;
51
@@ -626,6 +642,18 @@
52
        obs_source_media_previous(ss->source);
53
 }
54
 
55
+static void current_slide_proc(void *data, calldata_t *cd)
56
+{
57
+   struct slideshow *ss = data;
58
+   calldata_set_int(cd, "current_index", ss->cur_item);
59
+}
60
+
61
+static void total_slides_proc(void *data, calldata_t *cd)
62
+{
63
+   struct slideshow *ss = data;
64
+   calldata_set_int(cd, "total_files", ss->files.num);
65
+}
66
+
67
 static void ss_destroy(void *data)
68
 {
69
    struct slideshow *ss = data;
70
@@ -639,6 +667,7 @@
71
 static void *ss_create(obs_data_t *settings, obs_source_t *source)
72
 {
73
    struct slideshow *ss = bzalloc(sizeof(*ss));
74
+   proc_handler_t *ph = obs_source_get_proc_handler(source);
75
 
76
    ss->source = source;
77
 
78
@@ -667,6 +696,9 @@
79
        obs_module_text("SlideShow.PreviousSlide"),
80
        previous_slide_hotkey, ss);
81
 
82
+   proc_handler_add(ph, "int current_index()", current_slide_proc, ss);
83
+   proc_handler_add(ph, "int total_files()", total_slides_proc, ss);
84
+
85
    pthread_mutex_init_value(&ss->mutex);
86
    if (pthread_mutex_init(&ss->mutex, NULL) != 0)
87
        goto error;
88
@@ -742,20 +774,7 @@
89
            return;
90
        }
91
 
92
-       if (ss->randomize) {
93
-           size_t next = ss->cur_item;
94
-           if (ss->files.num > 1) {
95
-               while (next == ss->cur_item)
96
-                   next = random_file(ss);
97
-           }
98
-           ss->cur_item = next;
99
-
100
-       } else if (++ss->cur_item >= ss->files.num) {
101
-           ss->cur_item = 0;
102
-       }
103
-
104
-       if (ss->files.num)
105
-           do_transition(ss, false);
106
+       obs_source_media_next(ss->source);
107
    }
108
 }
109
 
110
@@ -922,7 +941,7 @@
111
        obs_property_list_add_string(p, aspectsi, aspectsi);
112
 
113
    char str32;
114
-   snprintf(str, 32, "%dx%d", cx, cy);
115
+   snprintf(str, sizeof(str), "%dx%d", cx, cy);
116
    obs_property_list_add_string(p, str, str);
117
 
118
    if (ss) {
119
@@ -987,7 +1006,10 @@
120
        const char *path = obs_data_get_string(file, "value");
121
 
122
        if (strcmp(path, orig_path) == 0) {
123
-           obs_data_set_string(file, "value", new_path);
124
+           if (new_path && *new_path)
125
+               obs_data_set_string(file, "value", new_path);
126
+           else
127
+               obs_data_array_erase(files, i);
128
 
129
            obs_data_release(file);
130
            break;
131
obs-studio-28.1.2.tar.xz/plugins/linux-capture/xshm-input.c -> obs-studio-29.0.0.tar.xz/plugins/linux-capture/xshm-input.c Changed
16
 
1
@@ -365,7 +365,13 @@
2
            x11_screen_geo(xcb, i, &w, &h);
3
 
4
        if (name == NULL) {
5
-           sprintf(name_tmp, "%" PRIuFAST32, i);
6
+           int ret = snprintf(name_tmp, sizeof(name_tmp),
7
+                      "%" PRIuFAST32, i);
8
+           if (ret >= sizeof(name_tmp))
9
+               blog(LOG_DEBUG,
10
+                    "linux-capture: A format truncation may have occurred."
11
+                    " This can be ignored since it is quite improbable.");
12
+
13
            name = name_tmp;
14
        }
15
 
16
obs-studio-28.1.2.tar.xz/plugins/linux-v4l2/v4l2-input.c -> obs-studio-29.0.0.tar.xz/plugins/linux-v4l2/v4l2-input.c Changed
18
 
1
@@ -416,8 +416,14 @@
2
 
3
        /* make sure device names are unique */
4
        char unique_device_name68;
5
-       sprintf(unique_device_name, "%s (%s)", video_cap.card,
6
-           video_cap.bus_info);
7
+       int ret = snprintf(unique_device_name,
8
+                  sizeof(unique_device_name), "%s (%s)",
9
+                  video_cap.card, video_cap.bus_info);
10
+       if (ret >= sizeof(unique_device_name))
11
+           blog(LOG_DEBUG,
12
+                "linux-v4l2: A format truncation may have occurred."
13
+                " This can be ignored since it is quite improbable.");
14
+
15
        obs_property_list_add_string(prop, unique_device_name,
16
                         device.array);
17
        blog(LOG_INFO, "Found device '%s' at %s", video_cap.card,
18
obs-studio-28.1.2.tar.xz/plugins/mac-avcapture/av-capture.mm -> obs-studio-29.0.0.tar.xz/plugins/mac-avcapture/av-capture.mm Changed
57
 
1
@@ -2205,34 +2205,32 @@
2
        OBS_COMBO_FORMAT_STRING);
3
    obs_property_list_add_string(dev_list, "", "");
4
 
5
-   NSArray *devices = nil;
6
-
7
-   AVCaptureDeviceDiscoverySession *mediaDeviceDiscoverySession = AVCaptureDeviceDiscoverySession
8
-       discoverySessionWithDeviceTypes:@
9
-           AVCaptureDeviceTypeBuiltInWideAngleCamera,
10
-           AVCaptureDeviceTypeExternalUnknown
11
-       
12
+   NSMutableArray *device_types = NSMutableArray
13
+       arrayWithObjects:AVCaptureDeviceTypeBuiltInWideAngleCamera,
14
+                AVCaptureDeviceTypeExternalUnknown, nil;
15
+#if __MAC_OS_X_VERSION_MAX_ALLOWED >= 130000
16
+   if (__builtin_available(macOS 13.0, *)) {
17
+       device_types addObject:AVCaptureDeviceTypeDeskViewCamera;
18
+   }
19
+#endif
20
+   AVCaptureDeviceDiscoverySession *video_discovery = AVCaptureDeviceDiscoverySession
21
+       discoverySessionWithDeviceTypes:device_types
22
                      mediaType:AVMediaTypeVideo
23
                       position:AVCaptureDevicePositionUnspecified;
24
-   NSArray *mediaDevices = mediaDeviceDiscoverySession devices;
25
+   for (AVCaptureDevice *dev in video_discovery devices) {
26
+       obs_property_list_add_string(dev_list,
27
+                        dev.localizedName.UTF8String,
28
+                        dev.uniqueID.UTF8String);
29
+   }
30
 
31
-   AVCaptureDeviceDiscoverySession *muxedDeviceDiscoverySession = AVCaptureDeviceDiscoverySession
32
-       discoverySessionWithDeviceTypes:@
33
-           AVCaptureDeviceTypeExternalUnknown
34
-       
35
+   AVCaptureDeviceDiscoverySession *muxed_discovery = AVCaptureDeviceDiscoverySession
36
+       discoverySessionWithDeviceTypes:device_types
37
                      mediaType:AVMediaTypeMuxed
38
                       position:AVCaptureDevicePositionUnspecified;
39
-   NSArray *muxedDevices = muxedDeviceDiscoverySession devices;
40
-
41
-   devices = mediaDevices arrayByAddingObjectsFromArray:muxedDevices;
42
-
43
-   for (AVCaptureDevice *dev in devices) {
44
-       if (dev hasMediaType:AVMediaTypeVideo ||
45
-           dev hasMediaType:AVMediaTypeMuxed) {
46
-           obs_property_list_add_string(
47
-               dev_list, dev.localizedName.UTF8String,
48
-               dev.uniqueID.UTF8String);
49
-       }
50
+   for (AVCaptureDevice *dev in muxed_discovery devices) {
51
+       obs_property_list_add_string(dev_list,
52
+                        dev.localizedName.UTF8String,
53
+                        dev.uniqueID.UTF8String);
54
    }
55
 
56
    obs_property_set_modified_callback(dev_list, properties_device_changed);
57
obs-studio-28.1.2.tar.xz/plugins/mac-capture/data/locale/en-US.ini -> obs-studio-29.0.0.tar.xz/plugins/mac-capture/data/locale/en-US.ini Changed
7
 
1
@@ -24,4 +24,5 @@
2
 SCK.Name="macOS Screen Capture"
3
 SCK.Name.Beta="macOS Screen Capture (BETA)"
4
 SCK.AudioUnavailable="Audio capture requires macOS 13 or newer."
5
+SCK.CaptureTypeUnavailable="Selected capture type requires macOS 13 or newer."
6
 SCK.Method="Method"
7
obs-studio-28.1.2.tar.xz/plugins/mac-capture/mac-display-capture.m -> obs-studio-29.0.0.tar.xz/plugins/mac-capture/mac-display-capture.m Changed
35
 
1
@@ -605,19 +605,20 @@
2
                    __attribute__((unused))) {
3
        char dimension_buffer412;
4
        char name_buffer256;
5
-       sprintf(dimension_buffer0, "%u",
6
-           (uint32_t)screen frame.size.width);
7
-       sprintf(dimension_buffer1, "%u",
8
-           (uint32_t)screen frame.size.height);
9
-       sprintf(dimension_buffer2, "%d",
10
-           (int32_t)screen frame.origin.x);
11
-       sprintf(dimension_buffer3, "%d",
12
-           (int32_t)screen frame.origin.y);
13
-
14
-       sprintf(name_buffer, "%.200s: %.12sx%.12s @ %.12s,%.12s",
15
-           screen localizedName UTF8String,
16
-           dimension_buffer0, dimension_buffer1,
17
-           dimension_buffer2, dimension_buffer3);
18
+       snprintf(dimension_buffer0, sizeof(dimension_buffer0), "%u",
19
+            (uint32_t)screen frame.size.width);
20
+       snprintf(dimension_buffer1, sizeof(dimension_buffer0), "%u",
21
+            (uint32_t)screen frame.size.height);
22
+       snprintf(dimension_buffer2, sizeof(dimension_buffer0), "%d",
23
+            (int32_t)screen frame.origin.x);
24
+       snprintf(dimension_buffer3, sizeof(dimension_buffer0), "%d",
25
+            (int32_t)screen frame.origin.y);
26
+
27
+       snprintf(name_buffer, sizeof(name_buffer),
28
+            "%.200s: %.12sx%.12s @ %.12s,%.12s",
29
+            screen localizedName UTF8String,
30
+            dimension_buffer0, dimension_buffer1,
31
+            dimension_buffer2, dimension_buffer3);
32
 
33
        obs_property_list_add_int(list, name_buffer, index);
34
    };
35
obs-studio-28.1.2.tar.xz/plugins/mac-capture/mac-screen-capture.m -> obs-studio-29.0.0.tar.xz/plugins/mac-capture/mac-screen-capture.m Changed
201
 
1
@@ -69,6 +69,8 @@
2
    NSString *application_id;
3
 };
4
 
5
+#pragma mark -
6
+
7
 static void destroy_screen_stream(struct screen_capture *sc)
8
 {
9
    if (sc->disp) {
10
@@ -365,33 +367,9 @@
11
    case ScreenCaptureDisplayStream: {
12
        SCDisplay *target_display = get_target_display();
13
 
14
-       if (@available(macOS 13.0, *)) {
15
-           content_filter = SCContentFilter alloc
16
-                initWithDisplay:target_display
17
-               excludingWindows:NSArray alloc init;
18
-       } else {
19
-           NSArray *excluded = sc->shareable_content.applications
20
-               filteredArrayUsingPredicate:
21
-                   NSPredicate predicateWithBlock:^BOOL(
22
-                                SCRunningApplication
23
-                                    *application,
24
-                                NSDictionary<
25
-                                    NSString *,
26
-                                    id>
27
-                                    *_Nullable bindings
28
-                                __attribute__((
29
-                                    unused))) {
30
-                       return application
31
-                               .bundleIdentifier
32
-                           isEqualToString:
33
-                               @"com.apple.controlcenter";
34
-                   };
35
-
36
-           content_filter = SCContentFilter alloc
37
-                     initWithDisplay:target_display
38
-               excludingApplications:excluded
39
-                    exceptingWindows:NSArray alloc init;
40
-       }
41
+       content_filter = SCContentFilter alloc
42
+            initWithDisplay:target_display
43
+           excludingWindows:NSArray alloc init;
44
 
45
        set_display_mode(sc, target_display);
46
    } break;
47
@@ -462,15 +440,24 @@
48
    os_sem_post(sc->shareable_content_available);
49
    sc->stream_properties setQueueDepth:8;
50
    sc->stream_properties setShowsCursor:!sc->hide_cursor;
51
-   sc->stream_properties setColorSpaceName:kCGColorSpaceSRGB;
52
-   sc->stream_properties setPixelFormat:'BGRA';
53
-#if __MAC_OS_X_VERSION_MAX_ALLOWED >= 130000
54
+   sc->stream_properties setColorSpaceName:kCGColorSpaceDisplayP3;
55
+   sc->stream_properties setPixelFormat:'l10r';
56
+
57
    if (@available(macOS 13.0, *)) {
58
+#if __MAC_OS_X_VERSION_MAX_ALLOWED >= 130000
59
        sc->stream_properties setCapturesAudio:TRUE;
60
        sc->stream_properties setExcludesCurrentProcessAudio:TRUE;
61
        sc->stream_properties setChannelCount:2;
62
-   }
63
 #endif
64
+   } else {
65
+       if (sc->capture_type != ScreenCaptureWindowStream) {
66
+           sc->disp = NULL;
67
+           os_event_init(&sc->disp_finished, OS_EVENT_TYPE_MANUAL);
68
+           os_event_init(&sc->stream_start_completed,
69
+                     OS_EVENT_TYPE_MANUAL);
70
+           return true;
71
+       }
72
+   }
73
 
74
    sc->disp = SCStream alloc initWithFilter:content_filter
75
                      configuration:sc->stream_properties
76
@@ -491,7 +478,7 @@
77
    }
78
 
79
 #if __MAC_OS_X_VERSION_MAX_ALLOWED >= 130000
80
-   if (__builtin_available(macOS 13.0, *)) {
81
+   if (@available(macOS 13.0, *)) {
82
        did_add_output = sc->disp
83
               addStreamOutput:sc->capture_delegate
84
                      type:SCStreamOutputTypeAudio
85
@@ -577,8 +564,8 @@
86
    sc->show_empty_names = obs_data_get_bool(settings, "show_empty_names");
87
    sc->show_hidden_windows =
88
        obs_data_get_bool(settings, "show_hidden_windows");
89
-   sc->window = obs_data_get_int(settings, "window");
90
-   sc->capture_type = obs_data_get_int(settings, "type");
91
+   sc->window = (CGWindowID)obs_data_get_int(settings, "window");
92
+   sc->capture_type = (unsigned int)obs_data_get_int(settings, "type");
93
 
94
    os_sem_init(&sc->shareable_content_available, 1);
95
    screen_capture_build_content_list(
96
@@ -591,7 +578,7 @@
97
    if (!sc->effect)
98
        goto fail;
99
 
100
-   sc->display = obs_data_get_int(settings, "display");
101
+   sc->display = (CGDirectDisplayID)obs_data_get_int(settings, "display");
102
    sc->application_id = NSString alloc
103
        initWithUTF8String:obs_data_get_string(settings,
104
                               "application");
105
@@ -649,18 +636,13 @@
106
    if (!sc->tex)
107
        return;
108
 
109
-   const bool linear_srgb = gs_get_linear_srgb();
110
-
111
    const bool previous = gs_framebuffer_srgb_enabled();
112
-   gs_enable_framebuffer_srgb(linear_srgb);
113
+   gs_enable_framebuffer_srgb(true);
114
 
115
    gs_eparam_t *param = gs_effect_get_param_by_name(sc->effect, "image");
116
-   if (linear_srgb)
117
-       gs_effect_set_texture_srgb(param, sc->tex);
118
-   else
119
-       gs_effect_set_texture(param, sc->tex);
120
+   gs_effect_set_texture(param, sc->tex);
121
 
122
-   while (gs_effect_loop(sc->effect, "Draw"))
123
+   while (gs_effect_loop(sc->effect, "DrawD65P3"))
124
        gs_draw_sprite(sc->tex, 0, 0, 0);
125
 
126
    gs_enable_framebuffer_srgb(previous);
127
@@ -704,10 +686,11 @@
128
        }
129
    }
130
 
131
-   obs_data_set_default_int(settings, "type", 0);
132
    obs_data_set_default_int(settings, "display", initial_display);
133
-   obs_data_set_default_int(settings, "window", kCGNullWindowID);
134
+
135
    obs_data_set_default_obj(settings, "application", NULL);
136
+   obs_data_set_default_int(settings, "type", ScreenCaptureDisplayStream);
137
+   obs_data_set_default_int(settings, "window", kCGNullWindowID);
138
    obs_data_set_default_bool(settings, "show_cursor", true);
139
    obs_data_set_default_bool(settings, "show_empty_names", false);
140
    obs_data_set_default_bool(settings, "show_hidden_windows", false);
141
@@ -718,7 +701,8 @@
142
    struct screen_capture *sc = data;
143
 
144
    CGWindowID old_window_id = sc->window;
145
-   CGWindowID new_window_id = obs_data_get_int(settings, "window");
146
+   CGWindowID new_window_id =
147
+       (CGWindowID)obs_data_get_int(settings, "window");
148
 
149
    if (new_window_id > 0 && new_window_id != old_window_id)
150
        sc->window = new_window_id;
151
@@ -771,6 +755,8 @@
152
    obs_leave_graphics();
153
 }
154
 
155
+#pragma mark - obs_properties
156
+
157
 static bool build_display_list(struct screen_capture *sc,
158
                   obs_properties_t *props)
159
 {
160
@@ -805,19 +791,20 @@
161
 
162
        char dimension_buffer412 = {};
163
        char name_buffer256 = {};
164
-       sprintf(dimension_buffer0, "%u",
165
-           (uint32_t)screen.frame.size.width);
166
-       sprintf(dimension_buffer1, "%u",
167
-           (uint32_t)screen.frame.size.height);
168
-       sprintf(dimension_buffer2, "%d",
169
-           (int32_t)screen.frame.origin.x);
170
-       sprintf(dimension_buffer3, "%d",
171
-           (int32_t)screen.frame.origin.y);
172
-
173
-       sprintf(name_buffer, "%.200s: %.12sx%.12s @ %.12s,%.12s",
174
-           screen.localizedName.UTF8String, dimension_buffer0,
175
-           dimension_buffer1, dimension_buffer2,
176
-           dimension_buffer3);
177
+       snprintf(dimension_buffer0, sizeof(dimension_buffer0), "%u",
178
+            (uint32_t)screen.frame.size.width);
179
+       snprintf(dimension_buffer1, sizeof(dimension_buffer0), "%u",
180
+            (uint32_t)screen.frame.size.height);
181
+       snprintf(dimension_buffer2, sizeof(dimension_buffer0), "%d",
182
+            (int32_t)screen.frame.origin.x);
183
+       snprintf(dimension_buffer3, sizeof(dimension_buffer0), "%d",
184
+            (int32_t)screen.frame.origin.y);
185
+
186
+       snprintf(name_buffer, sizeof(name_buffer),
187
+            "%.200s: %.12sx%.12s @ %.12s,%.12s",
188
+            screen.localizedName.UTF8String, dimension_buffer0,
189
+            dimension_buffer1, dimension_buffer2,
190
+            dimension_buffer3);
191
 
192
        obs_property_list_add_int(display_list, name_buffer,
193
                      display.displayID);
194
@@ -898,7 +885,8 @@
195
 {
196
    struct screen_capture *sc = data;
197
 
198
-   unsigned int capture_type_id = obs_data_get_int(settings, "type");
199
+   unsigned int capture_type_id =
200
+       (unsigned int)obs_data_get_int(settings, "type");
201
obs-studio-28.1.2.tar.xz/plugins/mac-videotoolbox/data/locale/en-US.ini -> obs-studio-29.0.0.tar.xz/plugins/mac-videotoolbox/data/locale/en-US.ini Changed
24
 
1
@@ -1,5 +1,11 @@
2
 VTH264EncHW="Apple VT H264 Hardware Encoder"
3
 VTH264EncSW="Apple VT H264 Software Encoder"
4
+VTHEVCEncHW="Apple VT HEVC Hardware Encoder"
5
+VTHEVCEncT2="Apple VT HEVC T2 Hardware Encoder"
6
+VTHEVCEncSW="Apple VT HEVC Software Encoder"
7
+VTProResEncHW="Apple VT ProRes Hardware Encoder"
8
+VTProResEncSW="Apple VT ProRes Software Encoder"
9
+VTEncoder="VideoToolbox Encoder"
10
 Bitrate="Bitrate"
11
 Quality="Quality"
12
 UseMaxBitrate="Limit bitrate"
13
@@ -9,4 +15,9 @@
14
 Profile="Profile"
15
 UseBFrames="Use B-Frames"
16
 RateControl="Rate Control"
17
-ColorFormatUnsupportedH264="The selected color format is not supported by the Apple VT H.264 encoder. Select a compatible color format in Settings -> Advanced or use a different encoder."
18
+ColorFormatUnsupported="The selected color format is not supported by the selected Apple VT encoder. Select a compatible color format in Settings -> Advanced or use a different encoder."
19
+ProResCodec="ProRes Codec"
20
+ProRes422Proxy="ProRes 422 Proxy"
21
+ProRes422LT="ProRes 422 LT"
22
+ProRes422="ProRes 422"
23
+ProRes422HQ="ProRes 422 HQ"
24
obs-studio-28.1.2.tar.xz/plugins/mac-videotoolbox/encoder.c -> obs-studio-29.0.0.tar.xz/plugins/mac-videotoolbox/encoder.c Changed
201
 
1
@@ -14,18 +14,28 @@
2
 
3
 #define VT_LOG(level, format, ...) \
4
    blog(level, "VideoToolbox encoder: " format, ##__VA_ARGS__)
5
-#define VT_LOG_ENCODER(encoder, level, format, ...)       \
6
-   blog(level, "VideoToolbox %s: 'h264': " format, \
7
-        obs_encoder_get_name(encoder), ##__VA_ARGS__)
8
-#define VT_BLOG(level, format, ...) \
9
-   VT_LOG_ENCODER(enc->encoder, level, format, ##__VA_ARGS__)
10
+#define VT_LOG_ENCODER(encoder, codec_type, level, format, ...) \
11
+   blog(level, "VideoToolbox %s: '%s': " format,         \
12
+        obs_encoder_get_name(encoder),                     \
13
+        codec_type_to_print_fmt(codec_type), ##__VA_ARGS__)
14
+#define VT_BLOG(level, format, ...)                                  \
15
+   VT_LOG_ENCODER(enc->encoder, enc->codec_type, level, format, \
16
+              ##__VA_ARGS__)
17
 
18
 struct vt_encoder_type_data {
19
    const char *disp_name;
20
    const char *id;
21
+   CMVideoCodecType codec_type;
22
    bool hardware_accelerated;
23
 };
24
 
25
+struct vt_prores_encoder_data {
26
+   FourCharCode codec_type;
27
+   CFStringRef encoder_id;
28
+};
29
+static DARRAY(struct vt_prores_encoder_data) vt_prores_hardware_encoder_list;
30
+static DARRAY(struct vt_prores_encoder_data) vt_prores_software_encoder_list;
31
+
32
 struct vt_encoder {
33
    obs_encoder_t *encoder;
34
 
35
@@ -42,6 +52,7 @@
36
    uint32_t rc_max_bitrate;
37
    float rc_max_bitrate_window;
38
    const char *profile;
39
+   CMVideoCodecType codec_type;
40
    bool bframes;
41
 
42
    int vt_pix_fmt;
43
@@ -54,6 +65,26 @@
44
    DARRAY(uint8_t) extra_data;
45
 };
46
 
47
+static const char *codec_type_to_print_fmt(CMVideoCodecType codec_type)
48
+{
49
+   switch (codec_type) {
50
+   case kCMVideoCodecType_H264:
51
+       return "h264";
52
+   case kCMVideoCodecType_HEVC:
53
+       return "hevc";
54
+   case kCMVideoCodecType_AppleProRes422Proxy:
55
+       return "apco";
56
+   case kCMVideoCodecType_AppleProRes422LT:
57
+       return "apcs";
58
+   case kCMVideoCodecType_AppleProRes422:
59
+       return "apcn";
60
+   case kCMVideoCodecType_AppleProRes422HQ:
61
+       return "apch";
62
+   default:
63
+       return "";
64
+   }
65
+}
66
+
67
 static void log_osstatus(int log_level, struct vt_encoder *enc,
68
             const char *context, OSStatus code)
69
 {
70
@@ -75,26 +106,134 @@
71
    CFRelease(err);
72
 }
73
 
74
-static CFStringRef obs_to_vt_profile(const char *profile)
75
+static CFStringRef obs_to_vt_profile(CMVideoCodecType codec_type,
76
+                    const char *profile,
77
+                    enum video_format format)
78
 {
79
-   if (strcmp(profile, "baseline") == 0)
80
+   if (codec_type == kCMVideoCodecType_H264) {
81
+       if (strcmp(profile, "baseline") == 0)
82
+           return kVTProfileLevel_H264_Baseline_AutoLevel;
83
+       else if (strcmp(profile, "main") == 0)
84
+           return kVTProfileLevel_H264_Main_AutoLevel;
85
+       else if (strcmp(profile, "high") == 0)
86
+           return kVTProfileLevel_H264_High_AutoLevel;
87
+       else
88
+           return kVTProfileLevel_H264_Main_AutoLevel;
89
+#ifdef ENABLE_HEVC
90
+   } else if (codec_type == kCMVideoCodecType_HEVC) {
91
+       if (strcmp(profile, "main") == 0) {
92
+           if (format == VIDEO_FORMAT_P010) {
93
+               VT_LOG(LOG_WARNING, "Forcing main10 for P010");
94
+               return kVTProfileLevel_HEVC_Main10_AutoLevel;
95
+           } else {
96
+               return kVTProfileLevel_HEVC_Main_AutoLevel;
97
+           }
98
+       }
99
+       if (strcmp(profile, "main10") == 0)
100
+           return kVTProfileLevel_HEVC_Main10_AutoLevel;
101
+#if __MAC_OS_X_VERSION_MAX_ALLOWED >= 120300 // macOS 12.3
102
+       if (__builtin_available(macOS 12.3, *)) {
103
+           if (strcmp(profile, "main42210") == 0)
104
+               return kVTProfileLevel_HEVC_Main42210_AutoLevel;
105
+       }
106
+#endif // macOS 12.3
107
+       return kVTProfileLevel_HEVC_Main_AutoLevel;
108
+#endif // ENABLE_HEVC
109
+   } else {
110
        return kVTProfileLevel_H264_Baseline_AutoLevel;
111
-   else if (strcmp(profile, "main") == 0)
112
-       return kVTProfileLevel_H264_Main_AutoLevel;
113
-   else if (strcmp(profile, "high") == 0)
114
-       return kVTProfileLevel_H264_High_AutoLevel;
115
-   else
116
-       return kVTProfileLevel_H264_Main_AutoLevel;
117
+   }
118
 }
119
 
120
 static CFStringRef obs_to_vt_colorspace(enum video_colorspace cs)
121
 {
122
-   if (cs == VIDEO_CS_709)
123
-       return kCVImageBufferYCbCrMatrix_ITU_R_709_2;
124
-   else if (cs == VIDEO_CS_601)
125
+   switch (cs) {
126
+   case VIDEO_CS_601:
127
        return kCVImageBufferYCbCrMatrix_ITU_R_601_4;
128
+   case VIDEO_CS_2100_PQ:
129
+   case VIDEO_CS_2100_HLG:
130
+       return kCVImageBufferYCbCrMatrix_ITU_R_2020;
131
+   default:
132
+       return kCVImageBufferYCbCrMatrix_ITU_R_709_2;
133
+   }
134
+}
135
 
136
-   return NULL;
137
+static CFStringRef obs_to_vt_primaries(enum video_colorspace cs)
138
+{
139
+   switch (cs) {
140
+   case VIDEO_CS_601:
141
+       return kCVImageBufferColorPrimaries_SMPTE_C;
142
+   case VIDEO_CS_2100_PQ:
143
+   case VIDEO_CS_2100_HLG:
144
+       return kCVImageBufferColorPrimaries_ITU_R_2020;
145
+   default:
146
+       return kCVImageBufferColorPrimaries_ITU_R_709_2;
147
+   }
148
+}
149
+
150
+static CFStringRef obs_to_vt_transfer(enum video_colorspace cs)
151
+{
152
+   switch (cs) {
153
+   case VIDEO_CS_SRGB:
154
+       return kCVImageBufferTransferFunction_sRGB;
155
+   case VIDEO_CS_2100_PQ:
156
+       return kCVImageBufferTransferFunction_SMPTE_ST_2084_PQ;
157
+   case VIDEO_CS_2100_HLG:
158
+       return kCVImageBufferTransferFunction_ITU_R_2100_HLG;
159
+   default:
160
+       return kCVImageBufferTransferFunction_ITU_R_709_2;
161
+   }
162
+}
163
+
164
+/* Adapted from Chromium GenerateMasteringDisplayColorVolume */
165
+static CFDataRef obs_to_vt_masteringdisplay(uint32_t hdr_nominal_peak_level)
166
+{
167
+   struct mastering_display_colour_volume {
168
+       uint16_t display_primaries32;
169
+       uint16_t white_point2;
170
+       uint32_t max_display_mastering_luminance;
171
+       uint32_t min_display_mastering_luminance;
172
+   };
173
+   static_assert(sizeof(struct mastering_display_colour_volume) == 24,
174
+             "May need to adjust struct packing");
175
+
176
+   struct mastering_display_colour_volume mdcv;
177
+   mdcv.display_primaries00 = __builtin_bswap16(13250);
178
+   mdcv.display_primaries01 = __builtin_bswap16(34500);
179
+   mdcv.display_primaries10 = __builtin_bswap16(7500);
180
+   mdcv.display_primaries11 = __builtin_bswap16(3000);
181
+   mdcv.display_primaries20 = __builtin_bswap16(34000);
182
+   mdcv.display_primaries21 = __builtin_bswap16(16000);
183
+   mdcv.white_point0 = __builtin_bswap16(15635);
184
+   mdcv.white_point1 = __builtin_bswap16(16450);
185
+   mdcv.max_display_mastering_luminance =
186
+       __builtin_bswap32(hdr_nominal_peak_level * 10000);
187
+   mdcv.min_display_mastering_luminance = 0;
188
+
189
+   UInt8 bytessizeof(struct mastering_display_colour_volume);
190
+   memcpy(bytes, &mdcv, sizeof(bytes));
191
+   return CFDataCreate(NULL, bytes, sizeof(bytes));
192
+}
193
+
194
+/* Adapted from Chromium GenerateContentLightLevelInfo */
195
+static CFDataRef
196
+obs_to_vt_contentlightlevelinfo(uint16_t hdr_nominal_peak_level)
197
+{
198
+   struct content_light_level_info {
199
+       uint16_t max_content_light_level;
200
+       uint16_t max_pic_average_light_level;
201
obs-studio-28.1.2.tar.xz/plugins/mac-virtualcam/src/dal-plugin/OBSDALMachClient.mm -> obs-studio-29.0.0.tar.xz/plugins/mac-virtualcam/src/dal-plugin/OBSDALMachClient.mm Changed
32
 
1
@@ -10,7 +10,6 @@
2
 #import "Logging.h"
3
 
4
 @interface OBSDALMachClient () <NSPortDelegate> {
5
-   uint32_t _seed;
6
    NSPort *_receivePort;
7
 }
8
 @end
9
@@ -123,11 +122,20 @@
10
                return;
11
            }
12
 
13
-           IOSurfaceLock(surface, 0, &_seed);
14
+           /*
15
+            * IOSurfaceLocks are only necessary on non Apple Silicon devices, as those have
16
+            * unified memory. On Intel machines, the lock ensures that the IOSurface is copied back
17
+            * from GPU memory to CPU memory so we can process the pixel buffer.
18
+            */
19
+#ifndef __aarch64__
20
+           IOSurfaceLock(surface, kIOSurfaceLockReadOnly, NULL);
21
+#endif
22
            CVPixelBufferRef frame;
23
            CVPixelBufferCreateWithIOSurface(kCFAllocatorDefault,
24
                             surface, NULL, &frame);
25
-           IOSurfaceUnlock(surface, 0, &_seed);
26
+#ifndef __aarch64__
27
+           IOSurfaceUnlock(surface, kIOSurfaceLockReadOnly, NULL);
28
+#endif
29
            CFRelease(surface);
30
 
31
            uint64_t timestamp;
32
obs-studio-28.1.2.tar.xz/plugins/mac-virtualcam/src/obs-plugin/OBSDALMachServer.mm -> obs-studio-29.0.0.tar.xz/plugins/mac-virtualcam/src/obs-plugin/OBSDALMachServer.mm Changed
25
 
1
@@ -15,7 +15,6 @@
2
 @property NSPort *port;
3
 @property NSMutableSet *clientPorts;
4
 @property NSRunLoop *runLoop;
5
-@property uint32_t seed;
6
 @end
7
 
8
 @implementation OBSDALMachServer
9
@@ -156,7 +155,6 @@
10
            return;
11
        }
12
 
13
-       IOSurfaceLock(surface, 0, &_seed);
14
        mach_port_t framePort = IOSurfaceCreateMachPort(surface);
15
 
16
        if (!framePort) {
17
@@ -176,7 +174,6 @@
18
                     ;
19
 
20
        mach_port_deallocate(mach_task_self(), framePort);
21
-       IOSurfaceUnlock(surface, 0, &_seed);
22
    }
23
 }
24
 
25
obs-studio-28.1.2.tar.xz/plugins/mac-virtualcam/src/obs-plugin/plugin-main.mm -> obs-studio-29.0.0.tar.xz/plugins/mac-virtualcam/src/obs-plugin/plugin-main.mm Changed
14
 
1
@@ -84,11 +84,9 @@
2
                    copyPluginCmd;
3
 
4
            NSDictionary *errorDict;
5
-           NSAppleEventDescriptor *returnDescriptor = NULL;
6
            NSAppleScript *scriptObject =
7
                NSAppleScript alloc initWithSource:copyCmd;
8
-           returnDescriptor =
9
-               scriptObject executeAndReturnError:&errorDict;
10
+           scriptObject executeAndReturnError:&errorDict;
11
            if (errorDict != nil) {
12
                const char *errorMessage = errorDict
13
                    objectForKey:@"NSAppleScriptErrorMessage"
14
obs-studio-28.1.2.tar.xz/plugins/obs-browser/browser-version.h -> obs-studio-29.0.0.tar.xz/plugins/obs-browser/browser-version.h Changed
12
 
1
@@ -1,8 +1,8 @@
2
 #pragma once
3
 
4
 #define OBS_BROWSER_VERSION_MAJOR 2
5
-#define OBS_BROWSER_VERSION_MINOR 18
6
-#define OBS_BROWSER_VERSION_PATCH 7
7
+#define OBS_BROWSER_VERSION_MINOR 19
8
+#define OBS_BROWSER_VERSION_PATCH 0
9
 
10
 #ifndef MAKE_SEMANTIC_VERSION
11
 #define MAKE_SEMANTIC_VERSION(major, minor, patch) \
12
obs-studio-28.1.2.tar.xz/plugins/obs-browser/data/locale/en-US.ini -> obs-studio-29.0.0.tar.xz/plugins/obs-browser/data/locale/en-US.ini Changed
10
 
1
@@ -11,6 +11,8 @@
2
 BrowserSource="Browser"
3
 CustomFrameRate="Use custom frame rate"
4
 RerouteAudio="Control audio via OBS"
5
+Inspect="Inspect"
6
+DevTools="Inspect Browser Dock '%1'"
7
 WebpageControlLevel="Page permissions"
8
 WebpageControlLevel.Level.None="No access to OBS"
9
 WebpageControlLevel.Level.ReadObs="Read access to OBS status information"
10
obs-studio-28.1.2.tar.xz/plugins/obs-browser/obs-browser-plugin.cpp -> obs-studio-29.0.0.tar.xz/plugins/obs-browser/obs-browser-plugin.cpp Changed
25
 
1
@@ -306,8 +306,14 @@
2
    CefMainArgs args(cmdline_args.argc, cmdline_args.argv);
3
 #endif
4
 
5
+   BPtr<char> conf_path = obs_module_config_path("");
6
+   os_mkdir(conf_path);
7
+
8
    CefSettings settings;
9
    settings.log_severity = LOGSEVERITY_DISABLE;
10
+   BPtr<char> log_path = obs_module_config_path("debug.log");
11
+   BPtr<char> log_path_abs = os_get_abs_path_ptr(log_path);
12
+   CefString(&settings.log_file) = log_path_abs;
13
    settings.windowless_rendering_enabled = true;
14
    settings.no_sandbox = true;
15
 
16
@@ -358,8 +364,6 @@
17
        accepted_languages = "en-US,en";
18
    }
19
 
20
-   BPtr<char> conf_path = obs_module_config_path("");
21
-   os_mkdir(conf_path);
22
    BPtr<char> conf_path_abs = os_get_abs_path_ptr(conf_path);
23
    CefString(&settings.locale) = obs_get_locale();
24
    CefString(&settings.accept_language_list) = accepted_languages;
25
obs-studio-28.1.2.tar.xz/plugins/obs-browser/panel/browser-panel-client.cpp -> obs-studio-29.0.0.tar.xz/plugins/obs-browser/panel/browser-panel-client.cpp Changed
130
 
1
@@ -19,6 +19,9 @@
2
 #include <X11/Xlib.h>
3
 #endif
4
 
5
+#define MENU_ITEM_DEVTOOLS MENU_ID_CUSTOM_FIRST
6
+#define MENU_ITEM_MUTE MENU_ID_CUSTOM_FIRST + 1
7
+
8
 /* CefClient */
9
 CefRefPtr<CefLoadHandler> QCefBrowserClient::GetLoadHandler()
10
 {
11
@@ -70,6 +73,9 @@
12
        QMetaObject::invokeMethod(widget, "titleChanged",
13
                      Q_ARG(QString, qt_title));
14
    } else { /* handle popup title */
15
+       if (title.compare("DevTools") == 0)
16
+           return;
17
+
18
        CefWindowHandle handl = browser->GetHost()->GetWindowHandle();
19
 #ifdef _WIN32
20
        std::wstring str_title = title;
21
@@ -228,7 +234,7 @@
22
    }
23
 }
24
 
25
-void QCefBrowserClient::OnBeforeContextMenu(CefRefPtr<CefBrowser>,
26
+void QCefBrowserClient::OnBeforeContextMenu(CefRefPtr<CefBrowser> browser,
27
                        CefRefPtr<CefFrame>,
28
                        CefRefPtr<CefContextMenuParams>,
29
                        CefRefPtr<CefMenuModel> model)
30
@@ -243,6 +249,15 @@
31
    if (model->IsVisible(MENU_ID_PRINT)) {
32
        model->Remove(MENU_ID_PRINT);
33
    }
34
+   if (model->IsVisible(MENU_ID_VIEW_SOURCE)) {
35
+       model->Remove(MENU_ID_VIEW_SOURCE);
36
+   }
37
+   model->AddSeparator();
38
+   model->InsertItemAt(model->GetCount(), MENU_ITEM_DEVTOOLS,
39
+               obs_module_text("Inspect"));
40
+   model->InsertCheckItemAt(model->GetCount(), MENU_ITEM_MUTE,
41
+                QObject::tr("Mute").toUtf8().constData());
42
+   model->SetChecked(MENU_ITEM_MUTE, browser->GetHost()->IsAudioMuted());
43
 }
44
 
45
 #if defined(_WIN32)
46
@@ -251,12 +266,13 @@
47
    CefRefPtr<CefContextMenuParams>, CefRefPtr<CefMenuModel> model,
48
    CefRefPtr<CefRunContextMenuCallback> callback)
49
 {
50
-   std::vector<std::tuple<std::string, int, bool, int>> menu_items;
51
+   std::vector<std::tuple<std::string, int, bool, int, bool>> menu_items;
52
    menu_items.reserve(model->GetCount());
53
    for (int i = 0; i < model->GetCount(); i++) {
54
        menu_items.push_back(
55
            {model->GetLabelAt(i), model->GetCommandIdAt(i),
56
-            model->IsEnabledAt(i), model->GetTypeAt(i)});
57
+            model->IsEnabledAt(i), model->GetTypeAt(i),
58
+            model->IsCheckedAt(i)});
59
    }
60
 
61
    QMetaObject::invokeMethod(
62
@@ -267,16 +283,22 @@
63
            int command_id;
64
            bool enabled;
65
            int type_id;
66
+           bool check;
67
 
68
-           for (const std::tuple<std::string, int, bool, int>
69
+           for (const std::tuple<std::string, int, bool, int, bool>
70
                     &menu_item : menu_items) {
71
-               std::tie(name, command_id, enabled, type_id) =
72
-                   menu_item;
73
+               std::tie(name, command_id, enabled, type_id,
74
+                    check) = menu_item;
75
                switch (type_id) {
76
+               case MENUITEMTYPE_CHECK:
77
                case MENUITEMTYPE_COMMAND: {
78
                    QAction *item =
79
                        new QAction(name.c_str());
80
                    item->setEnabled(enabled);
81
+                   if (type_id == MENUITEMTYPE_CHECK) {
82
+                       item->setCheckable(true);
83
+                       item->setChecked(check);
84
+                   }
85
                    item->setProperty("cmd_id", command_id);
86
                    contextMenu.addAction(item);
87
                } break;
88
@@ -299,6 +321,41 @@
89
 }
90
 #endif
91
 
92
+bool QCefBrowserClient::OnContextMenuCommand(
93
+   CefRefPtr<CefBrowser> browser, CefRefPtr<CefFrame>,
94
+   CefRefPtr<CefContextMenuParams> params, int command_id,
95
+   CefContextMenuHandler::EventFlags)
96
+{
97
+   if (command_id < MENU_ID_CUSTOM_FIRST)
98
+       return false;
99
+   CefRefPtr<CefBrowserHost> host = browser->GetHost();
100
+   CefWindowInfo windowInfo;
101
+   QPoint pos;
102
+   QString title;
103
+   switch (command_id) {
104
+   case MENU_ITEM_DEVTOOLS:
105
+#if defined(_WIN32)
106
+       title = QString(obs_module_text("DevTools"))
107
+               .arg(widget->parentWidget()->windowTitle());
108
+       windowInfo.SetAsPopup(host->GetWindowHandle(),
109
+                     title.toUtf8().constData());
110
+#endif
111
+       pos = widget->mapToGlobal(QPoint(0, 0));
112
+       windowInfo.bounds.x = pos.x();
113
+       windowInfo.bounds.y = pos.y() + 30;
114
+       windowInfo.bounds.width = 900;
115
+       windowInfo.bounds.height = 700;
116
+       host->ShowDevTools(
117
+           windowInfo, host->GetClient(), CefBrowserSettings(),
118
+           {params.get()->GetXCoord(), params.get()->GetYCoord()});
119
+       return true;
120
+   case MENU_ITEM_MUTE:
121
+       host->SetAudioMuted(!host->IsAudioMuted());
122
+       return true;
123
+   }
124
+   return false;
125
+}
126
+
127
 void QCefBrowserClient::OnLoadEnd(CefRefPtr<CefBrowser>,
128
                  CefRefPtr<CefFrame> frame, int)
129
 {
130
obs-studio-28.1.2.tar.xz/plugins/obs-browser/panel/browser-panel-client.hpp -> obs-studio-29.0.0.tar.xz/plugins/obs-browser/panel/browser-panel-client.hpp Changed
13
 
1
@@ -89,6 +89,11 @@
2
               CefRefPtr<CefRunContextMenuCallback> callback) override;
3
 #endif
4
 
5
+   virtual bool OnContextMenuCommand(
6
+       CefRefPtr<CefBrowser> browser, CefRefPtr<CefFrame> frame,
7
+       CefRefPtr<CefContextMenuParams> params, int command_id,
8
+       CefContextMenuHandler::EventFlags event_flags) override;
9
+
10
    /* CefLoadHandler */
11
    virtual void OnLoadEnd(CefRefPtr<CefBrowser> browser,
12
                   CefRefPtr<CefFrame> frame,
13
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/CMakeLists.txt -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/CMakeLists.txt Changed
15
 
1
@@ -119,9 +119,11 @@
2
             obs-ffmpeg.rc)
3
 
4
 elseif(OS_POSIX AND NOT OS_MACOS)
5
+  find_package(Libva REQUIRED)
6
   find_package(Libpci REQUIRED)
7
-  target_sources(obs-ffmpeg PRIVATE obs-ffmpeg-vaapi.c)
8
-  target_link_libraries(obs-ffmpeg PRIVATE LIBPCI::LIBPCI)
9
+  target_sources(obs-ffmpeg PRIVATE obs-ffmpeg-vaapi.c vaapi-utils.c
10
+                                    vaapi-utils.h)
11
+  target_link_libraries(obs-ffmpeg PRIVATE Libva::va LIBPCI::LIBPCI)
12
 endif()
13
 
14
 setup_plugin_target(obs-ffmpeg)
15
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/data/locale/en-US.ini -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/data/locale/en-US.ini Changed
18
 
1
@@ -14,7 +14,7 @@
2
 Level="Level"
3
 
4
 AMFOpts="AMF/FFmpeg Options"
5
-AMFOpts.ToolTip="Use to specify custom AMF or FFmpeg options. For example, \"level=5.2 profile=main BPicturesPattern=3\""
6
+AMFOpts.ToolTip="Use to specify custom AMF or FFmpeg options. For example, \"level=5.2 profile=main\". Check the AMF encoder docs for more details."
7
 
8
 BFrames="Max B-frames"
9
 
10
@@ -48,6 +48,7 @@
11
 AMF.Preset.speed="Speed"
12
 AMF.Preset.balanced="Balanced"
13
 AMF.Preset.quality="Quality"
14
+AMF.Preset.highQuality="High Quality"
15
 
16
 FFmpegSource="Media Source"
17
 LocalFile="Local File"
18
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/external/AMF/include/components/Component.h -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/external/AMF/include/components/Component.h Changed
9
 
1
@@ -414,6 +414,7 @@
2
     AMF_STREAM_CODEC_ID_VP9         = 11, // AMFVideoDecoderHW_VP9
3
     AMF_STREAM_CODEC_ID_VP9_10BIT   = 12, // AMFVideoDecoderHW_VP9_10BIT   
4
     AMF_STREAM_CODEC_ID_AV1         = 13, // AMFVideoDecoderHW_AV1 
5
+    AMF_STREAM_CODEC_ID_AV1_12BIT   = 14, // AMFVideoDecoderHW_AV1_12BIT
6
 } AMF_STREAM_CODEC_ID_ENUM;
7
 
8
 // common stream properties
9
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/external/AMF/include/components/CursorCapture.h -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/external/AMF/include/components/CursorCapture.h Changed
9
 
1
@@ -43,6 +43,7 @@
2
     class AMFCursorCapture : public AMFInterface
3
     {
4
     public:
5
+        AMF_DECLARE_IID(0x166efa1a, 0x19b8, 0x42f2, 0x86, 0x0f, 0x56, 0x69, 0xca, 0x7a, 0x85, 0x4c)
6
         virtual AMF_RESULT AMF_STD_CALL AcquireCursor(amf::AMFSurface** pSurface) = 0;
7
         virtual AMF_RESULT AMF_STD_CALL Reset() = 0;
8
     };
9
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/external/AMF/include/components/FFMPEGFileMuxer.h -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/external/AMF/include/components/FFMPEGFileMuxer.h Changed
8
 
1
@@ -49,5 +49,6 @@
2
 #define FFMPEG_MUXER_ENABLE_AUDIO             L"EnableAudio"              // bool (default = false)
3
 #define FFMPEG_MUXER_CURRENT_TIME_INTERFACE   L"CurrentTimeInterface"
4
 #define FFMPEG_MUXER_VIDEO_ROTATION           L"VideoRotation"            // amf_int64 (0, 90, 180, 270, default = 0)
5
+#define FFMPEG_MUXER_USAGE_IS_TRIM            L"UsageIsTrim"              // bool (default = false)
6
 
7
 #endif //#ifndef AMF_FileMuxerFFMPEG_h
8
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/external/AMF/include/components/HQScaler.h -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/external/AMF/include/components/HQScaler.h Changed
17
 
1
@@ -44,6 +44,8 @@
2
     AMF_HQ_SCALER_ALGORITHM_BILINEAR = 0,
3
     AMF_HQ_SCALER_ALGORITHM_BICUBIC = 1,
4
     AMF_HQ_SCALER_ALGORITHM_FSR = 2,
5
+    AMF_HQ_SCALER_ALGORITHM_POINT = 3,
6
+    AMF_HQ_SCALER_ALGORITHM_FSR1_1 = 4,
7
 
8
 };
9
 
10
@@ -60,5 +62,6 @@
11
 #define AMF_HQ_SCALER_FILL_COLOR         L"FillColor"               // AMFColor 
12
 #define AMF_HQ_SCALER_FROM_SRGB          L"FromSRGB"                   //  bool (default=true) Convert to SRGB. 
13
 
14
+#define AMF_HQ_SCALER_SHARPNESS          L"HQScalerSharpness"   // Float in the range of 0.0, 2.0
15
 
16
 #endif //#ifndef AMFHQScaler_h
17
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/external/AMF/include/components/PreAnalysis.h -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/external/AMF/include/components/PreAnalysis.h Changed
88
 
1
@@ -1,4 +1,4 @@
2
-// 
3
+//
4
 // Notice Regarding Standards.  AMD does not provide a license or sublicense to
5
 // any Intellectual Property Rights relating to any standards, including but not
6
 // limited to any audio and/or video codec technologies such as MPEG-2, MPEG-4;
7
@@ -6,9 +6,9 @@
8
 // (collectively, the "Media Technologies"). For clarity, you will pay any
9
 // royalties due for such third party technologies, which may include the Media
10
 // Technologies that are owed as a result of AMD providing the Software to you.
11
-// 
12
-// MIT license 
13
-// 
14
+//
15
+// MIT license
16
+//
17
 // Copyright (c) 2019 Advanced Micro Devices, Inc. All rights reserved.
18
 //
19
 // Permission is hereby granted, free of charge, to any person obtaining a copy
20
@@ -48,7 +48,7 @@
21
 
22
 
23
 enum  AMF_PA_STATIC_SCENE_DETECTION_SENSITIVITY_ENUM
24
-{  
25
+{
26
    AMF_PA_STATIC_SCENE_DETECTION_SENSITIVITY_LOW    = 0,
27
    AMF_PA_STATIC_SCENE_DETECTION_SENSITIVITY_MEDIUM = 1,
28
    AMF_PA_STATIC_SCENE_DETECTION_SENSITIVITY_HIGH   = 2
29
@@ -69,6 +69,26 @@
30
     AMF_PA_CAQ_STRENGTH_HIGH   = 2
31
 };
32
 
33
+// Perceptual adaptive quantization mode
34
+enum AMF_PA_PAQ_MODE_ENUM
35
+{
36
+    AMF_PA_PAQ_MODE_NONE = 0,
37
+    AMF_PA_PAQ_MODE_CAQ = 1
38
+};
39
+
40
+// Temporal adaptive quantization mode
41
+enum  AMF_PA_TAQ_MODE_ENUM
42
+{
43
+    AMF_PA_TAQ_MODE_NONE = 0,
44
+    AMF_PA_TAQ_MODE_1 = 1,
45
+    AMF_PA_TAQ_MODE_2 = 2
46
+};
47
+
48
+enum AMF_PA_HIGH_MOTION_QUALITY_BOOST_MODE_ENUM
49
+{
50
+    AMF_PA_HIGH_MOTION_QUALITY_BOOST_MODE_NONE = 0, //default
51
+    AMF_PA_HIGH_MOTION_QUALITY_BOOST_MODE_AUTO = 1
52
+};
53
 
54
 
55
 // PA object properties
56
@@ -77,17 +97,19 @@
57
 
58
 #define AMF_PA_SCENE_CHANGE_DETECTION_ENABLE        L"PASceneChangeDetectionEnable"         // bool       (default : True)                                          - Enable Scene Change Detection GPU algorithm
59
 #define AMF_PA_SCENE_CHANGE_DETECTION_SENSITIVITY   L"PASceneChangeDetectionSensitivity"   // AMF_PA_SCENE_CHANGE_DETECTION_SENSITIVITY_ENUM (default : Medium)    - Scene Change Detection Sensitivity
60
-#define AMF_PA_STATIC_SCENE_DETECTION_ENABLE        L"PAStaticSceneDetectionEnable"         // bool       (default : True)                                          - Enable Skip Detection GPU algorithm
61
+#define AMF_PA_STATIC_SCENE_DETECTION_ENABLE        L"PAStaticSceneDetectionEnable"         // bool       (default : False)                                         - Enable Skip Detection GPU algorithm
62
 #define AMF_PA_STATIC_SCENE_DETECTION_SENSITIVITY   L"PAStaticSceneDetectionSensitivity"   // AMF_PA_STATIC_SCENE_DETECTION_SENSITIVITY_ENUM (default : High)      - Allowable absolute difference between pixels (sample counts)
63
 #define AMF_PA_FRAME_SAD_ENABLE                     L"PAFrameSadEnable"                        // bool       (default : True)                                          - Enable Frame SAD algorithm
64
 #define AMF_PA_ACTIVITY_TYPE                        L"PAActivityType"                       // AMF_PA_ACTIVITY_TYPE_ENUM (default : Calculate on Y)                 - Block activity calculation mode
65
-#define AMF_PA_LTR_ENABLE                           L"PALongTermReferenceEnable"            // bool       (default : True)                                          - Enable Automatic Long Term Reference frame management
66
-
67
-
68
+#define AMF_PA_LTR_ENABLE                           L"PALongTermReferenceEnable"            // bool       (default : False)                                         - Enable Automatic Long Term Reference frame management
69
+#define AMF_PA_LOOKAHEAD_BUFFER_DEPTH               L"PALookAheadBufferDepth"               // amf_uint64 (default : 0)           Values: 0, MAX_LOOKAHEAD_DEPTH  - PA lookahead buffer size
70
+#define AMF_PA_PAQ_MODE                             L"PAPerceptualAQMode"                   // AMF_PA_PAQ_MODE_ENUM     (default : AMF_PA_PAQ_MODE_NONE)            - Perceptual AQ mode
71
+#define AMF_PA_TAQ_MODE                             L"PATemporalAQMode"                     // AMF_PA_TAQ_MODE_ENUM (default: AMF_PA_TAQ_MODE_NONE)                 - Temporal AQ mode
72
+#define AMF_PA_HIGH_MOTION_QUALITY_BOOST_MODE       L"PAHighMotionQualityBoostMode"         // AMF_PA_HIGH_MOTION_QUALITY_BOOST_MODE_ENUM (default: None)           - High motion quality boost mode
73
 
74
 ///////////////////////////////////////////
75
-// the following properties are available 
76
-// only through the Encoder - trying to 
77
+// the following properties are available
78
+// only through the Encoder - trying to
79
 // access/set them when PA is standalone
80
 // will fail
81
 
82
@@ -107,4 +129,5 @@
83
 #define AMF_PA_SCENE_CHANGE_DETECT                  L"PASceneChangeDetect"                  // bool                                                                 - True/False - available if AMF_PA_SCENE_CHANGE_DETECTION_ENABLE was set to True
84
 #define AMF_PA_STATIC_SCENE_DETECT                  L"PAStaticSceneDetect"                  // bool                                                                 - True/False - available if AMF_PA_STATIC_SCENE_DETECTION_ENABLE was set to True
85
 
86
+
87
 #endif //#ifndef AMFPreAnalysis_h
88
obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/external/AMF/include/components/VQEnhancer.h Added
49
 
1
@@ -0,0 +1,47 @@
2
+//
3
+// Notice Regarding Standards.  AMD does not provide a license or sublicense to
4
+// any Intellectual Property Rights relating to any standards, including but not
5
+// limited to any audio and/or video codec technologies such as MPEG-2, MPEG-4;
6
+// AVC/H.264; HEVC/H.265; AAC decode/FFMPEG; AAC encode/FFMPEG; VC-1; and MP3
7
+// (collectively, the "Media Technologies"). For clarity, you will pay any
8
+// royalties due for such third party technologies, which may include the Media
9
+// Technologies that are owed as a result of AMD providing the Software to you.
10
+//
11
+// MIT license
12
+//
13
+// Copyright (c) 2021 Advanced Micro Devices, Inc. All rights reserved.
14
+//
15
+// Permission is hereby granted, free of charge, to any person obtaining a copy
16
+// of this software and associated documentation files (the "Software"), to deal
17
+// in the Software without restriction, including without limitation the rights
18
+// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
19
+// copies of the Software, and to permit persons to whom the Software is
20
+// furnished to do so, subject to the following conditions:
21
+//
22
+// The above copyright notice and this permission notice shall be included in
23
+// all copies or substantial portions of the Software.
24
+//
25
+// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
26
+// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
27
+// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL THE
28
+// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
29
+// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
30
+// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
31
+// THE SOFTWARE.
32
+//
33
+
34
+#ifndef AMFVQEnhancer_h
35
+#define AMFVQEnhancer_h
36
+
37
+#pragma once
38
+
39
+#define VE_FCR_DEFAULT_ATTENUATION 0.1
40
+
41
+#define AMFVQEnhancer L"AMFVQEnhancer"
42
+
43
+#define AMF_VIDEO_ENHANCER_ENGINE_TYPE       L"AMF_VIDEI_ENHANCER_ENGINE_TYPE"        // AMF_MEMORY_TYPE (DX11, DX12, OPENCL, VULKAN default : DX11)"                    - determines how the object is initialized and what kernels to use
44
+#define AMF_VIDEO_ENHANCER_OUTPUT_SIZE       L"AMF_VIDEO_ENHANCER_OUTPUT_SIZE"        // AMFSize
45
+#define AMF_VE_FCR_ATTENUATION               L"AMF_VE_FCR_ATTENUATION"                // Float in the range of 0.02, 0.4, default : 0.1
46
+#define AMF_VE_FCR_RADIUS                    L"AMF_VE_FCR_RADIUS"                     // int  in the range of 1, 4
47
+
48
+#endif //#ifndef AMFVQEnhancer_h
49
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/external/AMF/include/components/VideoDecoderUVD.h -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/external/AMF/include/components/VideoDecoderUVD.h Changed
23
 
1
@@ -53,6 +53,7 @@
2
 #define AMFVideoDecoderHW_VP9                        L"AMFVideoDecoderHW_VP9"
3
 #define AMFVideoDecoderHW_VP9_10BIT                  L"AMFVideoDecoderHW_VP9_10BIT"
4
 #define AMFVideoDecoderHW_AV1                        L"AMFVideoDecoderHW_AV1"
5
+#define AMFVideoDecoderHW_AV1_12BIT                  L"AMFVideoDecoderHW_AV1_12BIT"
6
 
7
 enum AMF_VIDEO_DECODER_MODE_ENUM
8
 {
9
@@ -108,6 +109,7 @@
10
 #define AMF_VIDEO_DECODER_OUTPUT_COLOR_PRIMARIES                L"OutputColorPrimaries"       // amf_int64(AMF_COLOR_PRIMARIES_ENUM); default = AMF_COLOR_PRIMARIES_UNDEFINED, ISO/IEC 23001-8_2013   7.1 See ColorSpace.h for enum 
11
 #define AMF_VIDEO_DECODER_OUTPUT_HDR_METADATA                   L"OutHDRMetadata"           // AMFBuffer containing AMFHDRMetadata; default NULL
12
 
13
+#define AMF_VIDEO_DECODER_LOW_LATENCY                           L"LowLatencyDecode"         // amf_bool; default = false; true = low latency decode, false = regular decode
14
 
15
 #if defined(__ANDROID__)
16
 #define AMF_VIDEO_DECODER_NATIVEWINDOW                  L"AndroidNativeWindow"  // amf_int64; default = 0; pointer to native window
17
@@ -118,5 +120,4 @@
18
 #define AMF_VIDEO_DECODER_NATIVEWINDOW                  L"AppleNativeWindow"  // amf_int64; default = 0; pointer to native window
19
 #endif //__APPLE__
20
 
21
-
22
 #endif //#ifndef AMF_VideoDecoderUVD_h
23
obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/external/AMF/include/components/VideoEncoderAV1.h Added
201
 
1
@@ -0,0 +1,292 @@
2
+//
3
+// Copyright (c) 2021-2022 Advanced Micro Devices, Inc. All rights reserved.
4
+//
5
+// Permission is hereby granted, free of charge, to any person obtaining a copy
6
+// of this software and associated documentation files (the "Software"), to deal
7
+// in the Software without restriction, including without limitation the rights
8
+// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+// copies of the Software, and to permit persons to whom the Software is
10
+// furnished to do so, subject to the following conditions:
11
+//
12
+// The above copyright notice and this permission notice shall be included in
13
+// all copies or substantial portions of the Software.
14
+//
15
+// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL THE
18
+// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21
+// THE SOFTWARE.
22
+//
23
+
24
+//-------------------------------------------------------------------------------------------------
25
+//  VideoEncoderHW_AV1 interface declaration
26
+//-------------------------------------------------------------------------------------------------
27
+
28
+#ifndef AMF_VideoEncoderAV1_h
29
+#define AMF_VideoEncoderAV1_h
30
+#pragma once
31
+
32
+#include "Component.h"
33
+#include "ColorSpace.h"
34
+#include "PreAnalysis.h"
35
+
36
+#define AMFVideoEncoder_AV1 L"AMFVideoEncoderHW_AV1"
37
+
38
+enum AMF_VIDEO_ENCODER_AV1_ENCODING_LATENCY_MODE_ENUM
39
+{
40
+    AMF_VIDEO_ENCODER_AV1_ENCODING_LATENCY_MODE_NONE                    = 0,    // No encoding latency requirement. Encoder will balance encoding time and power consumption.
41
+    AMF_VIDEO_ENCODER_AV1_ENCODING_LATENCY_MODE_POWER_SAVING_REAL_TIME  = 1,    // Try the best to finish encoding a frame within 1/framerate sec. This mode may cause more power consumption
42
+    AMF_VIDEO_ENCODER_AV1_ENCODING_LATENCY_MODE_REAL_TIME               = 2,    // Try the best to finish encoding a frame within 1/(2 x framerate) sec. This mode will cause more power consumption than POWER_SAVING_REAL_TIME
43
+    AMF_VIDEO_ENCODER_AV1_ENCODING_LATENCY_MODE_LOWEST_LATENCY          = 3     // Encoding as fast as possible. This mode causes highest power consumption.
44
+};
45
+
46
+enum AMF_VIDEO_ENCODER_AV1_USAGE_ENUM
47
+{
48
+    AMF_VIDEO_ENCODER_AV1_USAGE_TRANSCODING     = 0,
49
+    AMF_VIDEO_ENCODER_AV1_USAGE_LOW_LATENCY     = 1
50
+};
51
+
52
+enum AMF_VIDEO_ENCODER_AV1_PROFILE_ENUM
53
+{
54
+    AMF_VIDEO_ENCODER_AV1_PROFILE_MAIN = 1
55
+};
56
+
57
+enum AMF_VIDEO_ENCODER_AV1_LEVEL_ENUM
58
+{
59
+    AMF_VIDEO_ENCODER_AV1_LEVEL_2_0 = 0,
60
+    AMF_VIDEO_ENCODER_AV1_LEVEL_2_1 = 1,
61
+    AMF_VIDEO_ENCODER_AV1_LEVEL_2_2 = 2,
62
+    AMF_VIDEO_ENCODER_AV1_LEVEL_2_3 = 3,
63
+    AMF_VIDEO_ENCODER_AV1_LEVEL_3_0 = 4,
64
+    AMF_VIDEO_ENCODER_AV1_LEVEL_3_1 = 5,
65
+    AMF_VIDEO_ENCODER_AV1_LEVEL_3_2 = 6,
66
+    AMF_VIDEO_ENCODER_AV1_LEVEL_3_3 = 7,
67
+    AMF_VIDEO_ENCODER_AV1_LEVEL_4_0 = 8,
68
+    AMF_VIDEO_ENCODER_AV1_LEVEL_4_1 = 9,
69
+    AMF_VIDEO_ENCODER_AV1_LEVEL_4_2 = 10,
70
+    AMF_VIDEO_ENCODER_AV1_LEVEL_4_3 = 11,
71
+    AMF_VIDEO_ENCODER_AV1_LEVEL_5_0 = 12,
72
+    AMF_VIDEO_ENCODER_AV1_LEVEL_5_1 = 13,
73
+    AMF_VIDEO_ENCODER_AV1_LEVEL_5_2 = 14,
74
+    AMF_VIDEO_ENCODER_AV1_LEVEL_5_3 = 15,
75
+    AMF_VIDEO_ENCODER_AV1_LEVEL_6_0 = 16,
76
+    AMF_VIDEO_ENCODER_AV1_LEVEL_6_1 = 17,
77
+    AMF_VIDEO_ENCODER_AV1_LEVEL_6_2 = 18,
78
+    AMF_VIDEO_ENCODER_AV1_LEVEL_6_3 = 19,
79
+    AMF_VIDEO_ENCODER_AV1_LEVEL_7_0 = 20,
80
+    AMF_VIDEO_ENCODER_AV1_LEVEL_7_1 = 21,
81
+    AMF_VIDEO_ENCODER_AV1_LEVEL_7_2 = 22,
82
+    AMF_VIDEO_ENCODER_AV1_LEVEL_7_3 = 23
83
+};
84
+
85
+enum AMF_VIDEO_ENCODER_AV1_RATE_CONTROL_METHOD_ENUM
86
+{
87
+    AMF_VIDEO_ENCODER_AV1_RATE_CONTROL_METHOD_UNKNOWN                   = -1,
88
+    AMF_VIDEO_ENCODER_AV1_RATE_CONTROL_METHOD_CONSTANT_QP               = 0,
89
+    AMF_VIDEO_ENCODER_AV1_RATE_CONTROL_METHOD_LATENCY_CONSTRAINED_VBR   = 1,
90
+    AMF_VIDEO_ENCODER_AV1_RATE_CONTROL_METHOD_PEAK_CONSTRAINED_VBR      = 2,
91
+    AMF_VIDEO_ENCODER_AV1_RATE_CONTROL_METHOD_CBR                       = 3,
92
+    AMF_VIDEO_ENCODER_AV1_RATE_CONTROL_METHOD_QUALITY_VBR               = 4,
93
+    AMF_VIDEO_ENCODER_AV1_RATE_CONTROL_METHOD_HIGH_QUALITY_VBR          = 5,
94
+    AMF_VIDEO_ENCODER_AV1_RATE_CONTROL_METHOD_HIGH_QUALITY_CBR          = 6
95
+};
96
+
97
+enum AMF_VIDEO_ENCODER_AV1_ALIGNMENT_MODE_ENUM
98
+{
99
+    AMF_VIDEO_ENCODER_AV1_ALIGNMENT_MODE_64X16_ONLY               = 1,
100
+    AMF_VIDEO_ENCODER_AV1_ALIGNMENT_MODE_64X16_1080P_CODED_1082   = 2,
101
+    AMF_VIDEO_ENCODER_AV1_ALIGNMENT_MODE_NO_RESTRICTIONS          = 3
102
+};
103
+
104
+enum AMF_VIDEO_ENCODER_AV1_FORCE_FRAME_TYPE_ENUM
105
+{
106
+    AMF_VIDEO_ENCODER_AV1_FORCE_FRAME_TYPE_NONE             = 0,
107
+    AMF_VIDEO_ENCODER_AV1_FORCE_FRAME_TYPE_KEY              = 1,
108
+    AMF_VIDEO_ENCODER_AV1_FORCE_FRAME_TYPE_INTRA_ONLY       = 2,
109
+    AMF_VIDEO_ENCODER_AV1_FORCE_FRAME_TYPE_SWITCH           = 3,
110
+    AMF_VIDEO_ENCODER_AV1_FORCE_FRAME_TYPE_SHOW_EXISTING    = 4
111
+};
112
+
113
+enum AMF_VIDEO_ENCODER_AV1_OUTPUT_FRAME_TYPE_ENUM
114
+{
115
+    AMF_VIDEO_ENCODER_AV1_OUTPUT_FRAME_TYPE_KEY             = 0,
116
+    AMF_VIDEO_ENCODER_AV1_OUTPUT_FRAME_TYPE_INTRA_ONLY      = 1,
117
+    AMF_VIDEO_ENCODER_AV1_OUTPUT_FRAME_TYPE_INTER           = 2,
118
+    AMF_VIDEO_ENCODER_AV1_OUTPUT_FRAME_TYPE_SWITCH          = 3,
119
+    AMF_VIDEO_ENCODER_AV1_OUTPUT_FRAME_TYPE_SHOW_EXISTING   = 4
120
+};
121
+
122
+enum AMF_VIDEO_ENCODER_AV1_QUALITY_PRESET_ENUM
123
+{
124
+    AMF_VIDEO_ENCODER_AV1_QUALITY_PRESET_HIGH_QUALITY   = 0,
125
+    AMF_VIDEO_ENCODER_AV1_QUALITY_PRESET_QUALITY        = 30,
126
+    AMF_VIDEO_ENCODER_AV1_QUALITY_PRESET_BALANCED       = 70,
127
+    AMF_VIDEO_ENCODER_AV1_QUALITY_PRESET_SPEED          = 100
128
+};
129
+
130
+enum AMF_VIDEO_ENCODER_AV1_HEADER_INSERTION_MODE_ENUM
131
+{
132
+    AMF_VIDEO_ENCODER_AV1_HEADER_INSERTION_MODE_NONE                = 0,
133
+    AMF_VIDEO_ENCODER_AV1_HEADER_INSERTION_MODE_GOP_ALIGNED         = 1,
134
+    AMF_VIDEO_ENCODER_AV1_HEADER_INSERTION_MODE_KEY_FRAME_ALIGNED   = 2
135
+};
136
+
137
+enum AMF_VIDEO_ENCODER_AV1_SWITCH_FRAME_INSERTION_MODE_ENUM
138
+{
139
+    AMF_VIDEO_ENCODER_AV1_SWITCH_FRAME_INSERTION_MODE_NONE              = 0,
140
+    AMF_VIDEO_ENCODER_AV1_SWITCH_FRAME_INSERTION_MODE_FIXED_INTERVAL    = 1
141
+};
142
+
143
+enum AMF_VIDEO_ENCODER_AV1_CDEF_MODE_ENUM
144
+{
145
+    AMF_VIDEO_ENCODER_AV1_CDEF_DISABLE          = 0,
146
+    AMF_VIDEO_ENCODER_AV1_CDEF_ENABLE_DEFAULT   = 1
147
+};
148
+
149
+enum AMF_VIDEO_ENCODER_AV1_CDF_FRAME_END_UPDATE_MODE_ENUM
150
+{
151
+    AMF_VIDEO_ENCODER_AV1_CDF_FRAME_END_UPDATE_MODE_DISABLE         = 0,
152
+    AMF_VIDEO_ENCODER_AV1_CDF_FRAME_END_UPDATE_MODE_ENABLE_DEFAULT  = 1
153
+};
154
+
155
+enum AMF_VIDEO_ENCODER_AV1_AQ_MODE_ENUM
156
+{
157
+    AMF_VIDEO_ENCODER_AV1_AQ_MODE_NONE = 0,
158
+    AMF_VIDEO_ENCODER_AV1_AQ_MODE_CAQ  = 1              // Content adaptive quantization mode
159
+};
160
+
161
+enum AMF_VIDEO_ENCODER_AV1_INTRA_REFRESH_MODE_ENUM
162
+{
163
+    AMF_VIDEO_ENCODER_AV1_INTRA_REFRESH_MODE__DISABLED = 0,
164
+    AMF_VIDEO_ENCODER_AV1_INTRA_REFRESH_MODE__GOP_ALIGNED = 1,
165
+    AMF_VIDEO_ENCODER_AV1_INTRA_REFRESH_MODE__CONTINUOUS = 2
166
+};
167
+
168
+
169
+// *** Static properties - can be set only before Init() ***
170
+
171
+// Encoder Engine Settings
172
+#define AMF_VIDEO_ENCODER_AV1_ENCODER_INSTANCE_INDEX                L"Av1EncoderInstanceIndex"          // amf_int64; default = 0; selected HW instance idx. The number of instances is queried by using AMF_VIDEO_ENCODER_AV1_CAP_NUM_OF_HW_INSTANCES
173
+#define AMF_VIDEO_ENCODER_AV1_ENCODING_LATENCY_MODE                 L"Av1EncodingLatencyMode"           // amf_int64(AMF_VIDEO_ENCODER_AV1_ENCODING_LATENCY_MODE_ENUM); default = depends on USAGE; The encoding latency mode.
174
+#define AMF_VIDEO_ENCODER_AV1_QUERY_TIMEOUT                         L"Av1QueryTimeout"                  // amf_int64; default = 0 (no wait); timeout for QueryOutput call in ms.
175
+
176
+// Usage Settings
177
+#define AMF_VIDEO_ENCODER_AV1_USAGE                                 L"Av1Usage"                         // amf_int64(AMF_VIDEO_ENCODER_AV1_USAGE_ENUM); default = N/A; Encoder usage. fully configures parameter set.
178
+
179
+// Session Configuration
180
+#define AMF_VIDEO_ENCODER_AV1_FRAMESIZE                             L"Av1FrameSize"                     // AMFSize; default = 0,0; Frame size
181
+#define AMF_VIDEO_ENCODER_AV1_COLOR_BIT_DEPTH                       L"Av1ColorBitDepth"                 // amf_int64(AMF_COLOR_BIT_DEPTH_ENUM); default = AMF_COLOR_BIT_DEPTH_8
182
+#define AMF_VIDEO_ENCODER_AV1_PROFILE                               L"Av1Profile"                       // amf_int64(AMF_VIDEO_ENCODER_AV1_PROFILE_ENUM) ; default = depends on USAGE; the codec profile of the coded bitstream
183
+#define AMF_VIDEO_ENCODER_AV1_LEVEL                                 L"Av1Level"                         // amf_int64 (AMF_VIDEO_ENCODER_AV1_LEVEL_ENUM); default = depends on USAGE; the codec level of the coded bitstream
184
+#define AMF_VIDEO_ENCODER_AV1_TILES_PER_FRAME                       L"Av1NumTilesPerFrame"              // amf_int64; default = 1; Number of tiles Per Frame. This is treated as suggestion. The actual number of tiles might be different due to compliance or encoder limitation.
185
+#define AMF_VIDEO_ENCODER_AV1_QUALITY_PRESET                        L"Av1QualityPreset"                 // amf_int64(AMF_VIDEO_ENCODER_AV1_QUALITY_PRESET_ENUM); default = depends on USAGE; Quality Preset
186
+
187
+// Codec Configuration
188
+#define AMF_VIDEO_ENCODER_AV1_SCREEN_CONTENT_TOOLS                  L"Av1ScreenContentTools"            // bool; default = depends on USAGE; If true, allow enabling screen content tools by AMF_VIDEO_ENCODER_AV1_PALETTE_MODE_ENABLE and AMF_VIDEO_ENCODER_AV1_FORCE_INTEGER_MV; if false, all screen content tools are disabled.
189
+#define AMF_VIDEO_ENCODER_AV1_ORDER_HINT                            L"Av1OrderHint"                     // bool; default = depends on USAGE; If true, code order hint; if false, don't code order hint
190
+#define AMF_VIDEO_ENCODER_AV1_FRAME_ID                              L"Av1FrameId"                       // bool; default = depends on USAGE; If true, code frame id; if false, don't code frame id
191
+#define AMF_VIDEO_ENCODER_AV1_TILE_GROUP_OBU                        L"Av1TileGroupObu"                  // bool; default = depends on USAGE; If true, code FrameHeaderObu + TileGroupObu and each TileGroupObu contains one tile; if false, code FrameObu.
192
+#define AMF_VIDEO_ENCODER_AV1_CDEF_MODE                             L"Av1CdefMode"                      // amd_int64(AMF_VIDEO_ENCODER_AV1_CDEF_MODE_ENUM); default = depends on USAGE; Cdef mode
193
+#define AMF_VIDEO_ENCODER_AV1_ERROR_RESILIENT_MODE                  L"Av1ErrorResilientMode"            // bool; default = depends on USAGE; If true, enable error resilient mode; if false, disable error resilient mode
194
+
195
+// Rate Control and Quality Enhancement
196
+#define AMF_VIDEO_ENCODER_AV1_RATE_CONTROL_METHOD                   L"Av1RateControlMethod"             // amf_int64(AMF_VIDEO_ENCODER_AV1_RATE_CONTROL_METHOD_ENUM); default = depends on USAGE; Rate Control Method
197
+#define AMF_VIDEO_ENCODER_AV1_QVBR_QUALITY_LEVEL                    L"Av1QvbrQualityLevel"              // amf_int64; default = 23; QVBR quality level; range = 1-51
198
+#define AMF_VIDEO_ENCODER_AV1_INITIAL_VBV_BUFFER_FULLNESS           L"Av1InitialVBVBufferFullness"      // amf_int64; default = depends on USAGE; Initial VBV Buffer Fullness 0=0% 64=100%
199
+
200
+// Alignment Mode Configuration
201
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/external/AMF/include/components/VideoEncoderHEVC.h -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/external/AMF/include/components/VideoEncoderHEVC.h Changed
52
 
1
@@ -80,7 +80,10 @@
2
     AMF_VIDEO_ENCODER_HEVC_RATE_CONTROL_METHOD_CONSTANT_QP = 0,
3
     AMF_VIDEO_ENCODER_HEVC_RATE_CONTROL_METHOD_LATENCY_CONSTRAINED_VBR,
4
     AMF_VIDEO_ENCODER_HEVC_RATE_CONTROL_METHOD_PEAK_CONSTRAINED_VBR,
5
-    AMF_VIDEO_ENCODER_HEVC_RATE_CONTROL_METHOD_CBR
6
+    AMF_VIDEO_ENCODER_HEVC_RATE_CONTROL_METHOD_CBR,
7
+    AMF_VIDEO_ENCODER_HEVC_RATE_CONTROL_METHOD_QUALITY_VBR,
8
+    AMF_VIDEO_ENCODER_HEVC_RATE_CONTROL_METHOD_HIGH_QUALITY_VBR,
9
+    AMF_VIDEO_ENCODER_HEVC_RATE_CONTROL_METHOD_HIGH_QUALITY_CBR
10
 };
11
 
12
 enum AMF_VIDEO_ENCODER_HEVC_PICTURE_TYPE_ENUM
13
@@ -145,9 +148,10 @@
14
 #define AMF_VIDEO_ENCODER_HEVC_QUALITY_PRESET                       L"HevcQualityPreset"            // amf_int64(AMF_VIDEO_ENCODER_HEVC_QUALITY_PRESET_ENUM); default = depends on USAGE; Quality Preset
15
 #define AMF_VIDEO_ENCODER_HEVC_EXTRADATA                            L"HevcExtraData"                // AMFInterface* - > AMFBuffer*; SPS/PPS buffer - read-only
16
 #define AMF_VIDEO_ENCODER_HEVC_ASPECT_RATIO                         L"HevcAspectRatio"              // AMFRatio; default = 1, 1
17
-#define AMF_VIDEO_ENCODER_HEVC_LOWLATENCY_MODE                     L"LowLatencyInternal"           // bool; default = false, enables low latency mode
18
+#define AMF_VIDEO_ENCODER_HEVC_LOWLATENCY_MODE                      L"LowLatencyInternal"           // bool; default = false, enables low latency mode
19
 #define AMF_VIDEO_ENCODER_HEVC_PRE_ANALYSIS_ENABLE                  L"HevcEnablePreAnalysis"        // bool; default = false; enables the pre-analysis module. Currently only works in AMF_VIDEO_ENCODER_HEVC_RATE_CONTROL_METHOD_PEAK_CONSTRAINED_VBR mode. Refer to AMF Video PreAnalysis API reference for more details.
20
-#define AMF_VIDEO_ENCODER_HEVC_NOMINAL_RANGE                       L"HevcNominalRange"             // amf_int64(AMF_VIDEO_ENCODER_HEVC_NOMINAL_RANGE); default = amf_int64(AMF_VIDEO_ENCODER_HEVC_NOMINAL_RANGE_STUDIO); property is bool but amf_int64 also works for backward compatibility.
21
+#define AMF_VIDEO_ENCODER_HEVC_NOMINAL_RANGE                        L"HevcNominalRange"                // amf_int64(AMF_VIDEO_ENCODER_HEVC_NOMINAL_RANGE); default = amf_int64(AMF_VIDEO_ENCODER_HEVC_NOMINAL_RANGE_STUDIO); property is bool but amf_int64 also works for backward compatibility.
22
+#define AMF_VIDEO_ENCODER_HEVC_MAX_NUM_TEMPORAL_LAYERS              L"HevcMaxNumOfTemporalLayers"   // amf_int64; default = 1; Max number of temporal layers.
23
 
24
 // Picture control properties
25
 #define AMF_VIDEO_ENCODER_HEVC_NUM_GOPS_PER_IDR                     L"HevcGOPSPerIDR"               // amf_int64; default = 1; The frequency to insert IDR as start of a GOP. 0 means no IDR will be inserted.
26
@@ -159,6 +163,7 @@
27
 
28
 // Rate control properties
29
 #define AMF_VIDEO_ENCODER_HEVC_RATE_CONTROL_METHOD                  L"HevcRateControlMethod"        // amf_int64(AMF_VIDEO_ENCODER_HEVC_RATE_CONTROL_METHOD_ENUM); default = depends on USAGE; Rate Control Method
30
+#define AMF_VIDEO_ENCODER_HEVC_QVBR_QUALITY_LEVEL                   L"HevcQvbrQualityLevel"         // amf_int64; default = 23; QVBR quality level; range = 1-51
31
 #define AMF_VIDEO_ENCODER_HEVC_VBV_BUFFER_SIZE                      L"HevcVBVBufferSize"            // amf_int64; default = depends on USAGE; VBV Buffer Size in bits
32
 #define AMF_VIDEO_ENCODER_HEVC_INITIAL_VBV_BUFFER_FULLNESS          L"HevcInitialVBVBufferFullness" // amf_int64; default =  64; Initial VBV Buffer Fullness 0=0% 64=100%
33
 #define AMF_VIDEO_ENCODER_HEVC_ENABLE_VBAQ                          L"HevcEnableVBAQ"               // // bool; default = depends on USAGE; Enable auto VBAQ
34
@@ -213,6 +218,9 @@
35
 #define AMF_VIDEO_ENCODER_HEVC_INPUT_HDR_METADATA                   L"HevcInHDRMetadata"            // AMFBuffer containing AMFHDRMetadata; default NULL
36
 //#define AMF_VIDEO_ENCODER_HEVC_OUTPUT_HDR_METADATA                  L"HevcOutHDRMetadata"            // AMFBuffer containing AMFHDRMetadata; default NULL
37
 
38
+// SVC
39
+#define AMF_VIDEO_ENCODER_HEVC_NUM_TEMPORAL_LAYERS                  L"HevcNumOfTemporalLayers"      // amf_int64; default = 1; Number of temporal layers. Can be changed at any time but the change is only applied when encoding next base layer frame.
40
+
41
 // DPB management
42
 #define AMF_VIDEO_ENCODER_HEVC_PICTURE_TRANSFER_MODE                L"HevcPicTransferMode"          // amf_int64(AMF_VIDEO_ENCODER_HEVC_PICTURE_TRANSFER_MODE_ENUM); default = AMF_VIDEO_ENCODER_HEVC_PICTURE_TRANSFER_MODE_OFF - whether to exchange reference/reconstructed pic between encoder and application
43
 
44
@@ -282,6 +290,7 @@
45
 #define AMF_VIDEO_ENCODER_HEVC_CAP_MAX_LEVEL                        L"HevcMaxLevel"                 // amf_int64 maximum profile level
46
 #define AMF_VIDEO_ENCODER_HEVC_CAP_MIN_REFERENCE_FRAMES             L"HevcMinReferenceFrames"       // amf_int64 minimum number of reference frames
47
 #define AMF_VIDEO_ENCODER_HEVC_CAP_MAX_REFERENCE_FRAMES             L"HevcMaxReferenceFrames"       // amf_int64 maximum number of reference frames
48
+#define AMF_VIDEO_ENCODER_HEVC_CAP_MAX_TEMPORAL_LAYERS              L"HevcMaxTemporalLayers"        // amf_int64 maximum number of temporal layers
49
 #define AMF_VIDEO_ENCODER_HEVC_CAP_NUM_OF_HW_INSTANCES              L"HevcNumOfHwInstances"         // amf_int64 number of HW encoder instances
50
 #define AMF_VIDEO_ENCODER_HEVC_CAP_COLOR_CONVERSION                 L"HevcColorConversion"          // amf_int64(AMF_ACCELERATION_TYPE) - type of supported color conversion. default AMF_ACCEL_GPU
51
 #define AMF_VIDEO_ENCODER_HEVC_CAP_PRE_ANALYSIS                     L"HevcPreAnalysis"              // amf_bool - pre analysis module is available for HEVC UVE encoder, n/a for the other encoders
52
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/external/AMF/include/components/VideoEncoderVCE.h -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/external/AMF/include/components/VideoEncoderVCE.h Changed
20
 
1
@@ -79,7 +79,9 @@
2
     AMF_VIDEO_ENCODER_RATE_CONTROL_METHOD_CBR,
3
     AMF_VIDEO_ENCODER_RATE_CONTROL_METHOD_PEAK_CONSTRAINED_VBR,
4
     AMF_VIDEO_ENCODER_RATE_CONTROL_METHOD_LATENCY_CONSTRAINED_VBR,
5
-    AMF_VIDEO_ENCODER_RATE_CONTROL_METHOD_QUALITY_VBR
6
+    AMF_VIDEO_ENCODER_RATE_CONTROL_METHOD_QUALITY_VBR,
7
+    AMF_VIDEO_ENCODER_RATE_CONTROL_METHOD_HIGH_QUALITY_VBR,
8
+    AMF_VIDEO_ENCODER_RATE_CONTROL_METHOD_HIGH_QUALITY_CBR
9
 };
10
 
11
 enum AMF_VIDEO_ENCODER_QUALITY_PRESET_ENUM
12
@@ -164,6 +166,7 @@
13
 #define AMF_VIDEO_ENCODER_RATE_CONTROL_PREANALYSIS_ENABLE       L"RateControlPreanalysisEnable"     // amf_int64(AMF_VIDEO_ENCODER_PREENCODE_MODE_ENUM); default =  AMF_VIDEO_ENCODER_PREENCODE_DISABLED; enables pre-encode assisted rate control. Deprecated, please use AMF_VIDEO_ENCODER_PREENCODE_ENABLE instead.
14
 #define AMF_VIDEO_ENCODER_RATE_CONTROL_METHOD                   L"RateControlMethod"        // amf_int64(AMF_VIDEO_ENCODER_RATE_CONTROL_METHOD_ENUM); default = depends on USAGE; Rate Control Method
15
 #define AMF_VIDEO_ENCODER_QVBR_QUALITY_LEVEL                    L"QvbrQualityLevel"         // amf_int64; default = 23; QVBR quality level; range = 1-51
16
+#define AMF_VIDEO_ENCODER_MAX_NUM_TEMPORAL_LAYERS               L"MaxNumOfTemporalLayers"   // amf_int64; default = 1; Max number of temporal layers.
17
 #if !defined(__GNUC__) && !defined(__clang__)
18
     #pragma deprecated("AMF_VIDEO_ENCODER_RATE_CONTROL_PREANALYSIS_ENABLE")
19
 #endif
20
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/external/AMF/include/core/Version.h -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/external/AMF/include/core/Version.h Changed
10
 
1
@@ -51,7 +51,7 @@
2
 
3
 #define AMF_VERSION_MAJOR       1
4
 #define AMF_VERSION_MINOR       4
5
-#define AMF_VERSION_RELEASE     24
6
+#define AMF_VERSION_RELEASE     29
7
 #define AMF_VERSION_BUILD_NUM   0
8
 
9
 #define AMF_FULL_VERSION AMF_MAKE_FULL_VERSION(AMF_VERSION_MAJOR, AMF_VERSION_MINOR, AMF_VERSION_RELEASE, AMF_VERSION_BUILD_NUM)
10
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/ffmpeg-mux/ffmpeg-mux.c -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/ffmpeg-mux/ffmpeg-mux.c Changed
93
 
1
@@ -99,9 +99,11 @@
2
    int color_trc;
3
    int colorspace;
4
    int color_range;
5
+   int chroma_sample_location;
6
    int max_luminance;
7
    char *acodec;
8
    char *muxer_settings;
9
+   int codec_tag;
10
 };
11
 
12
 struct audio_params {
13
@@ -368,6 +370,9 @@
14
        if (!get_opt_int(argc, argv, &params->color_range,
15
                 "video color range"))
16
            return false;
17
+       if (!get_opt_int(argc, argv, &params->chroma_sample_location,
18
+                "video chroma sample location"))
19
+           return false;
20
        if (!get_opt_int(argc, argv, &params->max_luminance,
21
                 "video max luminance"))
22
            return false;
23
@@ -375,6 +380,9 @@
24
            return false;
25
        if (!get_opt_int(argc, argv, &params->fps_den, "video fps den"))
26
            return false;
27
+       if (!get_opt_int(argc, argv, &params->codec_tag,
28
+                "video codec tag"))
29
+           params->codec_tag = 0;
30
    }
31
 
32
    if (params->tracks) {
33
@@ -445,6 +453,7 @@
34
    context = avcodec_alloc_context3(NULL);
35
    context->codec_type = codec->type;
36
    context->codec_id = codec->id;
37
+   context->codec_tag = ffm->params.codec_tag;
38
    context->bit_rate = (int64_t)ffm->params.vbitrate * 1000;
39
    context->width = ffm->params.width;
40
    context->height = ffm->params.height;
41
@@ -454,10 +463,7 @@
42
    context->color_trc = ffm->params.color_trc;
43
    context->colorspace = ffm->params.colorspace;
44
    context->color_range = ffm->params.color_range;
45
-   context->chroma_sample_location =
46
-       (ffm->params.colorspace == AVCOL_SPC_BT2020_NCL)
47
-           ? AVCHROMA_LOC_TOPLEFT
48
-           : AVCHROMA_LOC_LEFT;
49
+   context->chroma_sample_location = ffm->params.chroma_sample_location;
50
    context->extradata = extradata;
51
    context->extradata_size = ffm->video_header.size;
52
    context->time_base =
53
@@ -515,6 +521,7 @@
54
    AVStream *stream;
55
    void *extradata = NULL;
56
    const char *name = ffm->params.acodec;
57
+   int channels;
58
 
59
    const AVCodecDescriptor *codec = avcodec_descriptor_get_by_name(name);
60
    if (!codec) {
61
@@ -538,7 +545,10 @@
62
    context->codec_type = codec->type;
63
    context->codec_id = codec->id;
64
    context->bit_rate = (int64_t)ffm->audioidx.abitrate * 1000;
65
-   context->channels = ffm->audioidx.channels;
66
+   channels = ffm->audioidx.channels;
67
+#if LIBAVUTIL_VERSION_INT < AV_VERSION_INT(57, 24, 100)
68
+   context->channels = channels;
69
+#endif
70
    context->sample_rate = ffm->audioidx.sample_rate;
71
    context->frame_size = ffm->audioidx.frame_size;
72
    context->sample_fmt = AV_SAMPLE_FMT_S16;
73
@@ -546,15 +556,14 @@
74
    context->extradata = extradata;
75
    context->extradata_size = ffm->audio_headeridx.size;
76
 #if LIBAVCODEC_VERSION_INT < AV_VERSION_INT(59, 24, 100)
77
-   context->channel_layout =
78
-       av_get_default_channel_layout(context->channels);
79
+   context->channel_layout = av_get_default_channel_layout(channels);
80
    //avutil default channel layout for 5 channels is 5.0 ; fix for 4.1
81
-   if (context->channels == 5)
82
+   if (channels == 5)
83
        context->channel_layout = av_get_channel_layout("4.1");
84
 #else
85
-   av_channel_layout_default(&context->ch_layout, context->channels);
86
+   av_channel_layout_default(&context->ch_layout, channels);
87
    //avutil default channel layout for 5 channels is 5.0 ; fix for 4.1
88
-   if (context->channels == 5)
89
+   if (channels == 5)
90
        context->ch_layout = (AVChannelLayout)AV_CHANNEL_LAYOUT_4POINT1;
91
 #endif
92
    if (ffm->output->oformat->flags & AVFMT_GLOBALHEADER)
93
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/jim-nvenc-helpers.c -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/jim-nvenc-helpers.c Changed
45
 
1
@@ -2,6 +2,7 @@
2
 #include <util/platform.h>
3
 #include <util/threading.h>
4
 #include <util/config-file.h>
5
+#include <util/windows/device-enum.h>
6
 #include <util/dstr.h>
7
 #include <util/pipe.h>
8
 
9
@@ -222,14 +223,26 @@
10
 #endif
11
 extern struct obs_encoder_info av1_nvenc_info;
12
 
13
+static bool enum_luids(void *param, uint32_t idx, uint64_t luid)
14
+{
15
+   struct dstr *cmd = param;
16
+   dstr_catf(cmd, " %llX", luid);
17
+   UNUSED_PARAMETER(idx);
18
+   return true;
19
+}
20
+
21
 static bool av1_supported(void)
22
 {
23
    char *test_exe = os_get_executable_path_ptr("obs-nvenc-test.exe");
24
+   struct dstr cmd = {0};
25
    struct dstr caps_str = {0};
26
    bool av1_supported = false;
27
    config_t *config = NULL;
28
 
29
-   os_process_pipe_t *pp = os_process_pipe_create(test_exe, "r");
30
+   dstr_copy(&cmd, test_exe);
31
+   enum_graphics_device_luids(enum_luids, &cmd);
32
+
33
+   os_process_pipe_t *pp = os_process_pipe_create(cmd.array, "r");
34
    if (!pp) {
35
        blog(LOG_WARNING, "NVENC Failed to launch the NVENC "
36
                  "test process I guess");
37
@@ -278,6 +291,7 @@
38
    if (config)
39
        config_close(config);
40
    dstr_free(&caps_str);
41
+   dstr_free(&cmd);
42
    if (test_exe)
43
        bfree(test_exe);
44
 
45
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/jim-nvenc.c -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/jim-nvenc.c Changed
154
 
1
@@ -431,7 +431,7 @@
2
 }
3
 
4
 static bool init_encoder_base(struct nvenc_data *enc, obs_data_t *settings,
5
-                 int bf, bool psycho_aq, bool *lossless)
6
+                 int bf, bool compatibility, bool *lossless)
7
 {
8
    const char *rc = obs_data_get_string(settings, "rate_control");
9
    int bitrate = (int)obs_data_get_int(settings, "bitrate");
10
@@ -445,6 +445,8 @@
11
    const char *profile = obs_data_get_string(settings, "profile");
12
    bool lookahead = obs_data_get_bool(settings, "lookahead");
13
    bool vbr = astrcmpi(rc, "VBR") == 0;
14
+   bool psycho_aq = !compatibility &&
15
+            obs_data_get_bool(settings, "psycho_aq");
16
    NVENCSTATUS err;
17
 
18
    video_t *video = obs_encoder_video(enc->encoder);
19
@@ -458,7 +460,9 @@
20
 
21
    GUID nv_preset = get_nv_preset2(preset2);
22
    NV_ENC_TUNING_INFO nv_tuning = get_nv_tuning(tuning);
23
-   NV_ENC_MULTI_PASS nv_multipass = get_nv_multipass(multipass);
24
+   NV_ENC_MULTI_PASS nv_multipass = compatibility
25
+                        ? NV_ENC_MULTI_PASS_DISABLED
26
+                        : get_nv_multipass(multipass);
27
 
28
    if (obs_data_has_user_value(settings, "preset") &&
29
        !obs_data_has_user_value(settings, "preset2") &&
30
@@ -627,12 +631,14 @@
31
    }
32
 
33
    /* psycho aq */
34
-   if (nv_get_cap(enc, NV_ENC_CAPS_SUPPORT_TEMPORAL_AQ)) {
35
-       config->rcParams.enableAQ = psycho_aq;
36
-       config->rcParams.aqStrength = 8;
37
-       config->rcParams.enableTemporalAQ = psycho_aq;
38
-   } else if (psycho_aq) {
39
-       warn("Ignoring Psycho Visual Tuning request since GPU is not capable");
40
+   if (!compatibility) {
41
+       if (nv_get_cap(enc, NV_ENC_CAPS_SUPPORT_TEMPORAL_AQ)) {
42
+           config->rcParams.enableAQ = psycho_aq;
43
+           config->rcParams.aqStrength = 8;
44
+           config->rcParams.enableTemporalAQ = psycho_aq;
45
+       } else {
46
+           warn("Ignoring Psycho Visual Tuning request since GPU is not capable");
47
+       }
48
    }
49
 
50
    /* -------------------------- */
51
@@ -694,14 +700,14 @@
52
 }
53
 
54
 static bool init_encoder_h264(struct nvenc_data *enc, obs_data_t *settings,
55
-                 int bf, bool psycho_aq)
56
+                 int bf, bool compatibility)
57
 {
58
    const char *rc = obs_data_get_string(settings, "rate_control");
59
    int keyint_sec = (int)obs_data_get_int(settings, "keyint_sec");
60
    const char *profile = obs_data_get_string(settings, "profile");
61
    bool lossless;
62
 
63
-   if (!init_encoder_base(enc, settings, bf, psycho_aq, &lossless)) {
64
+   if (!init_encoder_base(enc, settings, bf, compatibility, &lossless)) {
65
        return false;
66
    }
67
 
68
@@ -779,14 +785,14 @@
69
 }
70
 
71
 static bool init_encoder_hevc(struct nvenc_data *enc, obs_data_t *settings,
72
-                 int bf, bool psycho_aq)
73
+                 int bf, bool compatibility)
74
 {
75
    const char *rc = obs_data_get_string(settings, "rate_control");
76
    int keyint_sec = (int)obs_data_get_int(settings, "keyint_sec");
77
    const char *profile = obs_data_get_string(settings, "profile");
78
    bool lossless;
79
 
80
-   if (!init_encoder_base(enc, settings, bf, psycho_aq, &lossless)) {
81
+   if (!init_encoder_base(enc, settings, bf, compatibility, &lossless)) {
82
        return false;
83
    }
84
 
85
@@ -880,13 +886,13 @@
86
 }
87
 
88
 static bool init_encoder_av1(struct nvenc_data *enc, obs_data_t *settings,
89
-                int bf, bool psycho_aq)
90
+                int bf, bool compatibility)
91
 {
92
    const char *rc = obs_data_get_string(settings, "rate_control");
93
    int keyint_sec = (int)obs_data_get_int(settings, "keyint_sec");
94
    bool lossless;
95
 
96
-   if (!init_encoder_base(enc, settings, bf, psycho_aq, &lossless)) {
97
+   if (!init_encoder_base(enc, settings, bf, compatibility, &lossless)) {
98
        return false;
99
    }
100
 
101
@@ -987,15 +993,15 @@
102
 static void nvenc_destroy(void *data);
103
 
104
 static bool init_specific_encoder(struct nvenc_data *enc, obs_data_t *settings,
105
-                 int bf, bool psycho_aq)
106
+                 int bf, bool compatibility)
107
 {
108
    switch (enc->codec) {
109
    case CODEC_HEVC:
110
-       return init_encoder_hevc(enc, settings, bf, psycho_aq);
111
+       return init_encoder_hevc(enc, settings, bf, compatibility);
112
    case CODEC_H264:
113
-       return init_encoder_h264(enc, settings, bf, psycho_aq);
114
+       return init_encoder_h264(enc, settings, bf, compatibility);
115
    case CODEC_AV1:
116
-       return init_encoder_av1(enc, settings, bf, psycho_aq);
117
+       return init_encoder_av1(enc, settings, bf, compatibility);
118
    }
119
 
120
    return false;
121
@@ -1005,7 +1011,6 @@
122
             obs_data_t *settings, obs_encoder_t *encoder)
123
 {
124
    int bf = (int)obs_data_get_int(settings, "bf");
125
-   const bool psycho_aq = obs_data_get_bool(settings, "psycho_aq");
126
    const bool support_10bit =
127
        nv_get_cap(enc, NV_ENC_CAPS_SUPPORT_10BIT_ENCODE);
128
    const int bf_max = nv_get_cap(enc, NV_ENC_CAPS_NUM_MAX_BFRAMES);
129
@@ -1038,12 +1043,9 @@
130
        bf = bf_max;
131
    }
132
 
133
-   if (!init_specific_encoder(enc, settings, bf, psycho_aq)) {
134
-       if (!psycho_aq)
135
-           return false;
136
-
137
+   if (!init_specific_encoder(enc, settings, bf, false)) {
138
        blog(LOG_WARNING, "jim-nvenc init_specific_encoder failed, "
139
-                 "trying again without Psycho Visual Tuning");
140
+                 "trying again with compatibility options");
141
 
142
        nv.nvEncDestroyEncoder(enc->session);
143
        enc->session = NULL;
144
@@ -1051,7 +1053,8 @@
145
        if (!init_session(enc)) {
146
            return false;
147
        }
148
-       if (!init_specific_encoder(enc, settings, bf, false)) {
149
+       /* try without multipass and psycho aq */
150
+       if (!init_specific_encoder(enc, settings, bf, true)) {
151
            return false;
152
        }
153
    }
154
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/obs-amf-test/obs-amf-test.cpp -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/obs-amf-test/obs-amf-test.cpp Changed
115
 
1
@@ -2,6 +2,7 @@
2
 #include "../external/AMF/include/core/Trace.h"
3
 #include "../external/AMF/include/components/VideoEncoderVCE.h"
4
 #include "../external/AMF/include/components/VideoEncoderHEVC.h"
5
+#include "../external/AMF/include/components/VideoEncoderAV1.h"
6
 
7
 #include <util/windows/ComPtr.hpp>
8
 
9
@@ -9,6 +10,7 @@
10
 #include <d3d11.h>
11
 #include <d3d11_1.h>
12
 
13
+#include <vector>
14
 #include <string>
15
 #include <map>
16
 
17
@@ -24,9 +26,11 @@
18
    bool is_amd = false;
19
    bool supports_avc = false;
20
    bool supports_hevc = false;
21
+   bool supports_av1 = false;
22
 };
23
 
24
 static AMFFactory *amf_factory = nullptr;
25
+static std::vector<uint64_t> luid_order;
26
 static std::map<uint32_t, adapter_caps> adapter_info;
27
 
28
 static bool has_encoder(AMFContextPtr &amf_context, const wchar_t *encoder_name)
29
@@ -37,6 +41,17 @@
30
    return res == AMF_OK;
31
 }
32
 
33
+static inline uint32_t get_adapter_idx(uint32_t adapter_idx, LUID luid)
34
+{
35
+   for (size_t i = 0; i < luid_order.size(); i++) {
36
+       if (luid_orderi == *(uint64_t *)&luid) {
37
+           return (uint32_t)i;
38
+       }
39
+   }
40
+
41
+   return adapter_idx;
42
+}
43
+
44
 static bool get_adapter_caps(IDXGIFactory *factory, uint32_t adapter_idx)
45
 {
46
    AMF_RESULT res;
47
@@ -47,21 +62,17 @@
48
    if (FAILED(hr))
49
        return false;
50
 
51
-   adapter_caps &caps = adapter_infoadapter_idx;
52
-
53
    DXGI_ADAPTER_DESC desc;
54
    adapter->GetDesc(&desc);
55
 
56
+   uint32_t luid_idx = get_adapter_idx(adapter_idx, desc.AdapterLuid);
57
+   adapter_caps &caps = adapter_infoluid_idx;
58
+
59
    if (desc.VendorId != AMD_VENDOR_ID)
60
        return true;
61
 
62
    caps.is_amd = true;
63
 
64
-   ComPtr<IDXGIOutput> output;
65
-   hr = adapter->EnumOutputs(0, &output);
66
-   if (FAILED(hr))
67
-       return true;
68
-
69
    ComPtr<ID3D11Device> device;
70
    ComPtr<ID3D11DeviceContext> context;
71
    hr = D3D11CreateDevice(adapter, D3D_DRIVER_TYPE_UNKNOWN, nullptr, 0,
72
@@ -81,6 +92,7 @@
73
 
74
    caps.supports_avc = has_encoder(amf_context, AMFVideoEncoderVCE_AVC);
75
    caps.supports_hevc = has_encoder(amf_context, AMFVideoEncoder_HEVC);
76
+   caps.supports_av1 = has_encoder(amf_context, AMFVideoEncoder_AV1);
77
 
78
    return true;
79
 }
80
@@ -98,7 +110,7 @@
81
    return 0;
82
 }
83
 
84
-int main(void)
85
+int main(int argc, char *argv)
86
 try {
87
    ComPtr<IDXGIFactory> factory;
88
    AMF_RESULT res;
89
@@ -130,6 +142,16 @@
90
    if (res != AMF_OK)
91
        throw "AMFInit failed";
92
 
93
+   /* --------------------------------------------------------- */
94
+   /* parse expected LUID order                                 */
95
+
96
+   for (int i = 1; i < argc; i++) {
97
+       luid_order.push_back(strtoull(argvi, NULL, 16));
98
+   }
99
+
100
+   /* --------------------------------------------------------- */
101
+   /* obtain adapter compatibility information                  */
102
+
103
    hr = CreateDXGIFactory1(__uuidof(IDXGIFactory), (void **)&factory);
104
    if (FAILED(hr))
105
        throw "CreateDXGIFactory1 failed";
106
@@ -145,6 +167,8 @@
107
               caps.supports_avc ? "true" : "false");
108
        printf("supports_hevc=%s\n",
109
               caps.supports_hevc ? "true" : "false");
110
+       printf("supports_av1=%s\n",
111
+              caps.supports_av1 ? "true" : "false");
112
    }
113
 
114
    return 0;
115
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-audio-encoders.c -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-audio-encoders.c Changed
111
 
1
@@ -116,6 +116,7 @@
2
 static bool initialize_codec(struct enc_encoder *enc)
3
 {
4
    int ret;
5
+   int channels;
6
 
7
    enc->aframe = av_frame_alloc();
8
    if (!enc->aframe) {
9
@@ -134,7 +135,12 @@
10
        return false;
11
    }
12
    enc->aframe->format = enc->context->sample_fmt;
13
+#if LIBAVUTIL_VERSION_INT < AV_VERSION_INT(57, 24, 100)
14
    enc->aframe->channels = enc->context->channels;
15
+   channels = enc->context->channels;
16
+#else
17
+   channels = enc->context->ch_layout.nb_channels;
18
+#endif
19
 #if LIBAVCODEC_VERSION_INT < AV_VERSION_INT(59, 24, 100)
20
    enc->aframe->channel_layout = enc->context->channel_layout;
21
 #else
22
@@ -148,8 +154,8 @@
23
 
24
    enc->frame_size_bytes = enc->frame_size * (int)enc->audio_size;
25
 
26
-   ret = av_samples_alloc(enc->samples, NULL, enc->context->channels,
27
-                  enc->frame_size, enc->context->sample_fmt, 0);
28
+   ret = av_samples_alloc(enc->samples, NULL, channels, enc->frame_size,
29
+                  enc->context->sample_fmt, 0);
30
    if (ret < 0) {
31
        warn("Failed to create audio buffer: %s", av_err2str(ret));
32
        return false;
33
@@ -216,16 +222,21 @@
34
    enc->context->bit_rate = bitrate * 1000;
35
    const struct audio_output_info *aoi;
36
    aoi = audio_output_get_info(audio);
37
+
38
+#if LIBAVUTIL_VERSION_INT < AV_VERSION_INT(57, 24, 100)
39
    enc->context->channels = (int)audio_output_get_channels(audio);
40
+#endif
41
+
42
 #if LIBAVCODEC_VERSION_INT < AV_VERSION_INT(59, 24, 100)
43
    enc->context->channel_layout = convert_speaker_layout(aoi->speakers);
44
 #else
45
    av_channel_layout_default(&enc->context->ch_layout,
46
-                 enc->context->channels);
47
+                 (int)audio_output_get_channels(audio));
48
    if (aoi->speakers == SPEAKERS_4POINT1)
49
        enc->context->ch_layout =
50
            (AVChannelLayout)AV_CHANNEL_LAYOUT_4POINT1;
51
 #endif
52
+
53
    enc->context->sample_rate = audio_output_get_sample_rate(audio);
54
    enc->context->sample_fmt = enc->codec->sample_fmts
55
                       ? enc->codec->sample_fmts0
56
@@ -264,7 +275,7 @@
57
    av_channel_layout_describe(&enc->context->ch_layout, buf, 256);
58
    info("bitrate: %" PRId64 ", channels: %d, channel_layout: %s\n",
59
         (int64_t)enc->context->bit_rate / 1000,
60
-        (int)enc->context->channels, buf);
61
+        (int)enc->context->ch_layout.nb_channels, buf);
62
 #endif
63
    init_sizes(enc, audio);
64
 
65
@@ -298,6 +309,7 @@
66
    AVPacket avpacket = {0};
67
    int got_packet;
68
    int ret;
69
+   int channels;
70
 
71
    enc->aframe->nb_samples = enc->frame_size;
72
    enc->aframe->pts = av_rescale_q(
73
@@ -305,11 +317,14 @@
74
        enc->context->time_base);
75
 #if LIBAVUTIL_VERSION_INT >= AV_VERSION_INT(57, 24, 100)
76
    enc->aframe->ch_layout = enc->context->ch_layout;
77
+   channels = enc->context->ch_layout.nb_channels;
78
+#else
79
+   channels = enc->context->channels;
80
 #endif
81
-   ret = avcodec_fill_audio_frame(
82
-       enc->aframe, enc->context->channels, enc->context->sample_fmt,
83
-       enc->samples0, enc->frame_size_bytes * enc->context->channels,
84
-       1);
85
+   ret = avcodec_fill_audio_frame(enc->aframe, channels,
86
+                      enc->context->sample_fmt,
87
+                      enc->samples0,
88
+                      enc->frame_size_bytes * channels, 1);
89
    if (ret < 0) {
90
        warn("avcodec_fill_audio_frame failed: %s", av_err2str(ret));
91
        return false;
92
@@ -391,10 +406,16 @@
93
 static void enc_audio_info(void *data, struct audio_convert_info *info)
94
 {
95
    struct enc_encoder *enc = data;
96
+   int channels;
97
+#if LIBAVUTIL_VERSION_INT >= AV_VERSION_INT(57, 24, 100)
98
+   channels = enc->context->ch_layout.nb_channels;
99
+#else
100
+   channels = enc->context->channels;
101
+#endif
102
    info->format = convert_ffmpeg_sample_format(enc->context->sample_fmt);
103
    info->samples_per_sec = (uint32_t)enc->context->sample_rate;
104
-   if (enc->context->channels != 7 && enc->context->channels <= 8)
105
-       info->speakers = (enum speaker_layout)(enc->context->channels);
106
+   if (channels != 7 && channels <= 8)
107
+       info->speakers = (enum speaker_layout)(channels);
108
    else
109
        info->speakers = SPEAKERS_UNKNOWN;
110
 }
111
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-av1.c -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-av1.c Changed
171
 
1
@@ -26,9 +26,14 @@
2
 #define info(format, ...) do_log(LOG_INFO, format, ##__VA_ARGS__)
3
 #define debug(format, ...) do_log(LOG_DEBUG, format, ##__VA_ARGS__)
4
 
5
+enum av1_encoder_type {
6
+   AV1_ENCODER_TYPE_AOM,
7
+   AV1_ENCODER_TYPE_SVT,
8
+};
9
+
10
 struct av1_encoder {
11
    struct ffmpeg_video_encoder ffve;
12
-   bool svtav1;
13
+   enum av1_encoder_type type;
14
 
15
    DARRAY(uint8_t) header;
16
 };
17
@@ -66,6 +71,7 @@
18
    int cqp = (int)obs_data_get_int(settings, "cqp");
19
    int keyint_sec = (int)obs_data_get_int(settings, "keyint_sec");
20
    int preset = (int)obs_data_get_int(settings, "preset");
21
+   AVDictionary *svtav1_opts = NULL;
22
 
23
    video_t *video = obs_encoder_video(enc->ffve.encoder);
24
    const struct video_output_info *voi = video_output_get_info(video);
25
@@ -78,9 +84,18 @@
26
    enc->ffve.context->thread_count = 0;
27
 
28
    av1_video_info(enc, &info);
29
-   av_opt_set_int(enc->ffve.context->priv_data,
30
-              enc->svtav1 ? "preset" : "cpu-used", preset, 0);
31
-   if (!enc->svtav1) {
32
+
33
+   if (enc->type == AV1_ENCODER_TYPE_SVT) {
34
+       av_opt_set_int(enc->ffve.context->priv_data, "preset", preset,
35
+                  0);
36
+#if LIBAVCODEC_VERSION_INT >= AV_VERSION_INT(59, 37, 100)
37
+       av_dict_set_int(&svtav1_opts, "rc", 1, 0);
38
+#else
39
+       av_opt_set(enc->ffve.context->priv_data, "rc", "vbr", 0);
40
+#endif
41
+   } else if (enc->type == AV1_ENCODER_TYPE_AOM) {
42
+       av_opt_set_int(enc->ffve.context->priv_data, "cpu-used", preset,
43
+                  0);
44
        av_opt_set(enc->ffve.context->priv_data, "usage", "realtime",
45
               0);
46
 #if 0
47
@@ -94,31 +109,53 @@
48
        av_opt_set_int(enc->ffve.context->priv_data, "row-mt", 1, 0);
49
    }
50
 
51
-   if (enc->svtav1)
52
-       av_opt_set(enc->ffve.context->priv_data, "rc", "vbr", 0);
53
-
54
    if (astrcmpi(rc, "cqp") == 0) {
55
        bitrate = 0;
56
-       enc->ffve.context->global_quality = cqp;
57
+       av_opt_set_int(enc->ffve.context->priv_data, "crf", cqp, 0);
58
 
59
-       if (enc->svtav1)
60
+       if (enc->type == AV1_ENCODER_TYPE_SVT) {
61
+#if LIBAVCODEC_VERSION_INT >= AV_VERSION_INT(59, 37, 100)
62
+           av_dict_set_int(&svtav1_opts, "rc", 0, 0);
63
+           av_opt_set_int(enc->ffve.context->priv_data, "qp", cqp,
64
+                      0);
65
+#else
66
            av_opt_set(enc->ffve.context->priv_data, "rc", "cqp",
67
                   0);
68
+           av_opt_set_int(enc->ffve.context->priv_data, "qp", cqp,
69
+                      0);
70
+#endif
71
+       }
72
 
73
    } else if (astrcmpi(rc, "vbr") != 0) { /* CBR by default */
74
        const int64_t rate = bitrate * INT64_C(1000);
75
-       enc->ffve.context->rc_max_rate = rate;
76
        enc->ffve.context->rc_min_rate = rate;
77
        cqp = 0;
78
 
79
-       if (enc->svtav1)
80
+       if (enc->type == AV1_ENCODER_TYPE_SVT) {
81
+#if LIBAVCODEC_VERSION_INT >= AV_VERSION_INT(59, 37, 100)
82
+           av_dict_set_int(&svtav1_opts, "rc", 2, 0);
83
+           av_dict_set_int(&svtav1_opts, "pred-struct", 1, 0);
84
+           av_dict_set_int(&svtav1_opts, "bias-pct", 0, 0);
85
+           av_dict_set_int(&svtav1_opts, "tbr", rate, 0);
86
+#else
87
+           enc->ffve.context->rc_max_rate = rate;
88
            av_opt_set(enc->ffve.context->priv_data, "rc", "cvbr",
89
                   0);
90
+#endif
91
+       } else {
92
+           enc->ffve.context->rc_max_rate = rate;
93
+       }
94
+   }
95
+
96
+   if (enc->type == AV1_ENCODER_TYPE_SVT) {
97
+       av_opt_set_dict_val(enc->ffve.context->priv_data, "svtav1_opts",
98
+                   svtav1_opts, 0);
99
    }
100
 
101
    const char *ffmpeg_opts = obs_data_get_string(settings, "ffmpeg_opts");
102
    ffmpeg_video_encoder_update(&enc->ffve, bitrate, keyint_sec, voi, &info,
103
                    ffmpeg_opts);
104
+   av_dict_free(&svtav1_opts);
105
 
106
    info("settings:\n"
107
         "\tencoder:      %s\n"
108
@@ -150,7 +187,7 @@
109
 {
110
    struct av1_encoder *enc = data;
111
 
112
-   if (enc->svtav1) {
113
+   if (enc->type == AV1_ENCODER_TYPE_SVT) {
114
        da_copy_array(enc->header, enc->ffve.context->extradata,
115
                  enc->ffve.context->extradata_size);
116
    } else {
117
@@ -192,7 +229,9 @@
118
    struct av1_encoder *enc = bzalloc(sizeof(*enc));
119
 
120
    if (strcmp(enc_lib, "libsvtav1") == 0)
121
-       enc->svtav1 = true;
122
+       enc->type = AV1_ENCODER_TYPE_SVT;
123
+   else if (strcmp(enc_lib, "libaom-av1") == 0)
124
+       enc->type = AV1_ENCODER_TYPE_AOM;
125
 
126
    if (!ffmpeg_video_encoder_init(&enc->ffve, enc, encoder, enc_lib, NULL,
127
                       enc_name, NULL, on_first_packet))
128
@@ -250,7 +289,7 @@
129
    return true;
130
 }
131
 
132
-obs_properties_t *av1_properties(bool svtav1)
133
+obs_properties_t *av1_properties(enum av1_encoder_type type)
134
 {
135
    obs_properties_t *props = obs_properties_create();
136
    obs_property_t *p;
137
@@ -280,7 +319,7 @@
138
    p = obs_properties_add_list(props, "preset", obs_module_text("Preset"),
139
                    OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT);
140
 
141
-   if (svtav1) {
142
+   if (type == AV1_ENCODER_TYPE_SVT) {
143
        obs_property_list_add_int(p, "Very likely too slow (6)", 6);
144
        obs_property_list_add_int(p, "Probably too slow (7)", 7);
145
        obs_property_list_add_int(p, "Seems okay (8)", 8);
146
@@ -290,7 +329,7 @@
147
        obs_property_list_add_int(
148
            p, "Whoa, although quality might be not so great (12)",
149
            12);
150
-   } else {
151
+   } else if (type == AV1_ENCODER_TYPE_AOM) {
152
        obs_property_list_add_int(p, "Probably too slow (7)", 7);
153
        obs_property_list_add_int(p, "Okay (8)", 8);
154
        obs_property_list_add_int(p, "Fast (9)", 9);
155
@@ -307,13 +346,13 @@
156
 obs_properties_t *aom_av1_properties(void *unused)
157
 {
158
    UNUSED_PARAMETER(unused);
159
-   return av1_properties(false);
160
+   return av1_properties(AV1_ENCODER_TYPE_AOM);
161
 }
162
 
163
 obs_properties_t *svt_av1_properties(void *unused)
164
 {
165
    UNUSED_PARAMETER(unused);
166
-   return av1_properties(true);
167
+   return av1_properties(AV1_ENCODER_TYPE_SVT);
168
 }
169
 
170
 static bool av1_extra_data(void *data, uint8_t **extra_data, size_t *size)
171
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-formats.h -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-formats.h Changed
66
 
1
@@ -1,6 +1,7 @@
2
 #pragma once
3
 
4
 #include <libavcodec/avcodec.h>
5
+#include <libavutil/pixdesc.h>
6
 
7
 static inline int64_t rescale_ts(int64_t val, AVCodecContext *context,
8
                 AVRational new_base)
9
@@ -47,11 +48,9 @@
10
        return AV_PIX_FMT_YUVA422P;
11
    case VIDEO_FORMAT_YUVA:
12
        return AV_PIX_FMT_YUVA444P;
13
-   case VIDEO_FORMAT_YA2L:
14
 #if LIBAVUTIL_BUILD >= AV_VERSION_INT(56, 31, 100)
15
+   case VIDEO_FORMAT_YA2L:
16
        return AV_PIX_FMT_YUVA444P12LE;
17
-#else
18
-       return AV_PIX_FMT_NONE;
19
 #endif
20
    case VIDEO_FORMAT_I010:
21
        return AV_PIX_FMT_YUV420P10LE;
22
@@ -59,11 +58,41 @@
23
        return AV_PIX_FMT_P010LE;
24
    case VIDEO_FORMAT_NONE:
25
    case VIDEO_FORMAT_AYUV:
26
-       /* not supported by FFmpeg */
27
+   default:
28
        return AV_PIX_FMT_NONE;
29
    }
30
+}
31
+
32
+static enum AVChromaLocation
33
+determine_chroma_location(enum AVPixelFormat pix_fmt,
34
+             enum AVColorSpace colorspace)
35
+{
36
+   const AVPixFmtDescriptor *const desc = av_pix_fmt_desc_get(pix_fmt);
37
+   if (desc) {
38
+       const unsigned log_chroma_w = desc->log2_chroma_w;
39
+       const unsigned log_chroma_h = desc->log2_chroma_h;
40
+       switch (log_chroma_h) {
41
+       case 0:
42
+           switch (log_chroma_w) {
43
+           case 0:
44
+               /* 4:4:4 */
45
+               return AVCHROMA_LOC_CENTER;
46
+           case 1:
47
+               /* 4:2:2 */
48
+               return AVCHROMA_LOC_LEFT;
49
+           }
50
+           break;
51
+       case 1:
52
+           if (log_chroma_w == 1) {
53
+               /* 4:2:0 */
54
+               return (colorspace == AVCOL_SPC_BT2020_NCL)
55
+                          ? AVCHROMA_LOC_TOPLEFT
56
+                          : AVCHROMA_LOC_LEFT;
57
+           }
58
+       }
59
+   }
60
 
61
-   return AV_PIX_FMT_NONE;
62
+   return AVCHROMA_LOC_UNSPECIFIED;
63
 }
64
 
65
 static inline enum audio_format
66
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-mpegts.c -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-mpegts.c Changed
201
 
1
@@ -31,8 +31,8 @@
2
 #include <libavutil/mastering_display_metadata.h>
3
 
4
 /* ------------------------------------------------------------------------- */
5
-#define do_log(level, format, ...)                             \
6
-   blog(level, "obs-ffmpeg mpegts muxer: '%s' " format, \
7
+#define do_log(level, format, ...)                              \
8
+   blog(level, "obs-ffmpeg mpegts muxer: '%s': " format, \
9
         obs_output_get_name(stream->output), ##__VA_ARGS__)
10
 
11
 #define warn(format, ...) do_log(LOG_WARNING, format, ##__VA_ARGS__)
12
@@ -151,7 +151,7 @@
13
    const char *name = data->config.video_encoder;
14
    const AVCodecDescriptor *codec = avcodec_descriptor_get_by_name(name);
15
    if (!codec) {
16
-       error("Couldn't find codec '%s'\n", name);
17
+       error("Couldn't find codec '%s'", name);
18
        return false;
19
    }
20
    if (!new_stream(data, &data->video, name))
21
@@ -206,10 +206,8 @@
22
    context->color_primaries = data->config.color_primaries;
23
    context->color_trc = data->config.color_trc;
24
    context->colorspace = data->config.colorspace;
25
-   context->chroma_sample_location =
26
-       (data->config.colorspace == AVCOL_SPC_BT2020_NCL)
27
-           ? AVCHROMA_LOC_TOPLEFT
28
-           : AVCHROMA_LOC_LEFT;
29
+   context->chroma_sample_location = determine_chroma_location(
30
+       data->config.format, data->config.colorspace);
31
    context->thread_count = 0;
32
 
33
    data->video->time_base = context->time_base;
34
@@ -235,10 +233,11 @@
35
    void *extradata = NULL;
36
    struct obs_audio_info aoi;
37
    const char *name = data->config.audio_encoder;
38
+   int channels;
39
 
40
    const AVCodecDescriptor *codec = avcodec_descriptor_get_by_name(name);
41
    if (!codec) {
42
-       warn("Couldn't find codec '%s'\n", name);
43
+       warn("Couldn't find codec '%s'", name);
44
        return false;
45
    }
46
 
47
@@ -255,14 +254,24 @@
48
    context->codec_id = codec->id;
49
    context->bit_rate = (int64_t)data->config.audio_bitrate * 1000;
50
    context->time_base = (AVRational){1, aoi.samples_per_sec};
51
+   channels = get_audio_channels(aoi.speakers);
52
+#if LIBAVUTIL_VERSION_INT < AV_VERSION_INT(57, 24, 100)
53
    context->channels = get_audio_channels(aoi.speakers);
54
+#endif
55
    context->sample_rate = aoi.samples_per_sec;
56
+
57
+#if LIBAVCODEC_VERSION_INT < AV_VERSION_INT(59, 24, 100)
58
    context->channel_layout =
59
        av_get_default_channel_layout(context->channels);
60
 
61
    //avutil default channel layout for 5 channels is 5.0 ; fix for 4.1
62
    if (aoi.speakers == SPEAKERS_4POINT1)
63
        context->channel_layout = av_get_channel_layout("4.1");
64
+#else
65
+   av_channel_layout_default(&context->ch_layout, channels);
66
+   if (aoi.speakers == SPEAKERS_4POINT1)
67
+       context->ch_layout = (AVChannelLayout)AV_CHANNEL_LAYOUT_4POINT1;
68
+#endif
69
 
70
    context->sample_fmt = AV_SAMPLE_FMT_S16;
71
    context->frame_size = data->config.frame_size;
72
@@ -316,14 +325,14 @@
73
    const char *url = stream->ff_data.config.url;
74
    if (!ff_network_init()) {
75
        ffmpeg_mpegts_log_error(LOG_ERROR, &stream->ff_data,
76
-                   "Can not initialize network.");
77
+                   "Couldn't initialize network");
78
        return AVERROR(EIO);
79
    }
80
 
81
    URLContext *uc = av_mallocz(sizeof(URLContext) + strlen(url) + 1);
82
    if (!uc) {
83
        ffmpeg_mpegts_log_error(LOG_ERROR, &stream->ff_data,
84
-                   "Can not allocate memory.");
85
+                   "Couldn't allocate memory");
86
        goto fail;
87
    }
88
    uc->url = (char *)url;
89
@@ -333,14 +342,53 @@
90
                : av_mallocz(sizeof(SRTContext));
91
    if (!uc->priv_data) {
92
        ffmpeg_mpegts_log_error(LOG_ERROR, &stream->ff_data,
93
-                   "Can not allocate memory.");
94
+                   "Couldn't allocate memory");
95
        goto fail;
96
    }
97
+   /* For SRT, pass streamid & passphrase; for RIST, pass passphrase, username
98
+    * & password.
99
+    */
100
+   if (!is_rist) {
101
+       SRTContext *context = (SRTContext *)uc->priv_data;
102
+       context->streamid = NULL;
103
+       if (stream->ff_data.config.key != NULL) {
104
+           if (strlen(stream->ff_data.config.key))
105
+               context->streamid =
106
+                   av_strdup(stream->ff_data.config.key);
107
+       }
108
+       context->passphrase = NULL;
109
+       if (stream->ff_data.config.password != NULL) {
110
+           if (strlen(stream->ff_data.config.password))
111
+               context->passphrase = av_strdup(
112
+                   stream->ff_data.config.password);
113
+       }
114
+   } else {
115
+       RISTContext *context = (RISTContext *)uc->priv_data;
116
+       context->secret = NULL;
117
+       if (stream->ff_data.config.key != NULL) {
118
+           if (strlen(stream->ff_data.config.key))
119
+               context->secret =
120
+                   bstrdup(stream->ff_data.config.key);
121
+       }
122
+       context->username = NULL;
123
+       if (stream->ff_data.config.username != NULL) {
124
+           if (strlen(stream->ff_data.config.username))
125
+               context->username = bstrdup(
126
+                   stream->ff_data.config.username);
127
+       }
128
+       context->password = NULL;
129
+       if (stream->ff_data.config.password != NULL) {
130
+           if (strlen(stream->ff_data.config.password))
131
+               context->password = bstrdup(
132
+                   stream->ff_data.config.password);
133
+       }
134
+   }
135
    stream->h = uc;
136
    if (is_rist)
137
        err = librist_open(uc, uc->url);
138
    else
139
        err = libsrt_open(uc, uc->url);
140
+
141
    if (err < 0)
142
        goto fail;
143
    return 0;
144
@@ -408,9 +456,9 @@
145
                        "=", " ", 0))) {
146
            ffmpeg_mpegts_log_error(
147
                LOG_WARNING, data,
148
-               "Failed to parse protocol settings: %s\n%s",
149
-               av_err2str(ret),
150
-               data->config.protocol_settings);
151
+               "Failed to parse protocol settings: %s, %s",
152
+               data->config.protocol_settings,
153
+               av_err2str(ret));
154
 
155
            av_dict_free(&dict);
156
            return OBS_OUTPUT_INVALID_STREAM;
157
@@ -444,7 +492,7 @@
158
        ret = avio_open2(&data->output->pb, data->config.url,
159
                 AVIO_FLAG_WRITE, NULL, &dict);
160
    } else {
161
-       info("ffmpeg mpegts muxer: Invalid protocol: %s",
162
+       info("ffmpeg mpegts muxer: Invalid protocol: %s",
163
             data->config.url);
164
        return OBS_OUTPUT_BAD_PATH;
165
    }
166
@@ -452,7 +500,7 @@
167
    if (ret < 0) {
168
        if ((rist || srt) && (ret == OBS_OUTPUT_CONNECT_FAILED ||
169
                      ret == OBS_OUTPUT_INVALID_STREAM)) {
170
-           error("failed to open the url or invalid stream");
171
+           error("Failed to open the url or invalid stream");
172
        } else {
173
            ffmpeg_mpegts_log_error(LOG_WARNING, data,
174
                        "Couldn't open '%s', %s",
175
@@ -477,7 +525,7 @@
176
                dstr_catf(&str, "\n\t%s=%s", entry->key,
177
                      entry->value);
178
 
179
-           info("ffmpeg mpegts muxer:Invalid protocol settings: %s",
180
+           info("ffmpeg mpegts muxer: Invalid protocol settings: %s",
181
                 str.array);
182
            dstr_free(&str);
183
        }
184
@@ -485,7 +533,7 @@
185
    } else {
186
        ret = allocate_custom_aviocontext(stream, rist);
187
        if (ret < 0) {
188
-           info("Couldn't allocate custom avio_context for rist or srt'%s', %s\n",
189
+           info("Couldn't allocate custom avio_context for url: '%s', %s",
190
                 data->config.url, av_err2str(ret));
191
            return OBS_OUTPUT_INVALID_STREAM;
192
        }
193
@@ -541,7 +589,7 @@
194
    avio_context_free(&stream->s);
195
 
196
    if (err)
197
-       info("ffmpeg mpegts muxer: Error closing URL %s",
198
+       info("ffmpeg mpegts muxer: Error closing URL %s",
199
             stream->ff_data.config.url);
200
 }
201
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-mux.c -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-mux.c Changed
43
 
1
@@ -16,6 +16,7 @@
2
 ******************************************************************************/
3
 #include "ffmpeg-mux/ffmpeg-mux.h"
4
 #include "obs-ffmpeg-mux.h"
5
+#include "obs-ffmpeg-formats.h"
6
 
7
 #ifdef _WIN32
8
 #include "util/windows/win-version.h"
9
@@ -142,6 +143,14 @@
10
    video_t *video = obs_get_video();
11
    const struct video_output_info *info = video_output_get_info(video);
12
 
13
+   int codec_tag = (int)obs_data_get_int(settings, "codec_type");
14
+#if __BYTE_ORDER == __LITTLE_ENDIAN
15
+   codec_tag = ((codec_tag >> 24) & 0x000000FF) |
16
+           ((codec_tag << 8) & 0x00FF0000) |
17
+           ((codec_tag >> 8) & 0x0000FF00) |
18
+           ((codec_tag << 24) & 0xFF000000);
19
+#endif
20
+
21
    obs_data_release(settings);
22
 
23
    enum AVColorPrimaries pri = AVCOL_PRI_UNSPECIFIED;
24
@@ -184,12 +193,15 @@
25
            ? (int)obs_get_video_hdr_nominal_peak_level()
26
            : ((trc == AVCOL_TRC_ARIB_STD_B67) ? 1000 : 0);
27
 
28
-   dstr_catf(cmd, "%s %d %d %d %d %d %d %d %d %d %d ",
29
+   dstr_catf(cmd, "%s %d %d %d %d %d %d %d %d %d %d %d %d ",
30
          obs_encoder_get_codec(vencoder), bitrate,
31
          obs_output_get_width(stream->output),
32
          obs_output_get_height(stream->output), (int)pri, (int)trc,
33
-         (int)spc, (int)range, max_luminance, (int)info->fps_num,
34
-         (int)info->fps_den);
35
+         (int)spc, (int)range,
36
+         (int)determine_chroma_location(
37
+             obs_to_ffmpeg_video_format(info->format), spc),
38
+         max_luminance, (int)info->fps_num, (int)info->fps_den,
39
+         (int)codec_tag);
40
 }
41
 
42
 static void add_audio_encoder_params(struct dstr *cmd, obs_encoder_t *aencoder)
43
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-nvenc.c -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-nvenc.c Changed
10
 
1
@@ -428,7 +428,7 @@
2
    obs_data_set_default_int(settings, "keyint_sec", 0);
3
    obs_data_set_default_int(settings, "cqp", 20);
4
    obs_data_set_default_string(settings, "rate_control", "CBR");
5
-   obs_data_set_default_string(settings, "preset2", "p6");
6
+   obs_data_set_default_string(settings, "preset2", "p5");
7
    obs_data_set_default_string(settings, "multipass", "qres");
8
    obs_data_set_default_string(settings, "tune", "hq");
9
    obs_data_set_default_string(settings, "profile",
10
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-output.c -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-output.c Changed
114
 
1
@@ -152,10 +152,8 @@
2
    data->vframe->color_primaries = data->config.color_primaries;
3
    data->vframe->color_trc = data->config.color_trc;
4
    data->vframe->colorspace = data->config.colorspace;
5
-   data->vframe->chroma_location =
6
-       (data->config.colorspace == AVCOL_SPC_BT2020_NCL)
7
-           ? AVCHROMA_LOC_TOPLEFT
8
-           : AVCHROMA_LOC_LEFT;
9
+   data->vframe->chroma_location = determine_chroma_location(
10
+       context->pix_fmt, data->config.colorspace);
11
 
12
    ret = av_frame_get_buffer(data->vframe, base_get_alignment());
13
    if (ret < 0) {
14
@@ -262,10 +260,8 @@
15
    context->color_primaries = data->config.color_primaries;
16
    context->color_trc = data->config.color_trc;
17
    context->colorspace = data->config.colorspace;
18
-   context->chroma_sample_location =
19
-       (data->config.colorspace == AVCOL_SPC_BT2020_NCL)
20
-           ? AVCHROMA_LOC_TOPLEFT
21
-           : AVCHROMA_LOC_LEFT;
22
+   context->chroma_sample_location = determine_chroma_location(
23
+       closest_format, data->config.colorspace);
24
    context->thread_count = 0;
25
 
26
    data->video->time_base = context->time_base;
27
@@ -294,6 +290,7 @@
28
    AVCodecContext *const context = data->audio_infosidx.ctx;
29
    char **opts = strlist_split(data->config.audio_settings, ' ', false);
30
    int ret;
31
+   int channels;
32
 
33
    if (opts) {
34
        parse_params(context, opts);
35
@@ -308,11 +305,13 @@
36
    }
37
 
38
    data->aframeidx->format = context->sample_fmt;
39
-   data->aframeidx->channels = context->channels;
40
 #if LIBAVUTIL_VERSION_INT < AV_VERSION_INT(57, 24, 100)
41
+   data->aframeidx->channels = context->channels;
42
    data->aframeidx->channel_layout = context->channel_layout;
43
+   channels = context->channels;
44
 #else
45
    data->aframeidx->ch_layout = context->ch_layout;
46
+   channels = context->ch_layout.nb_channels;
47
 #endif
48
    data->aframeidx->sample_rate = context->sample_rate;
49
    context->strict_std_compliance = -2;
50
@@ -327,7 +326,7 @@
51
 
52
    data->frame_size = context->frame_size ? context->frame_size : 1024;
53
 
54
-   ret = av_samples_alloc(data->samplesidx, NULL, context->channels,
55
+   ret = av_samples_alloc(data->samplesidx, NULL, channels,
56
                   data->frame_size, context->sample_fmt, 0);
57
    if (ret < 0) {
58
        ffmpeg_log_error(LOG_WARNING, data,
59
@@ -349,6 +348,7 @@
60
    AVCodecContext *context;
61
    AVStream *stream;
62
    struct obs_audio_info aoi;
63
+   int channels;
64
 
65
    if (!obs_get_audio_info(&aoi)) {
66
        ffmpeg_log_error(LOG_WARNING, data, "No active audio");
67
@@ -367,17 +367,20 @@
68
 #endif
69
    context->bit_rate = (int64_t)data->config.audio_bitrate * 1000;
70
    context->time_base = (AVRational){1, aoi.samples_per_sec};
71
+#if LIBAVUTIL_VERSION_INT < AV_VERSION_INT(57, 24, 100)
72
    context->channels = get_audio_channels(aoi.speakers);
73
+#endif
74
+   channels = get_audio_channels(aoi.speakers);
75
+
76
    context->sample_rate = aoi.samples_per_sec;
77
 #if LIBAVCODEC_VERSION_INT < AV_VERSION_INT(59, 24, 100)
78
-   context->channel_layout =
79
-       av_get_default_channel_layout(context->channels);
80
+   context->channel_layout = av_get_default_channel_layout(channels);
81
 
82
    //avutil default channel layout for 5 channels is 5.0 ; fix for 4.1
83
    if (aoi.speakers == SPEAKERS_4POINT1)
84
        context->channel_layout = av_get_channel_layout("4.1");
85
 #else
86
-   av_channel_layout_default(&context->ch_layout, context->channels);
87
+   av_channel_layout_default(&context->ch_layout, channels);
88
    if (aoi.speakers == SPEAKERS_4POINT1)
89
        context->ch_layout = (AVChannelLayout)AV_CHANNEL_LAYOUT_4POINT1;
90
 #endif
91
@@ -865,14 +868,20 @@
92
 
93
    AVPacket *packet = NULL;
94
    int ret, got_packet;
95
-   size_t total_size = data->frame_size * block_size * context->channels;
96
+   int channels;
97
+#if LIBAVUTIL_VERSION_INT >= AV_VERSION_INT(57, 24, 100)
98
+   channels = context->ch_layout.nb_channels;
99
+#else
100
+   channels = context->channels;
101
+#endif
102
+   size_t total_size = data->frame_size * block_size * channels;
103
 
104
    data->aframeidx->nb_samples = data->frame_size;
105
    data->aframeidx->pts = av_rescale_q(
106
        data->total_samplesidx, (AVRational){1, context->sample_rate},
107
        context->time_base);
108
 
109
-   ret = avcodec_fill_audio_frame(data->aframeidx, context->channels,
110
+   ret = avcodec_fill_audio_frame(data->aframeidx, channels,
111
                       context->sample_fmt,
112
                       data->samplesidx0, (int)total_size,
113
                       1);
114
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-output.h -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-output.h Changed
11
 
1
@@ -37,6 +37,9 @@
2
    int width;
3
    int height;
4
    int frame_size; // audio frame size
5
+   const char *username;
6
+   const char *password;
7
+   const char *key;
8
 };
9
 
10
 struct ffmpeg_audio_info {
11
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-rist.h -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-rist.h Changed
157
 
1
@@ -44,7 +44,8 @@
2
    int fifo_shift;
3
    bool overrun_nonfatal;
4
    char *secret;
5
-
6
+   char *username;
7
+   char *password;
8
    struct rist_logging_settings logging_settings;
9
    struct rist_peer_config peer_config;
10
 
11
@@ -100,14 +101,19 @@
12
 {
13
    RISTContext *s = h->priv_data;
14
    int ret = 0;
15
-
16
+   if (s->secret)
17
+       bfree(s->secret);
18
+   if (s->username)
19
+       bfree(s->username);
20
+   if (s->password)
21
+       bfree(s->password);
22
    s->peer = NULL;
23
 
24
    if (s->ctx)
25
        ret = rist_destroy(s->ctx);
26
    if (ret < 0) {
27
        blog(LOG_ERROR,
28
-            "obs-ffmpeg mpegts muxer / librist : failed to close properly %s\n",
29
+            "obs-ffmpeg mpegts muxer / librist: Failed to close properly %s",
30
             h->url);
31
        return -1;
32
    }
33
@@ -122,11 +128,15 @@
34
    rist_log(&s->logging_settings, RIST_LOG_INFO, "%s\n",
35
         stats_container->stats_json);
36
    if (stats_container->stats_type == RIST_STATS_SENDER_PEER) {
37
+       blog(LOG_INFO, "---------------------------------");
38
        blog(LOG_DEBUG,
39
-            "obs-ffmpeg mpegts muxer / librist RIST STATS\n\n"
40
-            "bandwidth %.3f Mbps\npackets sent %llu\npkts received %llu\n"
41
-            "pkts retransmitted %llu\nquality (pkt sent over sent+retransmitted+skipped) %.2f \n"
42
-            "rtt %" PRIu32 " ms\n\n",
43
+            "obs-ffmpeg mpegts muxer / librist: Session Summary\n"
44
+            "\tbandwidth %.3f Mbps\n"
45
+            "\tpackets sent %lu\n"
46
+            "\tpkts received %lu\n"
47
+            "\tpkts retransmitted %lu\n"
48
+            "\tquality (pkt sent over sent+retransmitted+skipped) %.2f\n"
49
+            "\trtt %" PRIu32 " ms\n",
50
             (double)(stats_container->stats.sender_peer.bandwidth) /
51
                 1000000.0,
52
             stats_container->stats.sender_peer.sent,
53
@@ -150,7 +160,6 @@
54
    s->packet_size = 1316;
55
    s->log_level = RIST_LOG_INFO;
56
    s->encryption = 0;
57
-   s->secret = NULL;
58
    s->overrun_nonfatal = 0;
59
    s->fifo_shift = FF_LIBRIST_FIFO_DEFAULT_SHIFT;
60
    s->logging_settings =
61
@@ -161,12 +170,12 @@
62
                   NULL);
63
    if (ret < 0) {
64
        blog(LOG_ERROR,
65
-            "obs-ffmpeg mpegts muxer / librist : Failed to initialize logging settings.");
66
+            "obs-ffmpeg mpegts muxer / librist: Failed to initialize logging settings");
67
        return OBS_OUTPUT_CONNECT_FAILED;
68
    }
69
 
70
    blog(LOG_INFO,
71
-        "obs-ffmpeg mpegts muxer / librist : \n librist version %s & API = %s .",
72
+        "obs-ffmpeg mpegts muxer / librist: libRIST version: %s, API: %s",
73
         librist_version(), librist_api_version());
74
 
75
    h->max_packet_size = s->packet_size;
76
@@ -174,14 +183,14 @@
77
 
78
    if (ret < 0) {
79
        blog(LOG_ERROR,
80
-            "obs-ffmpeg mpegts muxer / librist : failed to create a sender \n");
81
+            "obs-ffmpeg mpegts muxer / librist: Failed to create a sender");
82
        goto err;
83
    }
84
 
85
    ret = rist_peer_config_defaults_set(peer_config);
86
    if (ret < 0) {
87
        blog(LOG_ERROR,
88
-            "obs-ffmpeg mpegts muxer / librist : failed to set peer config defaults.\n");
89
+            "obs-ffmpeg mpegts muxer / librist: Failed to set peer config defaults");
90
        goto err;
91
    }
92
 
93
@@ -193,7 +202,7 @@
94
 #endif
95
    if (ret < 0) {
96
        blog(LOG_ERROR,
97
-            "obs-ffmpeg mpegts muxer / librist : failed to parse the url %s\n",
98
+            "obs-ffmpeg mpegts muxer / librist: Failed to parse url: %s",
99
             uri);
100
        librist_close(h);
101
        return OBS_OUTPUT_INVALID_STREAM;
102
@@ -203,7 +212,7 @@
103
        ((peer_config->key_size == 128 || peer_config->key_size == 256) &&
104
         !peer_config->secret0)) {
105
        blog(LOG_ERROR,
106
-            "secret is mandatory if encryption is enabled\n");
107
+            "obs-ffmpeg mpegts muxer / librist: Secret is mandatory if encryption is enabled");
108
        librist_close(h);
109
        return OBS_OUTPUT_INVALID_STREAM;
110
    }
111
@@ -219,26 +228,30 @@
112
        peer_config->recovery_length_min = s->buffer_size;
113
        peer_config->recovery_length_max = s->buffer_size;
114
    }
115
+   if (s->username && peer_config->srp_username0 == 0)
116
+       av_strlcpy(peer_config->srp_username, s->username,
117
+              RIST_MAX_STRING_LONG);
118
+   if (s->password && peer_config->srp_password0 == 0)
119
+       av_strlcpy(peer_config->srp_password, s->password,
120
+              RIST_MAX_STRING_LONG);
121
 
122
    ret = rist_peer_create(s->ctx, &s->peer, &s->peer_config);
123
    if (ret < 0) {
124
        blog(LOG_ERROR,
125
-            "obs-ffmpeg mpegts muxer / librist : failed to create a peer. \n");
126
+            "obs-ffmpeg mpegts muxer / librist: Failed to create a peer.");
127
        goto err;
128
    }
129
 
130
    ret = rist_start(s->ctx);
131
    if (ret < 0) {
132
        blog(LOG_ERROR,
133
-            "obs-ffmpeg mpegts muxer / librist : rist failed to start \n");
134
+            "obs-ffmpeg mpegts muxer / librist: RIST failed to start");
135
        goto err;
136
    }
137
    if (rist_stats_callback_set(s->ctx, s->statsinterval, cb_stats,
138
-                   (void *)s) == -1) {
139
+                   (void *)s) == -1)
140
        rist_log(&s->logging_settings, RIST_LOG_ERROR,
141
-            "Could not enable stats callback\n");
142
-       ;
143
-   }
144
+            "Could not enable stats callback");
145
    return 0;
146
 
147
 err:
148
@@ -260,7 +273,7 @@
149
    ret = rist_sender_data_write(s->ctx, &data_block);
150
    if (ret < 0) {
151
        blog(LOG_WARNING,
152
-            "obs-ffmpeg mpegts muxer / librist : failed to send data of size %i bytes",
153
+            "obs-ffmpeg mpegts muxer / librist: Failed to send %i bytes",
154
             size);
155
        return risterr2ret(ret);
156
    }
157
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-srt.h -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-srt.h Changed
201
 
1
@@ -80,12 +80,24 @@
2
 
3
 static int libsrt_neterrno(URLContext *h)
4
 {
5
+   SRTContext *s = (SRTContext *)h->priv_data;
6
    int os_errno;
7
    int err = srt_getlasterror(&os_errno);
8
-   blog(LOG_ERROR, "obs-ffmpeg mpegts muxer / libsrt : %s\n",
9
+   blog(LOG_ERROR, "obs-ffmpeg mpegts muxer / libsrt: %s",
10
         srt_getlasterror_str());
11
    if (err == SRT_EASYNCRCV || err == SRT_EASYNCSND)
12
        return AVERROR(EAGAIN);
13
+   if (err == SRT_ECONNREJ) {
14
+       int errj = srt_getrejectreason(s->fd);
15
+       if (errj == SRT_REJ_BADSECRET)
16
+           blog(LOG_ERROR,
17
+                "obs-ffmpeg mpegts muxer / libsrt: Wrong password");
18
+       else
19
+           blog(LOG_ERROR,
20
+                "obs-ffmpeg mpegts muxer / libsrt: Connection rejected, %s",
21
+                srt_rejectreason_str(errj));
22
+   }
23
+
24
    return os_errno ? AVERROR(os_errno) : AVERROR_UNKNOWN;
25
 }
26
 
27
@@ -94,7 +106,7 @@
28
 {
29
    if (srt_getsockopt(fd, 0, optname, optval, optlen) < 0) {
30
        blog(LOG_INFO,
31
-            "obs-ffmpeg mpegts muxer / libsrt : failed to get option %s on socket: %s\n",
32
+            "obs-ffmpeg mpegts muxer / libsrt: Failed to get option %s on socket: %s",
33
             optnamestr, srt_getlasterror_str());
34
        return AVERROR(EIO);
35
    }
36
@@ -139,6 +151,21 @@
37
        ret = srt_epoll_wait(eid, ready, &len, error, &errlen,
38
                     POLLING_TIME, 0, 0, 0, 0);
39
    }
40
+   if (len == 1 && errlen == 1) {
41
+       /* Socket reported in wsock AND rsock signifies an error. */
42
+       int reason = srt_getrejectreason(*ready);
43
+
44
+       if (reason == SRT_REJ_BADSECRET || reason == SRT_REJ_UNSECURE ||
45
+           reason == SRT_REJ_TIMEOUT) {
46
+           blog(LOG_ERROR,
47
+                "obs-ffmpeg mpegts muxer / libsrt: Connection rejected, wrong password or invalid URL");
48
+           return OBS_OUTPUT_INVALID_STREAM;
49
+       } else {
50
+           blog(LOG_ERROR,
51
+                "obs-ffmpeg mpegts muxer / libsrt: Connection rejected, %s",
52
+                srt_rejectreason_str(reason));
53
+       }
54
+   }
55
    if (ret < 0) {
56
        if (srt_getlasterror(NULL) == SRT_ETIMEOUT)
57
            ret = AVERROR(EAGAIN);
58
@@ -190,7 +217,7 @@
59
    if (srt_setsockopt(fd, SOL_SOCKET, SRTO_REUSEADDR, &reuse,
60
               sizeof(reuse))) {
61
        blog(LOG_WARNING,
62
-            "obs-ffmpeg mpegts muxer / libsrt : setsockopt(SRTO_REUSEADDR) failed\n");
63
+            "obs-ffmpeg mpegts muxer / libsrt: setsockopt(SRTO_REUSEADDR) failed");
64
    }
65
    if (srt_bind(fd, addr, addrlen))
66
        return libsrt_neterrno(h);
67
@@ -208,12 +235,12 @@
68
        return libsrt_neterrno(h);
69
    if (libsrt_socket_nonblock(ret, 1) < 0)
70
        blog(LOG_DEBUG,
71
-            "obs-ffmpeg mpegts muxer / libsrt : libsrt_socket_nonblock failed\n");
72
+            "obs-ffmpeg mpegts muxer / libsrt: libsrt_socket_nonblock failed");
73
    if (!libsrt_getsockopt(h, ret, SRTO_STREAMID, "SRTO_STREAMID", streamid,
74
                   &streamid_len))
75
        /* Note: returned streamid_len doesn't count the terminating null character */
76
        blog(LOG_INFO,
77
-            "obs-ffmpeg mpegts muxer / libsrt : accept streamid %s, length %d\n",
78
+            "obs-ffmpeg mpegts muxer / libsrt: Accept streamid %s, length %d",
79
             streamid, streamid_len);
80
 
81
    return ret;
82
@@ -233,11 +260,11 @@
83
    if (ret < 0) {
84
        if (will_try_next) {
85
            blog(LOG_WARNING,
86
-                "obs-ffmpeg mpegts muxer / libsrt : Connection to %s failed (%s), trying next address\n",
87
+                "obs-ffmpeg mpegts muxer / libsrt: Connection to %s failed (%s), trying next address",
88
                 h->url, av_err2str(ret));
89
        } else {
90
            blog(LOG_ERROR,
91
-                "obs-ffmpeg mpegts muxer / libsrt : Connection to %s failed: %s\n",
92
+                "obs-ffmpeg mpegts muxer / libsrt: Connection to %s failed: %s",
93
                 h->url, av_err2str(ret));
94
        }
95
    }
96
@@ -250,7 +277,7 @@
97
 {
98
    if (srt_setsockopt(fd, 0, optname, optval, optlen) < 0) {
99
        blog(LOG_ERROR,
100
-            "obs-ffmpeg mpegts muxer / libsrt : failed to set option %s on socket: %s\n",
101
+            "obs-ffmpeg mpegts muxer / libsrt: Failed to set option %s on socket: %s",
102
             optnamestr, srt_getlasterror_str());
103
        return AVERROR(EIO);
104
    }
105
@@ -434,7 +461,7 @@
106
        return AVERROR(EINVAL);
107
    if (port <= 0 || port >= 65536) {
108
        blog(LOG_ERROR,
109
-            "obs-ffmpeg mpegts muxer / libsrt : Port missing in uri\n");
110
+            "obs-ffmpeg mpegts muxer / libsrt: Port missing in uri");
111
        return OBS_OUTPUT_CONNECT_FAILED;
112
    }
113
    p = strchr(uri, '?');
114
@@ -457,7 +484,7 @@
115
    ret = getaddrinfo(hostname0 ? hostname : NULL, portstr, &hints, &ai);
116
    if (ret) {
117
        blog(LOG_ERROR,
118
-            "obs-ffmpeg mpegts muxer / libsrt : Failed to resolve hostname %s: %s\n",
119
+            "obs-ffmpeg mpegts muxer / libsrt: Failed to resolve hostname %s: %s",
120
             hostname, gai_strerror(ret));
121
        return OBS_OUTPUT_CONNECT_FAILED;
122
    }
123
@@ -490,7 +517,7 @@
124
    }
125
    if (libsrt_socket_nonblock(fd, 1) < 0)
126
        blog(LOG_DEBUG,
127
-            "obs-ffmpeg mpegts muxer / libsrt : libsrt_socket_nonblock failed\n");
128
+            "obs-ffmpeg mpegts muxer / libsrt: libsrt_socket_nonblock failed");
129
 
130
    ret = write_eid = libsrt_epoll_create(h, fd, 1);
131
    if (ret < 0)
132
@@ -574,7 +601,6 @@
133
    s->payload_size = SRT_LIVE_DEFAULT_PAYLOAD_SIZE;
134
    s->maxbw = -1;
135
    s->pbkeylen = -1;
136
-   s->passphrase = NULL;
137
    s->mss = -1;
138
    s->ffs = -1;
139
    s->ipttl = -1;
140
@@ -592,7 +618,6 @@
141
    s->rcvbuf = -1;
142
    s->lossmaxttl = -1;
143
    s->minversion = -1;
144
-   s->streamid = NULL;
145
    s->smoother = NULL;
146
    s->messageapi = -1;
147
    s->transtype = SRTT_LIVE;
148
@@ -609,11 +634,11 @@
149
 
150
    if (srt_startup() < 0) {
151
        blog(LOG_ERROR,
152
-            "obs-ffmpeg mpegts muxer / libsrt : libsrt failed to load.\n");
153
+            "obs-ffmpeg mpegts muxer / libsrt: libsrt failed to load");
154
        return OBS_OUTPUT_CONNECT_FAILED;
155
    } else {
156
        blog(LOG_INFO,
157
-            "obs-ffmpeg mpegts muxer / libsrt : libsrt v.%s loaded.\n",
158
+            "obs-ffmpeg mpegts muxer / libsrt: libsrt version %s loaded",
159
             SRT_VERSION_STRING);
160
    }
161
    libsrt_set_defaults(s);
162
@@ -801,8 +826,8 @@
163
 #endif
164
        if (time > (s->time + 60.0)) {
165
            srt_bistats(s->fd, &perf, 0, 1);
166
-           blog(LOG_INFO,
167
-                "obs-ffmpeg mpegts muxer / libsrt : rtt %.2f ms, link bandwidth %.1f Mbps\n",
168
+           blog(LOG_DEBUG,
169
+                "obs-ffmpeg mpegts muxer / libsrt: RTT %.2f ms, Link Bandwidth %.1f Mbps",
170
                 perf.msRTT, perf.mbpsBandwidth);
171
            s->time = time;
172
        }
173
@@ -814,15 +839,21 @@
174
 static int libsrt_close(URLContext *h)
175
 {
176
    SRTContext *s = (SRTContext *)h->priv_data;
177
-
178
+   if (s->streamid)
179
+       av_freep(&s->streamid);
180
+   if (s->passphrase)
181
+       av_freep(&s->passphrase);
182
    /* Log stream stats. */
183
    SRT_TRACEBSTATS perf;
184
    srt_bstats(s->fd, &perf, 1);
185
+   blog(LOG_INFO, "---------------------------------");
186
    blog(LOG_INFO,
187
-        "obs-ffmpeg mpegts muxer / libsrt : Stream stats\n\n"
188
-        "time elapsed %.1f sec\nmean speed %.1f Mbp\n"
189
-        "total bytes sent %.1f MB\nbytes retransmitted %.1f %%\n"
190
-        "bytes dropped %.1f %%\n\n",
191
+        "obs-ffmpeg mpegts muxer / libsrt: Session Summary\n"
192
+        "\ttime elapsed %.1f sec\n"
193
+        "\tmean speed %.1f Mbp\n"
194
+        "\ttotal bytes sent %.1f MB\n"
195
+        "\tbytes retransmitted %.1f %%\n"
196
+        "\tbytes dropped %.1f %%\n",
197
         (double)perf.msTimeStamp / 1000.0, perf.mbpsSendRate,
198
         (double)perf.byteSentTotal / 1000000.0,
199
         perf.byteSentTotal
200
@@ -835,13 +866,14 @@
201
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-vaapi.c -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-vaapi.c Changed
201
 
1
@@ -38,10 +38,11 @@
2
 
3
 #include <pci/pci.h>
4
 
5
+#include "vaapi-utils.h"
6
 #include "obs-ffmpeg-formats.h"
7
 
8
 #define do_log(level, format, ...)                          \
9
-   blog(level, "FFMPEG VAAPI encoder: '%s' " format, \
10
+   blog(level, "FFmpeg VAAPI encoder: '%s' " format, \
11
         obs_encoder_get_name(enc->encoder), ##__VA_ARGS__)
12
 
13
 #define warn(format, ...) do_log(LOG_WARNING, format, ##__VA_ARGS__)
14
@@ -77,7 +78,7 @@
15
 static const char *vaapi_getname(void *unused)
16
 {
17
    UNUSED_PARAMETER(unused);
18
-   return "FFMPEG VAAPI H.264";
19
+   return "FFmpeg VAAPI H.264";
20
 }
21
 
22
 static inline bool valid_format(enum video_format format)
23
@@ -144,6 +145,7 @@
24
    enc->vframe->height = enc->context->height;
25
    enc->vframe->colorspace = enc->context->colorspace;
26
    enc->vframe->color_range = enc->context->color_range;
27
+   enc->vframe->chroma_location = enc->context->chroma_sample_location;
28
 
29
    ret = av_frame_get_buffer(enc->vframe, base_get_alignment());
30
    if (ret < 0) {
31
@@ -250,30 +252,37 @@
32
    enc->context->height = obs_encoder_get_height(enc->encoder);
33
 
34
    enc->context->time_base = (AVRational){voi->fps_den, voi->fps_num};
35
-   enc->context->pix_fmt = obs_to_ffmpeg_video_format(info.format);
36
+   const enum AVPixelFormat pix_fmt =
37
+       obs_to_ffmpeg_video_format(info.format);
38
+   enc->context->pix_fmt = pix_fmt;
39
    enc->context->color_range = info.range == VIDEO_RANGE_FULL
40
                        ? AVCOL_RANGE_JPEG
41
                        : AVCOL_RANGE_MPEG;
42
 
43
+   enum AVColorSpace colorspace = AVCOL_SPC_UNSPECIFIED;
44
    switch (info.colorspace) {
45
    case VIDEO_CS_601:
46
        enc->context->color_trc = AVCOL_TRC_SMPTE170M;
47
        enc->context->color_primaries = AVCOL_PRI_SMPTE170M;
48
-       enc->context->colorspace = AVCOL_SPC_SMPTE170M;
49
+       colorspace = AVCOL_SPC_SMPTE170M;
50
        break;
51
    case VIDEO_CS_DEFAULT:
52
    case VIDEO_CS_709:
53
        enc->context->color_trc = AVCOL_TRC_BT709;
54
        enc->context->color_primaries = AVCOL_PRI_BT709;
55
-       enc->context->colorspace = AVCOL_SPC_BT709;
56
+       colorspace = AVCOL_SPC_BT709;
57
        break;
58
    case VIDEO_CS_SRGB:
59
        enc->context->color_trc = AVCOL_TRC_IEC61966_2_1;
60
        enc->context->color_primaries = AVCOL_PRI_BT709;
61
-       enc->context->colorspace = AVCOL_SPC_BT709;
62
+       colorspace = AVCOL_SPC_BT709;
63
        break;
64
    }
65
 
66
+   enc->context->colorspace = colorspace;
67
+   enc->context->chroma_sample_location =
68
+       determine_chroma_location(pix_fmt, colorspace);
69
+
70
    if (keyint_sec > 0) {
71
        enc->context->gop_size =
72
            keyint_sec * voi->fps_num / voi->fps_den;
73
@@ -511,8 +520,9 @@
74
 
75
 static void vaapi_defaults(obs_data_t *settings)
76
 {
77
-   obs_data_set_default_string(settings, "vaapi_device",
78
-                   "/dev/dri/renderD128");
79
+   const char *device = vaapi_get_h264_default_device();
80
+
81
+   obs_data_set_default_string(settings, "vaapi_device", device);
82
    obs_data_set_default_int(settings, "profile",
83
                 FF_PROFILE_H264_CONSTRAINED_BASELINE);
84
    obs_data_set_default_int(settings, "level", 40);
85
@@ -520,9 +530,67 @@
86
    obs_data_set_default_int(settings, "keyint_sec", 0);
87
    obs_data_set_default_int(settings, "bf", 0);
88
    obs_data_set_default_int(settings, "rendermode", 0);
89
-   obs_data_set_default_string(settings, "rate_control", "CBR");
90
    obs_data_set_default_int(settings, "qp", 20);
91
    obs_data_set_default_int(settings, "maxrate", 0);
92
+
93
+   int drm_fd = -1;
94
+   VADisplay va_dpy = vaapi_open_device(&drm_fd, device, "vaapi_defaults");
95
+   if (!va_dpy)
96
+       return;
97
+
98
+   if (vaapi_device_rc_supported(VAProfileH264ConstrainedBaseline, va_dpy,
99
+                     VA_RC_CBR, device))
100
+       obs_data_set_default_string(settings, "rate_control", "CBR");
101
+   else if (vaapi_device_rc_supported(VAProfileH264ConstrainedBaseline,
102
+                      va_dpy, VA_RC_VBR, device))
103
+       obs_data_set_default_string(settings, "rate_control", "VBR");
104
+   else
105
+       obs_data_set_default_string(settings, "rate_control", "CQP");
106
+
107
+   vaapi_close_device(&drm_fd, va_dpy);
108
+}
109
+
110
+static bool vaapi_device_modified(obs_properties_t *ppts, obs_property_t *p,
111
+                 obs_data_t *settings)
112
+{
113
+   UNUSED_PARAMETER(p);
114
+
115
+   const char *device = obs_data_get_string(settings, "vaapi_device");
116
+   int drm_fd = -1;
117
+   VADisplay va_dpy =
118
+       vaapi_open_device(&drm_fd, device, "vaapi_device_modified");
119
+   int profile = obs_data_get_int(settings, "profile");
120
+   obs_property_t *rc_p = obs_properties_get(ppts, "rate_control");
121
+
122
+   obs_property_list_clear(rc_p);
123
+
124
+   if (!va_dpy || !vaapi_display_h264_supported(va_dpy, device))
125
+       goto fail;
126
+
127
+   switch (profile) {
128
+   case FF_PROFILE_H264_CONSTRAINED_BASELINE:
129
+       profile = VAProfileH264ConstrainedBaseline;
130
+       break;
131
+   case FF_PROFILE_H264_MAIN:
132
+       profile = VAProfileH264Main;
133
+       break;
134
+   case FF_PROFILE_H264_HIGH:
135
+       profile = VAProfileH264High;
136
+       break;
137
+   }
138
+
139
+   if (vaapi_device_rc_supported(profile, va_dpy, VA_RC_CBR, device))
140
+       obs_property_list_add_string(rc_p, "CBR (default)", "CBR");
141
+
142
+   if (vaapi_device_rc_supported(profile, va_dpy, VA_RC_VBR, device))
143
+       obs_property_list_add_string(rc_p, "VBR", "VBR");
144
+
145
+   if (vaapi_device_rc_supported(profile, va_dpy, VA_RC_CQP, device))
146
+       obs_property_list_add_string(rc_p, "CQP", "CQP");
147
+
148
+fail:
149
+   vaapi_close_device(&drm_fd, va_dpy);
150
+   return true;
151
 }
152
 
153
 static bool rate_control_modified(obs_properties_t *ppts, obs_property_t *p,
154
@@ -617,6 +685,10 @@
155
                bool name_found = get_device_name_from_pci(
156
                    pacc, pci_slot, namebuf,
157
                    sizeof(namebuf));
158
+
159
+               if (!vaapi_device_h264_supported(path))
160
+                   continue;
161
+
162
                if (!name_found)
163
                    obs_property_list_add_string(list, path,
164
                                     path);
165
@@ -629,12 +701,21 @@
166
        os_closedir(by_path_dir);
167
    }
168
    if (obs_property_list_item_count(list) == 0) {
169
-       char path32 = "/dev/dri/renderD1";
170
+       char path32;
171
        for (int i = 28;; i++) {
172
-           sprintf(path, "/dev/dri/renderD1%d", i);
173
+           snprintf(path, sizeof(path), "/dev/dri/renderD1%d", i);
174
            if (access(path, F_OK) == 0) {
175
-               char card128 = "Card: ";
176
-               sprintf(card, "Card%d: %s", i - 28, path);
177
+               char card128;
178
+               int ret = snprintf(card, sizeof(card),
179
+                          "Card%d: %s", i - 28, path);
180
+               if (ret >= (int)sizeof(card))
181
+                   blog(LOG_DEBUG,
182
+                        "obs-ffmpeg-vaapi: A format truncation may have occurred."
183
+                        " This can be ignored since it is quite improbable.");
184
+
185
+               if (!vaapi_device_h264_supported(path))
186
+                   continue;
187
+
188
                obs_property_list_add_string(list, card, path);
189
            } else {
190
                break;
191
@@ -642,6 +723,8 @@
192
        }
193
    }
194
 
195
+   obs_property_set_modified_callback(list, vaapi_device_modified);
196
+
197
    list = obs_properties_add_list(props, "profile",
198
                       obs_module_text("Profile"),
199
                       OBS_COMBO_TYPE_LIST,
200
@@ -651,6 +734,8 @@
201
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-video-encoders.c -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg-video-encoders.c Changed
61
 
1
@@ -68,46 +68,51 @@
2
                 const char *ffmpeg_opts)
3
 {
4
    const int rate = bitrate * 1000;
5
+   const enum AVPixelFormat pix_fmt =
6
+       obs_to_ffmpeg_video_format(info->format);
7
    enc->context->bit_rate = rate;
8
    enc->context->rc_buffer_size = rate;
9
    enc->context->width = obs_encoder_get_width(enc->encoder);
10
    enc->context->height = obs_encoder_get_height(enc->encoder);
11
    enc->context->time_base = (AVRational){voi->fps_den, voi->fps_num};
12
-   enc->context->pix_fmt = obs_to_ffmpeg_video_format(info->format);
13
+   enc->context->pix_fmt = pix_fmt;
14
    enc->context->color_range = info->range == VIDEO_RANGE_FULL
15
                        ? AVCOL_RANGE_JPEG
16
                        : AVCOL_RANGE_MPEG;
17
 
18
+   enum AVColorSpace colorspace = AVCOL_SPC_UNSPECIFIED;
19
    switch (info->colorspace) {
20
    case VIDEO_CS_601:
21
        enc->context->color_primaries = AVCOL_PRI_SMPTE170M;
22
        enc->context->color_trc = AVCOL_TRC_SMPTE170M;
23
-       enc->context->colorspace = AVCOL_SPC_SMPTE170M;
24
+       colorspace = AVCOL_SPC_SMPTE170M;
25
        break;
26
    case VIDEO_CS_DEFAULT:
27
    case VIDEO_CS_709:
28
        enc->context->color_primaries = AVCOL_PRI_BT709;
29
        enc->context->color_trc = AVCOL_TRC_BT709;
30
-       enc->context->colorspace = AVCOL_SPC_BT709;
31
+       colorspace = AVCOL_SPC_BT709;
32
        break;
33
    case VIDEO_CS_SRGB:
34
        enc->context->color_primaries = AVCOL_PRI_BT709;
35
        enc->context->color_trc = AVCOL_TRC_IEC61966_2_1;
36
-       enc->context->colorspace = AVCOL_SPC_BT709;
37
+       colorspace = AVCOL_SPC_BT709;
38
        break;
39
    case VIDEO_CS_2100_PQ:
40
        enc->context->color_primaries = AVCOL_PRI_BT2020;
41
        enc->context->color_trc = AVCOL_TRC_SMPTE2084;
42
-       enc->context->colorspace = AVCOL_SPC_BT2020_NCL;
43
-       enc->context->chroma_sample_location = AVCHROMA_LOC_TOPLEFT;
44
+       colorspace = AVCOL_SPC_BT2020_NCL;
45
        break;
46
    case VIDEO_CS_2100_HLG:
47
        enc->context->color_primaries = AVCOL_PRI_BT2020;
48
        enc->context->color_trc = AVCOL_TRC_ARIB_STD_B67;
49
-       enc->context->colorspace = AVCOL_SPC_BT2020_NCL;
50
-       enc->context->chroma_sample_location = AVCHROMA_LOC_TOPLEFT;
51
+       colorspace = AVCOL_SPC_BT2020_NCL;
52
    }
53
 
54
+   enc->context->colorspace = colorspace;
55
+   enc->context->chroma_sample_location =
56
+       determine_chroma_location(pix_fmt, colorspace);
57
+
58
    if (keyint_sec)
59
        enc->context->gop_size =
60
            keyint_sec * voi->fps_num / voi->fps_den;
61
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg.c -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/obs-ffmpeg.c Changed
67
 
1
@@ -14,6 +14,13 @@
2
 #include "jim-nvenc.h"
3
 #endif
4
 
5
+#if !defined(_WIN32) && !defined(__APPLE__) && \
6
+   LIBAVUTIL_VERSION_INT >= AV_VERSION_INT(55, 27, 100)
7
+#include "vaapi-utils.h"
8
+
9
+#define LIBAVUTIL_VAAPI_AVAILABLE
10
+#endif
11
+
12
 OBS_DECLARE_MODULE()
13
 OBS_MODULE_USE_DEFAULT_LOCALE("obs-ffmpeg", "en-US")
14
 MODULE_EXPORT const char *obs_module_description(void)
15
@@ -36,10 +43,6 @@
16
 extern struct obs_encoder_info svt_av1_encoder_info;
17
 extern struct obs_encoder_info aom_av1_encoder_info;
18
 
19
-#if LIBAVUTIL_VERSION_INT >= AV_VERSION_INT(55, 27, 100)
20
-#define LIBAVUTIL_VAAPI_AVAILABLE
21
-#endif
22
-
23
 #ifdef LIBAVUTIL_VAAPI_AVAILABLE
24
 extern struct obs_encoder_info vaapi_encoder_info;
25
 #endif
26
@@ -320,10 +323,16 @@
27
 #endif
28
 
29
 #ifdef LIBAVUTIL_VAAPI_AVAILABLE
30
-static bool vaapi_supported(void)
31
+static bool h264_vaapi_supported(void)
32
 {
33
    const AVCodec *vaenc = avcodec_find_encoder_by_name("h264_vaapi");
34
-   return !!vaenc;
35
+
36
+   if (!vaenc)
37
+       return false;
38
+
39
+   /* NOTE: If default device is NULL, it means there is no device
40
+    * that support H264. */
41
+   return vaapi_get_h264_default_device() != NULL;
42
 }
43
 #endif
44
 
45
@@ -403,10 +412,18 @@
46
    amf_load();
47
 #endif
48
 
49
-#if !defined(_WIN32) && defined(LIBAVUTIL_VAAPI_AVAILABLE)
50
-   if (vaapi_supported()) {
51
-       blog(LOG_INFO, "FFMPEG VAAPI supported");
52
+#ifdef LIBAVUTIL_VAAPI_AVAILABLE
53
+   const char *libva_env = getenv("LIBVA_DRIVER_NAME");
54
+   if (!!libva_env)
55
+       blog(LOG_WARNING,
56
+            "LIBVA_DRIVER_NAME variable is set,"
57
+            " this could prevent FFmpeg VAAPI from working correctly");
58
+
59
+   if (h264_vaapi_supported()) {
60
+       blog(LOG_INFO, "FFmpeg VAAPI H264 encoding supported");
61
        obs_register_encoder(&vaapi_encoder_info);
62
+   } else {
63
+       blog(LOG_INFO, "FFmpeg VAAPI H264 encoding not supported");
64
    }
65
 #endif
66
 #endif
67
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/obs-nvenc-test/jim-nvenc-test.c -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/obs-nvenc-test/jim-nvenc-test.c Changed
93
 
1
@@ -1,4 +1,5 @@
2
 #include <stdbool.h>
3
+#include <stdlib.h>
4
 #include <stdio.h>
5
 
6
 #include "../external/nvEncodeAPI.h"
7
@@ -21,6 +22,8 @@
8
 };
9
 
10
 #define MAX_CAPS 10
11
+static uint32_t luid_count = 0;
12
+static uint64_t luid_orderMAX_CAPS = {0};
13
 static struct nvenc_info adapter_infoMAX_CAPS = {0};
14
 
15
 bool load_nvenc_lib(void)
16
@@ -37,11 +40,21 @@
17
    return func_ptr;
18
 }
19
 
20
+static inline uint32_t get_adapter_idx(uint32_t adapter_idx, LUID luid)
21
+{
22
+   for (uint32_t i = 0; i < luid_count; i++) {
23
+       if (luid_orderi == *(uint64_t *)&luid) {
24
+           return i;
25
+       }
26
+   }
27
+
28
+   return adapter_idx;
29
+}
30
+
31
 static bool get_adapter_caps(IDXGIFactory *factory, uint32_t adapter_idx)
32
 {
33
-   struct nvenc_info *caps = &adapter_infoadapter_idx;
34
+   struct nvenc_info *caps;
35
    IDXGIAdapter *adapter = NULL;
36
-   IDXGIOutput *output = NULL;
37
    ID3D11Device *device = NULL;
38
    ID3D11DeviceContext *context = NULL;
39
    GUID *guids = NULL;
40
@@ -58,15 +71,12 @@
41
    DXGI_ADAPTER_DESC desc;
42
    adapter->lpVtbl->GetDesc(adapter, &desc);
43
 
44
+   caps = &adapter_infoget_adapter_idx(adapter_idx, desc.AdapterLuid);
45
    if (desc.VendorId != NVIDIA_VENDOR_ID)
46
        return true;
47
 
48
    caps->is_nvidia = true;
49
 
50
-   hr = adapter->lpVtbl->EnumOutputs(adapter, 0, &output);
51
-   if (FAILED(hr))
52
-       goto finish;
53
-
54
    hr = D3D11CreateDevice(adapter, D3D_DRIVER_TYPE_UNKNOWN, NULL, 0, NULL,
55
                   0, D3D11_SDK_VERSION, &device, NULL, &context);
56
    if (FAILED(hr))
57
@@ -111,8 +121,6 @@
58
        context->lpVtbl->Release(context);
59
    if (device)
60
        device->lpVtbl->Release(device);
61
-   if (output)
62
-       output->lpVtbl->Release(output);
63
    if (adapter)
64
        adapter->lpVtbl->Release(adapter);
65
    return true;
66
@@ -172,7 +180,7 @@
67
    return 0;
68
 }
69
 
70
-int main(void)
71
+int main(int argc, char *argv)
72
 {
73
    IDXGIFactory *factory = NULL;
74
    HRESULT hr;
75
@@ -193,6 +201,17 @@
76
    if (!init_nvenc_internal())
77
        return 0;
78
 
79
+   /* --------------------------------------------------------- */
80
+   /* parse expected LUID order                                 */
81
+
82
+   luid_count = argc - 1;
83
+   for (int i = 1; i < argc; i++) {
84
+       luid_orderi - 1 = strtoull(argvi, NULL, 16);
85
+   }
86
+
87
+   /* --------------------------------------------------------- */
88
+   /* obtain adapter compatibility information                  */
89
+
90
    hr = CreateDXGIFactory1(&IID_IDXGIFactory1, (void **)&factory);
91
    if (FAILED(hr))
92
        return 0;
93
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/texture-amf-opts.hpp -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/texture-amf-opts.hpp Changed
152
 
1
@@ -17,25 +17,40 @@
2
 {
3
    bool avc = enc->codec == amf_codec_type::AVC;
4
    bool hevc = enc->codec == amf_codec_type::HEVC;
5
+   bool av1 = enc->codec == amf_codec_type::AV1;
6
 
7
    if (strcmp(opt->name, "g") == 0 || strcmp(opt->name, "keyint") == 0) {
8
 
9
        int val = atoi(opt->value);
10
-       if (enc->codec == amf_codec_type::AVC)
11
+       if (avc)
12
            set_avc_opt(IDR_PERIOD, val);
13
-       else
14
+       else if (hevc)
15
            set_hevc_opt(NUM_GOPS_PER_IDR, val);
16
+       else if (av1)
17
+           set_av1_opt(GOP_SIZE, val);
18
 
19
    } else if (strcmp(opt->name, "usage") == 0) {
20
 
21
        if (strcmp(opt->value, "transcoding") == 0) {
22
-           set_enum_opt(USAGE, TRANSCONDING);
23
+           set_enum_opt(USAGE, TRANSCODING);
24
        } else if (strcmp(opt->value, "ultralowlatency") == 0) {
25
-           set_enum_opt(USAGE, ULTRA_LOW_LATENCY);
26
+           if (avc)
27
+               set_avc_enum(USAGE, ULTRA_LOW_LATENCY);
28
+           else if (hevc)
29
+               set_hevc_enum(USAGE, ULTRA_LOW_LATENCY);
30
+           else
31
+               warn("Invalid value for %s: %s", opt->name,
32
+                    opt->value);
33
        } else if (strcmp(opt->value, "lowlatency") == 0) {
34
            set_enum_opt(USAGE, LOW_LATENCY);
35
        } else if (strcmp(opt->value, "webcam") == 0) {
36
-           set_enum_opt(USAGE, WEBCAM);
37
+           if (avc)
38
+               set_avc_enum(USAGE, WEBCAM);
39
+           else if (hevc)
40
+               set_hevc_enum(USAGE, WEBCAM);
41
+           else
42
+               warn("Invalid value for %s: %s", opt->name,
43
+                    opt->value);
44
        } else {
45
            warn("Invalid value for %s: %s", opt->name, opt->value);
46
        }
47
@@ -69,7 +84,12 @@
48
            val.erase(pos, 1);
49
 
50
        int level = std::stoi(val);
51
-       set_opt(PROFILE_LEVEL, level);
52
+       if (avc)
53
+           set_avc_opt(PROFILE_LEVEL, level);
54
+       else if (hevc)
55
+           set_hevc_opt(PROFILE_LEVEL, level);
56
+       else
57
+           warn("Invalid value for %s: %s", opt->name, opt->value);
58
 
59
    } else if (strcmp(opt->name, "quality") == 0) {
60
 
61
@@ -106,42 +126,82 @@
62
    } else if (strcmp(opt->name, "filler_data") == 0) {
63
 
64
        bool val = str_to_bool(opt->value);
65
-       set_opt(FILLER_DATA_ENABLE, val);
66
+       if (avc)
67
+           set_avc_opt(FILLER_DATA_ENABLE, val);
68
+       else if (hevc)
69
+           set_hevc_opt(FILLER_DATA_ENABLE, val);
70
+       else
71
+           warn("Invalid value for %s: %s", opt->name, opt->value);
72
 
73
    } else if (strcmp(opt->name, "vbaq") == 0) {
74
 
75
        bool val = str_to_bool(opt->value);
76
-       set_opt(ENABLE_VBAQ, val);
77
+       if (avc)
78
+           set_avc_opt(ENABLE_VBAQ, val);
79
+       else if (hevc)
80
+           set_hevc_opt(ENABLE_VBAQ, val);
81
+       else
82
+           warn("Invalid value for %s: %s", opt->name, opt->value);
83
 
84
    } else if (strcmp(opt->name, "qp_i") == 0) {
85
 
86
        int val = atoi(opt->value);
87
-       set_opt(QP_I, val);
88
+       if (avc)
89
+           set_avc_opt(QP_I, val);
90
+       else if (hevc)
91
+           set_hevc_opt(QP_I, val);
92
+       else
93
+           warn("Invalid value for %s: %s", opt->name, opt->value);
94
 
95
    } else if (strcmp(opt->name, "qp_p") == 0) {
96
 
97
        int val = atoi(opt->value);
98
-       set_opt(QP_P, val);
99
+       if (avc)
100
+           set_avc_opt(QP_P, val);
101
+       else if (hevc)
102
+           set_hevc_opt(QP_P, val);
103
+       else
104
+           warn("Invalid value for %s: %s", opt->name, opt->value);
105
 
106
    } else if (strcmp(opt->name, "me_half_pel") == 0) {
107
 
108
        bool val = str_to_bool(opt->value);
109
-       set_opt(MOTION_HALF_PIXEL, val);
110
+       if (avc)
111
+           set_avc_opt(MOTION_HALF_PIXEL, val);
112
+       else if (hevc)
113
+           set_hevc_opt(MOTION_HALF_PIXEL, val);
114
+       else
115
+           warn("Invalid value for %s: %s", opt->name, opt->value);
116
 
117
    } else if (strcmp(opt->name, "me_quarter_pel") == 0) {
118
 
119
        bool val = str_to_bool(opt->value);
120
-       set_opt(MOTION_QUARTERPIXEL, val);
121
+       if (avc)
122
+           set_avc_opt(MOTION_QUARTERPIXEL, val);
123
+       else if (hevc)
124
+           set_hevc_opt(MOTION_QUARTERPIXEL, val);
125
+       else
126
+           warn("Invalid value for %s: %s", opt->name, opt->value);
127
 
128
    } else if (strcmp(opt->name, "aud") == 0) {
129
 
130
        bool val = str_to_bool(opt->value);
131
-       set_opt(INSERT_AUD, val);
132
+       if (avc)
133
+           set_avc_opt(INSERT_AUD, val);
134
+       else if (hevc)
135
+           set_hevc_opt(INSERT_AUD, val);
136
+       else
137
+           warn("Invalid value for %s: %s", opt->name, opt->value);
138
 
139
    } else if (strcmp(opt->name, "max_au_size") == 0) {
140
 
141
        int val = atoi(opt->value);
142
-       set_opt(MAX_AU_SIZE, val);
143
+       if (avc)
144
+           set_avc_opt(MAX_AU_SIZE, val);
145
+       else if (hevc)
146
+           set_hevc_opt(MAX_AU_SIZE, val);
147
+       else
148
+           warn("Invalid value for %s: %s", opt->name, opt->value);
149
 
150
    } else if (avc && strcmp(opt->name, "preanalysis") == 0) {
151
 
152
obs-studio-28.1.2.tar.xz/plugins/obs-ffmpeg/texture-amf.cpp -> obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/texture-amf.cpp Changed
201
 
1
@@ -8,7 +8,7 @@
2
 #include <unordered_map>
3
 #include <cstdlib>
4
 #include <memory>
5
-#include <string>
6
+#include <sstream>
7
 #include <vector>
8
 #include <mutex>
9
 #include <deque>
10
@@ -16,6 +16,7 @@
11
 
12
 #include "external/AMF/include/components/VideoEncoderHEVC.h"
13
 #include "external/AMF/include/components/VideoEncoderVCE.h"
14
+#include "external/AMF/include/components/VideoEncoderAV1.h"
15
 #include "external/AMF/include/core/Factory.h"
16
 #include "external/AMF/include/core/Trace.h"
17
 
18
@@ -23,6 +24,7 @@
19
 #include <d3d11.h>
20
 #include <d3d11_1.h>
21
 
22
+#include <util/windows/device-enum.h>
23
 #include <util/windows/HRError.hpp>
24
 #include <util/windows/ComPtr.hpp>
25
 #include <util/platform.h>
26
@@ -63,6 +65,7 @@
27
    bool is_amd = false;
28
    bool supports_avc = false;
29
    bool supports_hevc = false;
30
+   bool supports_av1 = false;
31
 };
32
 
33
 /* ------------------------------------------------------------------------- */
34
@@ -80,6 +83,7 @@
35
 enum class amf_codec_type {
36
    AVC,
37
    HEVC,
38
+   AV1,
39
 };
40
 
41
 struct amf_base {
42
@@ -213,25 +217,34 @@
43
    set_amf_property(enc, AMF_VIDEO_ENCODER_##name, value)
44
 #define set_hevc_property(enc, name, value) \
45
    set_amf_property(enc, AMF_VIDEO_ENCODER_HEVC_##name, value)
46
+#define set_av1_property(enc, name, value) \
47
+   set_amf_property(enc, AMF_VIDEO_ENCODER_AV1_##name, value)
48
 
49
 #define get_avc_property(enc, name, value) \
50
    get_amf_property(enc, AMF_VIDEO_ENCODER_##name, value)
51
 #define get_hevc_property(enc, name, value) \
52
    get_amf_property(enc, AMF_VIDEO_ENCODER_HEVC_##name, value)
53
+#define get_av1_property(enc, name, value) \
54
+   get_amf_property(enc, AMF_VIDEO_ENCODER_AV1_##name, value)
55
 
56
 #define get_opt_name(name)                                              \
57
    ((enc->codec == amf_codec_type::AVC) ? AMF_VIDEO_ENCODER_##name \
58
-                        : AMF_VIDEO_ENCODER_HEVC_##name)
59
+    : (enc->codec == amf_codec_type::HEVC)                         \
60
+        ? AMF_VIDEO_ENCODER_HEVC_##name                        \
61
+        : AMF_VIDEO_ENCODER_AV1_##name)
62
 #define set_opt(name, value) set_amf_property(enc, get_opt_name(name), value)
63
 #define get_opt(name, value) get_amf_property(enc, get_opt_name(name), value)
64
 #define set_avc_opt(name, value) set_avc_property(enc, name, value)
65
 #define set_hevc_opt(name, value) set_hevc_property(enc, name, value)
66
+#define set_av1_opt(name, value) set_av1_property(enc, name, value)
67
 #define set_enum_opt(name, value) \
68
    set_amf_property(enc, get_opt_name(name), get_opt_name(name##_##value))
69
 #define set_avc_enum(name, value) \
70
    set_avc_property(enc, name, AMF_VIDEO_ENCODER_##name##_##value)
71
 #define set_hevc_enum(name, value) \
72
    set_hevc_property(enc, name, AMF_VIDEO_ENCODER_HEVC_##name##_##value)
73
+#define set_av1_enum(name, value) \
74
+   set_av1_property(enc, name, AMF_VIDEO_ENCODER_AV1_##name##_##value)
75
 
76
 /* ------------------------------------------------------------------------- */
77
 /* Implementation                                                            */
78
@@ -397,32 +410,78 @@
79
        mb_frame * (amf_int64)enc->fps_num / (amf_int64)enc->fps_den;
80
 }
81
 
82
+static inline int get_avc_preset(amf_base *enc, const char *preset);
83
+#if ENABLE_HEVC
84
+static inline int get_hevc_preset(amf_base *enc, const char *preset);
85
+#endif
86
+static inline int get_av1_preset(amf_base *enc, const char *preset);
87
+
88
+static inline int get_preset(amf_base *enc, const char *preset)
89
+{
90
+   if (enc->codec == amf_codec_type::AVC)
91
+       return get_avc_preset(enc, preset);
92
+
93
+#if ENABLE_HEVC
94
+   else if (enc->codec == amf_codec_type::HEVC)
95
+       return get_hevc_preset(enc, preset);
96
+
97
+#endif
98
+   else if (enc->codec == amf_codec_type::AV1)
99
+       return get_av1_preset(enc, preset);
100
+
101
+   return 0;
102
+}
103
+
104
+static inline void refresh_throughput_caps(amf_base *enc, const char *&preset)
105
+{
106
+   AMF_RESULT res = AMF_OK;
107
+   AMFCapsPtr caps;
108
+
109
+   set_opt(QUALITY_PRESET, get_preset(enc, preset));
110
+   res = enc->amf_encoder->GetCaps(&caps);
111
+   if (res == AMF_OK) {
112
+       caps->GetProperty(get_opt_name(CAP_MAX_THROUGHPUT),
113
+                 &enc->max_throughput);
114
+   }
115
+}
116
+
117
 static inline void check_preset_compatibility(amf_base *enc,
118
                          const char *&preset)
119
 {
120
-   /* 1.8 * current base throughput == quality,
121
-    * 1.1 * current base throughput == balanced */
122
-   static constexpr amf_int64 throughput_quality_mul = 18;
123
-   static constexpr amf_int64 throughput_balanced_mul = 11;
124
+   /* The throughput depends on the current preset and the other static
125
+    * encoder properties. If the throughput is lower than the max
126
+    * throughput, switch to a lower preset. */
127
+
128
+   if (astrcmpi(preset, "highQuality") == 0) {
129
+       if (!enc->max_throughput) {
130
+           preset = "quality";
131
+           set_opt(QUALITY_PRESET, get_preset(enc, preset));
132
+       } else {
133
+           if (enc->max_throughput < enc->throughput) {
134
+               preset = "quality";
135
+               refresh_throughput_caps(enc, preset);
136
+           }
137
+       }
138
+   }
139
 
140
-   /* if the throughput * 1.8 is lower than the max throughput, switch to
141
-    * a lower preset */
142
    if (astrcmpi(preset, "quality") == 0) {
143
        if (!enc->max_throughput) {
144
            preset = "balanced";
145
+           set_opt(QUALITY_PRESET, get_preset(enc, preset));
146
        } else {
147
-           amf_int64 req_throughput =
148
-               enc->throughput * throughput_quality_mul / 10;
149
-           if (enc->max_throughput < req_throughput)
150
+           if (enc->max_throughput < enc->throughput) {
151
                preset = "balanced";
152
+               refresh_throughput_caps(enc, preset);
153
+           }
154
        }
155
    }
156
 
157
    if (astrcmpi(preset, "balanced") == 0) {
158
-       amf_int64 req_throughput =
159
-           enc->throughput * throughput_balanced_mul / 10;
160
-       if (enc->max_throughput && enc->max_throughput < req_throughput)
161
+       if (enc->max_throughput &&
162
+           enc->max_throughput < enc->throughput) {
163
            preset = "speed";
164
+           refresh_throughput_caps(enc, preset);
165
+       }
166
    }
167
 }
168
 
169
@@ -447,29 +506,62 @@
170
    enc->packet_data = AMFBufferPtr(data);
171
    data->GetProperty(L"PTS", &packet->pts);
172
 
173
-   bool hevc = enc->codec == amf_codec_type::HEVC;
174
-   const wchar_t *get_output_type =
175
-       hevc ? AMF_VIDEO_ENCODER_HEVC_OUTPUT_DATA_TYPE
176
-            : AMF_VIDEO_ENCODER_OUTPUT_DATA_TYPE;
177
-
178
-   uint64_t type;
179
-   data->GetProperty(get_output_type, &type);
180
-
181
-   switch (type) {
182
-   case AMF_VIDEO_ENCODER_OUTPUT_DATA_TYPE_IDR:
183
-       packet->priority = OBS_NAL_PRIORITY_HIGHEST;
184
-       break;
185
-   case AMF_VIDEO_ENCODER_OUTPUT_DATA_TYPE_I:
186
-       packet->priority = OBS_NAL_PRIORITY_HIGH;
187
+   const wchar_t *get_output_type;
188
+   switch (enc->codec) {
189
+   case amf_codec_type::AVC:
190
+       get_output_type = AMF_VIDEO_ENCODER_OUTPUT_DATA_TYPE;
191
        break;
192
-   case AMF_VIDEO_ENCODER_OUTPUT_DATA_TYPE_P:
193
-       packet->priority = OBS_NAL_PRIORITY_LOW;
194
+   case amf_codec_type::HEVC:
195
+       get_output_type = AMF_VIDEO_ENCODER_HEVC_OUTPUT_DATA_TYPE;
196
        break;
197
-   case AMF_VIDEO_ENCODER_OUTPUT_DATA_TYPE_B:
198
-       packet->priority = OBS_NAL_PRIORITY_DISPOSABLE;
199
+   case amf_codec_type::AV1:
200
+       get_output_type = AMF_VIDEO_ENCODER_AV1_OUTPUT_FRAME_TYPE;
201
obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/vaapi-utils.c Added
201
 
1
@@ -0,0 +1,262 @@
2
+// SPDX-FileCopyrightText: 2022 tytan652 <tytan652@tytanium.xyz>
3
+//
4
+// SPDX-License-Identifier: GPL-2.0-or-later
5
+
6
+#include "vaapi-utils.h"
7
+
8
+#include <util/bmem.h>
9
+#include <util/dstr.h>
10
+
11
+#include <va/va_drm.h>
12
+#include <va/va_str.h>
13
+
14
+#include <stdio.h>
15
+#include <unistd.h>
16
+#include <fcntl.h>
17
+
18
+static bool version_logged = false;
19
+
20
+inline static VADisplay vaapi_open_display_drm(int *fd, const char *device_path)
21
+{
22
+   VADisplay va_dpy;
23
+
24
+   if (!device_path)
25
+       return NULL;
26
+
27
+   *fd = open(device_path, O_RDWR);
28
+   if (*fd < 0) {
29
+       blog(LOG_ERROR, "VAAPI: Failed to open device '%s'",
30
+            device_path);
31
+       return NULL;
32
+   }
33
+
34
+   va_dpy = vaGetDisplayDRM(*fd);
35
+
36
+   if (!va_dpy) {
37
+       blog(LOG_ERROR, "VAAPI: Failed to initialize DRM display");
38
+       return NULL;
39
+   }
40
+
41
+   return va_dpy;
42
+}
43
+
44
+inline static void vaapi_close_display_drm(int *fd)
45
+{
46
+   if (*fd < 0)
47
+       return;
48
+
49
+   close(*fd);
50
+   *fd = -1;
51
+}
52
+
53
+static void vaapi_log_info_cb(void *user_context, const char *message)
54
+{
55
+   UNUSED_PARAMETER(user_context);
56
+
57
+   // Libva message always ends with a newline
58
+   struct dstr m;
59
+   dstr_init_copy(&m, message);
60
+   dstr_depad(&m);
61
+
62
+   blog(LOG_DEBUG, "Libva: %s", m.array);
63
+
64
+   dstr_free(&m);
65
+}
66
+
67
+static void vaapi_log_error_cb(void *user_context, const char *message)
68
+{
69
+   UNUSED_PARAMETER(user_context);
70
+
71
+   // Libva message always ends with a newline
72
+   struct dstr m;
73
+   dstr_init_copy(&m, message);
74
+   dstr_depad(&m);
75
+
76
+   blog(LOG_DEBUG, "Libva error: %s", m.array);
77
+
78
+   dstr_free(&m);
79
+}
80
+
81
+VADisplay vaapi_open_device(int *fd, const char *device_path,
82
+               const char *func_name)
83
+{
84
+   VADisplay va_dpy;
85
+   VAStatus va_status;
86
+   int major, minor;
87
+   const char *driver;
88
+
89
+   va_dpy = vaapi_open_display_drm(fd, device_path);
90
+   if (!va_dpy)
91
+       return NULL;
92
+
93
+   blog(LOG_DEBUG, "VAAPI: Initializing display in %s", func_name);
94
+
95
+   vaSetInfoCallback(va_dpy, vaapi_log_info_cb, NULL);
96
+   vaSetErrorCallback(va_dpy, vaapi_log_error_cb, NULL);
97
+
98
+   va_status = vaInitialize(va_dpy, &major, &minor);
99
+
100
+   if (va_status != VA_STATUS_SUCCESS) {
101
+       blog(LOG_ERROR, "VAAPI: Failed to initialize display in %s",
102
+            func_name);
103
+       return NULL;
104
+   }
105
+
106
+   blog(LOG_DEBUG, "VAAPI: Display initialized");
107
+
108
+   if (!version_logged) {
109
+       blog(LOG_INFO, "VAAPI: API version %d.%d", major, minor);
110
+       version_logged = true;
111
+   }
112
+
113
+   driver = vaQueryVendorString(va_dpy);
114
+
115
+   blog(LOG_DEBUG, "VAAPI: '%s' in use for device '%s'", driver,
116
+        device_path);
117
+
118
+   return va_dpy;
119
+}
120
+
121
+void vaapi_close_device(int *fd, VADisplay dpy)
122
+{
123
+   vaTerminate(dpy);
124
+   vaapi_close_display_drm(fd);
125
+}
126
+
127
+static uint32_t vaapi_display_ep_combo_rate_controls(VAProfile profile,
128
+                            VAEntrypoint entrypoint,
129
+                            VADisplay dpy,
130
+                            const char *device_path)
131
+{
132
+   bool ret = false;
133
+   VAStatus va_status;
134
+   VAConfigAttrib attrib1;
135
+   attrib->type = VAConfigAttribRateControl;
136
+
137
+   va_status = vaGetConfigAttributes(dpy, profile, entrypoint, attrib, 1);
138
+
139
+   switch (va_status) {
140
+   case VA_STATUS_SUCCESS:
141
+       return attrib->value;
142
+   case VA_STATUS_ERROR_UNSUPPORTED_PROFILE:
143
+       blog(LOG_DEBUG, "VAAPI: %s is not supported by the device '%s'",
144
+            vaProfileStr(profile), device_path);
145
+       return 0;
146
+   case VA_STATUS_ERROR_UNSUPPORTED_ENTRYPOINT:
147
+       blog(LOG_DEBUG,
148
+            "VAAPI: %s %s is not supported by the device '%s'",
149
+            vaProfileStr(profile), vaEntrypointStr(entrypoint),
150
+            device_path);
151
+       return 0;
152
+   default:
153
+       blog(LOG_ERROR,
154
+            "VAAPI: Fail to get RC attribute from the %s %s of the device '%s'",
155
+            vaProfileStr(profile), vaEntrypointStr(entrypoint),
156
+            device_path);
157
+       return 0;
158
+   }
159
+}
160
+
161
+static bool vaapi_display_ep_combo_supported(VAProfile profile,
162
+                        VAEntrypoint entrypoint,
163
+                        VADisplay dpy,
164
+                        const char *device_path)
165
+{
166
+   uint32_t ret = vaapi_display_ep_combo_rate_controls(profile, entrypoint,
167
+                               dpy, device_path);
168
+   if (ret & VA_RC_CBR || ret & VA_RC_CQP || ret & VA_RC_VBR)
169
+       return true;
170
+
171
+   return false;
172
+}
173
+
174
+bool vaapi_device_rc_supported(VAProfile profile, VADisplay dpy, uint32_t rc,
175
+                  const char *device_path)
176
+{
177
+   uint32_t ret = vaapi_display_ep_combo_rate_controls(
178
+       profile, VAEntrypointEncSlice, dpy, device_path);
179
+   if (ret & rc)
180
+       return true;
181
+   ret = vaapi_display_ep_combo_rate_controls(
182
+       profile, VAEntrypointEncSliceLP, dpy, device_path);
183
+   if (ret & rc)
184
+       return true;
185
+
186
+   return false;
187
+}
188
+
189
+#define CHECK_PROFILE(ret, profile, va_dpy, device_path)                      \
190
+   if (vaapi_display_ep_combo_supported(profile, VAEntrypointEncSlice,   \
191
+                        va_dpy, device_path)) {          \
192
+       blog(LOG_DEBUG, "'%s' support encoding with %s", device_path, \
193
+            vaProfileStr(profile));                                  \
194
+       ret |= true;                                                  \
195
+   }
196
+
197
+#define CHECK_PROFILE_LP(ret, profile, va_dpy, device_path)                   \
198
+   if (vaapi_display_ep_combo_supported(profile, VAEntrypointEncSliceLP, \
199
+                        va_dpy, device_path)) {          \
200
+       blog(LOG_DEBUG, "'%s' support low power encoding with %s",    \
201
obs-studio-29.0.0.tar.xz/plugins/obs-ffmpeg/vaapi-utils.h Added
24
 
1
@@ -0,0 +1,22 @@
2
+// SPDX-FileCopyrightText: 2022 tytan652 <tytan652@tytanium.xyz>
3
+//
4
+// SPDX-License-Identifier: GPL-2.0-or-later
5
+
6
+#pragma once
7
+
8
+#include <util/base.h>
9
+
10
+#include <va/va.h>
11
+
12
+VADisplay vaapi_open_device(int *fd, const char *device_path,
13
+               const char *func_name);
14
+void vaapi_close_device(int *fd, VADisplay dpy);
15
+
16
+bool vaapi_device_rc_supported(VAProfile profile, VADisplay dpy, uint32_t rc,
17
+                  const char *device_path);
18
+
19
+bool vaapi_display_h264_supported(VADisplay dpy, const char *device_path);
20
+
21
+bool vaapi_device_h264_supported(const char *device_path);
22
+
23
+const char *vaapi_get_h264_default_device(void);
24
obs-studio-28.1.2.tar.xz/plugins/obs-filters/CMakeLists.txt -> obs-studio-29.0.0.tar.xz/plugins/obs-filters/CMakeLists.txt Changed
9
 
1
@@ -129,6 +129,7 @@
2
           color-key-filter.c
3
           color-grade-filter.c
4
           sharpness-filter.c
5
+          eq-filter.c
6
           gain-filter.c
7
           noise-gate-filter.c
8
           mask-filter.c
9
obs-studio-28.1.2.tar.xz/plugins/obs-filters/data/locale/en-US.ini -> obs-studio-29.0.0.tar.xz/plugins/obs-filters/data/locale/en-US.ini Changed
21
 
1
@@ -87,6 +87,7 @@
2
 NoiseSuppress.Method.Nvafx.Denoiser="NVIDIA Noise Removal"
3
 NoiseSuppress.Method.Nvafx.Dereverb="NVIDIA Room Echo Removal"
4
 NoiseSuppress.Method.Nvafx.DenoiserPlusDereverb="NVIDIA Noise Removal + Room Echo Removal"
5
+NoiseSuppress.Method.Nvafx.Deprecation="WARNING: Please upgrade both NVIDIA Video & Audio SDK. Your current version of Audio SDK is outdated."
6
 Saturation="Saturation"
7
 HueShift="Hue Shift"
8
 Amount="Amount"
9
@@ -124,3 +125,11 @@
10
 Greenscreen.Quality="Quality (higher GPU usage, better quality)"
11
 Greenscreen.Performance="Performance (lower GPU usage, good quality)"
12
 Greenscreen.Threshold="Threshold"
13
+Greenscreen.Deprecation="WARNING: Please upgrade both NVIDIA Video & Audio SDK. Your current version of Video SDK is outdated."
14
+Greenscreen.Processing="Mask refresh frequency in frames"
15
+Greenscreen.Processing.Hint="This alleviates GPU load by generating a mask every N frames only (2 on default)."
16
+Upward.Compressor="Upward Compressor"
17
+3BandEq="3-Band Equalizer"
18
+3BandEq.low="Low"
19
+3BandEq.mid="Mid"
20
+3BandEq.high="High"
21
obs-studio-29.0.0.tar.xz/plugins/obs-filters/eq-filter.c Added
163
 
1
@@ -0,0 +1,161 @@
2
+#include <media-io/audio-math.h>
3
+#include <util/circlebuf.h>
4
+#include <util/darray.h>
5
+#include <obs-module.h>
6
+
7
+#include <math.h>
8
+
9
+#define LOW_FREQ 800.0f
10
+#define HIGH_FREQ 5000.0f
11
+
12
+struct eq_channel_state {
13
+   float lf_delay0;
14
+   float lf_delay1;
15
+   float lf_delay2;
16
+   float lf_delay3;
17
+
18
+   float hf_delay0;
19
+   float hf_delay1;
20
+   float hf_delay2;
21
+   float hf_delay3;
22
+
23
+   float sample_delay1;
24
+   float sample_delay2;
25
+   float sample_delay3;
26
+};
27
+
28
+struct eq_data {
29
+   obs_source_t *context;
30
+   size_t channels;
31
+   struct eq_channel_state eqsMAX_AUDIO_CHANNELS;
32
+   float lf;
33
+   float hf;
34
+   float low_gain;
35
+   float mid_gain;
36
+   float high_gain;
37
+};
38
+
39
+static const char *eq_name(void *unused)
40
+{
41
+   UNUSED_PARAMETER(unused);
42
+   return obs_module_text("3BandEq");
43
+}
44
+
45
+static void eq_update(void *data, obs_data_t *settings)
46
+{
47
+   struct eq_data *eq = data;
48
+   eq->low_gain = db_to_mul((float)obs_data_get_double(settings, "low"));
49
+   eq->mid_gain = db_to_mul((float)obs_data_get_double(settings, "mid"));
50
+   eq->high_gain = db_to_mul((float)obs_data_get_double(settings, "high"));
51
+}
52
+
53
+static void eq_defaults(obs_data_t *defaults)
54
+{
55
+   obs_data_set_default_double(defaults, "low", 0.0);
56
+   obs_data_set_default_double(defaults, "mid", 0.0);
57
+   obs_data_set_default_double(defaults, "high", 0.0);
58
+}
59
+
60
+static obs_properties_t *eq_properties(void *unused)
61
+{
62
+   obs_properties_t *props = obs_properties_create();
63
+   obs_property_t *p;
64
+
65
+#define make_db_slider(name)                                                  \
66
+   p = obs_properties_add_float_slider(props, name,                      \
67
+                       obs_module_text("3BandEq." name), \
68
+                       -20.0f, 20.0, 0.1);               \
69
+   obs_property_float_set_suffix(p, " dB");
70
+
71
+   make_db_slider("high");
72
+   make_db_slider("mid");
73
+   make_db_slider("low");
74
+#undef make_db_slider
75
+
76
+   UNUSED_PARAMETER(unused);
77
+   return props;
78
+}
79
+
80
+static void *eq_create(obs_data_t *settings, obs_source_t *filter)
81
+{
82
+   struct eq_data *eq = bzalloc(sizeof(*eq));
83
+   eq->channels = audio_output_get_channels(obs_get_audio());
84
+   eq->context = filter;
85
+
86
+   float freq = (float)audio_output_get_sample_rate(obs_get_audio());
87
+   eq->lf = 2.0f * sinf(M_PI * LOW_FREQ / freq);
88
+   eq->hf = 2.0f * sinf(M_PI * HIGH_FREQ / freq);
89
+
90
+   eq_update(eq, settings);
91
+   return eq;
92
+}
93
+
94
+static void eq_destroy(void *data)
95
+{
96
+   struct eq_data *eq = data;
97
+   bfree(eq);
98
+}
99
+
100
+#define EQ_EPSILON (1.0f / 4294967295.0f)
101
+
102
+static inline float eq_process(struct eq_data *eq, struct eq_channel_state *c,
103
+                  float sample)
104
+{
105
+   float l, m, h;
106
+
107
+   c->lf_delay0 += eq->lf * (sample - c->lf_delay0) + EQ_EPSILON;
108
+   c->lf_delay1 += eq->lf * (c->lf_delay0 - c->lf_delay1);
109
+   c->lf_delay2 += eq->lf * (c->lf_delay1 - c->lf_delay2);
110
+   c->lf_delay3 += eq->lf * (c->lf_delay2 - c->lf_delay3);
111
+
112
+   l = c->lf_delay3;
113
+
114
+   c->hf_delay0 += eq->hf * (sample - c->hf_delay0) + EQ_EPSILON;
115
+   c->hf_delay1 += eq->hf * (c->hf_delay0 - c->hf_delay1);
116
+   c->hf_delay2 += eq->hf * (c->hf_delay1 - c->hf_delay2);
117
+   c->hf_delay3 += eq->hf * (c->hf_delay2 - c->hf_delay3);
118
+
119
+   h = c->sample_delay3 - c->hf_delay3;
120
+   m = c->sample_delay3 - (h + l);
121
+
122
+   l *= eq->low_gain;
123
+   m *= eq->mid_gain;
124
+   h *= eq->high_gain;
125
+
126
+   c->sample_delay3 = c->sample_delay2;
127
+   c->sample_delay2 = c->sample_delay1;
128
+   c->sample_delay1 = sample;
129
+
130
+   return l + m + h;
131
+}
132
+
133
+static struct obs_audio_data *eq_filter_audio(void *data,
134
+                         struct obs_audio_data *audio)
135
+{
136
+   struct eq_data *eq = data;
137
+   const uint32_t frames = audio->frames;
138
+
139
+   for (size_t c = 0; c < eq->channels; c++) {
140
+       float *adata = (float *)audio->datac;
141
+       struct eq_channel_state *channel = &eq->eqsc;
142
+
143
+       for (size_t i = 0; i < frames; i++) {
144
+           adatai = eq_process(eq, channel, adatai);
145
+       }
146
+   }
147
+
148
+   return audio;
149
+}
150
+
151
+struct obs_source_info eq_filter = {
152
+   .id = "basic_eq_filter",
153
+   .type = OBS_SOURCE_TYPE_FILTER,
154
+   .output_flags = OBS_SOURCE_AUDIO,
155
+   .get_name = eq_name,
156
+   .create = eq_create,
157
+   .destroy = eq_destroy,
158
+   .update = eq_update,
159
+   .filter_audio = eq_filter_audio,
160
+   .get_defaults = eq_defaults,
161
+   .get_properties = eq_properties,
162
+};
163
obs-studio-28.1.2.tar.xz/plugins/obs-filters/expander-filter.c -> obs-studio-29.0.0.tar.xz/plugins/obs-filters/expander-filter.c Changed
201
 
1
@@ -10,8 +10,8 @@
2
 
3
 /* -------------------------------------------------------- */
4
 
5
-#define do_log(level, format, ...)              \
6
-   blog(level, "expander: '%s' " format, \
7
+#define do_log(level, format, ...)                                     \
8
+   blog(level, "expander/gate/upward compressor: '%s' " format, \
9
         obs_source_get_name(cd->context), ##__VA_ARGS__)
10
 
11
 #define warn(format, ...) do_log(LOG_WARNING, format, ##__VA_ARGS__)
12
@@ -51,6 +51,8 @@
13
 
14
 #define MIN_RATIO                       1.0f
15
 #define MAX_RATIO                       20.0f
16
+#define MIN_RATIO_UPW                   0.0f
17
+#define MAX_RATIO_UPW                   1.0f
18
 #define MIN_THRESHOLD_DB                -60.0f
19
 #define MAX_THRESHOLD_DB                0.0f
20
 #define MIN_OUTPUT_GAIN_DB              -32.0f
21
@@ -87,11 +89,12 @@
22
    bool is_gate;
23
    float *runaverageMAX_AUDIO_CHANNELS;
24
    size_t runaverage_len;
25
-   float *gaindBMAX_AUDIO_CHANNELS;
26
-   size_t gaindB_len;
27
-   float gaindB_bufMAX_AUDIO_CHANNELS;
28
+   float *gain_dbMAX_AUDIO_CHANNELS;
29
+   size_t gain_db_len;
30
+   float gain_db_bufMAX_AUDIO_CHANNELS;
31
    float *env_in;
32
    size_t env_in_len;
33
+   bool is_upwcomp;
34
 };
35
 
36
 enum {
37
@@ -125,12 +128,12 @@
38
    cd->env_in = brealloc(cd->env_in, cd->env_in_len * sizeof(float));
39
 }
40
 
41
-static void resize_gaindB_buffer(struct expander_data *cd, size_t len)
42
+static void resize_gain_db_buffer(struct expander_data *cd, size_t len)
43
 {
44
-   cd->gaindB_len = len;
45
+   cd->gain_db_len = len;
46
    for (int i = 0; i < MAX_AUDIO_CHANNELS; i++)
47
-       cd->gaindBi =
48
-           brealloc(cd->gaindBi, cd->gaindB_len * sizeof(float));
49
+       cd->gain_dbi = brealloc(cd->gain_dbi,
50
+                     cd->gain_db_len * sizeof(float));
51
 }
52
 
53
 static inline float gain_coefficient(uint32_t sample_rate, float time)
54
@@ -144,6 +147,12 @@
55
    return obs_module_text("Expander");
56
 }
57
 
58
+static const char *upward_compressor_name(void *unused)
59
+{
60
+   UNUSED_PARAMETER(unused);
61
+   return obs_module_text("Upward.Compressor");
62
+}
63
+
64
 static void expander_defaults(obs_data_t *s)
65
 {
66
    const char *presets = obs_data_get_string(s, S_PRESETS);
67
@@ -162,21 +171,33 @@
68
    obs_data_set_default_string(s, S_DETECTOR, "RMS");
69
 }
70
 
71
+static void upward_compressor_defaults(obs_data_t *s)
72
+{
73
+   obs_data_set_default_double(s, S_RATIO, 0.5);
74
+   obs_data_set_default_double(s, S_THRESHOLD, -20.0f);
75
+   obs_data_set_default_int(s, S_ATTACK_TIME, 10);
76
+   obs_data_set_default_int(s, S_RELEASE_TIME, 50);
77
+   obs_data_set_default_double(s, S_OUTPUT_GAIN, 0.0);
78
+   obs_data_set_default_string(s, S_DETECTOR, "RMS");
79
+}
80
+
81
 static void expander_update(void *data, obs_data_t *s)
82
 {
83
    struct expander_data *cd = data;
84
-   const char *presets = obs_data_get_string(s, S_PRESETS);
85
-   if (strcmp(presets, "expander") == 0 && cd->is_gate) {
86
-       obs_data_clear(s);
87
-       obs_data_set_string(s, S_PRESETS, "expander");
88
-       expander_defaults(s);
89
-       cd->is_gate = false;
90
-   }
91
-   if (strcmp(presets, "gate") == 0 && !cd->is_gate) {
92
-       obs_data_clear(s);
93
-       obs_data_set_string(s, S_PRESETS, "gate");
94
-       expander_defaults(s);
95
-       cd->is_gate = true;
96
+   if (!cd->is_upwcomp) {
97
+       const char *presets = obs_data_get_string(s, S_PRESETS);
98
+       if (strcmp(presets, "expander") == 0 && cd->is_gate) {
99
+           obs_data_clear(s);
100
+           obs_data_set_string(s, S_PRESETS, "expander");
101
+           expander_defaults(s);
102
+           cd->is_gate = false;
103
+       }
104
+       if (strcmp(presets, "gate") == 0 && !cd->is_gate) {
105
+           obs_data_clear(s);
106
+           obs_data_set_string(s, S_PRESETS, "gate");
107
+           expander_defaults(s);
108
+           cd->is_gate = true;
109
+       }
110
    }
111
 
112
    const uint32_t sample_rate =
113
@@ -213,28 +234,41 @@
114
        resize_runaverage_buffer(cd, sample_len);
115
    if (cd->env_in_len == 0)
116
        resize_env_in_buffer(cd, sample_len);
117
-   if (cd->gaindB_len == 0)
118
-       resize_gaindB_buffer(cd, sample_len);
119
+   if (cd->gain_db_len == 0)
120
+       resize_gain_db_buffer(cd, sample_len);
121
 }
122
 
123
-static void *expander_create(obs_data_t *settings, obs_source_t *filter)
124
+static void *compressor_expander_create(obs_data_t *settings,
125
+                   obs_source_t *filter,
126
+                   bool is_compressor)
127
 {
128
    struct expander_data *cd = bzalloc(sizeof(struct expander_data));
129
    cd->context = filter;
130
    for (int i = 0; i < MAX_AUDIO_CHANNELS; i++) {
131
        cd->runavei = 0;
132
        cd->envelopei = 0;
133
-       cd->gaindB_bufi = 0;
134
+       cd->gain_db_bufi = 0;
135
    }
136
    cd->is_gate = false;
137
    const char *presets = obs_data_get_string(settings, S_PRESETS);
138
    if (strcmp(presets, "gate") == 0)
139
        cd->is_gate = true;
140
-
141
+   cd->is_upwcomp = is_compressor;
142
    expander_update(cd, settings);
143
    return cd;
144
 }
145
 
146
+static void *expander_create(obs_data_t *settings, obs_source_t *filter)
147
+{
148
+   return compressor_expander_create(settings, filter, false);
149
+}
150
+
151
+static void *upward_compressor_create(obs_data_t *settings,
152
+                     obs_source_t *filter)
153
+{
154
+   return compressor_expander_create(settings, filter, true);
155
+}
156
+
157
 static void expander_destroy(void *data)
158
 {
159
    struct expander_data *cd = data;
160
@@ -242,7 +276,7 @@
161
    for (int i = 0; i < MAX_AUDIO_CHANNELS; i++) {
162
        bfree(cd->envelope_bufi);
163
        bfree(cd->runaveragei);
164
-       bfree(cd->gaindBi);
165
+       bfree(cd->gain_dbi);
166
    }
167
    bfree(cd->env_in);
168
    bfree(cd);
169
@@ -304,59 +338,93 @@
170
    }
171
 }
172
 
173
+static inline void process_sample(size_t idx, float *samples, float *env_buf,
174
+                 float *gain_db, bool is_upwcomp,
175
+                 float channel_gain, float threshold,
176
+                 float slope, float attack_gain,
177
+                 float inv_attack_gain, float release_gain,
178
+                 float inv_release_gain, float output_gain)
179
+{
180
+   /* --------------------------------- */
181
+   /* gain stage of expansion           */
182
+
183
+   float env_db = mul_to_db(env_bufidx);
184
+   float diff = threshold - env_db;
185
+
186
+   if (is_upwcomp && env_db <= -60.0f)
187
+       diff = 0.0f;
188
+
189
+   float gain = diff > 0.0f ? fmaxf(slope * diff, -60.0f) : 0.0f;
190
+   float prev_gain = gain_dbidx - 1;
191
+
192
+   /* --------------------------------- */
193
+   /* ballistics (attack/release)       */
194
+
195
+   if (idx > 0) {
196
+       if (gain > prev_gain)
197
+           gain_dbidx = attack_gain * prev_gain +
198
+                      inv_attack_gain * gain;
199
+       else
200
+           gain_dbidx = release_gain * prev_gain +
201
obs-studio-28.1.2.tar.xz/plugins/obs-filters/noise-suppress-filter.c -> obs-studio-29.0.0.tar.xz/plugins/obs-filters/noise-suppress-filter.c Changed
94
 
1
@@ -59,6 +59,8 @@
2
 #define TEXT_METHOD_NVAFX_DEREVERB MT_("NoiseSuppress.Method.Nvafx.Dereverb")
3
 #define TEXT_METHOD_NVAFX_DEREVERB_DENOISER \
4
    MT_("NoiseSuppress.Method.Nvafx.DenoiserPlusDereverb")
5
+#define TEXT_METHOD_NVAFX_DEPRECATION \
6
+   MT_("NoiseSuppress.Method.Nvafx.Deprecation")
7
 
8
 #define MAX_PREPROC_CHANNELS 8
9
 
10
@@ -623,9 +625,23 @@
11
 bool load_nvafx(void)
12
 {
13
 #ifdef LIBNVAFX_ENABLED
14
+   unsigned int version = get_lib_version();
15
+   uint8_t major = (version >> 24) & 0xff;
16
+   uint8_t minor = (version >> 16) & 0x00ff;
17
+   uint8_t build = (version >> 8) & 0x0000ff;
18
+   uint8_t revision = (version >> 0) & 0x000000ff;
19
+   if (version) {
20
+       blog(LOG_INFO,
21
+            "noise suppress: NVIDIA AUDIO FX version: %i.%i.%i.%i",
22
+            major, minor, build, revision);
23
+       if (version < (MIN_AFX_SDK_VERSION)) {
24
+           blog(LOG_INFO,
25
+                "noise suppress: NVIDIA AUDIO Effects SDK is outdated. Please update both audio & video SDK.");
26
+       }
27
+   }
28
    if (!load_lib()) {
29
        blog(LOG_INFO,
30
-            "noise suppress: NVIDIA RTX denoiser disabled, redistributable not found");
31
+            "noise suppress: NVIDIA denoiser disabled, redistributable not found or could not be loaded.");
32
        return false;
33
    }
34
 
35
@@ -699,10 +715,10 @@
36
    if (err != NVAFX_STATUS_SUCCESS) {
37
        if (err == NVAFX_STATUS_GPU_UNSUPPORTED) {
38
            blog(LOG_INFO,
39
-                "noise suppress: NVIDIA RTX AUDIO FX disabled: unsupported GPU");
40
+                "noise suppress: NVIDIA AUDIO FX disabled: unsupported GPU");
41
        } else {
42
            blog(LOG_ERROR,
43
-                "noise suppress: NVIDIA RTX AUDIO FX disabled, error %i",
44
+                "noise suppress: NVIDIA AUDIO FX disabled, error %i",
45
                 err);
46
        }
47
        goto unload_everything;
48
@@ -711,18 +727,18 @@
49
    err = NvAFX_DestroyEffect(h);
50
    if (err != NVAFX_STATUS_SUCCESS) {
51
        blog(LOG_ERROR,
52
-            "noise suppress: NVIDIA RTX AUDIO FX disabled, error %i",
53
+            "noise suppress: NVIDIA AUDIO FX disabled, error %i",
54
             err);
55
        goto unload_everything;
56
    }
57
 
58
    nvafx_loaded = true;
59
-   blog(LOG_INFO, "noise suppress: NVIDIA RTX AUDIO FX enabled");
60
+   blog(LOG_INFO, "noise suppress: NVIDIA AUDIO FX enabled");
61
    return true;
62
 
63
 cuda_errors:
64
    blog(LOG_ERROR,
65
-        "noise suppress: NVIDIA RTX AUDIO FX disabled, CUDA error %i",
66
+        "noise suppress: NVIDIA AUDIO FX disabled, CUDA error %i",
67
         cudaerr);
68
 unload_everything:
69
    release_lib();
70
@@ -1205,10 +1221,19 @@
71
 #endif
72
 
73
 #ifdef LIBNVAFX_ENABLED
74
-   obs_properties_add_float_slider(ppts, S_NVAFX_INTENSITY,
75
-                   TEXT_NVAFX_INTENSITY, 0.0f, 1.0f,
76
-                   0.01f);
77
-
78
+   if (ng->nvafx_enabled) {
79
+       obs_properties_add_float_slider(ppts, S_NVAFX_INTENSITY,
80
+                       TEXT_NVAFX_INTENSITY, 0.0f,
81
+                       1.0f, 0.01f);
82
+   }
83
+   unsigned int version = get_lib_version();
84
+   if (version < (MIN_AFX_SDK_VERSION)) {
85
+       obs_property_t *warning = obs_properties_add_text(
86
+           ppts, "deprecation", NULL, OBS_TEXT_INFO);
87
+       obs_property_text_set_info_type(warning, OBS_TEXT_INFO_WARNING);
88
+       obs_property_set_long_description(
89
+           warning, TEXT_METHOD_NVAFX_DEPRECATION);
90
+   }
91
 #if defined(LIBRNNOISE_ENABLED) && defined(LIBSPEEXDSP_ENABLED)
92
    if (!nvafx_loaded) {
93
        obs_property_list_item_disable(method, 2, true);
94
obs-studio-28.1.2.tar.xz/plugins/obs-filters/nvafx-load.h -> obs-studio-29.0.0.tar.xz/plugins/obs-filters/nvafx-load.h Changed
33
 
1
@@ -4,10 +4,12 @@
2
 #include <stddef.h>
3
 #include <stdint.h>
4
 #include <util/platform.h>
5
+#include <util/windows/win-version.h>
6
 
7
 #define NVAFX_API
8
 
9
 #ifdef LIBNVAFX_ENABLED
10
+#define MIN_AFX_SDK_VERSION 1 << 24 | 2 << 16 | 13 << 0
11
 static HMODULE nv_audiofx = NULL;
12
 static HMODULE nv_cuda = NULL;
13
 
14
@@ -307,4 +309,18 @@
15
    nv_cuda = LoadLibrary(L"nvcuda.dll");
16
    return !!nv_audiofx && !!nv_cuda;
17
 }
18
+
19
+static unsigned int get_lib_version(void)
20
+{
21
+   char pathMAX_PATH;
22
+   if (!nvafx_get_sdk_path(path, sizeof(path)))
23
+       return 0;
24
+
25
+   SetDllDirectoryA(path);
26
+   struct win_version_info nto_ver = {0};
27
+   get_dll_ver(L"NVAudioEffects.dll", &nto_ver);
28
+   unsigned int version = nto_ver.major << 24 | nto_ver.minor << 16 |
29
+                  nto_ver.build << 8 | nto_ver.revis << 0;
30
+   return version;
31
+}
32
 #endif
33
obs-studio-28.1.2.tar.xz/plugins/obs-filters/nvidia-greenscreen-filter.c -> obs-studio-29.0.0.tar.xz/plugins/obs-filters/nvidia-greenscreen-filter.c Changed
201
 
1
@@ -6,9 +6,9 @@
2
 #include "nvvfx-load.h"
3
 /* -------------------------------------------------------- */
4
 
5
-#define do_log(level, format, ...)                                             \
6
-   blog(level,                                                            \
7
-        "NVIDIA RTX AI Greenscreen (Background removal): '%s' " format, \
8
+#define do_log(level, format, ...)                                         \
9
+   blog(level,                                                        \
10
+        "NVIDIA AI Greenscreen (Background removal): '%s' " format, \
11
         obs_source_get_name(filter->context), ##__VA_ARGS__)
12
 
13
 #define warn(format, ...) do_log(LOG_WARNING, format, ##__VA_ARGS__)
14
@@ -27,14 +27,19 @@
15
 #define S_MODE_PERF 1
16
 #define S_THRESHOLDFX "threshold"
17
 #define S_THRESHOLDFX_DEFAULT 1.0
18
+#define S_PROCESSING "processing_interval"
19
 
20
 #define MT_ obs_module_text
21
 #define TEXT_MODE MT_("Greenscreen.Mode")
22
 #define TEXT_MODE_QUALITY MT_("Greenscreen.Quality")
23
 #define TEXT_MODE_PERF MT_("Greenscreen.Performance")
24
 #define TEXT_MODE_THRESHOLD MT_("Greenscreen.Threshold")
25
+#define TEXT_DEPRECATION MT_("Greenscreen.Deprecation")
26
+#define TEXT_PROCESSING MT_("Greenscreen.Processing")
27
+#define TEXT_PROCESSING_HINT MT_("Greenscreen.Processing.Hint")
28
 
29
 bool nvvfx_loaded = false;
30
+bool nvvfx_new_sdk = false;
31
 struct nv_greenscreen_data {
32
    obs_source_t *context;
33
    bool images_allocated;
34
@@ -54,6 +59,8 @@
35
    NvCVImage *A_dst_img;   // mask img on GPU
36
    NvCVImage *dst_img;     // mask texture
37
    NvCVImage *stage;       // planar stage img used for transfer to texture
38
+   unsigned int version;
39
+   NvVFX_StateObjectHandle stateObjectHandle;
40
 
41
    /* alpha mask effect */
42
    gs_effect_t *effect;
43
@@ -68,6 +75,13 @@
44
    gs_eparam_t *threshold_param;
45
    gs_eparam_t *multiplier_param;
46
    float threshold;
47
+
48
+   /* Every nth frame is processed through the FX (where n =
49
+    * processing_interval) so that the same mask is used for n consecutive
50
+    * frames. This is to alleviate the GPU load. Default: 1 (process every frame).
51
+    */
52
+   int processing_interval;
53
+   int processing_counter;
54
 };
55
 
56
 static const char *nv_greenscreen_filter_name(void *unused)
57
@@ -89,6 +103,8 @@
58
            error("Error loading AI Greenscreen FX %i", vfxErr);
59
    }
60
    filter->threshold = (float)obs_data_get_double(settings, S_THRESHOLDFX);
61
+   filter->processing_interval =
62
+       (int)obs_data_get_int(settings, S_PROCESSING);
63
 }
64
 
65
 static void nv_greenscreen_filter_actual_destroy(void *data)
66
@@ -117,8 +133,13 @@
67
        NvVFX_CudaStreamDestroy(filter->stream);
68
    }
69
    if (filter->handle) {
70
+       if (filter->stateObjectHandle) {
71
+           NvVFX_DeallocateState(filter->handle,
72
+                         filter->stateObjectHandle);
73
+       }
74
        NvVFX_DestroyEffect(filter->handle);
75
    }
76
+
77
    if (filter->effect) {
78
        obs_enter_graphics();
79
        gs_effect_destroy(filter->effect);
80
@@ -145,6 +166,10 @@
81
        NvVFX_CudaStreamDestroy(filter->stream);
82
    }
83
    if (filter->handle) {
84
+       if (filter->stateObjectHandle) {
85
+           NvVFX_DeallocateState(filter->handle,
86
+                         filter->stateObjectHandle);
87
+       }
88
        NvVFX_DestroyEffect(filter->handle);
89
    }
90
    // recreate
91
@@ -410,6 +435,8 @@
92
    filter->initial_render = false;
93
    os_atomic_set_bool(&filter->processing_stop, false);
94
    filter->handler = NULL;
95
+   filter->processing_interval = 1;
96
+   filter->processing_counter = 0;
97
 
98
    /* 1. Create FX */
99
    vfxErr = NvVFX_CreateEffect(NVVFX_FX_GREEN_SCREEN, &filter->handle);
100
@@ -445,13 +472,15 @@
101
        nv_greenscreen_filter_destroy(filter);
102
        return NULL;
103
    }
104
-   /* log sdk version */
105
-   unsigned int version;
106
-   if (NvVFX_GetVersion(&version) == NVCV_SUCCESS) {
107
-       uint8_t major = (version >> 24) & 0xff;
108
-       uint8_t minor = (version >> 16) & 0x00ff;
109
-       uint8_t build = (version >> 8) & 0x0000ff;
110
-       info("RTX VIDEO FX version: %i.%i.%i", major, minor, build);
111
+   /* check sdk version */
112
+   if (NvVFX_GetVersion(&filter->version) == NVCV_SUCCESS) {
113
+       uint8_t major = (filter->version >> 24) & 0xff;
114
+       uint8_t minor = (filter->version >> 16) & 0x00ff;
115
+       uint8_t build = (filter->version >> 8) & 0x0000ff;
116
+       uint8_t revision = (filter->version >> 0) & 0x000000ff;
117
+       // sanity check
118
+       nvvfx_new_sdk = filter->version >= (MIN_VFX_SDK_VERSION) &&
119
+               nvvfx_new_sdk;
120
    }
121
 
122
    /* 3. Load alpha mask effect. */
123
@@ -472,6 +501,29 @@
124
    }
125
    obs_leave_graphics();
126
 
127
+   /* 4. Allocate state for the effect */
128
+   if (nvvfx_new_sdk) {
129
+       vfxErr = NvVFX_AllocateState(filter->handle,
130
+                        &filter->stateObjectHandle);
131
+       if (NVCV_SUCCESS != vfxErr) {
132
+           const char *errString =
133
+               NvCV_GetErrorStringFromCode(vfxErr);
134
+           error("Error allocating FX state %i", vfxErr);
135
+           nv_greenscreen_filter_destroy(filter);
136
+           return NULL;
137
+       }
138
+       vfxErr = NvVFX_SetStateObjectHandleArray(
139
+           filter->handle, NVVFX_STATE,
140
+           &filter->stateObjectHandle);
141
+       if (NVCV_SUCCESS != vfxErr) {
142
+           const char *errString =
143
+               NvCV_GetErrorStringFromCode(vfxErr);
144
+           error("Error setting FX state %i", vfxErr);
145
+           nv_greenscreen_filter_destroy(filter);
146
+           return NULL;
147
+       }
148
+   }
149
+
150
    if (!filter->effect) {
151
        nv_greenscreen_filter_destroy(filter);
152
        return NULL;
153
@@ -486,6 +538,7 @@
154
 
155
 static obs_properties_t *nv_greenscreen_filter_properties(void *data)
156
 {
157
+   struct nv_greenscreen_data *filter = (struct nv_greenscreen_data *)data;
158
    obs_properties_t *props = obs_properties_create();
159
    obs_property_t *mode = obs_properties_add_list(props, S_MODE, TEXT_MODE,
160
                               OBS_COMBO_TYPE_LIST,
161
@@ -494,8 +547,17 @@
162
    obs_property_list_add_int(mode, TEXT_MODE_PERF, S_MODE_PERF);
163
    obs_property_t *threshold = obs_properties_add_float_slider(
164
        props, S_THRESHOLDFX, TEXT_MODE_THRESHOLD, 0, 1, 0.05);
165
+   obs_property_t *partial = obs_properties_add_int_slider(
166
+       props, S_PROCESSING, TEXT_PROCESSING, 1, 4, 1);
167
+   obs_property_set_long_description(partial, TEXT_PROCESSING_HINT);
168
+   unsigned int version = get_lib_version();
169
+   if (version < (MIN_VFX_SDK_VERSION)) {
170
+       obs_property_t *warning = obs_properties_add_text(
171
+           props, "deprecation", NULL, OBS_TEXT_INFO);
172
+       obs_property_text_set_info_type(warning, OBS_TEXT_INFO_WARNING);
173
+       obs_property_set_long_description(warning, TEXT_DEPRECATION);
174
+   }
175
 
176
-   UNUSED_PARAMETER(data);
177
    return props;
178
 }
179
 
180
@@ -504,7 +566,9 @@
181
    obs_data_set_default_int(settings, S_MODE, S_MODE_QUALITY);
182
    obs_data_set_default_double(settings, S_THRESHOLDFX,
183
                    S_THRESHOLDFX_DEFAULT);
184
+   obs_data_set_default_int(settings, S_PROCESSING, 1);
185
 }
186
+
187
 static struct obs_source_frame *
188
 nv_greenscreen_filter_video(void *data, struct obs_source_frame *frame)
189
 {
190
@@ -660,7 +724,7 @@
191
 
192
    if (parent && !filter->handler) {
193
        filter->handler = obs_source_get_signal_handler(parent);
194
-       signal_handler_connect(filter->handler, "update_properties",
195
+       signal_handler_connect(filter->handler, "update",
196
                       nv_greenscreen_filter_reset, filter);
197
    }
198
 
199
@@ -799,7 +863,14 @@
200
    if (filter->initial_render && filter->images_allocated) {
201
obs-studio-28.1.2.tar.xz/plugins/obs-filters/nvvfx-load.h -> obs-studio-29.0.0.tar.xz/plugins/obs-filters/nvvfx-load.h Changed
144
 
1
@@ -6,6 +6,7 @@
2
 #include <stdint.h>
3
 #include <util/platform.h>
4
 #include <dxgitype.h>
5
+#include <util/windows/win-version.h>
6
 
7
 #ifdef __cplusplus
8
 extern "C" {
9
@@ -38,7 +39,7 @@
10
 #define CUDARTAPI
11
 
12
 #ifdef LIBNVVFX_ENABLED
13
-// allows for future loading of a second fx
14
+#define MIN_VFX_SDK_VERSION 0 << 24 | 7 << 16 | 1 << 8 | 0 << 0
15
 static HMODULE nv_videofx = NULL;
16
 static HMODULE nv_cvimage = NULL;
17
 static HMODULE nv_cudart = NULL;
18
@@ -139,8 +140,14 @@
19
 #define NVVFX_CUDA_STREAM "CudaStream" //!< The CUDA stream to use
20
 #define NVVFX_CUDA_GRAPH "CudaGraph"   //!< Enable CUDA graph to use
21
 #define NVVFX_INFO "Info"              //!< Get info about the effects
22
-#define NVVFX_SCALE "Scale"            //!< Scale factor
23
-#define NVVFX_STRENGTH "Strength"      //!< Strength for different filters
24
+#define NVVFX_MAX_INPUT_WIDTH \
25
+   "MaxInputWidth" //!< Maximum width of the input supported
26
+#define NVVFX_MAX_INPUT_HEIGHT \
27
+   "MaxInputHeight" //!< Maximum height of the input supported
28
+#define NVVFX_MAX_NUMBER_STREAMS \
29
+   "MaxNumberStreams"  //!< Maximum number of concurrent input streams
30
+#define NVVFX_SCALE "Scale" //!< Scale factor
31
+#define NVVFX_STRENGTH "Strength" //!< Strength for different filters
32
 #define NVVFX_STRENGTH_LEVELS "StrengthLevels" //!< Number of strength levels
33
 #define NVVFX_MODE "Mode"                      //!< Mode for different filters
34
 #define NVVFX_TEMPORAL "Temporal"    //!< Temporal mode: 0=image, 1=video
35
@@ -149,6 +156,8 @@
36
 #define NVVFX_MODEL_BATCH "ModelBatch"
37
 #define NVVFX_STATE "State"          //!< State variable
38
 #define NVVFX_STATE_SIZE "StateSize" //!< Number of bytes needed to store state
39
+#define NVVFX_STATE_COUNT \
40
+   "NumStateObjects" //!< Number of active state object handles
41
 
42
 //! The format of pixels in an image.
43
 typedef enum NvCVImage_PixelFormat {
44
@@ -324,6 +333,8 @@
45
 typedef const char *NvVFX_EffectSelector;
46
 typedef const char *NvVFX_ParameterSelector;
47
 typedef void *NvVFX_Handle;
48
+/* requires sdk version >= 0.7.0 */
49
+typedef void *NvVFX_StateObjectHandle;
50
 
51
 /* nvvfx functions */
52
 typedef NvCV_Status NvVFX_API (*NvVFX_GetVersion_t)(unsigned int *version);
53
@@ -346,6 +357,11 @@
54
                    unsigned long long val);
55
 typedef NvCV_Status NvVFX_API (*NvVFX_SetObject_t)(
56
    NvVFX_Handle effect, NvVFX_ParameterSelector param_name, void *ptr);
57
+/* requires sdk version >= 0.7.0 */
58
+typedef NvCV_Status NvVFX_API (*NvVFX_SetStateObjectHandleArray_t)(
59
+   NvVFX_Handle effect, NvVFX_ParameterSelector param_name,
60
+   NvVFX_StateObjectHandle *handle);
61
+/* ----------------------------- */
62
 typedef NvCV_Status
63
    NvVFX_API (*NvVFX_SetCudaStream_t)(NvVFX_Handle effect,
64
                       NvVFX_ParameterSelector param_name,
65
@@ -387,6 +403,17 @@
66
 typedef NvCV_Status NvVFX_API (*NvVFX_CudaStreamCreate_t)(CUstream *stream);
67
 typedef NvCV_Status NvVFX_API (*NvVFX_CudaStreamDestroy_t)(CUstream stream);
68
 
69
+/* requires sdk version >= 0.7.0 */
70
+typedef NvCV_Status
71
+   NvVFX_API (*NvVFX_AllocateState_t)(NvVFX_Handle effect,
72
+                      NvVFX_StateObjectHandle *handle);
73
+typedef NvCV_Status
74
+   NvVFX_API (*NvVFX_DeallocateState_t)(NvVFX_Handle effect,
75
+                        NvVFX_StateObjectHandle handle);
76
+typedef NvCV_Status
77
+   NvVFX_API (*NvVFX_ResetState_t)(NvVFX_Handle effect,
78
+                   NvVFX_StateObjectHandle handle);
79
+
80
 /* NvCVImage functions */
81
 typedef NvCV_Status NvCV_API (*NvCVImage_Init_t)(
82
    NvCVImage *im, unsigned width, unsigned height, int pitch, void *pixels,
83
@@ -643,6 +670,12 @@
84
 static NvVFX_CudaStreamCreate_t NvVFX_CudaStreamCreate = NULL;
85
 static NvVFX_CudaStreamDestroy_t NvVFX_CudaStreamDestroy = NULL;
86
 
87
+/* nvvfx sdk >= 0.7.0 */
88
+static NvVFX_SetStateObjectHandleArray_t NvVFX_SetStateObjectHandleArray = NULL;
89
+static NvVFX_AllocateState_t NvVFX_AllocateState = NULL;
90
+static NvVFX_DeallocateState_t NvVFX_DeallocateState = NULL;
91
+static NvVFX_ResetState_t NvVFX_ResetState = NULL;
92
+
93
 /*nvcvimage */
94
 static NvCVImage_Init_t NvCVImage_Init = NULL;
95
 static NvCVImage_InitView_t NvCVImage_InitView = NULL;
96
@@ -707,6 +740,10 @@
97
    NvVFX_SetString = NULL;
98
    NvVFX_SetU32 = NULL;
99
    NvVFX_SetU64 = NULL;
100
+   NvVFX_SetStateObjectHandleArray = NULL;
101
+   NvVFX_AllocateState = NULL;
102
+   NvVFX_DeallocateState = NULL;
103
+   NvVFX_ResetState = NULL;
104
    if (nv_videofx) {
105
        FreeLibrary(nv_videofx);
106
        nv_videofx = NULL;
107
@@ -761,8 +798,7 @@
108
 
109
        size_t max_len = sizeof(path) / sizeof(char);
110
        snprintf(buffer, max_len,
111
-            "%s\\NVIDIA Corporation\\NVIDIA Video Effects\\",
112
-            path);
113
+            "%s\\NVIDIA Corporation\\NVIDIA Video Effects", path);
114
    }
115
 }
116
 
117
@@ -771,6 +807,7 @@
118
    char fullPathMAX_PATH;
119
    nvvfx_get_sdk_path(fullPath, MAX_PATH);
120
    SetDllDirectoryA(fullPath);
121
+
122
    nv_videofx = LoadLibrary(L"NVVideoEffects.dll");
123
    nv_cvimage = LoadLibrary(L"NVCVImage.dll");
124
    nv_cudart = LoadLibrary(L"cudart64_110.dll");
125
@@ -778,6 +815,18 @@
126
    return !!nv_videofx && !!nv_cvimage && !!nv_cudart;
127
 }
128
 
129
+static unsigned int get_lib_version(void)
130
+{
131
+   char pathMAX_PATH;
132
+   nvvfx_get_sdk_path(path, sizeof(path));
133
+
134
+   SetDllDirectoryA(path);
135
+   struct win_version_info nto_ver = {0};
136
+   get_dll_ver(L"NVVideoEffects.dll", &nto_ver);
137
+   unsigned int version = nto_ver.major << 24 | nto_ver.minor << 16 |
138
+                  nto_ver.build << 8 | nto_ver.revis << 0;
139
+   return version;
140
+}
141
 #endif
142
 
143
 #ifdef __cplusplus
144
obs-studio-28.1.2.tar.xz/plugins/obs-filters/obs-filters.c -> obs-studio-29.0.0.tar.xz/plugins/obs-filters/obs-filters.c Changed
33
 
1
@@ -11,6 +11,7 @@
2
 extern struct obs_source_info mask_filter_v2;
3
 extern struct obs_source_info crop_filter;
4
 extern struct obs_source_info gain_filter;
5
+extern struct obs_source_info eq_filter;
6
 extern struct obs_source_info hdr_tonemap_filter;
7
 extern struct obs_source_info color_filter;
8
 extern struct obs_source_info color_filter_v2;
9
@@ -36,6 +37,7 @@
10
 extern struct obs_source_info compressor_filter;
11
 extern struct obs_source_info limiter_filter;
12
 extern struct obs_source_info expander_filter;
13
+extern struct obs_source_info upward_compressor_filter;
14
 extern struct obs_source_info luma_key_filter;
15
 extern struct obs_source_info luma_key_filter_v2;
16
 #ifdef LIBNVVFX_ENABLED
17
@@ -50,6 +52,7 @@
18
    obs_register_source(&mask_filter_v2);
19
    obs_register_source(&crop_filter);
20
    obs_register_source(&gain_filter);
21
+   obs_register_source(&eq_filter);
22
    obs_register_source(&hdr_tonemap_filter);
23
    obs_register_source(&color_filter);
24
    obs_register_source(&color_filter_v2);
25
@@ -77,6 +80,7 @@
26
    obs_register_source(&compressor_filter);
27
    obs_register_source(&limiter_filter);
28
    obs_register_source(&expander_filter);
29
+   obs_register_source(&upward_compressor_filter);
30
    obs_register_source(&luma_key_filter);
31
    obs_register_source(&luma_key_filter_v2);
32
 #ifdef LIBNVVFX_ENABLED
33
obs-studio-28.1.2.tar.xz/plugins/obs-filters/scale-filter.c -> obs-studio-29.0.0.tar.xz/plugins/obs-filters/scale-filter.c Changed
11
 
1
@@ -540,7 +540,8 @@
2
 
3
    for (size_t i = 0; i < NUM_DOWNSCALES; i++) {
4
        char str32;
5
-       snprintf(str, 32, "%dx%d", downscalesi.cx, downscalesi.cy);
6
+       snprintf(str, sizeof(str), "%dx%d", downscalesi.cx,
7
+            downscalesi.cy);
8
        obs_property_list_add_string(p, str, str);
9
    }
10
 
11
obs-studio-28.1.2.tar.xz/plugins/obs-outputs/librtmp/amf.c -> obs-studio-29.0.0.tar.xz/plugins/obs-outputs/librtmp/amf.c Changed
38
 
1
@@ -831,7 +831,7 @@
2
     if (name.av_len > 18)
3
         name.av_len = 18;
4
 
5
-    snprintf(strRes, 255, "Name: %18.*s, ", name.av_len, name.av_val);
6
+    snprintf(strRes, sizeof(strRes), "Name: %18.*s, ", name.av_len, name.av_val);
7
 
8
     if (prop->p_type == AMF_OBJECT)
9
     {
10
@@ -855,22 +855,22 @@
11
     switch (prop->p_type)
12
     {
13
     case AMF_NUMBER:
14
-        snprintf(str, 255, "NUMBER:\t%.2f", prop->p_vu.p_number);
15
+        snprintf(str, sizeof(str), "NUMBER:\t%.2f", prop->p_vu.p_number);
16
         break;
17
     case AMF_BOOLEAN:
18
-        snprintf(str, 255, "BOOLEAN:\t%s",
19
+        snprintf(str, sizeof(str), "BOOLEAN:\t%s",
20
                  prop->p_vu.p_number != 0.0 ? "TRUE" : "FALSE");
21
         break;
22
     case AMF_STRING:
23
-        snprintf(str, 255, "STRING:\t%.*s", prop->p_vu.p_aval.av_len,
24
+        snprintf(str, sizeof(str), "STRING:\t%.*s", prop->p_vu.p_aval.av_len,
25
                  prop->p_vu.p_aval.av_val);
26
         break;
27
     case AMF_DATE:
28
-        snprintf(str, 255, "DATE:\ttimestamp: %.2f, UTC offset: %d",
29
+        snprintf(str, sizeof(str), "DATE:\ttimestamp: %.2f, UTC offset: %d",
30
                  prop->p_vu.p_number, prop->p_UTCoffset);
31
         break;
32
     default:
33
-        snprintf(str, 255, "INVALID TYPE 0x%02x", (unsigned char)prop->p_type);
34
+        snprintf(str, sizeof(str), "INVALID TYPE 0x%02x", (unsigned char)prop->p_type);
35
     }
36
 
37
     RTMP_Log(RTMP_LOGDEBUG, "Property: <%s%s>", strRes, str);
38
obs-studio-28.1.2.tar.xz/plugins/obs-outputs/librtmp/hashswf.c -> obs-studio-29.0.0.tar.xz/plugins/obs-outputs/librtmp/hashswf.c Changed
53
 
1
@@ -152,12 +152,12 @@
2
     if (sb.sb_socket == INVALID_SOCKET)
3
         return HTTPRES_LOST_CONNECTION;
4
     i =
5
-        sprintf(sb.sb_buf,
6
+        snprintf(sb.sb_buf, RTMP_BUFFER_CACHE_SIZE,
7
                 "GET %s HTTP/1.0\r\nUser-Agent: %s\r\nHost: %s\r\nReferer: %.*s\r\n",
8
                 path, AGENT, host, (int)(path - url + 1), url);
9
     if (http->date0)
10
-        i += sprintf(sb.sb_buf + i, "If-Modified-Since: %s\r\n", http->date);
11
-    i += sprintf(sb.sb_buf + i, "\r\n");
12
+        i += snprintf(sb.sb_buf + i, RTMP_BUFFER_CACHE_SIZE, "If-Modified-Since: %s\r\n", http->date);
13
+    i += snprintf(sb.sb_buf + i, RTMP_BUFFER_CACHE_SIZE, "\r\n");
14
 
15
     if (connect
16
             (sb.sb_socket, (struct sockaddr *)&sa, sizeof(struct sockaddr)) < 0)
17
@@ -455,12 +455,12 @@
18
  * Weekday, DD-MMM-YYYY HH:MM:SS GMT
19
  */
20
 static void
21
-strtime(time_t * t, char *s)
22
+strtime(time_t * t, char *s, size_t size)
23
 {
24
     struct tm *tm;
25
 
26
     tm = gmtime((time_t *) t);
27
-    sprintf(s, "%s, %02d %s %d %02d:%02d:%02d GMT",
28
+    snprintf(s, size, "%s, %02d %s %d %02d:%02d:%02d GMT",
29
             daystm->tm_wday, tm->tm_mday, monthtabtm->tm_mon,
30
             tm->tm_year + 1900, tm->tm_hour, tm->tm_min, tm->tm_sec);
31
 }
32
@@ -516,8 +516,9 @@
33
      * These fields must be present in this order. All fields
34
      * besides URL are fixed size.
35
      */
36
-    path = malloc(hpre.av_len + home.av_len + sizeof(DIRSEP ".swfinfo"));
37
-    sprintf(path, "%s%s" DIRSEP ".swfinfo", hpre.av_val, home.av_val);
38
+    size_t path_size = hpre.av_len + home.av_len + sizeof(DIRSEP ".swfinfo");
39
+    path = malloc(path_size);
40
+    snprintf(path, path_size, "%s%s" DIRSEP ".swfinfo", hpre.av_val, home.av_val);
41
 
42
     f = fopen(path, "r+");
43
     while (f)
44
@@ -651,7 +652,7 @@
45
 
46
             fprintf(f, "url: %.*s\n", i, url);
47
         }
48
-        strtime(&cnow, cctim);
49
+        strtime(&cnow, cctim, sizeof(cctim));
50
         fprintf(f, "ctim: %s\n", cctim);
51
 
52
         if (!in.first)
53
obs-studio-28.1.2.tar.xz/plugins/obs-outputs/librtmp/rtmp.c -> obs-studio-29.0.0.tar.xz/plugins/obs-outputs/librtmp/rtmp.c Changed
146
 
1
@@ -488,7 +488,7 @@
2
     r->Link.curStreamIdx = 0;
3
     r->Link.nStreams = 0;
4
     r->Link.receiveTimeout = 30;
5
-    r->Link.sendTimeout = 6;
6
+    r->Link.sendTimeout = 15;
7
     r->Link.swfAge = 30;
8
 }
9
 
10
@@ -781,7 +781,7 @@
11
 
12
     char portStr8;
13
 
14
-    sprintf(portStr, "%d", port);
15
+    snprintf(portStr, sizeof(portStr), "%d", port);
16
 
17
     int err = getaddrinfo(hostname, portStr, &hints, &result);
18
 
19
@@ -2623,12 +2623,12 @@
20
 static const AVal av_authmod_adobe = AVC("authmod=adobe");
21
 static const AVal av_authmod_llnw  = AVC("authmod=llnw");
22
 
23
-static void hexenc(unsigned char *inbuf, int len, char *dst)
24
+static void hexenc(unsigned char *inbuf, int len, char *dst, size_t size)
25
 {
26
     char *ptr = dst;
27
     while(len--)
28
     {
29
-        sprintf(ptr, "%02x", *inbuf++);
30
+        snprintf(ptr, size, "%02x", *inbuf++);
31
         ptr += 2;
32
     }
33
     *ptr = '\0';
34
@@ -2676,8 +2676,9 @@
35
             }
36
             else if(r->Link.pubUser.av_len && r->Link.pubPasswd.av_len)
37
             {
38
-                pubToken.av_val = malloc(r->Link.pubUser.av_len + av_authmod_adobe.av_len + 8);
39
-                pubToken.av_len = sprintf(pubToken.av_val, "?%s&user=%s",
40
+                size_t val_size = r->Link.pubUser.av_len + av_authmod_adobe.av_len + 8;
41
+                pubToken.av_val = malloc(val_size);
42
+                pubToken.av_len = snprintf(pubToken.av_val, val_size, "?%s&user=%s",
43
                                           av_authmod_adobe.av_val,
44
                                           r->Link.pubUser.av_val);
45
                 RTMP_Log(RTMP_LOGDEBUG, "%s, pubToken1: %s", __FUNCTION__, pubToken.av_val);
46
@@ -2777,8 +2778,9 @@
47
             RTMP_Log(RTMP_LOGDEBUG, "%s, b64(md5_2) = %s", __FUNCTION__, response);
48
 
49
             /* have all hashes, create auth token for the end of app */
50
-            pubToken.av_val = malloc(32 + B64INT_LEN + B64DIGEST_LEN + opaque.av_len);
51
-            pubToken.av_len = sprintf(pubToken.av_val,
52
+            size_t val_size = 32 + B64INT_LEN + B64DIGEST_LEN + opaque.av_len;
53
+            pubToken.av_val = malloc(val_size);
54
+            pubToken.av_len = snprintf(pubToken.av_val, val_size,
55
                                       "&challenge=%s&response=%s&opaque=%s",
56
                                       challenge2,
57
                                       response,
58
@@ -2845,8 +2847,9 @@
59
             }
60
             else if(r->Link.pubUser.av_len && r->Link.pubPasswd.av_len)
61
             {
62
-                pubToken.av_val = malloc(r->Link.pubUser.av_len + av_authmod_llnw.av_len + 8);
63
-                pubToken.av_len = sprintf(pubToken.av_val, "?%s&user=%s",
64
+                size_t val_size = r->Link.pubUser.av_len + av_authmod_llnw.av_len + 8;
65
+                pubToken.av_val = malloc(val_size);
66
+                pubToken.av_len = snprintf(pubToken.av_val, val_size, "?%s&user=%s",
67
                                           av_authmod_llnw.av_val,
68
                                           r->Link.pubUser.av_val);
69
                 RTMP_Log(RTMP_LOGDEBUG, "%s, pubToken1: %s", __FUNCTION__, pubToken.av_val);
70
@@ -2923,8 +2926,8 @@
71
 
72
             /* FIXME: handle case where user==NULL or nonce==NULL */
73
 
74
-            sprintf(nchex, "%08x", nc);
75
-            sprintf(cnonce, "%08x", rand());
76
+            snprintf(nchex, sizeof(nchex), "%08x", nc);
77
+            snprintf(cnonce, sizeof(cnonce), "%08x", rand());
78
 
79
             /* hash1 = hexenc(md5(user + ":" + realm + ":" + password)) */
80
             MD5_Init(&md5ctx);
81
@@ -2937,7 +2940,7 @@
82
             RTMP_Log(RTMP_LOGDEBUG, "%s, md5(%s:%s:%s) =>", __FUNCTION__,
83
                      user.av_val, realm, r->Link.pubPasswd.av_val);
84
             RTMP_LogHexString(RTMP_LOGDEBUG, md5sum_val, MD5_DIGEST_LENGTH);
85
-            hexenc(md5sum_val, MD5_DIGEST_LENGTH, hash1);
86
+            hexenc(md5sum_val, MD5_DIGEST_LENGTH, hash1, sizeof(hash1));
87
 
88
             /* hash2 = hexenc(md5(method + ":/" + app + "/" + appInstance)) */
89
             /* Extract appname + appinstance without query parameters */
90
@@ -2956,7 +2959,7 @@
91
             RTMP_Log(RTMP_LOGDEBUG, "%s, md5(%s:/%.*s) =>", __FUNCTION__,
92
                      method, apptmp.av_len, apptmp.av_val);
93
             RTMP_LogHexString(RTMP_LOGDEBUG, md5sum_val, MD5_DIGEST_LENGTH);
94
-            hexenc(md5sum_val, MD5_DIGEST_LENGTH, hash2);
95
+            hexenc(md5sum_val, MD5_DIGEST_LENGTH, hash2, sizeof(hash2));
96
 
97
             /* hash3 = hexenc(md5(hash1 + ":" + nonce + ":" + nchex + ":" + cnonce + ":" + qop + ":" + hash2)) */
98
             MD5_Init(&md5ctx);
99
@@ -2975,13 +2978,14 @@
100
             RTMP_Log(RTMP_LOGDEBUG, "%s, md5(%s:%s:%s:%s:%s:%s) =>", __FUNCTION__,
101
                      hash1, nonce.av_val, nchex, cnonce, qop, hash2);
102
             RTMP_LogHexString(RTMP_LOGDEBUG, md5sum_val, MD5_DIGEST_LENGTH);
103
-            hexenc(md5sum_val, MD5_DIGEST_LENGTH, hash3);
104
+            hexenc(md5sum_val, MD5_DIGEST_LENGTH, hash3, sizeof(hash3));
105
 
106
             /* pubToken = &authmod=<authmod>&user=<username>&nonce=<nonce>&cnonce=<cnonce>&nc=<nchex>&response=<hash3> */
107
             /* Append nonces and response to query string which already contains
108
              * user + authmod */
109
-            pubToken.av_val = malloc(64 + sizeof(authmod)-1 + user.av_len + nonce.av_len + sizeof(cnonce)-1 + sizeof(nchex)-1 + HEXHASH_LEN);
110
-            sprintf(pubToken.av_val,
111
+            size_t token_size = 64 + sizeof(authmod)-1 + user.av_len + nonce.av_len + sizeof(cnonce)-1 + sizeof(nchex)-1 + HEXHASH_LEN;
112
+            pubToken.av_val = malloc(token_size);
113
+            snprintf(pubToken.av_val, token_size,
114
                     "&nonce=%s&cnonce=%s&nc=%s&response=%s",
115
                     nonce.av_val, cnonce, nchex, hash3);
116
             pubToken.av_len = (int)strlen(pubToken.av_val);
117
@@ -3487,23 +3491,23 @@
118
             DumpMetaData(&prop->p_vu.p_object);
119
             break;
120
         case AMF_NUMBER:
121
-            snprintf(str, 255, "%.2f", prop->p_vu.p_number);
122
+            snprintf(str, sizeof(str), "%.2f", prop->p_vu.p_number);
123
             break;
124
         case AMF_BOOLEAN:
125
-            snprintf(str, 255, "%s",
126
+            snprintf(str, sizeof(str), "%s",
127
                      prop->p_vu.p_number != 0. ? "TRUE" : "FALSE");
128
             break;
129
         case AMF_STRING:
130
-            len = snprintf(str, 255, "%.*s", prop->p_vu.p_aval.av_len,
131
+            len = snprintf(str, sizeof(str), "%.*s", prop->p_vu.p_aval.av_len,
132
                            prop->p_vu.p_aval.av_val);
133
             if (len >= 1 && strlen-1 == '\n')
134
                 strlen-1 = '\0';
135
             break;
136
         case AMF_DATE:
137
-            snprintf(str, 255, "timestamp:%.2f", prop->p_vu.p_number);
138
+            snprintf(str, sizeof(str), "timestamp:%.2f", prop->p_vu.p_number);
139
             break;
140
         default:
141
-            snprintf(str, 255, "INVALID TYPE 0x%02x",
142
+            snprintf(str, sizeof(str), "INVALID TYPE 0x%02x",
143
                      (unsigned char)prop->p_type);
144
         }
145
         if (str0 && prop->p_name.av_len)
146
obs-studio-28.1.2.tar.xz/plugins/obs-outputs/rtmp-stream.c -> obs-studio-29.0.0.tar.xz/plugins/obs-outputs/rtmp-stream.c Changed
10
 
1
@@ -33,7 +33,7 @@
2
 #endif
3
 
4
 /* dynamic bitrate coefficients */
5
-#define DBR_INC_TIMER (30ULL * SEC_TO_NSEC)
6
+#define DBR_INC_TIMER (4ULL * SEC_TO_NSEC)
7
 #define DBR_TRIGGER_USEC (200ULL * MSEC_TO_USEC)
8
 #define MIN_ESTIMATE_DURATION_MS 1000
9
 #define MAX_ESTIMATE_DURATION_MS 2000
10
obs-studio-28.1.2.tar.xz/plugins/obs-qsv11/CMakeLists.txt -> obs-studio-29.0.0.tar.xz/plugins/obs-qsv11/CMakeLists.txt Changed
19
 
1
@@ -94,6 +94,8 @@
2
 target_compile_definitions(obs-qsv11 PRIVATE DX11_D3D)
3
 
4
 if(OS_WINDOWS)
5
+  add_subdirectory(obs-qsv-test)
6
+
7
   set(MODULE_DESCRIPTION "OBS QSV encoder")
8
   configure_file(${CMAKE_SOURCE_DIR}/cmake/bundle/windows/obs-module.rc.in
9
                  obs-qsv11.rc)
10
@@ -105,7 +107,7 @@
11
                       _CRT_NONSTDC_NO_WARNINGS)
12
 endif()
13
 
14
-set_target_properties(obs-qsv11 PROPERTIES FOLDER "plugins")
15
+set_target_properties(obs-qsv11 PROPERTIES FOLDER "plugins/obs-qsv11")
16
 
17
 file(GLOB _OBS_QSV11_SOURCE_FILES ${CMAKE_CURRENT_SOURCE_DIR}/*.c
18
      ${CMAKE_CURRENT_SOURCE_DIR}/*.cpp)
19
obs-studio-28.1.2.tar.xz/plugins/obs-qsv11/QSV_Encoder.cpp -> obs-studio-29.0.0.tar.xz/plugins/obs-qsv11/QSV_Encoder.cpp Changed
190
 
1
@@ -74,117 +74,44 @@
2
 mfxVersion ver = {{0, 1}}; // for backward compatibility
3
 std::atomic<bool> is_active{false};
4
 
5
-bool prefer_current_or_igpu_enc(int *iGPUIndex)
6
+void qsv_encoder_version(unsigned short *major, unsigned short *minor)
7
 {
8
-   IDXGIAdapter *pAdapter;
9
-   bool hasIGPU = false;
10
-   bool hasDGPU = false;
11
-   bool hasCurrent = false;
12
-
13
-   HMODULE hDXGI = LoadLibrary(L"dxgi.dll");
14
-   if (hDXGI == NULL) {
15
-       return false;
16
-   }
17
-
18
-   typedef HRESULT(WINAPI * LPCREATEDXGIFACTORY)(REFIID riid,
19
-                             void **ppFactory);
20
-
21
-   LPCREATEDXGIFACTORY pCreateDXGIFactory =
22
-       (LPCREATEDXGIFACTORY)GetProcAddress(hDXGI,
23
-                           "CreateDXGIFactory1");
24
-   if (pCreateDXGIFactory == NULL) {
25
-       pCreateDXGIFactory = (LPCREATEDXGIFACTORY)GetProcAddress(
26
-           hDXGI, "CreateDXGIFactory");
27
+   *major = ver.Major;
28
+   *minor = ver.Minor;
29
+}
30
 
31
-       if (pCreateDXGIFactory == NULL) {
32
-           FreeLibrary(hDXGI);
33
-           return false;
34
-       }
35
-   }
36
+qsv_t *qsv_encoder_open(qsv_param_t *pParams, enum qsv_codec codec)
37
+{
38
+   mfxIMPL impl_list4 = {MFX_IMPL_HARDWARE, MFX_IMPL_HARDWARE2,
39
+               MFX_IMPL_HARDWARE3, MFX_IMPL_HARDWARE4};
40
 
41
-   IDXGIFactory *pFactory = NULL;
42
-   if (FAILED((*pCreateDXGIFactory)(__uuidof(IDXGIFactory),
43
-                    (void **)(&pFactory)))) {
44
-       FreeLibrary(hDXGI);
45
-       return false;
46
-   }
47
+   obs_video_info ovi;
48
+   obs_get_video_info(&ovi);
49
+   size_t adapter_idx = ovi.adapter;
50
 
51
-   LUID luid;
52
-   bool hasLuid = false;
53
-   obs_enter_graphics();
54
-   {
55
-       ID3D11Device *pDevice = (ID3D11Device *)gs_get_device_obj();
56
-       Microsoft::WRL::ComPtr<IDXGIDevice> dxgiDevice;
57
-       if (SUCCEEDED(pDevice->QueryInterface<IDXGIDevice>(
58
-               dxgiDevice.GetAddressOf()))) {
59
-           Microsoft::WRL::ComPtr<IDXGIAdapter> dxgiAdapter;
60
-           if (SUCCEEDED(dxgiDevice->GetAdapter(
61
-                   dxgiAdapter.GetAddressOf()))) {
62
-               DXGI_ADAPTER_DESC desc;
63
-               hasLuid =
64
-                   SUCCEEDED(dxgiAdapter->GetDesc(&desc));
65
-               if (hasLuid) {
66
-                   luid = desc.AdapterLuid;
67
-               }
68
+   // Select current adapter - will be iGPU if exists due to adapter reordering
69
+   if (codec == QSV_CODEC_AV1 && !adaptersadapter_idx.supports_av1) {
70
+       for (size_t i = 0; i < 4; i++) {
71
+           if (adaptersi.supports_av1) {
72
+               adapter_idx = i;
73
+               break;
74
            }
75
        }
76
-   }
77
-   obs_leave_graphics();
78
-
79
-   // Check for i+I cases (Intel discrete + Intel integrated graphics on the same system). Default will be integrated.
80
-   for (int adapterIndex = 0;
81
-        SUCCEEDED(pFactory->EnumAdapters(adapterIndex, &pAdapter));
82
-        ++adapterIndex) {
83
-       DXGI_ADAPTER_DESC AdapterDesc = {};
84
-       const HRESULT hr = pAdapter->GetDesc(&AdapterDesc);
85
-       pAdapter->Release();
86
-
87
-       if (SUCCEEDED(hr) && (AdapterDesc.VendorId == 0x8086)) {
88
-           if (hasLuid &&
89
-               (AdapterDesc.AdapterLuid.LowPart == luid.LowPart) &&
90
-               (AdapterDesc.AdapterLuid.HighPart ==
91
-                luid.HighPart)) {
92
-               hasCurrent = true;
93
-               *iGPUIndex = adapterIndex;
94
+   } else if (!adaptersadapter_idx.is_intel) {
95
+       for (size_t i = 0; i < 4; i++) {
96
+           if (adaptersi.is_intel) {
97
+               adapter_idx = i;
98
                break;
99
            }
100
-
101
-           if (AdapterDesc.DedicatedVideoMemory <=
102
-               512 * 1024 * 1024) {
103
-               hasIGPU = true;
104
-               if (iGPUIndex != NULL) {
105
-                   *iGPUIndex = adapterIndex;
106
-               }
107
-           } else {
108
-               hasDGPU = true;
109
-           }
110
        }
111
    }
112
 
113
-   pFactory->Release();
114
-   FreeLibrary(hDXGI);
115
-
116
-   return hasCurrent || (hasIGPU && hasDGPU);
117
-}
118
-
119
-void qsv_encoder_version(unsigned short *major, unsigned short *minor)
120
-{
121
-   *major = ver.Major;
122
-   *minor = ver.Minor;
123
-}
124
-
125
-qsv_t *qsv_encoder_open(qsv_param_t *pParams)
126
-{
127
-   mfxIMPL impl_list4 = {MFX_IMPL_HARDWARE, MFX_IMPL_HARDWARE2,
128
-               MFX_IMPL_HARDWARE3, MFX_IMPL_HARDWARE4};
129
-   int igpu_index = -1;
130
-   if (prefer_current_or_igpu_enc(&igpu_index) &&
131
-       (igpu_index < _countof(impl_list))) {
132
-       impl = impl_listigpu_index;
133
-   }
134
+   bool isDGPU = adaptersadapter_idx.is_dgpu;
135
+   impl = impl_listadapter_idx;
136
 
137
-   QSV_Encoder_Internal *pEncoder = new QSV_Encoder_Internal(impl, ver);
138
-   mfxStatus sts = pEncoder->Open(pParams);
139
+   QSV_Encoder_Internal *pEncoder =
140
+       new QSV_Encoder_Internal(impl, ver, isDGPU);
141
+   mfxStatus sts = pEncoder->Open(pParams, codec);
142
    if (sts != MFX_ERR_NONE) {
143
 
144
 #define WARN_ERR_IMPL(err, str, err_name)                   \
145
@@ -272,6 +199,12 @@
146
    return (qsv_t *)pEncoder;
147
 }
148
 
149
+bool qsv_encoder_is_dgpu(qsv_t *pContext)
150
+{
151
+   QSV_Encoder_Internal *pEncoder = (QSV_Encoder_Internal *)pContext;
152
+   return pEncoder->IsDGPU();
153
+}
154
+
155
 int qsv_encoder_headers(qsv_t *pContext, uint8_t **pSPS, uint8_t **pPPS,
156
            uint16_t *pnSPS, uint16_t *pnPPS)
157
 {
158
@@ -349,12 +282,12 @@
159
 int qsv_encoder_reconfig(qsv_t *pContext, qsv_param_t *pParams)
160
 {
161
    QSV_Encoder_Internal *pEncoder = (QSV_Encoder_Internal *)pContext;
162
-   mfxStatus sts = pEncoder->Reset(pParams);
163
+   pEncoder->UpdateParams(pParams);
164
+   mfxStatus sts = pEncoder->ReconfigureEncoder();
165
 
166
-   if (sts == MFX_ERR_NONE)
167
-       return 0;
168
-   else
169
-       return -1;
170
+   if (sts != MFX_ERR_NONE)
171
+       return false;
172
+   return true;
173
 }
174
 
175
 enum qsv_cpu_platform qsv_get_cpu_platform()
176
@@ -438,3 +371,13 @@
177
    //assume newer revisions are at least as capable as Haswell
178
    return QSV_CPU_PLATFORM_INTEL;
179
 }
180
+
181
+int qsv_hevc_encoder_headers(qsv_t *pContext, uint8_t **pVPS, uint8_t **pSPS,
182
+                uint8_t **pPPS, uint16_t *pnVPS, uint16_t *pnSPS,
183
+                uint16_t *pnPPS)
184
+{
185
+   QSV_Encoder_Internal *pEncoder = (QSV_Encoder_Internal *)pContext;
186
+   pEncoder->GetVpsSpsPps(pVPS, pSPS, pPPS, pnVPS, pnSPS, pnPPS);
187
+
188
+   return 0;
189
+}
190
obs-studio-28.1.2.tar.xz/plugins/obs-qsv11/QSV_Encoder.h -> obs-studio-29.0.0.tar.xz/plugins/obs-qsv11/QSV_Encoder.h Changed
93
 
1
@@ -58,6 +58,7 @@
2
 
3
 #include <Windows.h>
4
 #include "mfxstructures.h"
5
+#include "mfxadapter.h"
6
 #include <stdint.h>
7
 
8
 #ifdef __cplusplus
9
@@ -73,7 +74,13 @@
10
    {"CBR", false},   {"VBR", false}, {"VCM", true},    {"CQP", false},
11
    {"AVBR", false},  {"ICQ", true},  {"LA_ICQ", true}, {"LA_CBR", true},
12
    {"LA_VBR", true}, {0, false}};
13
+
14
+static const struct qsv_rate_control_info qsv_av1_ratecontrols =
15
+   {{"CBR", false}, {"VBR", false}, {"CQP", false}, {0, false}};
16
+
17
 static const char *const qsv_profile_names = {"high", "main", "baseline", 0};
18
+static const char *const qsv_profile_names_av1 = {"main", 0};
19
+static const char *const qsv_profile_names_hevc = {"main", "main10", 0};
20
 static const char *const qsv_usage_names = {"quality",  "balanced", "speed",
21
                          "veryslow", "slower",   "slow",
22
                          "medium",   "fast",     "faster",
23
@@ -82,6 +89,23 @@
24
                        0};
25
 typedef struct qsv_t qsv_t;
26
 
27
+struct adapter_info {
28
+   bool is_intel;
29
+   bool is_dgpu;
30
+   bool supports_av1;
31
+   bool supports_hevc;
32
+};
33
+
34
+enum qsv_codec {
35
+   QSV_CODEC_AVC,
36
+   QSV_CODEC_AV1,
37
+   QSV_CODEC_HEVC,
38
+};
39
+
40
+#define MAX_ADAPTERS 10
41
+extern struct adapter_info adaptersMAX_ADAPTERS;
42
+extern size_t adapter_count;
43
+
44
 typedef struct {
45
    mfxU16 nTargetUsage; /* 1 through 7, 1 being best quality and 7
46
                being the best speed */
47
@@ -103,8 +127,24 @@
48
    mfxU16 nKeyIntSec;
49
    mfxU16 nbFrames;
50
    mfxU16 nICQQuality;
51
+   mfxU16 VideoFormat;
52
+   mfxU16 VideoFullRange;
53
+   mfxU16 ColourPrimaries;
54
+   mfxU16 TransferCharacteristics;
55
+   mfxU16 MatrixCoefficients;
56
+   mfxU16 ChromaSampleLocTypeTopField;
57
+   mfxU16 ChromaSampleLocTypeBottomField;
58
+   mfxU16 DisplayPrimariesX3;
59
+   mfxU16 DisplayPrimariesY3;
60
+   mfxU16 WhitePointX;
61
+   mfxU16 WhitePointY;
62
+   mfxU32 MaxDisplayMasteringLuminance;
63
+   mfxU32 MinDisplayMasteringLuminance;
64
+   mfxU16 MaxContentLightLevel;
65
+   mfxU16 MaxPicAverageLightLevel;
66
    bool bMBBRC;
67
    bool bCQM;
68
+   bool video_fmt_10bit;
69
 } qsv_param_t;
70
 
71
 enum qsv_cpu_platform {
72
@@ -132,7 +172,8 @@
73
                 const char *tune);
74
 int qsv_encoder_reconfig(qsv_t *, qsv_param_t *);
75
 void qsv_encoder_version(unsigned short *major, unsigned short *minor);
76
-qsv_t *qsv_encoder_open(qsv_param_t *);
77
+qsv_t *qsv_encoder_open(qsv_param_t *, enum qsv_codec codec);
78
+bool qsv_encoder_is_dgpu(qsv_t *);
79
 int qsv_encoder_encode(qsv_t *, uint64_t, uint8_t *, uint8_t *, uint32_t,
80
               uint32_t, mfxBitstream **pBS);
81
 int qsv_encoder_encode_tex(qsv_t *, uint64_t, uint32_t, uint64_t, uint64_t *,
82
@@ -142,6 +183,10 @@
83
 enum qsv_cpu_platform qsv_get_cpu_platform();
84
 bool prefer_igpu_enc(int *iGPUIndex);
85
 
86
+int qsv_hevc_encoder_headers(qsv_t *pContext, uint8_t **vVPS, uint8_t **pSPS,
87
+                uint8_t **pPPS, uint16_t *pnVPS, uint16_t *pnSPS,
88
+                uint16_t *pnPPS);
89
+
90
 #ifdef __cplusplus
91
 }
92
 #endif
93
obs-studio-28.1.2.tar.xz/plugins/obs-qsv11/QSV_Encoder_Internal.cpp -> obs-studio-29.0.0.tar.xz/plugins/obs-qsv11/QSV_Encoder_Internal.cpp Changed
201
 
1
@@ -71,7 +71,8 @@
2
 mfxHDL QSV_Encoder_Internal::g_DX_Handle = NULL;
3
 mfxU16 QSV_Encoder_Internal::g_numEncodersOpen = 0;
4
 
5
-QSV_Encoder_Internal::QSV_Encoder_Internal(mfxIMPL &impl, mfxVersion &version)
6
+QSV_Encoder_Internal::QSV_Encoder_Internal(mfxIMPL &impl, mfxVersion &version,
7
+                      bool isDGPU)
8
    : m_pmfxSurfaces(NULL),
9
      m_pmfxENC(NULL),
10
      m_nSPSBufferSize(1024),
11
@@ -80,7 +81,8 @@
12
      m_pTaskPool(NULL),
13
      m_nTaskIdx(0),
14
      m_nFirstSyncTask(0),
15
-     m_outBitstream()
16
+     m_outBitstream(),
17
+     m_isDGPU(isDGPU)
18
 {
19
    mfxIMPL tempImpl;
20
    mfxStatus sts;
21
@@ -148,7 +150,7 @@
22
        ClearData();
23
 }
24
 
25
-mfxStatus QSV_Encoder_Internal::Open(qsv_param_t *pParams)
26
+mfxStatus QSV_Encoder_Internal::Open(qsv_param_t *pParams, enum qsv_codec codec)
27
 {
28
    mfxStatus sts = MFX_ERR_NONE;
29
 
30
@@ -167,7 +169,8 @@
31
 
32
    m_pmfxENC = new MFXVideoENCODE(m_session);
33
 
34
-   InitParams(pParams);
35
+   sts = InitParams(pParams, codec);
36
+   MSDK_CHECK_RESULT(sts, MFX_ERR_NONE, sts);
37
 
38
    sts = m_pmfxENC->Query(&m_mfxEncParams, &m_mfxEncParams);
39
    MSDK_IGNORE_MFX_STS(sts, MFX_WRN_INCOMPATIBLE_VIDEO_PARAM);
40
@@ -179,7 +182,7 @@
41
    sts = m_pmfxENC->Init(&m_mfxEncParams);
42
    MSDK_CHECK_RESULT(sts, MFX_ERR_NONE, sts);
43
 
44
-   sts = GetVideoParam();
45
+   sts = GetVideoParam(codec);
46
    MSDK_CHECK_RESULT(sts, MFX_ERR_NONE, sts);
47
 
48
    sts = InitBitstream();
49
@@ -191,28 +194,53 @@
50
    return sts;
51
 }
52
 
53
-bool QSV_Encoder_Internal::InitParams(qsv_param_t *pParams)
54
+mfxStatus QSV_Encoder_Internal::InitParams(qsv_param_t *pParams,
55
+                      enum qsv_codec codec)
56
 {
57
    memset(&m_mfxEncParams, 0, sizeof(m_mfxEncParams));
58
 
59
-   m_mfxEncParams.mfx.CodecId = MFX_CODEC_AVC;
60
+   if (codec == QSV_CODEC_AVC)
61
+       m_mfxEncParams.mfx.CodecId = MFX_CODEC_AVC;
62
+   else if (codec == QSV_CODEC_AV1)
63
+       m_mfxEncParams.mfx.CodecId = MFX_CODEC_AV1;
64
+   else if (codec == QSV_CODEC_HEVC)
65
+       m_mfxEncParams.mfx.CodecId = MFX_CODEC_HEVC;
66
+
67
    m_mfxEncParams.mfx.GopOptFlag = MFX_GOP_STRICT;
68
-   m_mfxEncParams.mfx.NumSlice = 1;
69
+   if (codec == QSV_CODEC_HEVC) {
70
+       m_mfxEncParams.mfx.NumSlice = 0;
71
+       m_mfxEncParams.mfx.IdrInterval = 1;
72
+   } else {
73
+       m_mfxEncParams.mfx.NumSlice = 1;
74
+   }
75
    m_mfxEncParams.mfx.TargetUsage = pParams->nTargetUsage;
76
    m_mfxEncParams.mfx.CodecProfile = pParams->nCodecProfile;
77
    m_mfxEncParams.mfx.FrameInfo.FrameRateExtN = pParams->nFpsNum;
78
    m_mfxEncParams.mfx.FrameInfo.FrameRateExtD = pParams->nFpsDen;
79
-   m_mfxEncParams.mfx.FrameInfo.FourCC = MFX_FOURCC_NV12;
80
+   if (pParams->video_fmt_10bit) {
81
+       m_mfxEncParams.mfx.FrameInfo.FourCC = MFX_FOURCC_P010;
82
+       m_mfxEncParams.mfx.FrameInfo.BitDepthChroma = 10;
83
+       m_mfxEncParams.mfx.FrameInfo.BitDepthLuma = 10;
84
+       m_mfxEncParams.mfx.FrameInfo.Shift = 1;
85
+   } else {
86
+       m_mfxEncParams.mfx.FrameInfo.FourCC = MFX_FOURCC_NV12;
87
+   }
88
    m_mfxEncParams.mfx.FrameInfo.ChromaFormat = MFX_CHROMAFORMAT_YUV420;
89
    m_mfxEncParams.mfx.FrameInfo.PicStruct = MFX_PICSTRUCT_PROGRESSIVE;
90
    m_mfxEncParams.mfx.FrameInfo.CropX = 0;
91
    m_mfxEncParams.mfx.FrameInfo.CropY = 0;
92
    m_mfxEncParams.mfx.FrameInfo.CropW = pParams->nWidth;
93
    m_mfxEncParams.mfx.FrameInfo.CropH = pParams->nHeight;
94
-   m_mfxEncParams.mfx.GopRefDist = pParams->nbFrames + 1;
95
+   if (codec == QSV_CODEC_AV1)
96
+       m_mfxEncParams.mfx.GopRefDist = 1;
97
+   else
98
+       m_mfxEncParams.mfx.GopRefDist = pParams->nbFrames + 1;
99
+
100
+   if (codec == QSV_CODEC_HEVC)
101
+       m_mfxEncParams.mfx.LowPower = MFX_CODINGOPTION_OFF;
102
 
103
    enum qsv_cpu_platform qsv_platform = qsv_get_cpu_platform();
104
-   if ((qsv_platform >= QSV_CPU_PLATFORM_ICL ||
105
+   if ((m_isDGPU || qsv_platform >= QSV_CPU_PLATFORM_ICL ||
106
         qsv_platform == QSV_CPU_PLATFORM_UNKNOWN) &&
107
        (pParams->nbFrames == 0) &&
108
        (m_ver.Major == 1 && m_ver.Minor >= 31)) {
109
@@ -228,11 +256,17 @@
110
    switch (pParams->nRateControl) {
111
    case MFX_RATECONTROL_CBR:
112
        m_mfxEncParams.mfx.TargetKbps = pParams->nTargetBitRate;
113
+       m_mfxEncParams.mfx.BufferSizeInKB = pParams->nTargetBitRate * 2;
114
+       m_mfxEncParams.mfx.InitialDelayInKB =
115
+           pParams->nTargetBitRate * 1;
116
        break;
117
    case MFX_RATECONTROL_VBR:
118
    case MFX_RATECONTROL_VCM:
119
        m_mfxEncParams.mfx.TargetKbps = pParams->nTargetBitRate;
120
        m_mfxEncParams.mfx.MaxKbps = pParams->nMaxBitRate;
121
+       m_mfxEncParams.mfx.BufferSizeInKB = pParams->nTargetBitRate * 2;
122
+       m_mfxEncParams.mfx.InitialDelayInKB =
123
+           pParams->nTargetBitRate * 1;
124
        break;
125
    case MFX_RATECONTROL_CQP:
126
        m_mfxEncParams.mfx.QPI = pParams->nQPI;
127
@@ -266,13 +300,12 @@
128
        (mfxU16)(pParams->nKeyIntSec * pParams->nFpsNum /
129
             (float)pParams->nFpsDen);
130
 
131
-   static mfxExtBuffer *extendedBuffers3;
132
-   int iBuffers = 0;
133
-
134
    if (m_ver.Major == 1 && m_ver.Minor >= 8) {
135
        memset(&m_co2, 0, sizeof(mfxExtCodingOption2));
136
        m_co2.Header.BufferId = MFX_EXTBUFF_CODING_OPTION2;
137
        m_co2.Header.BufferSz = sizeof(m_co2);
138
+       if (codec != QSV_CODEC_AVC)
139
+           m_co2.RepeatPPS = MFX_CODINGOPTION_ON;
140
        if (pParams->nRateControl == MFX_RATECONTROL_LA_ICQ ||
141
            pParams->nRateControl == MFX_RATECONTROL_LA)
142
            m_co2.LookAheadDepth = pParams->nLADEPTH;
143
@@ -281,34 +314,109 @@
144
        if (pParams->nbFrames > 1)
145
            m_co2.BRefType = MFX_B_REF_PYRAMID;
146
        if (m_mfxEncParams.mfx.LowPower == MFX_CODINGOPTION_ON) {
147
-           m_co2.RepeatPPS = MFX_CODINGOPTION_OFF;
148
+           if (codec == QSV_CODEC_AVC)
149
+               m_co2.RepeatPPS = MFX_CODINGOPTION_OFF;
150
            if (pParams->nRateControl == MFX_RATECONTROL_CBR ||
151
                pParams->nRateControl == MFX_RATECONTROL_VBR) {
152
                m_co2.LookAheadDepth = pParams->nLADEPTH;
153
            }
154
        }
155
-       extendedBuffersiBuffers++ = (mfxExtBuffer *)&m_co2;
156
+       extendedBuffers.push_back((mfxExtBuffer *)&m_co2);
157
    }
158
 
159
-   if (m_mfxEncParams.mfx.LowPower == MFX_CODINGOPTION_ON) {
160
+   if ((m_mfxEncParams.mfx.LowPower == MFX_CODINGOPTION_ON) ||
161
+       (pParams->bCQM && m_ver.Major == 1 && m_ver.Minor >= 16)) {
162
        memset(&m_co3, 0, sizeof(mfxExtCodingOption3));
163
        m_co3.Header.BufferId = MFX_EXTBUFF_CODING_OPTION3;
164
        m_co3.Header.BufferSz = sizeof(m_co3);
165
        m_co3.ScenarioInfo = MFX_SCENARIO_GAME_STREAMING;
166
-       extendedBuffersiBuffers++ = (mfxExtBuffer *)&m_co3;
167
-   } else if (pParams->bCQM) {
168
-       if (m_ver.Major == 1 && m_ver.Minor >= 16) {
169
-           memset(&m_co3, 0, sizeof(mfxExtCodingOption3));
170
-           m_co3.Header.BufferId = MFX_EXTBUFF_CODING_OPTION3;
171
-           m_co3.Header.BufferSz = sizeof(m_co3);
172
-           m_co3.ScenarioInfo = 7; // MFX_SCENARIO_GAME_STREAMING
173
-           extendedBuffersiBuffers++ = (mfxExtBuffer *)&m_co3;
174
+       extendedBuffers.push_back((mfxExtBuffer *)&m_co3);
175
+   }
176
+
177
+   if (codec == QSV_CODEC_HEVC) {
178
+       if ((pParams->nWidth & 15) || (pParams->nHeight & 15)) {
179
+           memset(&m_ExtHEVCParam, 0, sizeof(m_ExtHEVCParam));
180
+           m_ExtHEVCParam.Header.BufferId = MFX_EXTBUFF_HEVC_PARAM;
181
+           m_ExtHEVCParam.Header.BufferSz = sizeof(m_ExtHEVCParam);
182
+           m_ExtHEVCParam.PicWidthInLumaSamples = pParams->nWidth;
183
+           m_ExtHEVCParam.PicHeightInLumaSamples =
184
+               pParams->nHeight;
185
+           extendedBuffers.push_back(
186
+               (mfxExtBuffer *)&m_ExtHEVCParam);
187
        }
188
    }
189
 
190
-   if (iBuffers > 0) {
191
-       m_mfxEncParams.ExtParam = extendedBuffers;
192
-       m_mfxEncParams.NumExtParam = (mfxU16)iBuffers;
193
+   memset(&m_ExtVideoSignalInfo, 0, sizeof(m_ExtVideoSignalInfo));
194
+   m_ExtVideoSignalInfo.Header.BufferId = MFX_EXTBUFF_VIDEO_SIGNAL_INFO;
195
+   m_ExtVideoSignalInfo.Header.BufferSz = sizeof(m_ExtVideoSignalInfo);
196
+   m_ExtVideoSignalInfo.VideoFormat = pParams->VideoFormat;
197
+   m_ExtVideoSignalInfo.VideoFullRange = pParams->VideoFullRange;
198
+   m_ExtVideoSignalInfo.ColourDescriptionPresent = 1;
199
+   m_ExtVideoSignalInfo.ColourPrimaries = pParams->ColourPrimaries;
200
+   m_ExtVideoSignalInfo.TransferCharacteristics =
201
obs-studio-28.1.2.tar.xz/plugins/obs-qsv11/QSV_Encoder_Internal.h -> obs-studio-29.0.0.tar.xz/plugins/obs-qsv11/QSV_Encoder_Internal.h Changed
80
 
1
@@ -59,14 +59,18 @@
2
 #include "QSV_Encoder.h"
3
 #include "common_utils.h"
4
 
5
+#include <vector>
6
+
7
 class QSV_Encoder_Internal {
8
 public:
9
-   QSV_Encoder_Internal(mfxIMPL &impl, mfxVersion &version);
10
+   QSV_Encoder_Internal(mfxIMPL &impl, mfxVersion &version, bool isDGPU);
11
    ~QSV_Encoder_Internal();
12
 
13
-   mfxStatus Open(qsv_param_t *pParams);
14
+   mfxStatus Open(qsv_param_t *pParams, enum qsv_codec codec);
15
    void GetSPSPPS(mfxU8 **pSPSBuf, mfxU8 **pPPSBuf, mfxU16 *pnSPSBuf,
16
               mfxU16 *pnPPSBuf);
17
+   void GetVpsSpsPps(mfxU8 **pVPSBuf, mfxU8 **pSPSBuf, mfxU8 **pPPSBuf,
18
+             mfxU16 *pnVPSBuf, mfxU16 *pnSPSBuf, mfxU16 *pnPPSBuf);
19
    mfxStatus Encode(uint64_t ts, uint8_t *pDataY, uint8_t *pDataUV,
20
             uint32_t strideY, uint32_t strideUV,
21
             mfxBitstream **pBS);
22
@@ -74,16 +78,23 @@
23
                 uint64_t lock_key, uint64_t *next_key,
24
                 mfxBitstream **pBS);
25
    mfxStatus ClearData();
26
-   mfxStatus Reset(qsv_param_t *pParams);
27
+   mfxStatus Reset(qsv_param_t *pParams, enum qsv_codec codec);
28
+   mfxStatus ReconfigureEncoder();
29
+   bool UpdateParams(qsv_param_t *pParams);
30
+
31
+   bool IsDGPU() const { return m_isDGPU; }
32
 
33
 protected:
34
-   bool InitParams(qsv_param_t *pParams);
35
+   mfxStatus InitParams(qsv_param_t *pParams, enum qsv_codec codec);
36
    mfxStatus AllocateSurfaces();
37
-   mfxStatus GetVideoParam();
38
+   mfxStatus GetVideoParam(enum qsv_codec codec);
39
    mfxStatus InitBitstream();
40
    mfxStatus LoadNV12(mfxFrameSurface1 *pSurface, uint8_t *pDataY,
41
               uint8_t *pDataUV, uint32_t strideY,
42
               uint32_t strideUV);
43
+   mfxStatus LoadP010(mfxFrameSurface1 *pSurface, uint8_t *pDataY,
44
+              uint8_t *pDataUV, uint32_t strideY,
45
+              uint32_t strideUV);
46
    mfxStatus Drain();
47
    int GetFreeTaskIndex(Task *pTaskPool, mfxU16 nPoolSize);
48
 
49
@@ -97,14 +108,22 @@
50
    mfxFrameSurface1 **m_pmfxSurfaces;
51
    mfxU16 m_nSurfNum;
52
    MFXVideoENCODE *m_pmfxENC;
53
+   mfxU8 m_VPSBuffer1024;
54
    mfxU8 m_SPSBuffer1024;
55
    mfxU8 m_PPSBuffer1024;
56
+   mfxU16 m_nVPSBufferSize;
57
    mfxU16 m_nSPSBufferSize;
58
    mfxU16 m_nPPSBufferSize;
59
    mfxVideoParam m_parameter;
60
+   std::vector<mfxExtBuffer *> extendedBuffers;
61
    mfxExtCodingOption3 m_co3;
62
    mfxExtCodingOption2 m_co2;
63
    mfxExtCodingOption m_co;
64
+   mfxExtHEVCParam m_ExtHEVCParam{};
65
+   mfxExtVideoSignalInfo m_ExtVideoSignalInfo{};
66
+   mfxExtChromaLocInfo m_ExtChromaLocInfo{};
67
+   mfxExtMasteringDisplayColourVolume m_ExtMasteringDisplayColourVolume{};
68
+   mfxExtContentLightLevelInfo m_ExtContentLightLevelInfo{};
69
    mfxU16 m_nTaskPool;
70
    Task *m_pTaskPool;
71
    int m_nTaskIdx;
72
@@ -113,6 +132,7 @@
73
    bool m_bIsWindows8OrGreater;
74
    bool m_bUseD3D11;
75
    bool m_bD3D9HACK;
76
+   bool m_isDGPU;
77
    static mfxU16 g_numEncodersOpen;
78
    static mfxHDL
79
        g_DX_Handle; // we only want one handle for all instances to use;
80
obs-studio-28.1.2.tar.xz/plugins/obs-qsv11/common_directx11.cpp -> obs-studio-29.0.0.tar.xz/plugins/obs-qsv11/common_directx11.cpp Changed
25
 
1
@@ -169,6 +169,8 @@
2
         request->Info
3
             .FourCC) //|| MFX_FOURCC_P8_TEXTURE == request->Info.FourCC
4
        format = DXGI_FORMAT_P8;
5
+   else if (MFX_FOURCC_P010 == request->Info.FourCC)
6
+       format = DXGI_FORMAT_P010;
7
    else
8
        format = DXGI_FORMAT_UNKNOWN;
9
 
10
@@ -389,6 +391,14 @@
11
        ptr->U = 0;
12
        ptr->V = 0;
13
        break;
14
+   case DXGI_FORMAT_P010:
15
+       ptr->Pitch = (mfxU16)lockedRect.RowPitch;
16
+       ptr->PitchHigh = 0;
17
+       ptr->Y = (mfxU8 *)lockedRect.pData;
18
+       ptr->U = (mfxU8 *)lockedRect.pData +
19
+            desc.Height * lockedRect.RowPitch;
20
+       ptr->V = ptr->U + 2;
21
+       break;
22
    default:
23
        return MFX_ERR_LOCK_MEMORY;
24
    }
25
obs-studio-29.0.0.tar.xz/plugins/obs-qsv11/obs-qsv-test Added
2
 
1
+(directory)
2
obs-studio-29.0.0.tar.xz/plugins/obs-qsv11/obs-qsv-test/CMakeLists.txt Added
13
 
1
@@ -0,0 +1,11 @@
2
+project(obs-qsv-test)
3
+
4
+include_directories(SYSTEM ${CMAKE_SOURCE_DIR}/libobs)
5
+
6
+add_executable(obs-qsv-test)
7
+target_sources(obs-qsv-test PRIVATE obs-qsv-test.cpp)
8
+target_link_libraries(obs-qsv-test d3d11 dxgi dxguid OBS::libmfx)
9
+
10
+set_target_properties(obs-qsv-test PROPERTIES FOLDER "plugins/obs-qsv11")
11
+
12
+setup_binary_target(obs-qsv-test)
13
obs-studio-29.0.0.tar.xz/plugins/obs-qsv11/obs-qsv-test/obs-qsv-test.cpp Added
162
 
1
@@ -0,0 +1,160 @@
2
+#include "mfxstructures.h"
3
+#include "mfxadapter.h"
4
+#include "mfxvideo++.h"
5
+#include "../common_utils.h"
6
+
7
+#include <util/windows/ComPtr.hpp>
8
+
9
+#include <dxgi.h>
10
+#include <d3d11.h>
11
+#include <d3d11_1.h>
12
+
13
+#include <vector>
14
+#include <string>
15
+#include <map>
16
+
17
+#ifdef _MSC_VER
18
+extern "C" __declspec(dllexport) DWORD NvOptimusEnablement = 1;
19
+extern "C" __declspec(dllexport) int AmdPowerXpressRequestHighPerformance = 1;
20
+#endif
21
+
22
+#define INTEL_VENDOR_ID 0x8086
23
+
24
+struct adapter_caps {
25
+   bool is_intel = false;
26
+   bool is_dgpu = false;
27
+   bool supports_av1 = false;
28
+   bool supports_hevc = false;
29
+};
30
+
31
+static std::vector<uint64_t> luid_order;
32
+static std::map<uint32_t, adapter_caps> adapter_info;
33
+
34
+static bool has_encoder(mfxIMPL impl, mfxU32 codec_id)
35
+{
36
+   MFXVideoSession session;
37
+   mfxInitParam init_param = {};
38
+   init_param.Implementation = impl;
39
+   init_param.Version.Major = 1;
40
+   init_param.Version.Minor = 0;
41
+   mfxStatus sts = session.InitEx(init_param);
42
+
43
+   mfxVideoParam video_param;
44
+   memset(&video_param, 0, sizeof(video_param));
45
+   video_param.mfx.CodecId = codec_id;
46
+   video_param.mfx.FrameInfo.FourCC = MFX_FOURCC_NV12;
47
+   video_param.mfx.FrameInfo.ChromaFormat = MFX_CHROMAFORMAT_YUV420;
48
+   video_param.mfx.FrameInfo.Width = MSDK_ALIGN16(1280);
49
+   video_param.mfx.FrameInfo.Height = MSDK_ALIGN16(720);
50
+   sts = MFXVideoENCODE_Query(session, &video_param, &video_param);
51
+   session.Close();
52
+
53
+   return sts == MFX_ERR_NONE;
54
+}
55
+
56
+static inline uint32_t get_adapter_idx(uint32_t adapter_idx, LUID luid)
57
+{
58
+   for (size_t i = 0; i < luid_order.size(); i++) {
59
+       if (luid_orderi == *(uint64_t *)&luid) {
60
+           return (uint32_t)i;
61
+       }
62
+   }
63
+
64
+   return adapter_idx;
65
+}
66
+
67
+static bool get_adapter_caps(IDXGIFactory *factory, uint32_t adapter_idx)
68
+{
69
+   mfxIMPL impls4 = {MFX_IMPL_HARDWARE, MFX_IMPL_HARDWARE2,
70
+               MFX_IMPL_HARDWARE3, MFX_IMPL_HARDWARE4};
71
+   HRESULT hr;
72
+
73
+   ComPtr<IDXGIAdapter> adapter;
74
+   hr = factory->EnumAdapters(adapter_idx, &adapter);
75
+   if (FAILED(hr))
76
+       return false;
77
+
78
+   DXGI_ADAPTER_DESC desc;
79
+   adapter->GetDesc(&desc);
80
+
81
+   uint32_t luid_idx = get_adapter_idx(adapter_idx, desc.AdapterLuid);
82
+   adapter_caps &caps = adapter_infoluid_idx;
83
+   if (desc.VendorId != INTEL_VENDOR_ID)
84
+       return true;
85
+
86
+   bool dgpu = desc.DedicatedVideoMemory > 512 * 1024 * 1024;
87
+   mfxIMPL impl = implsadapter_idx;
88
+
89
+   caps.is_intel = true;
90
+   caps.is_dgpu = dgpu;
91
+   caps.supports_av1 = has_encoder(impl, MFX_CODEC_AV1);
92
+#if ENABLE_HEVC
93
+   caps.supports_hevc = has_encoder(impl, MFX_CODEC_HEVC);
94
+#endif
95
+
96
+   return true;
97
+}
98
+
99
+#define CHECK_TIMEOUT_MS 10000
100
+
101
+DWORD WINAPI TimeoutThread(LPVOID param)
102
+{
103
+   HANDLE hMainThread = (HANDLE)param;
104
+
105
+   DWORD ret = WaitForSingleObject(hMainThread, CHECK_TIMEOUT_MS);
106
+   if (ret == WAIT_TIMEOUT)
107
+       TerminateProcess(GetCurrentProcess(), STATUS_TIMEOUT);
108
+
109
+   CloseHandle(hMainThread);
110
+
111
+   return 0;
112
+}
113
+
114
+int main(int argc, char *argv)
115
+try {
116
+   ComPtr<IDXGIFactory> factory;
117
+   HRESULT hr;
118
+
119
+   HANDLE hMainThread;
120
+   DuplicateHandle(GetCurrentProcess(), GetCurrentThread(),
121
+           GetCurrentProcess(), &hMainThread, 0, FALSE,
122
+           DUPLICATE_SAME_ACCESS);
123
+   DWORD threadId;
124
+   HANDLE hThread;
125
+   hThread =
126
+       CreateThread(NULL, 0, TimeoutThread, hMainThread, 0, &threadId);
127
+   CloseHandle(hThread);
128
+
129
+   /* --------------------------------------------------------- */
130
+   /* parse expected LUID order                                 */
131
+
132
+   for (int i = 1; i < argc; i++) {
133
+       luid_order.push_back(strtoull(argvi, NULL, 16));
134
+   }
135
+
136
+   /* --------------------------------------------------------- */
137
+   /* query qsv support                                         */
138
+
139
+   hr = CreateDXGIFactory1(__uuidof(IDXGIFactory), (void **)&factory);
140
+   if (FAILED(hr))
141
+       throw "CreateDXGIFactory1 failed";
142
+
143
+   uint32_t idx = 0;
144
+   while (get_adapter_caps(factory, idx++) && idx < 4)
145
+       ;
146
+
147
+   for (auto &idx, caps : adapter_info) {
148
+       printf("%u\n", idx);
149
+       printf("is_intel=%s\n", caps.is_intel ? "true" : "false");
150
+       printf("is_dgpu=%s\n", caps.is_dgpu ? "true" : "false");
151
+       printf("supports_av1=%s\n",
152
+              caps.supports_av1 ? "true" : "false");
153
+       printf("supports_hevc=%s\n",
154
+              caps.supports_hevc ? "true" : "false");
155
+   }
156
+
157
+   return 0;
158
+} catch (const char *text) {
159
+   printf("error\nstring=%s\n", text);
160
+   return 0;
161
+}
162
obs-studio-28.1.2.tar.xz/plugins/obs-qsv11/obs-qsv11-plugin-main.c -> obs-studio-29.0.0.tar.xz/plugins/obs-qsv11/obs-qsv11-plugin-main.c Changed
155
 
1
@@ -54,39 +54,136 @@
2
 OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
3
 */
4
 #include <obs-module.h>
5
-#include "mfxsession.h"
6
+#include <util/windows/device-enum.h>
7
+#include <util/config-file.h>
8
+#include <util/platform.h>
9
+#include <util/pipe.h>
10
+#include <util/dstr.h>
11
+#include "QSV_Encoder.h"
12
 
13
 OBS_DECLARE_MODULE()
14
 OBS_MODULE_USE_DEFAULT_LOCALE("obs-qsv11", "en-US")
15
 MODULE_EXPORT const char *obs_module_description(void)
16
 {
17
-   return "Intel Quick Sync Video H.264 encoder support for Windows";
18
+   return "Intel Quick Sync Video support for Windows";
19
 }
20
 
21
 extern struct obs_encoder_info obs_qsv_encoder;
22
+extern struct obs_encoder_info obs_qsv_encoder_v2;
23
 extern struct obs_encoder_info obs_qsv_encoder_tex;
24
+extern struct obs_encoder_info obs_qsv_encoder_tex_v2;
25
+extern struct obs_encoder_info obs_qsv_av1_encoder_tex;
26
+extern struct obs_encoder_info obs_qsv_av1_encoder;
27
+extern struct obs_encoder_info obs_qsv_hevc_encoder_tex;
28
+extern struct obs_encoder_info obs_qsv_hevc_encoder;
29
+
30
+extern bool av1_supported(mfxIMPL impl);
31
+
32
+struct adapter_info adaptersMAX_ADAPTERS = {0};
33
+size_t adapter_count = 0;
34
+
35
+static bool enum_luids(void *param, uint32_t idx, uint64_t luid)
36
+{
37
+   struct dstr *cmd = param;
38
+   dstr_catf(cmd, " %llX", luid);
39
+   UNUSED_PARAMETER(idx);
40
+   return true;
41
+}
42
 
43
 bool obs_module_load(void)
44
 {
45
-   mfxIMPL impl = MFX_IMPL_HARDWARE_ANY | MFX_IMPL_VIA_D3D11;
46
-   mfxVersion ver = {{0, 1}};
47
-   mfxSession session;
48
-   mfxStatus sts;
49
+   char *test_exe = os_get_executable_path_ptr("obs-qsv-test.exe");
50
+   struct dstr cmd = {0};
51
+   struct dstr caps_str = {0};
52
+   os_process_pipe_t *pp = NULL;
53
+   config_t *config = NULL;
54
+
55
+   dstr_copy(&cmd, test_exe);
56
+   enum_graphics_device_luids(enum_luids, &cmd);
57
+
58
+   pp = os_process_pipe_create(cmd.array, "r");
59
+   if (!pp) {
60
+       blog(LOG_INFO, "Failed to launch the QSV test process I guess");
61
+       goto fail;
62
+   }
63
 
64
-   sts = MFXInit(impl, &ver, &session);
65
+   for (;;) {
66
+       char data2048;
67
+       size_t len =
68
+           os_process_pipe_read(pp, (uint8_t *)data, sizeof(data));
69
+       if (!len)
70
+           break;
71
 
72
-   if (sts == MFX_ERR_NONE) {
73
-       obs_register_encoder(&obs_qsv_encoder);
74
+       dstr_ncat(&caps_str, data, len);
75
+   }
76
+
77
+   if (dstr_is_empty(&caps_str)) {
78
+       blog(LOG_INFO, "Seems the QSV test subprocess crashed. "
79
+                  "Better there than here I guess. "
80
+                  "Let's just skip loading QSV then I suppose.");
81
+       goto fail;
82
+   }
83
+
84
+   if (config_open_string(&config, caps_str.array) != 0) {
85
+       blog(LOG_INFO, "Couldn't open QSV configuration string");
86
+       goto fail;
87
+   }
88
+
89
+   const char *error = config_get_string(config, "error", "string");
90
+   if (error) {
91
+       blog(LOG_INFO, "Error querying QSV support: %s", error);
92
+       goto fail;
93
+   }
94
+
95
+   adapter_count = config_num_sections(config);
96
+   bool avc_supported = false;
97
+   bool av1_supported = false;
98
+   bool hevc_supported = false;
99
+
100
+   if (adapter_count > MAX_ADAPTERS)
101
+       adapter_count = MAX_ADAPTERS;
102
+
103
+   for (size_t i = 0; i < adapter_count; i++) {
104
+       char section16;
105
+       snprintf(section, sizeof(section), "%d", (int)i);
106
+
107
+       struct adapter_info *adapter = &adaptersi;
108
+       adapter->is_intel =
109
+           config_get_bool(config, section, "is_intel");
110
+       adapter->is_dgpu = config_get_bool(config, section, "is_dgpu");
111
+       adapter->supports_av1 =
112
+           config_get_bool(config, section, "supports_av1");
113
+       adapter->supports_hevc =
114
+           config_get_bool(config, section, "supports_hevc");
115
+
116
+       avc_supported |= adapter->is_intel;
117
+       av1_supported |= adapter->supports_av1;
118
+       hevc_supported |= adapter->supports_hevc;
119
+   }
120
+
121
+   if (avc_supported) {
122
+       obs_register_encoder(&obs_qsv_encoder_tex_v2);
123
        obs_register_encoder(&obs_qsv_encoder_tex);
124
-       MFXClose(session);
125
-   } else {
126
-       impl = MFX_IMPL_HARDWARE_ANY | MFX_IMPL_VIA_D3D9;
127
-       sts = MFXInit(impl, &ver, &session);
128
-       if (sts == MFX_ERR_NONE) {
129
-           obs_register_encoder(&obs_qsv_encoder);
130
-           MFXClose(session);
131
-       }
132
+       obs_register_encoder(&obs_qsv_encoder_v2);
133
+       obs_register_encoder(&obs_qsv_encoder);
134
+   }
135
+   if (av1_supported) {
136
+       obs_register_encoder(&obs_qsv_av1_encoder_tex);
137
+       obs_register_encoder(&obs_qsv_av1_encoder);
138
+   }
139
+#if ENABLE_HEVC
140
+   if (hevc_supported) {
141
+       obs_register_encoder(&obs_qsv_hevc_encoder_tex);
142
+       obs_register_encoder(&obs_qsv_hevc_encoder);
143
    }
144
+#endif
145
+
146
+fail:
147
+   config_close(config);
148
+   dstr_free(&caps_str);
149
+   dstr_free(&cmd);
150
+   os_process_pipe_destroy(pp);
151
+   bfree(test_exe);
152
 
153
    return true;
154
 }
155
obs-studio-28.1.2.tar.xz/plugins/obs-qsv11/obs-qsv11.c -> obs-studio-29.0.0.tar.xz/plugins/obs-qsv11/obs-qsv11.c Changed
201
 
1
@@ -59,6 +59,7 @@
2
 #include <util/darray.h>
3
 #include <util/platform.h>
4
 #include <obs-module.h>
5
+#include <obs-hevc.h>
6
 #include <obs-avc.h>
7
 #include <d3d11.h>
8
 #include <dxgi1_2.h>
9
@@ -84,6 +85,8 @@
10
 struct obs_qsv {
11
    obs_encoder_t *encoder;
12
 
13
+   enum qsv_codec codec;
14
+
15
    qsv_param_t params;
16
    qsv_t *context;
17
 
18
@@ -107,12 +110,30 @@
19
 static int64_t g_prevDts;
20
 static bool g_bFirst;
21
 
22
+static const char *obs_qsv_getname_v1(void *type_data)
23
+{
24
+   UNUSED_PARAMETER(type_data);
25
+   return "QuickSync H.264 (v1 deprecated)";
26
+}
27
+
28
 static const char *obs_qsv_getname(void *type_data)
29
 {
30
    UNUSED_PARAMETER(type_data);
31
    return "QuickSync H.264";
32
 }
33
 
34
+static const char *obs_qsv_getname_av1(void *type_data)
35
+{
36
+   UNUSED_PARAMETER(type_data);
37
+   return "QuickSync AV1";
38
+}
39
+
40
+static const char *obs_qsv_getname_hevc(void *type_data)
41
+{
42
+   UNUSED_PARAMETER(type_data);
43
+   return "QuickSync HEVC";
44
+}
45
+
46
 static void obs_qsv_stop(void *data);
47
 
48
 static void clear_data(struct obs_qsv *obsqsv)
49
@@ -143,16 +164,21 @@
50
    }
51
 }
52
 
53
-static void obs_qsv_defaults(obs_data_t *settings)
54
+static void obs_qsv_defaults(obs_data_t *settings, int ver,
55
+                enum qsv_codec codec)
56
 {
57
    obs_data_set_default_string(settings, "target_usage", "balanced");
58
    obs_data_set_default_int(settings, "bitrate", 2500);
59
    obs_data_set_default_int(settings, "max_bitrate", 3000);
60
-   obs_data_set_default_string(settings, "profile", "high");
61
+   obs_data_set_default_string(settings, "profile",
62
+                   codec == QSV_CODEC_AVC ? "high" : "main");
63
    obs_data_set_default_string(settings, "rate_control", "CBR");
64
 
65
+   obs_data_set_default_int(settings, "__ver", ver);
66
+
67
    obs_data_set_default_int(settings, "accuracy", 1000);
68
    obs_data_set_default_int(settings, "convergence", 1);
69
+   obs_data_set_default_int(settings, "cqp", 23);
70
    obs_data_set_default_int(settings, "qpi", 23);
71
    obs_data_set_default_int(settings, "qpp", 23);
72
    obs_data_set_default_int(settings, "qpb", 23);
73
@@ -164,6 +190,26 @@
74
    obs_data_set_default_bool(settings, "enhancements", false);
75
 }
76
 
77
+static void obs_qsv_defaults_h264_v1(obs_data_t *settings)
78
+{
79
+   obs_qsv_defaults(settings, 1, QSV_CODEC_AVC);
80
+}
81
+
82
+static void obs_qsv_defaults_h264_v2(obs_data_t *settings)
83
+{
84
+   obs_qsv_defaults(settings, 2, QSV_CODEC_AVC);
85
+}
86
+
87
+static void obs_qsv_defaults_av1(obs_data_t *settings)
88
+{
89
+   obs_qsv_defaults(settings, 2, QSV_CODEC_AV1);
90
+}
91
+
92
+static void obs_qsv_defaults_hevc(obs_data_t *settings)
93
+{
94
+   obs_qsv_defaults(settings, 2, QSV_CODEC_HEVC);
95
+}
96
+
97
 static inline void add_strings(obs_property_t *list, const char *const *strings)
98
 {
99
    while (*strings) {
100
@@ -287,11 +333,17 @@
101
 
102
    bVisible = astrcmpi(rate_control, "CQP") == 0;
103
    p = obs_properties_get(ppts, "qpi");
104
-   obs_property_set_visible(p, bVisible);
105
+   if (p)
106
+       obs_property_set_visible(p, bVisible);
107
    p = obs_properties_get(ppts, "qpb");
108
-   obs_property_set_visible(p, bVisible);
109
+   if (p)
110
+       obs_property_set_visible(p, bVisible);
111
    p = obs_properties_get(ppts, "qpp");
112
-   obs_property_set_visible(p, bVisible);
113
+   if (p)
114
+       obs_property_set_visible(p, bVisible);
115
+   p = obs_properties_get(ppts, "cqp");
116
+   if (p)
117
+       obs_property_set_visible(p, bVisible);
118
 
119
    bVisible = astrcmpi(rate_control, "ICQ") == 0 ||
120
           astrcmpi(rate_control, "LA_ICQ") == 0;
121
@@ -335,13 +387,45 @@
122
    }
123
 }
124
 
125
-static obs_properties_t *obs_qsv_props(void *unused)
126
+static obs_properties_t *obs_qsv_props(enum qsv_codec codec, void *unused,
127
+                      int ver)
128
 {
129
    UNUSED_PARAMETER(unused);
130
 
131
    obs_properties_t *props = obs_properties_create();
132
    obs_property_t *prop;
133
 
134
+   prop = obs_properties_add_list(props, "rate_control", TEXT_RATE_CONTROL,
135
+                      OBS_COMBO_TYPE_LIST,
136
+                      OBS_COMBO_FORMAT_STRING);
137
+
138
+   if (codec == QSV_CODEC_AVC || codec == QSV_CODEC_HEVC)
139
+       add_rate_controls(prop, qsv_ratecontrols);
140
+   else if (codec == QSV_CODEC_AV1)
141
+       add_rate_controls(prop, qsv_av1_ratecontrols);
142
+
143
+   obs_property_set_modified_callback(prop, rate_control_modified);
144
+
145
+   prop = obs_properties_add_int(props, "bitrate", TEXT_TARGET_BITRATE, 50,
146
+                     10000000, 50);
147
+   obs_property_int_set_suffix(prop, " Kbps");
148
+
149
+   prop = obs_properties_add_int(props, "max_bitrate", TEXT_MAX_BITRATE,
150
+                     50, 10000000, 50);
151
+   obs_property_int_set_suffix(prop, " Kbps");
152
+
153
+   if (ver >= 2) {
154
+       obs_properties_add_int(props, "cqp", "CQP", 1,
155
+                      codec == QSV_CODEC_AV1 ? 63 : 51, 1);
156
+   } else {
157
+       obs_properties_add_int(props, "qpi", "QPI", 1, 51, 1);
158
+       obs_properties_add_int(props, "qpp", "QPP", 1, 51, 1);
159
+       obs_properties_add_int(props, "qpb", "QPB", 1, 51, 1);
160
+   }
161
+
162
+   obs_properties_add_int(props, "icq_quality", TEXT_ICQ_QUALITY, 1, 51,
163
+                  1);
164
+
165
    prop = obs_properties_add_list(props, "target_usage", TEXT_SPEED,
166
                       OBS_COMBO_TYPE_LIST,
167
                       OBS_COMBO_FORMAT_STRING);
168
@@ -350,7 +434,13 @@
169
    prop = obs_properties_add_list(props, "profile", TEXT_PROFILE,
170
                       OBS_COMBO_TYPE_LIST,
171
                       OBS_COMBO_FORMAT_STRING);
172
-   add_strings(prop, qsv_profile_names);
173
+
174
+   if (codec == QSV_CODEC_AVC)
175
+       add_strings(prop, qsv_profile_names);
176
+   else if (codec == QSV_CODEC_AV1)
177
+       add_strings(prop, qsv_profile_names_av1);
178
+   else if (codec == QSV_CODEC_HEVC)
179
+       add_strings(prop, qsv_profile_names_hevc);
180
 
181
    obs_property_set_modified_callback(prop, profile_modified);
182
 
183
@@ -358,28 +448,9 @@
184
                      20, 1);
185
    obs_property_int_set_suffix(prop, " s");
186
 
187
-   prop = obs_properties_add_list(props, "rate_control", TEXT_RATE_CONTROL,
188
-                      OBS_COMBO_TYPE_LIST,
189
-                      OBS_COMBO_FORMAT_STRING);
190
-   add_rate_controls(prop, qsv_ratecontrols);
191
-   obs_property_set_modified_callback(prop, rate_control_modified);
192
-
193
-   prop = obs_properties_add_int(props, "bitrate", TEXT_TARGET_BITRATE, 50,
194
-                     10000000, 50);
195
-   obs_property_int_set_suffix(prop, " Kbps");
196
-
197
-   prop = obs_properties_add_int(props, "max_bitrate", TEXT_MAX_BITRATE,
198
-                     50, 10000000, 50);
199
-   obs_property_int_set_suffix(prop, " Kbps");
200
-
201
obs-studio-28.1.2.tar.xz/plugins/obs-vst/VSTPlugin.cpp -> obs-studio-29.0.0.tar.xz/plugins/obs-vst/VSTPlugin.cpp Changed
76
 
1
@@ -96,14 +96,14 @@
2
    numChannels = (std::max)((size_t)0, count);
3
 
4
    if (numChannels > 0) {
5
-       inputs = (float **)malloc(sizeof(float *) * numChannels);
6
-       outputs = (float **)malloc(sizeof(float *) * numChannels);
7
-       channelrefs = (float **)malloc(sizeof(float *) * numChannels);
8
+       inputs = (float **)bmalloc(sizeof(float *) * numChannels);
9
+       outputs = (float **)bmalloc(sizeof(float *) * numChannels);
10
+       channelrefs = (float **)bmalloc(sizeof(float *) * numChannels);
11
        for (size_t channel = 0; channel < numChannels; channel++) {
12
            inputschannel =
13
-               (float *)malloc(sizeof(float) * blocksize);
14
+               (float *)bmalloc(sizeof(float) * blocksize);
15
            outputschannel =
16
-               (float *)malloc(sizeof(float) * blocksize);
17
+               (float *)bmalloc(sizeof(float) * blocksize);
18
        }
19
    }
20
 }
21
@@ -112,30 +112,30 @@
22
 {
23
    for (size_t channel = 0; channel < numChannels; channel++) {
24
        if (inputs && inputschannel) {
25
-           free(inputschannel);
26
+           bfree(inputschannel);
27
            inputschannel = NULL;
28
        }
29
        if (outputs && outputschannel) {
30
-           free(outputschannel);
31
+           bfree(outputschannel);
32
            outputschannel = NULL;
33
        }
34
    }
35
    if (inputs) {
36
-       free(inputs);
37
+       bfree(inputs);
38
        inputs = NULL;
39
    }
40
    if (outputs) {
41
-       free(outputs);
42
+       bfree(outputs);
43
        outputs = NULL;
44
    }
45
    if (channelrefs) {
46
-       free(channelrefs);
47
+       bfree(channelrefs);
48
        channelrefs = NULL;
49
    }
50
    numChannels = 0;
51
 }
52
 
53
-void VSTPlugin::loadEffectFromPath(std::string path)
54
+void VSTPlugin::loadEffectFromPath(const std::string &path)
55
 {
56
    if (this->pluginPath.compare(path) != 0) {
57
        unloadEffect();
58
@@ -309,7 +309,7 @@
59
 
60
    unloadLibrary();
61
 
62
-   pluginPath = "";
63
+   pluginPath.clear();
64
 }
65
 
66
 bool VSTPlugin::isEditorOpen()
67
@@ -403,7 +403,7 @@
68
    }
69
 }
70
 
71
-void VSTPlugin::setChunk(std::string data)
72
+void VSTPlugin::setChunk(const std::string &data)
73
 {
74
    if (!effect) {
75
        return;
76
obs-studio-28.1.2.tar.xz/plugins/obs-vst/headers/VSTPlugin.h -> obs-studio-29.0.0.tar.xz/plugins/obs-vst/headers/VSTPlugin.h Changed
15
 
1
@@ -88,11 +88,11 @@
2
 public:
3
    VSTPlugin(obs_source_t *sourceContext);
4
    ~VSTPlugin();
5
-   void loadEffectFromPath(std::string path);
6
+   void loadEffectFromPath(const std::string &path);
7
    void unloadEffect();
8
    std::string getEffectPath();
9
    std::string getChunk();
10
-   void setChunk(std::string data);
11
+   void setChunk(const std::string &data);
12
    void setProgram(const int programNumber);
13
    int getProgram();
14
    void getSourceNames();
15
obs-studio-28.1.2.tar.xz/plugins/obs-vst/obs-vst.cpp -> obs-studio-29.0.0.tar.xz/plugins/obs-vst/obs-vst.cpp Changed
37
 
1
@@ -111,7 +111,20 @@
2
 
3
    const char *path = obs_data_get_string(settings, "plugin_path");
4
 
5
-   if (strcmp(path, "") == 0) {
6
+#ifdef __linux__
7
+   // Migrate freedesktop.org Flatpak runtime 21.08 VST paths to 22.08.
8
+   if (QFile::exists("/.flatpak-info") &&
9
+       QString(path).startsWith("/app/extensions/Plugins/lxvst")) {
10
+       QString newPath(path);
11
+       newPath.replace("/app/extensions/Plugins/lxvst",
12
+               "/app/extensions/Plugins/vst");
13
+       obs_data_set_string(settings, "plugin_path",
14
+                   newPath.toStdString().c_str());
15
+       path = obs_data_get_string(settings, "plugin_path");
16
+   }
17
+#endif
18
+
19
+   if (!*path) {
20
        vstPlugin->unloadEffect();
21
        return;
22
    }
23
@@ -121,10 +134,10 @@
24
    const char *chunkHash = obs_data_get_string(settings, "chunk_hash");
25
    const char *chunkData = obs_data_get_string(settings, "chunk_data");
26
 
27
-   bool chunkHashesMatch = chunkHash && strlen(chunkHash) > 0 &&
28
+   bool chunkHashesMatch = chunkHash && *chunkHash &&
29
                hash.compare(chunkHash) == 0;
30
-   if (chunkData && strlen(chunkData) > 0 &&
31
-       (chunkHashesMatch || !chunkHash || strlen(chunkHash) == 0)) {
32
+   if (chunkData && *chunkData &&
33
+       (chunkHashesMatch || !chunkHash || !*chunkHash)) {
34
        vstPlugin->setChunk(std::string(chunkData));
35
    }
36
 }
37
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/.github/ISSUE_TEMPLATE/bug_report.yaml -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/.github/ISSUE_TEMPLATE/bug_report.yaml Changed
43
 
1
@@ -28,10 +28,8 @@
2
     - macOS 10.14
3
     - macOS 10.13
4
     - Ubuntu 22.04 LTS
5
-    - Ubuntu 21.04
6
     - Ubuntu 20.10
7
     - Ubuntu 20.04 LTS
8
-    - Ubuntu 18.04 LTS
9
     - Other
10
   validations:
11
     required: true
12
@@ -49,16 +47,10 @@
13
     label: OBS Studio Version
14
     description: What version of OBS Studio are you using?
15
     options:
16
-    - 28.0.0
17
+    - 29.0.x
18
+    - 28.1.x
19
+    - 28.0.x
20
     - 27.2.4
21
-    - 27.2.3
22
-    - 27.2.2
23
-    - 27.2.1
24
-    - 27.2.0
25
-    - 27.1.3
26
-    - 27.1.1
27
-    - 27.1.0
28
-    - 27.0.1
29
     - Git
30
     - Other
31
   validations:
32
@@ -76,9 +68,9 @@
33
     label: obs-websocket Version
34
     description: What version of obs-websocket are you using?
35
     options:
36
+    - 5.1.0
37
     - 5.0.1
38
     - 5.0.0
39
-    - 5.0.0-beta1
40
     - Git
41
   validations:
42
     required: true
43
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/CMakeLists.txt -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/CMakeLists.txt Changed
7
 
1
@@ -1,4 +1,4 @@
2
-project(obs-websocket VERSION 5.0.1)
3
+project(obs-websocket VERSION 5.1.0)
4
 set(OBS_WEBSOCKET_RPC_VERSION 1)
5
 
6
 option(ENABLE_WEBSOCKET "Enable building OBS with websocket plugin" ON)
7
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/README.md -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/README.md Changed
32
 
1
@@ -31,21 +31,30 @@
2
 
3
 ### Client software
4
 
5
+- Macro Deck(https://www.macrodeck.org/)
6
 - Touch Portal(https://www.touch-portal.com/)
7
 - Twitchat(https://twitchat.fr/)
8
 - OBS-web(https://github.com/Niek/obs-web) - hosted client at obs-web.niek.tv/(http://obs-web.niek.tv/)
9
+- Streamer.bot(https://streamer.bot/)
10
+- Deckboard(https://deckboard.app/)
11
+- OBS Blade(https://github.com/Kounex/obs_blade)
12
+- Aitum(https://aitum.tv/)
13
+- Kruiz Control(https://github.com/Kruiser8/Kruiz-Control)
14
+- Bitfocus Companion Module(https://bitfocus.io/companion/)
15
 
16
 ### Client libraries (for developers)
17
 
18
 Here's a list of available language APIs for obs-websocket:
19
 
20
 - Python 3.7+ (Asyncio): simpleobsws(https://github.com/IRLToolkit/simpleobsws/tree/master) by IRLToolkit
21
+- Python 3.10+ (Non-Asyncio): obsws-python(https://pypi.org/project/obsws-python) by aatikturk and onyx-and-iris
22
 - Rust: obws(https://github.com/dnaka91/obws) by dnaka91
23
 - Godot 3.4.x: obs-websocket-gd(https://github.com/you-win/obs-websocket-gd) by you-win
24
 - Javascript (Node and web): obs-websocket-js(https://github.com/obs-websocket-community-projects/obs-websocket-js) by OBS Websocket Community
25
   - C (uses obs-websocket-js): v8-libwebsocket-obs-websocket(https://github.com/dgatwood/v8-libwebsocket-obs-websocket)
26
 - Go: goobs(https://github.com/andreykaipov/goobs) by andreykaipov
27
 - Dart/Flutter (can target all supported platforms): obs_websocket(https://github.com/faithoflifedev/obs_websocket) by faithoflifedev
28
+- Java: obs-websocket-java(https://github.com/obs-websocket-community-projects/obs-websocket-java) by OBS Websocket Community
29
 
30
 The 5.x server is a typical WebSocket server running by default on port 4455 (the port number can be changed in the Settings dialog under `Tools`).
31
 The protocol we use is documented in PROTOCOL.md(docs/generated/protocol.md).
32
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/data/locale/en-US.ini -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/data/locale/en-US.ini Changed
9
 
1
@@ -1,6 +1,6 @@
2
 OBSWebSocket.Plugin.Description="Remote-control of OBS Studio through WebSocket"
3
 
4
-OBSWebSocket.Settings.DialogTitle="obs-websocket Settings"
5
+OBSWebSocket.Settings.DialogTitle="WebSocket Server Settings"
6
 
7
 OBSWebSocket.Settings.PluginSettingsTitle="Plugin Settings"
8
 OBSWebSocket.Settings.ServerEnable="Enable WebSocket server"
9
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/data/locale/fa-IR.ini -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/data/locale/fa-IR.ini Changed
45
 
1
@@ -1,3 +1,43 @@
2
+OBSWebSocket.Plugin.Description="کنترل از راه دور OBS Studio از طریق WebSocket"
3
+OBSWebSocket.Settings.DialogTitle="تنظیمات obs-سوکت وب"
4
+OBSWebSocket.Settings.PluginSettingsTitle="تنظیمات پلاگین"
5
+OBSWebSocket.Settings.ServerEnable="سرور سوکت وب را فعال کنید"
6
+OBSWebSocket.Settings.AlertsEnable="هشدارهای سینی سیستم را فعال کنید"
7
+OBSWebSocket.Settings.DebugEnable="فعال کردن گزارش اشکال زدایی"
8
+OBSWebSocket.Settings.DebugEnableHoverText="ثبت اشکال زدایی را برای نمونه فعلی OBS فعال می کند. در بارگذاری ادامه نمییابد.\n برای فعال کردن در بارگذاری از --websocket_debug استفاده کنید."
9
+OBSWebSocket.Settings.ServerSettingsTitle="تنظیمات سرور"
10
+OBSWebSocket.Settings.AuthRequired="فعال کردن احراز هویت"
11
+OBSWebSocket.Settings.Password="رمز سرور"
12
+OBSWebSocket.Settings.GeneratePassword="ایجاد رمز عبور"
13
+OBSWebSocket.Settings.ServerPort="پورت سرور"
14
+OBSWebSocket.Settings.ShowConnectInfo="نمایش اطلاعات اتصال"
15
+OBSWebSocket.Settings.ShowConnectInfoWarningTitle="هشدار: در حال حاضر زنده است"
16
+OBSWebSocket.Settings.ShowConnectInfoWarningMessage="به نظر می رسد که یک خروجی (جریان، ضبط و غیره) در حال حاضر فعال است."
17
+OBSWebSocket.Settings.ShowConnectInfoWarningInfoText="آیا مطمئن هستید که می خواهید اطلاعات اتصال خود را نشان دهید؟"
18
+OBSWebSocket.Settings.Save.UserPasswordWarningTitle="هشدار: مشکل امنیتی احتمالی"
19
+OBSWebSocket.Settings.Save.UserPasswordWarningMessage="obs-websocket رمز عبور سرور را به صورت متن ساده ذخیره می کند. استفاده از رمز عبور تولید شده توسط obs-websocket بسیار توصیه می شود."
20
+OBSWebSocket.Settings.Save.UserPasswordWarningInfoText="آیا مطمئن هستید که می خواهید از رمز عبور خود استفاده کنید؟"
21
+OBSWebSocket.Settings.Save.PasswordInvalidErrorTitle="خطا: پیکربندی نامعتبر است"
22
+OBSWebSocket.Settings.Save.PasswordInvalidErrorMessage="باید از رمز عبور 6 کاراکتر یا بیشتر استفاده کنید."
23
+OBSWebSocket.SessionTable.Title="جلسات سوکت وب متصل"
24
+OBSWebSocket.SessionTable.RemoteAddressColumnTitle="آدرس از راه دور"
25
+OBSWebSocket.SessionTable.SessionDurationColumnTitle="مدت زمان جلسه"
26
+OBSWebSocket.SessionTable.MessagesInOutColumnTitle="پیام های ورودی/خارجی"
27
+OBSWebSocket.SessionTable.IdentifiedTitle="تایید هویت شده"
28
 OBSWebSocket.SessionTable.KickButtonColumnTitle="لگد زدن؟"
29
 OBSWebSocket.SessionTable.KickButtonText="اخراج"
30
+OBSWebSocket.ConnectInfo.DialogTitle="اطلاعات اتصال سوکت وب"
31
 OBSWebSocket.ConnectInfo.CopyText="رونوشت"
32
+OBSWebSocket.ConnectInfo.ServerIp="IP سرور (بهترین حدس)"
33
+OBSWebSocket.ConnectInfo.ServerPort="پورت سرور"
34
+OBSWebSocket.ConnectInfo.ServerPassword="رمز سرور"
35
+OBSWebSocket.ConnectInfo.ServerPasswordPlaceholderText="احراز غیر فعال"
36
+OBSWebSocket.ConnectInfo.QrTitle="QR را وصل کنید"
37
+OBSWebSocket.TrayNotification.Identified.Title="اتصال سوکت وب جدید"
38
+OBSWebSocket.TrayNotification.Identified.Body="سرویس گیرنده %1 شناسایی شد."
39
+OBSWebSocket.TrayNotification.AuthenticationFailed.Title="خرابی تأیید اعتبار سوکت وب"
40
+OBSWebSocket.TrayNotification.AuthenticationFailed.Body="سرویس گیرنده %1 احراز هویت نشد."
41
+OBSWebSocket.TrayNotification.Disconnected.Title="سرویس گیرنده سوکت وب قطع شد"
42
+OBSWebSocket.TrayNotification.Disconnected.Body="سرویس گیرنده %1 قطع شد."
43
+OBSWebSocket.Server.StartFailed.Title="خرابی سرور سوکت وب"
44
+OBSWebSocket.Server.StartFailed.Message="سرور WebSocket راهاندازی نشد. درگاه TCP % 1 ممکن است قبلاً در جای دیگری در این سیستم توسط برنامه دیگری استفاده شده باشد. یک پورت TCP دیگر را در تنظیمات سرور WebSocket تنظیم کنید، یا هر برنامهای را که میتواند از این پورت استفاده کند متوقف کنید.\n پیام خطا: %2"
45
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/data/locale/pt-BR.ini -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/data/locale/pt-BR.ini Changed
10
 
1
@@ -22,7 +22,7 @@
2
 OBSWebSocket.SessionTable.Title="Sessões WebSocket Conectadas"
3
 OBSWebSocket.SessionTable.RemoteAddressColumnTitle="Endereço Remoto"
4
 OBSWebSocket.SessionTable.SessionDurationColumnTitle="Duração de Sessão"
5
-OBSWebSocket.SessionTable.MessagesInOutColumnTitle="Mengsagens de Entrada/Saída"
6
+OBSWebSocket.SessionTable.MessagesInOutColumnTitle="Mensagens"
7
 OBSWebSocket.SessionTable.IdentifiedTitle="Identificada"
8
 OBSWebSocket.SessionTable.KickButtonColumnTitle="Expulsar?"
9
 OBSWebSocket.SessionTable.KickButtonText="Expulsar"
10
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/data/locale/ro-RO.ini -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/data/locale/ro-RO.ini Changed
16
 
1
@@ -12,11 +12,11 @@
2
 OBSWebSocket.Settings.ServerPort="Portul serverului"
3
 OBSWebSocket.Settings.ShowConnectInfo="Afișează informațiile de conectare"
4
 OBSWebSocket.Settings.ShowConnectInfoWarningTitle="Avertisment: În prezent în direct"
5
-OBSWebSocket.Settings.ShowConnectInfoWarningMessage="Se pare că o ieșire (flux, înregistrare etc.) este activă în prezent."
6
-OBSWebSocket.Settings.ShowConnectInfoWarningInfoText="Ești sigur că vrei să-ți arăți informațiile de conectare?"
7
+OBSWebSocket.Settings.ShowConnectInfoWarningMessage="Se pare că un output (transmisiune, înregistrare etc.) este activ în prezent."
8
+OBSWebSocket.Settings.ShowConnectInfoWarningInfoText="Sigur vrei să afișezi informațiile de conectare?"
9
 OBSWebSocket.Settings.Save.UserPasswordWarningTitle="Avertisment: Potențială problemă de securitate"
10
 OBSWebSocket.Settings.Save.UserPasswordWarningMessage="obs-websocket stochează parola serverului ca text simplu. Este foarte recomandat să folosiți o parolă generată de obs-websocket."
11
-OBSWebSocket.Settings.Save.UserPasswordWarningInfoText="Ești sigur că vrei să-ți folosești propria parolă?"
12
+OBSWebSocket.Settings.Save.UserPasswordWarningInfoText="Sigur vrei să-ți folosești propria parolă?"
13
 OBSWebSocket.Settings.Save.PasswordInvalidErrorTitle="Eroare: Configurație invalidă"
14
 OBSWebSocket.Settings.Save.PasswordInvalidErrorMessage="Trebuie să folosești o parolă care să aibă 6 sau mai multe caractere."
15
 OBSWebSocket.SessionTable.Title="Sesiuni WebSocket conectate"
16
obs-studio-29.0.0.tar.xz/plugins/obs-websocket/data/locale/vi-VN.ini Added
9
 
1
@@ -0,0 +1,7 @@
2
+OBSWebSocket.Settings.PluginSettingsTitle="Thiết đặt trình cắm"
3
+OBSWebSocket.Settings.ServerEnable="Bật máy chủ WebSocket"
4
+OBSWebSocket.Settings.ServerSettingsTitle="Thiết đặt máy chủ"
5
+OBSWebSocket.Settings.Password="Mật khẩu máy chủ"
6
+OBSWebSocket.Settings.GeneratePassword="Tạo mật khẩu"
7
+OBSWebSocket.Settings.ServerPort="Cổng máy chủ"
8
+OBSWebSocket.Settings.ShowConnectInfo="Hiện thông tin kết nối"
9
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/docs/docs/generate_md.py -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/docs/docs/generate_md.py Changed
12
 
1
@@ -9,7 +9,9 @@
2
     'WebSocketCloseCode',
3
     'RequestBatchExecutionType',
4
     'RequestStatus',
5
-    'EventSubscription'
6
+    'EventSubscription',
7
+    'ObsMediaInputAction',
8
+    'ObsOutputState'
9
 
10
 
11
 categoryOrder = 
12
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/docs/docs/partials/introduction.md -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/docs/docs/partials/introduction.md Changed
36
 
1
@@ -1,5 +1,5 @@
2
 
3
-# obs-websocket 5.0.1 Protocol
4
+# obs-websocket 5.1.0 Protocol
5
 
6
 ## Main Table of Contents
7
 
8
@@ -19,6 +19,9 @@
9
   - RequestResponse (OpCode 7)(#requestresponse-opcode-7)
10
   - RequestBatch (OpCode 8)(#requestbatch-opcode-8)
11
   - RequestBatchResponse (OpCode 9)(#requestbatchresponse-opcode-9)
12
+- Enumerations(#enums)
13
+- Events(#events)
14
+- Requests(#requests)
15
 
16
 ## General Intro
17
 
18
@@ -143,7 +146,7 @@
19
 {
20
   "op": 0,
21
   "d": {
22
-    "obsWebSocketVersion": "5.0.1",
23
+    "obsWebSocketVersion": "5.1.0",
24
     "rpcVersion": 1,
25
     "authentication": {
26
       "challenge": "+IxH4CnCiqpX1rM9scsNynZzbOe4KhDeYcTNS3PDaeY=",
27
@@ -159,7 +162,7 @@
28
 {
29
   "op": 0,
30
   "d": {
31
-    "obsWebSocketVersion": "5.0.1",
32
+    "obsWebSocketVersion": "5.1.0",
33
     "rpcVersion": 1
34
   }
35
 }
36
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/docs/generated/protocol.json -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/docs/generated/protocol.json Changed
201
 
1
@@ -448,6 +448,83 @@
2
       
3
     },
4
     {
5
+      "enumType": "ObsOutputState",
6
+      "enumIdentifiers": 
7
+        {
8
+          "description": "Unknown state.",
9
+          "enumIdentifier": "OBS_WEBSOCKET_OUTPUT_UNKNOWN",
10
+          "rpcVersion": 1,
11
+          "deprecated": true,
12
+          "initialVersion": "5.0.0",
13
+          "enumValue": "OBS_WEBSOCKET_OUTPUT_UNKNOWN"
14
+        },
15
+        {
16
+          "description": "The output is starting.",
17
+          "enumIdentifier": "OBS_WEBSOCKET_OUTPUT_STARTING",
18
+          "rpcVersion": 1,
19
+          "deprecated": true,
20
+          "initialVersion": "5.0.0",
21
+          "enumValue": "OBS_WEBSOCKET_OUTPUT_STARTING"
22
+        },
23
+        {
24
+          "description": "The input has started.",
25
+          "enumIdentifier": "OBS_WEBSOCKET_OUTPUT_STARTED",
26
+          "rpcVersion": 1,
27
+          "deprecated": true,
28
+          "initialVersion": "5.0.0",
29
+          "enumValue": "OBS_WEBSOCKET_OUTPUT_STARTED"
30
+        },
31
+        {
32
+          "description": "The output is stopping.",
33
+          "enumIdentifier": "OBS_WEBSOCKET_OUTPUT_STOPPING",
34
+          "rpcVersion": 1,
35
+          "deprecated": true,
36
+          "initialVersion": "5.0.0",
37
+          "enumValue": "OBS_WEBSOCKET_OUTPUT_STOPPING"
38
+        },
39
+        {
40
+          "description": "The output has stopped.",
41
+          "enumIdentifier": "OBS_WEBSOCKET_OUTPUT_STOPPED",
42
+          "rpcVersion": 1,
43
+          "deprecated": true,
44
+          "initialVersion": "5.0.0",
45
+          "enumValue": "OBS_WEBSOCKET_OUTPUT_STOPPED"
46
+        },
47
+        {
48
+          "description": "The output has disconnected and is reconnecting.",
49
+          "enumIdentifier": "OBS_WEBSOCKET_OUTPUT_RECONNECTING",
50
+          "rpcVersion": 1,
51
+          "deprecated": true,
52
+          "initialVersion": "5.0.0",
53
+          "enumValue": "OBS_WEBSOCKET_OUTPUT_RECONNECTING"
54
+        },
55
+        {
56
+          "description": "The output has reconnected successfully.",
57
+          "enumIdentifier": "OBS_WEBSOCKET_OUTPUT_RECONNECTED",
58
+          "rpcVersion": 1,
59
+          "deprecated": true,
60
+          "initialVersion": "5.1.0",
61
+          "enumValue": "OBS_WEBSOCKET_OUTPUT_RECONNECTED"
62
+        },
63
+        {
64
+          "description": "The output is now paused.",
65
+          "enumIdentifier": "OBS_WEBSOCKET_OUTPUT_PAUSED",
66
+          "rpcVersion": 1,
67
+          "deprecated": true,
68
+          "initialVersion": "5.1.0",
69
+          "enumValue": "OBS_WEBSOCKET_OUTPUT_PAUSED"
70
+        },
71
+        {
72
+          "description": "The output has been resumed (unpaused).",
73
+          "enumIdentifier": "OBS_WEBSOCKET_OUTPUT_RESUMED",
74
+          "rpcVersion": 1,
75
+          "deprecated": true,
76
+          "initialVersion": "5.0.0",
77
+          "enumValue": "OBS_WEBSOCKET_OUTPUT_RESUMED"
78
+        }
79
+      
80
+    },
81
+    {
82
       "enumType": "ObsMediaInputAction",
83
       "enumIdentifiers": 
84
         {
85
@@ -1776,16 +1853,16 @@
86
           "valueType": "Number",
87
           "valueDescription": "Number of milliseconds to sleep for (if `SERIAL_REALTIME` mode)",
88
           "valueRestrictions": ">= 0, <= 50000",
89
-          "valueOptional": false,
90
-          "valueOptionalBehavior": null
91
+          "valueOptional": true,
92
+          "valueOptionalBehavior": "Unknown"
93
         },
94
         {
95
           "valueName": "sleepFrames",
96
           "valueType": "Number",
97
           "valueDescription": "Number of frames to sleep for (if `SERIAL_FRAME` mode)",
98
           "valueRestrictions": ">= 0, <= 10000",
99
-          "valueOptional": false,
100
-          "valueOptionalBehavior": null
101
+          "valueOptional": true,
102
+          "valueOptionalBehavior": "Unknown"
103
         }
104
       ,
105
       "responseFields": 
106
@@ -2481,7 +2558,7 @@
107
       
108
     },
109
     {
110
-      "description": "Presses a button in the properties of an input.\n\nNote: Use this in cases where there is a button in the properties of an input that cannot be accessed in any other way. For example, browser sources, where there is a refresh button.",
111
+      "description": "Presses a button in the properties of an input.\n\nSome known `propertyName` values are:\n\n- `refreshnocache` - Browser source reload button\n\nNote: Use this in cases where there is a button in the properties of an input that cannot be accessed in any other way. For example, browser sources, where there is a refresh button.",
112
       "requestType": "PressInputPropertiesButton",
113
       "complexity": 4,
114
       "rpcVersion": "1",
115
@@ -2777,7 +2854,13 @@
116
       "initialVersion": "5.0.0",
117
       "category": "outputs",
118
       "requestFields": ,
119
-      "responseFields": 
120
+      "responseFields": 
121
+        {
122
+          "valueName": "outputs",
123
+          "valueType": "Array<Object>",
124
+          "valueDescription": "Array of outputs"
125
+        }
126
+      
127
     },
128
     {
129
       "description": "Gets the status of an output.",
130
@@ -2976,7 +3059,7 @@
131
           "valueDescription": "Whether the output is active"
132
         },
133
         {
134
-          "valueName": "ouputPaused",
135
+          "valueName": "outputPaused",
136
           "valueType": "Boolean",
137
           "valueDescription": "Whether the output is paused"
138
         },
139
@@ -3096,7 +3179,7 @@
140
       
141
     },
142
     {
143
-      "description": "Basically GetSceneItemList, but for groups.\n\nUsing groups at all in OBS is discouraged, as they are very broken under the hood.\n\nGroups only",
144
+      "description": "Basically GetSceneItemList, but for groups.\n\nUsing groups at all in OBS is discouraged, as they are very broken under the hood. Please use nested scenes instead.\n\nGroups only",
145
       "requestType": "GetGroupSceneItemList",
146
       "complexity": 3,
147
       "rpcVersion": "1",
148
@@ -4955,7 +5038,7 @@
149
         {
150
           "valueName": "inputVolumeMul",
151
           "valueType": "Number",
152
-          "valueDescription": "New volume level in multimap"
153
+          "valueDescription": "New volume level multiplier"
154
         },
155
         {
156
           "valueName": "inputVolumeDb",
157
@@ -5639,6 +5722,23 @@
158
       
159
     },
160
     {
161
+      "description": "A screenshot has been saved.\n\nNote: Triggered for the screenshot feature available in `Settings -> Hotkeys -> Screenshot Output` ONLY.\nApplications using `Get/SaveSourceScreenshot` should implement a `CustomEvent` if this kind of inter-client\ncommunication is desired.",
162
+      "eventType": "ScreenshotSaved",
163
+      "eventSubscription": "Ui",
164
+      "complexity": 2,
165
+      "rpcVersion": "1",
166
+      "deprecated": false,
167
+      "initialVersion": "5.1.0",
168
+      "category": "ui",
169
+      "dataFields": 
170
+        {
171
+          "valueName": "savedScreenshotPath",
172
+          "valueType": "String",
173
+          "valueDescription": "Path of the saved image file"
174
+        }
175
+      
176
+    },
177
+    {
178
       "description": "An event has been emitted from a vendor.\n\nA vendor is a unique name registered by a third-party plugin or script, which allows for custom requests and events to be added to obs-websocket.\nIf a plugin or script implements vendor requests or events, documentation is expected to be provided with them.",
179
       "eventType": "VendorEvent",
180
       "eventSubscription": "Vendors",
181
@@ -5664,6 +5764,23 @@
182
           "valueDescription": "Vendor-provided event data. {} if event does not provide any data"
183
         }
184
       
185
+    },
186
+    {
187
+      "description": "Custom event emitted by `BroadcastCustomEvent`.",
188
+      "eventType": "CustomEvent",
189
+      "eventSubscription": "General",
190
+      "complexity": 1,
191
+      "rpcVersion": "1",
192
+      "deprecated": false,
193
+      "initialVersion": "5.0.0",
194
+      "category": "general",
195
+      "dataFields": 
196
+        {
197
+          "valueName": "eventData",
198
+          "valueType": "Object",
199
+          "valueDescription": "Custom event data"
200
+        }
201
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/docs/generated/protocol.md -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/docs/generated/protocol.md Changed
201
 
1
@@ -21,6 +21,9 @@
2
   - RequestResponse (OpCode 7)(#requestresponse-opcode-7)
3
   - RequestBatch (OpCode 8)(#requestbatch-opcode-8)
4
   - RequestBatchResponse (OpCode 9)(#requestbatchresponse-opcode-9)
5
+- Enumerations(#enums)
6
+- Events(#events)
7
+- Requests(#requests)
8
 
9
 ## General Intro
10
 
11
@@ -512,6 +515,24 @@
12
   - EventSubscription::InputActiveStateChanged(#eventsubscriptioninputactivestatechanged)
13
   - EventSubscription::InputShowStateChanged(#eventsubscriptioninputshowstatechanged)
14
   - EventSubscription::SceneItemTransformChanged(#eventsubscriptionsceneitemtransformchanged)
15
+- ObsMediaInputAction(#obsmediainputaction)
16
+  - ObsMediaInputAction::OBS_WEBSOCKET_MEDIA_INPUT_ACTION_NONE(#obsmediainputactionobs_websocket_media_input_action_none)
17
+  - ObsMediaInputAction::OBS_WEBSOCKET_MEDIA_INPUT_ACTION_PLAY(#obsmediainputactionobs_websocket_media_input_action_play)
18
+  - ObsMediaInputAction::OBS_WEBSOCKET_MEDIA_INPUT_ACTION_PAUSE(#obsmediainputactionobs_websocket_media_input_action_pause)
19
+  - ObsMediaInputAction::OBS_WEBSOCKET_MEDIA_INPUT_ACTION_STOP(#obsmediainputactionobs_websocket_media_input_action_stop)
20
+  - ObsMediaInputAction::OBS_WEBSOCKET_MEDIA_INPUT_ACTION_RESTART(#obsmediainputactionobs_websocket_media_input_action_restart)
21
+  - ObsMediaInputAction::OBS_WEBSOCKET_MEDIA_INPUT_ACTION_NEXT(#obsmediainputactionobs_websocket_media_input_action_next)
22
+  - ObsMediaInputAction::OBS_WEBSOCKET_MEDIA_INPUT_ACTION_PREVIOUS(#obsmediainputactionobs_websocket_media_input_action_previous)
23
+- ObsOutputState(#obsoutputstate)
24
+  - ObsOutputState::OBS_WEBSOCKET_OUTPUT_UNKNOWN(#obsoutputstateobs_websocket_output_unknown)
25
+  - ObsOutputState::OBS_WEBSOCKET_OUTPUT_STARTING(#obsoutputstateobs_websocket_output_starting)
26
+  - ObsOutputState::OBS_WEBSOCKET_OUTPUT_STARTED(#obsoutputstateobs_websocket_output_started)
27
+  - ObsOutputState::OBS_WEBSOCKET_OUTPUT_STOPPING(#obsoutputstateobs_websocket_output_stopping)
28
+  - ObsOutputState::OBS_WEBSOCKET_OUTPUT_STOPPED(#obsoutputstateobs_websocket_output_stopped)
29
+  - ObsOutputState::OBS_WEBSOCKET_OUTPUT_RECONNECTING(#obsoutputstateobs_websocket_output_reconnecting)
30
+  - ObsOutputState::OBS_WEBSOCKET_OUTPUT_RECONNECTED(#obsoutputstateobs_websocket_output_reconnected)
31
+  - ObsOutputState::OBS_WEBSOCKET_OUTPUT_PAUSED(#obsoutputstateobs_websocket_output_paused)
32
+  - ObsOutputState::OBS_WEBSOCKET_OUTPUT_RESUMED(#obsoutputstateobs_websocket_output_resumed)
33
 
34
 ## WebSocketOpCode
35
 
36
@@ -1294,6 +1315,182 @@
37
 - Latest Supported RPC Version: `1`
38
 - Added in v5.0.0
39
 
40
+## ObsMediaInputAction
41
+
42
+### ObsMediaInputAction::OBS_WEBSOCKET_MEDIA_INPUT_ACTION_NONE
43
+
44
+No action.
45
+
46
+- Identifier Value: `OBS_WEBSOCKET_MEDIA_INPUT_ACTION_NONE`
47
+- Latest Supported RPC Version: `1`
48
+- **⚠️ Deprecated. ⚠️**
49
+- Added in v5.0.0
50
+
51
+---
52
+
53
+### ObsMediaInputAction::OBS_WEBSOCKET_MEDIA_INPUT_ACTION_PLAY
54
+
55
+Play the media input.
56
+
57
+- Identifier Value: `OBS_WEBSOCKET_MEDIA_INPUT_ACTION_PLAY`
58
+- Latest Supported RPC Version: `1`
59
+- **⚠️ Deprecated. ⚠️**
60
+- Added in v5.0.0
61
+
62
+---
63
+
64
+### ObsMediaInputAction::OBS_WEBSOCKET_MEDIA_INPUT_ACTION_PAUSE
65
+
66
+Pause the media input.
67
+
68
+- Identifier Value: `OBS_WEBSOCKET_MEDIA_INPUT_ACTION_PAUSE`
69
+- Latest Supported RPC Version: `1`
70
+- **⚠️ Deprecated. ⚠️**
71
+- Added in v5.0.0
72
+
73
+---
74
+
75
+### ObsMediaInputAction::OBS_WEBSOCKET_MEDIA_INPUT_ACTION_STOP
76
+
77
+Stop the media input.
78
+
79
+- Identifier Value: `OBS_WEBSOCKET_MEDIA_INPUT_ACTION_STOP`
80
+- Latest Supported RPC Version: `1`
81
+- **⚠️ Deprecated. ⚠️**
82
+- Added in v5.0.0
83
+
84
+---
85
+
86
+### ObsMediaInputAction::OBS_WEBSOCKET_MEDIA_INPUT_ACTION_RESTART
87
+
88
+Restart the media input.
89
+
90
+- Identifier Value: `OBS_WEBSOCKET_MEDIA_INPUT_ACTION_RESTART`
91
+- Latest Supported RPC Version: `1`
92
+- **⚠️ Deprecated. ⚠️**
93
+- Added in v5.0.0
94
+
95
+---
96
+
97
+### ObsMediaInputAction::OBS_WEBSOCKET_MEDIA_INPUT_ACTION_NEXT
98
+
99
+Go to the next playlist item.
100
+
101
+- Identifier Value: `OBS_WEBSOCKET_MEDIA_INPUT_ACTION_NEXT`
102
+- Latest Supported RPC Version: `1`
103
+- **⚠️ Deprecated. ⚠️**
104
+- Added in v5.0.0
105
+
106
+---
107
+
108
+### ObsMediaInputAction::OBS_WEBSOCKET_MEDIA_INPUT_ACTION_PREVIOUS
109
+
110
+Go to the previous playlist item.
111
+
112
+- Identifier Value: `OBS_WEBSOCKET_MEDIA_INPUT_ACTION_PREVIOUS`
113
+- Latest Supported RPC Version: `1`
114
+- **⚠️ Deprecated. ⚠️**
115
+- Added in v5.0.0
116
+
117
+## ObsOutputState
118
+
119
+### ObsOutputState::OBS_WEBSOCKET_OUTPUT_UNKNOWN
120
+
121
+Unknown state.
122
+
123
+- Identifier Value: `OBS_WEBSOCKET_OUTPUT_UNKNOWN`
124
+- Latest Supported RPC Version: `1`
125
+- **⚠️ Deprecated. ⚠️**
126
+- Added in v5.0.0
127
+
128
+---
129
+
130
+### ObsOutputState::OBS_WEBSOCKET_OUTPUT_STARTING
131
+
132
+The output is starting.
133
+
134
+- Identifier Value: `OBS_WEBSOCKET_OUTPUT_STARTING`
135
+- Latest Supported RPC Version: `1`
136
+- **⚠️ Deprecated. ⚠️**
137
+- Added in v5.0.0
138
+
139
+---
140
+
141
+### ObsOutputState::OBS_WEBSOCKET_OUTPUT_STARTED
142
+
143
+The input has started.
144
+
145
+- Identifier Value: `OBS_WEBSOCKET_OUTPUT_STARTED`
146
+- Latest Supported RPC Version: `1`
147
+- **⚠️ Deprecated. ⚠️**
148
+- Added in v5.0.0
149
+
150
+---
151
+
152
+### ObsOutputState::OBS_WEBSOCKET_OUTPUT_STOPPING
153
+
154
+The output is stopping.
155
+
156
+- Identifier Value: `OBS_WEBSOCKET_OUTPUT_STOPPING`
157
+- Latest Supported RPC Version: `1`
158
+- **⚠️ Deprecated. ⚠️**
159
+- Added in v5.0.0
160
+
161
+---
162
+
163
+### ObsOutputState::OBS_WEBSOCKET_OUTPUT_STOPPED
164
+
165
+The output has stopped.
166
+
167
+- Identifier Value: `OBS_WEBSOCKET_OUTPUT_STOPPED`
168
+- Latest Supported RPC Version: `1`
169
+- **⚠️ Deprecated. ⚠️**
170
+- Added in v5.0.0
171
+
172
+---
173
+
174
+### ObsOutputState::OBS_WEBSOCKET_OUTPUT_RECONNECTING
175
+
176
+The output has disconnected and is reconnecting.
177
+
178
+- Identifier Value: `OBS_WEBSOCKET_OUTPUT_RECONNECTING`
179
+- Latest Supported RPC Version: `1`
180
+- **⚠️ Deprecated. ⚠️**
181
+- Added in v5.0.0
182
+
183
+---
184
+
185
+### ObsOutputState::OBS_WEBSOCKET_OUTPUT_RECONNECTED
186
+
187
+The output has reconnected successfully.
188
+
189
+- Identifier Value: `OBS_WEBSOCKET_OUTPUT_RECONNECTED`
190
+- Latest Supported RPC Version: `1`
191
+- **⚠️ Deprecated. ⚠️**
192
+- Added in v5.1.0
193
+
194
+---
195
+
196
+### ObsOutputState::OBS_WEBSOCKET_OUTPUT_PAUSED
197
+
198
+The output is now paused.
199
+
200
+- Identifier Value: `OBS_WEBSOCKET_OUTPUT_PAUSED`
201
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/src/eventhandler/EventHandler.cpp -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/src/eventhandler/EventHandler.cpp Changed
201
 
1
@@ -267,97 +267,10 @@
2
    switch (event) {
3
    // General
4
    case OBS_FRONTEND_EVENT_FINISHED_LOADING:
5
-       blog_debug(
6
-           "EventHandler::OnFrontendEvent OBS has finished loading. Connecting final handlers and enabling events...");
7
-
8
-       // Connect source signals and enable events only after OBS has fully loaded (to reduce extra logging).
9
-       eventHandler->_obsLoaded.store(true);
10
-
11
-       // In the case that plugins become hotloadable, this will have to go back into `EventHandler::EventHandler()`
12
-       // Enumerate inputs and connect each one
13
-       {
14
-           auto enumInputs = (void *param, obs_source_t *source) {
15
-               auto eventHandler = static_cast<EventHandler *>(param);
16
-               eventHandler->ConnectSourceSignals(source);
17
-               return true;
18
-           };
19
-           obs_enum_sources(enumInputs, private_data);
20
-       }
21
-
22
-       // Enumerate scenes and connect each one
23
-       {
24
-           auto enumScenes = (void *param, obs_source_t *source) {
25
-               auto eventHandler = static_cast<EventHandler *>(param);
26
-               eventHandler->ConnectSourceSignals(source);
27
-               return true;
28
-           };
29
-           obs_enum_scenes(enumScenes, private_data);
30
-       }
31
-
32
-       // Enumerate all scene transitions and connect each one
33
-       {
34
-           obs_frontend_source_list transitions = {};
35
-           obs_frontend_get_transitions(&transitions);
36
-           for (size_t i = 0; i < transitions.sources.num; i++) {
37
-               obs_source_t *transition = transitions.sources.arrayi;
38
-               eventHandler->ConnectSourceSignals(transition);
39
-           }
40
-           obs_frontend_source_list_free(&transitions);
41
-       }
42
-
43
-       blog_debug("EventHandler::OnFrontendEvent Finished.");
44
-
45
-       if (eventHandler->_obsLoadedCallback)
46
-           eventHandler->_obsLoadedCallback();
47
-
48
+       eventHandler->FrontendFinishedLoadingMultiHandler();
49
        break;
50
    case OBS_FRONTEND_EVENT_EXIT:
51
-       eventHandler->HandleExitStarted();
52
-
53
-       blog_debug("EventHandler::OnFrontendEvent OBS is unloading. Disabling events...");
54
-       // Disconnect source signals and disable events when OBS starts unloading (to reduce extra logging).
55
-       eventHandler->_obsLoaded.store(false);
56
-
57
-       // In the case that plugins become hotloadable, this will have to go back into `EventHandler::~EventHandler()`
58
-       // Enumerate inputs and disconnect each one
59
-       {
60
-           auto enumInputs = (void *param, obs_source_t *source) {
61
-               auto eventHandler = static_cast<EventHandler *>(param);
62
-               eventHandler->DisconnectSourceSignals(source);
63
-               return true;
64
-           };
65
-           obs_enum_sources(enumInputs, private_data);
66
-       }
67
-
68
-       // Enumerate scenes and disconnect each one
69
-       {
70
-           auto enumScenes = (void *param, obs_source_t *source) {
71
-               auto eventHandler = static_cast<EventHandler *>(param);
72
-               eventHandler->DisconnectSourceSignals(source);
73
-               return true;
74
-           };
75
-           obs_enum_scenes(enumScenes, private_data);
76
-       }
77
-
78
-       // Enumerate all scene transitions and disconnect each one
79
-       {
80
-           obs_frontend_source_list transitions = {};
81
-           obs_frontend_get_transitions(&transitions);
82
-           for (size_t i = 0; i < transitions.sources.num; i++) {
83
-               obs_source_t *transition = transitions.sources.arrayi;
84
-               eventHandler->DisconnectSourceSignals(transition);
85
-           }
86
-           obs_frontend_source_list_free(&transitions);
87
-       }
88
-
89
-       blog_debug("EventHandler::OnFrontendEvent Finished.");
90
-
91
-       break;
92
-   case OBS_FRONTEND_EVENT_STUDIO_MODE_ENABLED:
93
-       eventHandler->HandleStudioModeStateChanged(true);
94
-       break;
95
-   case OBS_FRONTEND_EVENT_STUDIO_MODE_DISABLED:
96
-       eventHandler->HandleStudioModeStateChanged(false);
97
+       eventHandler->FrontendExitMultiHandler();
98
        break;
99
 
100
    // Config
101
@@ -427,12 +340,31 @@
102
    // Outputs
103
    case OBS_FRONTEND_EVENT_STREAMING_STARTING:
104
        eventHandler->HandleStreamStateChanged(OBS_WEBSOCKET_OUTPUT_STARTING);
105
+       {
106
+           // Connect signals for stream output reconnects (hacky)
107
+           OBSOutputAutoRelease streamOutput = obs_frontend_get_streaming_output();
108
+           if (streamOutput) {
109
+               signal_handler_t *sh = obs_output_get_signal_handler(streamOutput);
110
+               signal_handler_connect(sh, "reconnect", StreamOutputReconnectHandler, private_data);
111
+               signal_handler_connect(sh, "reconnect_success", StreamOutputReconnectSuccessHandler, private_data);
112
+           }
113
+       }
114
        break;
115
    case OBS_FRONTEND_EVENT_STREAMING_STARTED:
116
        eventHandler->HandleStreamStateChanged(OBS_WEBSOCKET_OUTPUT_STARTED);
117
        break;
118
    case OBS_FRONTEND_EVENT_STREAMING_STOPPING:
119
        eventHandler->HandleStreamStateChanged(OBS_WEBSOCKET_OUTPUT_STOPPING);
120
+       {
121
+           // Disconnect signals for stream output reconnects
122
+           OBSOutputAutoRelease streamOutput = obs_frontend_get_streaming_output();
123
+           if (streamOutput) {
124
+               signal_handler_t *sh = obs_output_get_signal_handler(streamOutput);
125
+               signal_handler_disconnect(sh, "reconnect", StreamOutputReconnectHandler, private_data);
126
+               signal_handler_disconnect(sh, "reconnect_success", StreamOutputReconnectSuccessHandler,
127
+                             private_data);
128
+           }
129
+       }
130
        break;
131
    case OBS_FRONTEND_EVENT_STREAMING_STOPPED:
132
        eventHandler->HandleStreamStateChanged(OBS_WEBSOCKET_OUTPUT_STOPPED);
133
@@ -477,11 +409,112 @@
134
        eventHandler->HandleReplayBufferSaved();
135
        break;
136
 
137
+   // Ui
138
+   case OBS_FRONTEND_EVENT_STUDIO_MODE_ENABLED:
139
+       eventHandler->HandleStudioModeStateChanged(true);
140
+       break;
141
+   case OBS_FRONTEND_EVENT_STUDIO_MODE_DISABLED:
142
+       eventHandler->HandleStudioModeStateChanged(false);
143
+       break;
144
+   case OBS_FRONTEND_EVENT_SCREENSHOT_TAKEN:
145
+       eventHandler->HandleScreenshotSaved();
146
+       break;
147
+
148
    default:
149
        break;
150
    }
151
 }
152
 
153
+void EventHandler::FrontendFinishedLoadingMultiHandler()
154
+{
155
+   blog_debug(
156
+       "EventHandler::FrontendFinishedLoadingMultiHandler OBS has finished loading. Connecting final handlers and enabling events...");
157
+
158
+   // Connect source signals and enable events only after OBS has fully loaded (to reduce extra logging).
159
+   _obsLoaded.store(true);
160
+
161
+   // In the case that plugins become hotloadable, this will have to go back into `EventHandler::EventHandler()`
162
+   // Enumerate inputs and connect each one
163
+   {
164
+       auto enumInputs = (void *param, obs_source_t *source) {
165
+           auto eventHandler = static_cast<EventHandler *>(param);
166
+           eventHandler->ConnectSourceSignals(source);
167
+           return true;
168
+       };
169
+       obs_enum_sources(enumInputs, this);
170
+   }
171
+
172
+   // Enumerate scenes and connect each one
173
+   {
174
+       auto enumScenes = (void *param, obs_source_t *source) {
175
+           auto eventHandler = static_cast<EventHandler *>(param);
176
+           eventHandler->ConnectSourceSignals(source);
177
+           return true;
178
+       };
179
+       obs_enum_scenes(enumScenes, this);
180
+   }
181
+
182
+   // Enumerate all scene transitions and connect each one
183
+   {
184
+       obs_frontend_source_list transitions = {};
185
+       obs_frontend_get_transitions(&transitions);
186
+       for (size_t i = 0; i < transitions.sources.num; i++) {
187
+           obs_source_t *transition = transitions.sources.arrayi;
188
+           ConnectSourceSignals(transition);
189
+       }
190
+       obs_frontend_source_list_free(&transitions);
191
+   }
192
+
193
+   blog_debug("EventHandler::FrontendFinishedLoadingMultiHandler Finished.");
194
+
195
+   if (_obsLoadedCallback)
196
+       _obsLoadedCallback();
197
+}
198
+
199
+void EventHandler::FrontendExitMultiHandler()
200
+{
201
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/src/eventhandler/EventHandler.h -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/src/eventhandler/EventHandler.h Changed
33
 
1
@@ -61,6 +61,8 @@
2
 
3
    // Signal handler: frontend
4
    static void OnFrontendEvent(enum obs_frontend_event event, void *private_data);
5
+   void FrontendFinishedLoadingMultiHandler();
6
+   void FrontendExitMultiHandler();
7
 
8
    // Signal handler: libobs
9
    static void SourceCreatedMultiHandler(void *param, calldata_t *data);
10
@@ -76,9 +78,12 @@
11
    static void SourceMediaNextMultiHandler(void *param, calldata_t *data);
12
    static void SourceMediaPreviousMultiHandler(void *param, calldata_t *data);
13
 
14
+   // Signal handler: output
15
+   static void StreamOutputReconnectHandler(void *param, calldata_t *data);
16
+   static void StreamOutputReconnectSuccessHandler(void *param, calldata_t *data);
17
+
18
    // General
19
    void HandleExitStarted();
20
-   void HandleStudioModeStateChanged(bool enabled);
21
 
22
    // Config
23
    void HandleCurrentSceneCollectionChanging();
24
@@ -170,4 +175,8 @@
25
    static void HandleMediaInputPlaybackEnded(void *param,
26
                          calldata_t *data); // Direct callback
27
    void HandleMediaInputActionTriggered(obs_source_t *source, ObsMediaInputAction action);
28
+
29
+   // Ui
30
+   void HandleStudioModeStateChanged(bool enabled);
31
+   void HandleScreenshotSaved();
32
 };
33
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/src/eventhandler/EventHandler_Inputs.cpp -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/src/eventhandler/EventHandler_Inputs.cpp Changed
10
 
1
@@ -200,7 +200,7 @@
2
  * An input's volume level has changed.
3
  *
4
  * @dataField inputName      | String | Name of the input
5
- * @dataField inputVolumeMul | Number | New volume level in multimap
6
+ * @dataField inputVolumeMul | Number | New volume level multiplier
7
  * @dataField inputVolumeDb  | Number | New volume level in dB
8
  *
9
  * @eventType InputVolumeChanged
10
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/src/eventhandler/EventHandler_Outputs.cpp -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/src/eventhandler/EventHandler_Outputs.cpp Changed
14
 
1
@@ -24,10 +24,12 @@
2
    switch (state) {
3
    case OBS_WEBSOCKET_OUTPUT_STARTED:
4
    case OBS_WEBSOCKET_OUTPUT_RESUMED:
5
+   case OBS_WEBSOCKET_OUTPUT_RECONNECTED:
6
        return true;
7
    case OBS_WEBSOCKET_OUTPUT_STARTING:
8
    case OBS_WEBSOCKET_OUTPUT_STOPPING:
9
    case OBS_WEBSOCKET_OUTPUT_STOPPED:
10
+   case OBS_WEBSOCKET_OUTPUT_RECONNECTING:
11
    case OBS_WEBSOCKET_OUTPUT_PAUSED:
12
        return false;
13
    default:
14
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/src/eventhandler/EventHandler_Ui.cpp -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/src/eventhandler/EventHandler_Ui.cpp Changed
29
 
1
@@ -38,3 +38,27 @@
2
    eventData"studioModeEnabled" = enabled;
3
    BroadcastEvent(EventSubscription::Ui, "StudioModeStateChanged", eventData);
4
 }
5
+
6
+/**
7
+ * A screenshot has been saved.
8
+ *
9
+ * Note: Triggered for the screenshot feature available in `Settings -> Hotkeys -> Screenshot Output` ONLY.
10
+ * Applications using `Get/SaveSourceScreenshot` should implement a `CustomEvent` if this kind of inter-client
11
+ * communication is desired.
12
+ *
13
+ * @dataField savedScreenshotPath | String | Path of the saved image file
14
+ *
15
+ * @eventType ScreenshotSaved
16
+ * @eventSubscription Ui
17
+ * @complexity 2
18
+ * @rpcVersion -1
19
+ * @initialVersion 5.1.0
20
+ * @api events
21
+ * @category ui
22
+ */
23
+void EventHandler::HandleScreenshotSaved()
24
+{
25
+   json eventData;
26
+   eventData"savedScreenshotPath" = Utils::Obs::StringHelper::GetLastScreenshotFileName();
27
+   BroadcastEvent(EventSubscription::Ui, "ScreenshotSaved", eventData);
28
+}
29
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/src/forms/ConnectInfo.cpp -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/src/forms/ConnectInfo.cpp Changed
10
 
1
@@ -105,7 +105,7 @@
2
 
3
 void ConnectInfo::DrawQr(QString qrText)
4
 {
5
-   QPixmap map(230, 230);
6
+   QPixmap map(236, 236);
7
    map.fill(Qt::white);
8
    QPainter painter(&map);
9
 
10
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/src/forms/ConnectInfo.ui -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/src/forms/ConnectInfo.ui Changed
171
 
1
@@ -7,7 +7,7 @@
2
     <x>0</x>
3
     <y>0</y>
4
     <width>451</width>
5
-    <height>412</height>
6
+    <height>432</height>
7
    </rect>
8
   </property>
9
   <property name="minimumSize">
10
@@ -19,7 +19,7 @@
11
   <property name="maximumSize">
12
    <size>
13
     <width>451</width>
14
-    <height>412</height>
15
+    <height>432</height>
16
    </size>
17
   </property>
18
   <property name="windowTitle">
19
@@ -31,12 +31,21 @@
20
      <x>10</x>
21
      <y>10</y>
22
      <width>431</width>
23
-     <height>101</height>
24
+     <height>121</height>
25
     </rect>
26
    </property>
27
    <layout class="QFormLayout" name="formLayout">
28
+    <property name="formAlignment">
29
+     <set>Qt::AlignLeading|Qt::AlignLeft|Qt::AlignVCenter</set>
30
+    </property>
31
     <item row="0" column="0">
32
      <widget class="QLabel" name="serverIpLabel">
33
+      <property name="maximumSize">
34
+       <size>
35
+        <width>200</width>
36
+        <height>16777215</height>
37
+       </size>
38
+      </property>
39
       <property name="text">
40
        <string>OBSWebSocket.ConnectInfo.ServerIp</string>
41
       </property>
42
@@ -46,6 +55,12 @@
43
      <layout class="QHBoxLayout" name="horizontalLayout">
44
       <item>
45
        <widget class="QLineEdit" name="serverIpLineEdit">
46
+        <property name="focusPolicy">
47
+         <enum>Qt::NoFocus</enum>
48
+        </property>
49
+        <property name="alignment">
50
+         <set>Qt::AlignBottom|Qt::AlignLeading|Qt::AlignLeft</set>
51
+        </property>
52
         <property name="readOnly">
53
          <bool>true</bool>
54
         </property>
55
@@ -53,6 +68,12 @@
56
       </item>
57
       <item>
58
        <widget class="QPushButton" name="copyServerIpButton">
59
+        <property name="maximumSize">
60
+         <size>
61
+          <width>75</width>
62
+          <height>16777215</height>
63
+         </size>
64
+        </property>
65
         <property name="text">
66
          <string>OBSWebSocket.ConnectInfo.CopyText</string>
67
         </property>
68
@@ -62,6 +83,12 @@
69
     </item>
70
     <item row="1" column="0">
71
      <widget class="QLabel" name="serverPortLabel">
72
+      <property name="maximumSize">
73
+       <size>
74
+        <width>200</width>
75
+        <height>16777215</height>
76
+       </size>
77
+      </property>
78
       <property name="text">
79
        <string>OBSWebSocket.ConnectInfo.ServerPort</string>
80
       </property>
81
@@ -71,6 +98,12 @@
82
      <layout class="QHBoxLayout" name="horizontalLayout_2">
83
       <item>
84
        <widget class="QLineEdit" name="serverPortLineEdit">
85
+        <property name="focusPolicy">
86
+         <enum>Qt::NoFocus</enum>
87
+        </property>
88
+        <property name="alignment">
89
+         <set>Qt::AlignBottom|Qt::AlignLeading|Qt::AlignLeft</set>
90
+        </property>
91
         <property name="readOnly">
92
          <bool>true</bool>
93
         </property>
94
@@ -78,6 +111,12 @@
95
       </item>
96
       <item>
97
        <widget class="QPushButton" name="copyServerPortButton">
98
+        <property name="maximumSize">
99
+         <size>
100
+          <width>75</width>
101
+          <height>16777215</height>
102
+         </size>
103
+        </property>
104
         <property name="text">
105
          <string>OBSWebSocket.ConnectInfo.CopyText</string>
106
         </property>
107
@@ -87,6 +126,12 @@
108
     </item>
109
     <item row="2" column="0">
110
      <widget class="QLabel" name="serverPasswordLabel">
111
+      <property name="maximumSize">
112
+       <size>
113
+        <width>200</width>
114
+        <height>16777215</height>
115
+       </size>
116
+      </property>
117
       <property name="text">
118
        <string>OBSWebSocket.ConnectInfo.ServerPassword</string>
119
       </property>
120
@@ -96,9 +141,15 @@
121
      <layout class="QHBoxLayout" name="horizontalLayout_3">
122
       <item>
123
        <widget class="QLineEdit" name="serverPasswordLineEdit">
124
+        <property name="focusPolicy">
125
+         <enum>Qt::NoFocus</enum>
126
+        </property>
127
         <property name="text">
128
          <string>OBSWebSocket.ConnectInfo.ServerPasswordPlaceholderText</string>
129
         </property>
130
+        <property name="alignment">
131
+         <set>Qt::AlignBottom|Qt::AlignLeading|Qt::AlignLeft</set>
132
+        </property>
133
         <property name="readOnly">
134
          <bool>true</bool>
135
         </property>
136
@@ -106,6 +157,12 @@
137
       </item>
138
       <item>
139
        <widget class="QPushButton" name="copyServerPasswordButton">
140
+        <property name="maximumSize">
141
+         <size>
142
+          <width>75</width>
143
+          <height>16777215</height>
144
+         </size>
145
+        </property>
146
         <property name="text">
147
          <string>OBSWebSocket.ConnectInfo.CopyText</string>
148
         </property>
149
@@ -119,7 +176,7 @@
150
    <property name="geometry">
151
     <rect>
152
      <x>10</x>
153
-     <y>120</y>
154
+     <y>140</y>
155
      <width>431</width>
156
      <height>281</height>
157
     </rect>
158
@@ -131,9 +188,9 @@
159
     <property name="geometry">
160
      <rect>
161
       <x>100</x>
162
-      <y>40</y>
163
-      <width>230</width>
164
-      <height>230</height>
165
+      <y>30</y>
166
+      <width>236</width>
167
+      <height>236</height>
168
      </rect>
169
     </property>
170
     <property name="text">
171
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/src/forms/SettingsDialog.ui -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/src/forms/SettingsDialog.ui Changed
23
 
1
@@ -80,7 +80,7 @@
2
       <item row="3" column="1">
3
        <layout class="QHBoxLayout" name="horizontalLayout_2">
4
         <property name="spacing">
5
-         <number>0</number>
6
+         <number>2</number>
7
         </property>
8
         <item>
9
          <widget class="QCheckBox" name="enableDebugLoggingCheckBox">
10
@@ -97,6 +97,12 @@
11
           <property name="toolTip">
12
            <string>OBSWebSocket.Settings.DebugEnableHoverText</string>
13
           </property>
14
+          <property name="scaledContents">
15
+           <bool>true</bool>
16
+          </property>
17
+          <property name="alignment">
18
+           <set>Qt::AlignCenter</set>
19
+          </property>
20
          </widget>
21
         </item>
22
         <item>
23
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/src/requesthandler/RequestHandler_General.cpp -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/src/requesthandler/RequestHandler_General.cpp Changed
33
 
1
@@ -103,6 +103,20 @@
2
 }
3
 
4
 /**
5
+ * Custom event emitted by `BroadcastCustomEvent`.
6
+ * 
7
+ * @dataField eventData | Object | Custom event data
8
+ *
9
+ * @eventType CustomEvent
10
+ * @eventSubscription General
11
+ * @complexity 1
12
+ * @rpcVersion -1
13
+ * @initialVersion 5.0.0
14
+ * @category general
15
+ * @api events
16
+ */
17
+
18
+/**
19
  * Broadcasts a `CustomEvent` to all WebSocket clients. Receivers are clients which are identified and subscribed.
20
  *
21
  * @requestField eventData | Object | Data payload to emit to all receivers
22
@@ -311,8 +325,8 @@
23
 /**
24
  * Sleeps for a time duration or number of frames. Only available in request batches with types `SERIAL_REALTIME` or `SERIAL_FRAME`.
25
  *
26
- * @requestField sleepMillis | Number | Number of milliseconds to sleep for (if `SERIAL_REALTIME` mode) | >= 0, <= 50000
27
- * @requestField sleepFrames | Number | Number of frames to sleep for (if `SERIAL_FRAME` mode) | >= 0, <= 10000
28
+ * @requestField ?sleepMillis | Number | Number of milliseconds to sleep for (if `SERIAL_REALTIME` mode) | >= 0, <= 50000
29
+ * @requestField ?sleepFrames | Number | Number of frames to sleep for (if `SERIAL_FRAME` mode) | >= 0, <= 10000
30
  *
31
  * @requestType Sleep
32
  * @complexity 2
33
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/src/requesthandler/RequestHandler_Inputs.cpp -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/src/requesthandler/RequestHandler_Inputs.cpp Changed
12
 
1
@@ -889,6 +889,10 @@
2
 /**
3
  * Presses a button in the properties of an input.
4
  *
5
+ * Some known `propertyName` values are:
6
+ *
7
+ * - `refreshnocache` - Browser source reload button
8
+ *
9
  * Note: Use this in cases where there is a button in the properties of an input that cannot be accessed in any other way. For example, browser sources, where there is a refresh button.
10
  *
11
  * @requestField inputName    | String | Name of the input
12
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/src/requesthandler/RequestHandler_Outputs.cpp -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/src/requesthandler/RequestHandler_Outputs.cpp Changed
10
 
1
@@ -278,6 +278,8 @@
2
 
3
 /**
4
  * Gets the list of available outputs.
5
+ * 
6
+ * @responseField outputs | Array<Object> | Array of outputs
7
  *
8
  * @requestType GetOutputList
9
  * @complexity 4
10
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/src/requesthandler/RequestHandler_Record.cpp -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/src/requesthandler/RequestHandler_Record.cpp Changed
10
 
1
@@ -23,7 +23,7 @@
2
  * Gets the status of the record output.
3
  *
4
  * @responseField outputActive        | Boolean | Whether the output is active
5
- * @responseField ouputPaused         | Boolean | Whether the output is paused
6
+ * @responseField outputPaused        | Boolean | Whether the output is paused
7
  * @responseField outputTimecode      | String  | Current formatted timecode string for the output
8
  * @responseField outputDuration      | Number  | Current duration in milliseconds for the output
9
  * @responseField outputBytes         | Number  | Number of bytes sent by the output
10
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/src/requesthandler/RequestHandler_SceneItems.cpp -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/src/requesthandler/RequestHandler_SceneItems.cpp Changed
10
 
1
@@ -52,7 +52,7 @@
2
 /**
3
  * Basically GetSceneItemList, but for groups.
4
  *
5
- * Using groups at all in OBS is discouraged, as they are very broken under the hood.
6
+ * Using groups at all in OBS is discouraged, as they are very broken under the hood. Please use nested scenes instead.
7
  *
8
  * Groups only
9
  *
10
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/src/utils/Obs.h -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/src/utils/Obs.h Changed
124
 
1
@@ -67,22 +67,105 @@
2
 }
3
 
4
 enum ObsOutputState {
5
+   /**
6
+   * Unknown state.
7
+   *
8
+   * @enumIdentifier OBS_WEBSOCKET_OUTPUT_UNKNOWN
9
+   * @enumType ObsOutputState
10
+   * @rpcVersion 1
11
+   * @initialVersion 5.0.0
12
+   * @api enums
13
+   */
14
    OBS_WEBSOCKET_OUTPUT_UNKNOWN,
15
+   /**
16
+   * The output is starting.
17
+   *
18
+   * @enumIdentifier OBS_WEBSOCKET_OUTPUT_STARTING
19
+   * @enumType ObsOutputState
20
+   * @rpcVersion 1
21
+   * @initialVersion 5.0.0
22
+   * @api enums
23
+   */
24
    OBS_WEBSOCKET_OUTPUT_STARTING,
25
+   /**
26
+   * The input has started.
27
+   *
28
+   * @enumIdentifier OBS_WEBSOCKET_OUTPUT_STARTED
29
+   * @enumType ObsOutputState
30
+   * @rpcVersion 1
31
+   * @initialVersion 5.0.0
32
+   * @api enums
33
+   */
34
    OBS_WEBSOCKET_OUTPUT_STARTED,
35
+   /**
36
+   * The output is stopping.
37
+   *
38
+   * @enumIdentifier OBS_WEBSOCKET_OUTPUT_STOPPING
39
+   * @enumType ObsOutputState
40
+   * @rpcVersion 1
41
+   * @initialVersion 5.0.0
42
+   * @api enums
43
+   */
44
    OBS_WEBSOCKET_OUTPUT_STOPPING,
45
+   /**
46
+   * The output has stopped.
47
+   *
48
+   * @enumIdentifier OBS_WEBSOCKET_OUTPUT_STOPPED
49
+   * @enumType ObsOutputState
50
+   * @rpcVersion 1
51
+   * @initialVersion 5.0.0
52
+   * @api enums
53
+   */
54
    OBS_WEBSOCKET_OUTPUT_STOPPED,
55
+   /**
56
+   * The output has disconnected and is reconnecting.
57
+   *
58
+   * @enumIdentifier OBS_WEBSOCKET_OUTPUT_RECONNECTING
59
+   * @enumType ObsOutputState
60
+   * @rpcVersion 1
61
+   * @initialVersion 5.0.0
62
+   * @api enums
63
+   */
64
    OBS_WEBSOCKET_OUTPUT_RECONNECTING,
65
+   /**
66
+   * The output has reconnected successfully.
67
+   *
68
+   * @enumIdentifier OBS_WEBSOCKET_OUTPUT_RECONNECTED
69
+   * @enumType ObsOutputState
70
+   * @rpcVersion 1
71
+   * @initialVersion 5.1.0
72
+   * @api enums
73
+   */
74
+   OBS_WEBSOCKET_OUTPUT_RECONNECTED,
75
+   /**
76
+   * The output is now paused.
77
+   *
78
+   * @enumIdentifier OBS_WEBSOCKET_OUTPUT_PAUSED
79
+   * @enumType ObsOutputState
80
+   * @rpcVersion 1
81
+   * @initialVersion 5.1.0
82
+   * @api enums
83
+   */
84
    OBS_WEBSOCKET_OUTPUT_PAUSED,
85
+   /**
86
+   * The output has been resumed (unpaused).
87
+   *
88
+   * @enumIdentifier OBS_WEBSOCKET_OUTPUT_RESUMED
89
+   * @enumType ObsOutputState
90
+   * @rpcVersion 1
91
+   * @initialVersion 5.0.0
92
+   * @api enums
93
+   */
94
    OBS_WEBSOCKET_OUTPUT_RESUMED,
95
 };
96
-
97
 NLOHMANN_JSON_SERIALIZE_ENUM(ObsOutputState, {
98
                             {OBS_WEBSOCKET_OUTPUT_UNKNOWN, "OBS_WEBSOCKET_OUTPUT_UNKNOWN"},
99
                             {OBS_WEBSOCKET_OUTPUT_STARTING, "OBS_WEBSOCKET_OUTPUT_STARTING"},
100
                             {OBS_WEBSOCKET_OUTPUT_STARTED, "OBS_WEBSOCKET_OUTPUT_STARTED"},
101
                             {OBS_WEBSOCKET_OUTPUT_STOPPING, "OBS_WEBSOCKET_OUTPUT_STOPPING"},
102
                             {OBS_WEBSOCKET_OUTPUT_STOPPED, "OBS_WEBSOCKET_OUTPUT_STOPPED"},
103
+                            {OBS_WEBSOCKET_OUTPUT_RECONNECTING, "OBS_WEBSOCKET_OUTPUT_RECONNECTING"},
104
+                            {OBS_WEBSOCKET_OUTPUT_RECONNECTED, "OBS_WEBSOCKET_OUTPUT_RECONNECTED"},
105
                             {OBS_WEBSOCKET_OUTPUT_PAUSED, "OBS_WEBSOCKET_OUTPUT_PAUSED"},
106
                             {OBS_WEBSOCKET_OUTPUT_RESUMED, "OBS_WEBSOCKET_OUTPUT_RESUMED"},
107
                         })
108
@@ -159,7 +242,6 @@
109
    */
110
    OBS_WEBSOCKET_MEDIA_INPUT_ACTION_PREVIOUS,
111
 };
112
-
113
 NLOHMANN_JSON_SERIALIZE_ENUM(ObsMediaInputAction,
114
                 {
115
                     {OBS_WEBSOCKET_MEDIA_INPUT_ACTION_NONE, "OBS_WEBSOCKET_MEDIA_INPUT_ACTION_NONE"},
116
@@ -181,6 +263,7 @@
117
            std::string GetCurrentRecordOutputPath();
118
            std::string GetLastRecordFileName();
119
            std::string GetLastReplayBufferFileName();
120
+           std::string GetLastScreenshotFileName();
121
            std::string DurationToTimecode(uint64_t);
122
        }
123
 
124
obs-studio-28.1.2.tar.xz/plugins/obs-websocket/src/utils/Obs_StringHelper.cpp -> obs-studio-29.0.0.tar.xz/plugins/obs-websocket/src/utils/Obs_StringHelper.cpp Changed
33
 
1
@@ -94,20 +94,18 @@
2
 
3
 std::string Utils::Obs::StringHelper::GetLastReplayBufferFileName()
4
 {
5
-   OBSOutputAutoRelease output = obs_frontend_get_replay_buffer_output();
6
-   if (!output)
7
-       return "";
8
-
9
-   calldata_t cd = {0};
10
-   proc_handler_t *ph = obs_output_get_proc_handler(output);
11
-   proc_handler_call(ph, "get_last_replay", &cd);
12
-   const char *savedReplayPath = calldata_string(&cd, "path");
13
-   calldata_free(&cd);
14
-
15
-   if (!savedReplayPath)
16
-       return "";
17
+   char *replayBufferPath = obs_frontend_get_last_replay();
18
+   std::string ret = replayBufferPath;
19
+   bfree(replayBufferPath);
20
+   return ret;
21
+}
22
 
23
-   return savedReplayPath;
24
+std::string Utils::Obs::StringHelper::GetLastScreenshotFileName()
25
+{
26
+   char *screenshotPath = obs_frontend_get_last_screenshot();
27
+   std::string ret = screenshotPath;
28
+   bfree(screenshotPath);
29
+   return ret;
30
 }
31
 
32
 std::string Utils::Obs::StringHelper::DurationToTimecode(uint64_t ms)
33
obs-studio-28.1.2.tar.xz/plugins/obs-x264/obs-x264.c -> obs-studio-29.0.0.tar.xz/plugins/obs-x264/obs-x264.c Changed
10
 
1
@@ -361,7 +361,7 @@
2
    struct obs_x264 *obsx264 = param;
3
    char str1024;
4
 
5
-   vsnprintf(str, 1024, format, args);
6
+   vsnprintf(str, sizeof(str), format, args);
7
    info("%s", str);
8
 
9
    UNUSED_PARAMETER(level);
10
obs-studio-28.1.2.tar.xz/plugins/rtmp-services/CMakeLists.txt -> obs-studio-29.0.0.tar.xz/plugins/rtmp-services/CMakeLists.txt Changed
12
 
1
@@ -43,6 +43,10 @@
2
                  rtmp-services.rc)
3
 
4
   target_sources(rtmp-services PRIVATE rtmp-services.rc)
5
+
6
+  if(MSVC)
7
+    target_link_options(rtmp-services PRIVATE "LINKER:/IGNORE:4098")
8
+  endif()
9
 endif()
10
 
11
 set_target_properties(rtmp-services PROPERTIES FOLDER "plugins" PREFIX "")
12
obs-studio-28.1.2.tar.xz/plugins/rtmp-services/data/package.json -> obs-studio-29.0.0.tar.xz/plugins/rtmp-services/data/package.json Changed
16
 
1
@@ -1,11 +1,11 @@
2
 {
3
     "$schema": "schema/package-schema.json",
4
-    "url": "https://obsproject.com/obs2_update/rtmp-services",
5
-    "version": 211,
6
+    "url": "https://obsproject.com/obs2_update/rtmp-services/v4",
7
+    "version": 216,
8
     "files": 
9
         {
10
             "name": "services.json",
11
-            "version": 211
12
+            "version": 216
13
         }
14
     
15
 }
16
obs-studio-29.0.0.tar.xz/plugins/rtmp-services/data/schema/service-schema-v4.json Added
201
 
1
@@ -0,0 +1,239 @@
2
+{
3
+    "$schema": "http://json-schema.org/draft-07/schema#",
4
+    "type": "object",
5
+    "properties": {
6
+        "format_version": {
7
+            "type": "integer",
8
+            "description": "Identifier for parsing this file.\n- v4 introduced 'ffmpeg_mpegts_muxer' to services/recommended/output\n - v3 introduced 'ffmpeg_hls_muxer' to services/recommended/output\n - v2 introduced 'alt_names' to services"
9
+        },
10
+        "services": {
11
+            "type": "array",
12
+            "items": {
13
+                "type": "object",
14
+                "properties": {
15
+                    "name": {
16
+                        "type": "string",
17
+                        "description": "Name of the streaming service. Will be displayed in the Service dropdown.",
18
+                        "minLength": 1
19
+                    },
20
+                    "common": {
21
+                        "type": "boolean",
22
+                        "description": "Whether or not the service is shown in the list before it is expanded to all services by the user.",
23
+                        "default": false
24
+                    },
25
+                    "stream_key_link": {
26
+                        "$ref": "#/definitions/saneUrl",
27
+                        "description": "Link where a logged-in user can find the 'stream key', presented as a button alongside the stream key field."
28
+                    },
29
+                    "supported video codecs": {
30
+                        "type": "array",
31
+                        "description": "Video codecs that are supported by the service.",
32
+                        "items": {
33
+                            "type": "string",
34
+                            "description": "Short-form codec names.",
35
+                            "minLength": 1,
36
+                            "enum": 
37
+                                "h264",
38
+                                "hevc"
39
+                            
40
+                        }
41
+                    },
42
+                    "supported audio codecs": {
43
+                        "type": "array",
44
+                        "description": "Audio codecs that are supported by the service.",
45
+                        "items": {
46
+                            "type": "string",
47
+                            "description": "Short-form codec names.",
48
+                            "minLength": 1,
49
+                            "enum": 
50
+                                "aac",
51
+                                "opus"
52
+                            
53
+                        }
54
+                    },
55
+                    "servers": {
56
+                        "type": "array",
57
+                        "description": "List of servers.",
58
+                        "items": {
59
+                            "type": "object",
60
+                            "properties": {
61
+                                "name": {
62
+                                    "type": "string",
63
+                                    "description": "Name of the server (e.g. location, primary/backup), displayed in the Server dropdown.",
64
+                                    "minLength": 1
65
+                                },
66
+                                "url": {
67
+                                    "$ref": "#/definitions/serviceUri",
68
+                                    "description": "RTMP(S) or HLS URL of the ingest server.",
69
+                                    "minLength": 1
70
+                                }
71
+                            },
72
+                            "additionalProperties": false,
73
+                            "required": 
74
+                                "name",
75
+                                "url"
76
+                            
77
+                        },
78
+                        "default": 
79
+                            {
80
+                                "name": "",
81
+                                "url": ""
82
+                            }
83
+                        ,
84
+                        "minItems": 1,
85
+                        "additionalItems": true
86
+                    },
87
+                    "recommended": {
88
+                        "type": "object",
89
+                        "description": "Recommended service settings. Users will be unable to choose values outside of these by default, so choose recommended values carefully.",
90
+                        "properties": {
91
+                            "keyint": {
92
+                                "type": "integer",
93
+                                "description": "Keyframe interval (seconds)."
94
+                            },
95
+                            "max video bitrate": {
96
+                                "type": "integer",
97
+                                "description": "Highest supported video bitrate (kbps)."
98
+                            },
99
+                            "max audio bitrate": {
100
+                                "type": "integer",
101
+                                "description": "Highest supported audio bitrate (kbps)."
102
+                            },
103
+                            "x264opts": {
104
+                                "type": "string",
105
+                                "description": "Additional x264 encoder options. Space-separated.",
106
+                                "pattern": "^(\\S+=\\S+\\s*)+$"
107
+                            },
108
+                            "output": {
109
+                                "type": "string",
110
+                                "description": "OBS output module used.",
111
+                                "enum": 
112
+                                    "rtmp_output",
113
+                                    "ffmpeg_hls_muxer",
114
+                                    "ftl_output",
115
+                                    "ffmpeg_mpegts_muxer"
116
+                                
117
+                            },
118
+                            "profile": {
119
+                                "type": "string",
120
+                                "description": "H.264 Profile.",
121
+                                "minLength": 1,
122
+                                "enum": 
123
+                                    "high",
124
+                                    "main",
125
+                                    "baseline"
126
+                                
127
+                            },
128
+                            "bframes": {
129
+                                "type": "integer",
130
+                                "description": "Maximum allowed number of B-Frames."
131
+                            },
132
+                            "supported resolutions": {
133
+                                "type": "array",
134
+                                "description": "List of supported resolutions in format {width}x{height}",
135
+                                "items": {
136
+                                    "$ref": "#/definitions/resolution"
137
+                                },
138
+                                "minItems": 1,
139
+                                "additionalItems": true
140
+                            },
141
+                            "max fps": {
142
+                                "type": "integer",
143
+                                "description": "Maximum supported framerate."
144
+                            },
145
+                            "bitrate matrix": {
146
+                                "type": "array",
147
+                                "description": "List of resolutions and frame rate combinations with their recommended maximum bitrate.",
148
+                                "items": {
149
+                                    "type": "object",
150
+                                    "properties": {
151
+                                        "res": {
152
+                                            "$ref": "#/definitions/resolution",
153
+                                            "description": "Resolution in format {width}x{height}"
154
+                                        },
155
+                                        "fps": {
156
+                                            "type": "integer",
157
+                                            "description": "Frame rate"
158
+                                        },
159
+                                        "max bitrate": {
160
+                                            "type": "integer",
161
+                                            "description": "Maximum bitrate in kbps."
162
+                                        }
163
+                                    },
164
+                                    "minItems": 1,
165
+                                    "additionalProperties": false,
166
+                                    "required": 
167
+                                        "res",
168
+                                        "fps",
169
+                                        "max bitrate"
170
+                                    
171
+                                },
172
+                                "default": 
173
+                                    {
174
+                                        "res": "",
175
+                                        "fps": "",
176
+                                        "max bitrate": ""
177
+                                    }
178
+                                ,
179
+                                "additionalItems": true
180
+                            }
181
+                        },
182
+                        "additionalProperties": false
183
+                    },
184
+                    "more_info_link": {
185
+                        "$ref": "#/definitions/saneUrl",
186
+                        "description": "Link that provides additional info about the service, presented in the UI as a button next to the services dropdown."
187
+                    },
188
+                    "alt_names": {
189
+                        "type": "array",
190
+                        "description": "Previous names of the service used for migrating existing users to the updated entry.",
191
+                        "items": {
192
+                            "type": "string",
193
+                            "minLength": 1
194
+                        },
195
+                        "default": 
196
+                            ""
197
+                        
198
+                    }
199
+                },
200
+                "additionalProperties": false,
201
obs-studio-28.1.2.tar.xz/plugins/rtmp-services/data/services.json -> obs-studio-29.0.0.tar.xz/plugins/rtmp-services/data/services.json Changed
112
 
1
@@ -1,6 +1,6 @@
2
 {
3
-    "$schema": "schema/service-schema-v3.json",
4
-    "format_version": 3,
5
+    "$schema": "schema/service-schema-v4.json",
6
+    "format_version": 4,
7
     "services": 
8
         {
9
             "name": "Twitch",
10
@@ -2072,29 +2072,6 @@
11
             }
12
         },
13
         {
14
-            "name": "Brime Live",
15
-            "stream_key_link": "https://brime.tv/studio",
16
-            "servers": 
17
-                {
18
-                    "name": "North America - Ashburn, VA",
19
-                    "url": "rtmp://ingest-us-ashburn.brime.tv/live"
20
-                },
21
-                {
22
-                    "name": "North America - San Jose, CA",
23
-                    "url": "rtmp://ingest-us-sanjose.brime.tv/live"
24
-                },
25
-                {
26
-                    "name": "Europe / EMEA - Germany (Frankfurt)",
27
-                    "url": "rtmp://ingest-eu-frankfurt.brime.tv/live"
28
-                }
29
-            ,
30
-            "recommended": {
31
-                "max video bitrate": 30000,
32
-                "max audio bitrate": 320,
33
-                "x264opts": "scenecut=0"
34
-            }
35
-        },
36
-        {
37
             "name": "Bilibili Live - RTMP | 哔哩哔哩直播 - RTMP",
38
             "more_info_link": "https://link.bilibili.com/p/help/index#/tools-tutorial?id=9",
39
             "stream_key_link": "https://link.bilibili.com/p/center/index#/my-room/start-live",
40
@@ -2546,6 +2523,71 @@
41
                 "max video bitrate": 20000,
42
                 "max audio bitrate": 192
43
             }
44
+        },
45
+        {
46
+            "name": "Whowatch (ふわっち)",
47
+            "more_info_link": "https://whowatch.tv/help/encoder",
48
+            "stream_key_link": "https://whowatch.tv/publish",
49
+            "servers": 
50
+                {
51
+                    "name": "default",
52
+                    "url": "rtmp://live.whowatch.tv/live/"
53
+                }
54
+            ,
55
+            "recommended": {
56
+                "keyint": 2,
57
+                "max video bitrate": 1800,
58
+                "max audio bitrate": 192
59
+            }
60
+        },
61
+        {
62
+            "name": "IRLToolkit",
63
+            "stream_key_link": "https://irl.run/settings/ingest/",
64
+            "servers": 
65
+                {
66
+                    "name": "Global (Recommended)",
67
+                    "url": "rtmp://stream.global.irl.run/ingest"
68
+                },
69
+                {
70
+                    "name": "Los Angeles, US",
71
+                    "url": "rtmp://stream.lax.irl.run/ingest"
72
+                },
73
+                {
74
+                    "name": "New York, US",
75
+                    "url": "rtmp://stream.ewr.irl.run/ingest"
76
+                },
77
+                {
78
+                    "name": "Rotterdam, NL",
79
+                    "url": "rtmp://stream.rtm.irl.run/ingest"
80
+                },
81
+                {
82
+                    "name": "Singapore",
83
+                    "url": "rtmp://stream.sin.irl.run/ingest"
84
+                },
85
+                {
86
+                    "name": "Tokyo, JP",
87
+                    "url": "rtmp://stream.tyo.irl.run/ingest"
88
+                }
89
+            ,
90
+            "recommended": {
91
+                "keyint": 2,
92
+                "bframes": 2,
93
+                "max video bitrate": 20000
94
+            }
95
+        },
96
+        {
97
+            "name": "Bitmovin",
98
+            "more_info_link": "https://developer.bitmovin.com/docs/overview",
99
+            "stream_key_link": "https://bitmovin.com/dashboard/streams?streamsTab=LIVE",
100
+            "servers": 
101
+                {
102
+                    "name": "Streams Live",
103
+                    "url": "rtmp://live-input.bitmovin.com/streams"
104
+                }
105
+            ,
106
+            "recommended": {
107
+                "keyint": 2
108
+            }
109
         }
110
     
111
 }
112
obs-studio-28.1.2.tar.xz/plugins/rtmp-services/rtmp-format-ver.h -> obs-studio-29.0.0.tar.xz/plugins/rtmp-services/rtmp-format-ver.h Changed
6
 
1
@@ -1,3 +1,3 @@
2
 #pragma once
3
 
4
-#define RTMP_SERVICES_FORMAT_VERSION 3
5
+#define RTMP_SERVICES_FORMAT_VERSION 4
6
obs-studio-28.1.2.tar.xz/plugins/win-capture/cursor-capture.c -> obs-studio-29.0.0.tar.xz/plugins/win-capture/cursor-capture.c Changed
75
 
1
@@ -107,14 +107,19 @@
2
    bottom = bmp.bmWidthBytes * bmp.bmHeight;
3
 
4
    for (long i = 0; i < pixels; i++) {
5
-       uint8_t alpha = bit_to_alpha(mask, i, false);
6
-       uint8_t color = bit_to_alpha(mask + bottom, i, true);
7
-
8
-       if (!alpha) {
9
-           outputi * 4 + 3 = color;
10
+       uint8_t andMask = bit_to_alpha(mask, i, true);
11
+       uint8_t xorMask = bit_to_alpha(mask + bottom, i, true);
12
+
13
+       if (!andMask) {
14
+           // black in the AND mask
15
+           *(uint32_t *)&outputi * 4 =
16
+               !!xorMask ? 0x00FFFFFF /*always white*/
17
+                     : 0xFF000000 /*always black*/;
18
        } else {
19
-           *(uint32_t *)&outputi * 4 = !!color ? 0xFFFFFFFF
20
-                                 : 0xFF000000;
21
+           // white in the AND mask
22
+           *(uint32_t *)&outputi * 4 =
23
+               !!xorMask ? 0xFFFFFFFF /*source inverted*/
24
+                     : 0 /*transparent*/;
25
        }
26
    }
27
 
28
@@ -126,13 +131,16 @@
29
 }
30
 
31
 static inline uint8_t *cursor_capture_icon_bitmap(ICONINFO *ii, uint32_t *width,
32
-                         uint32_t *height)
33
+                         uint32_t *height,
34
+                         bool *monochrome)
35
 {
36
    uint8_t *output;
37
-
38
+   *monochrome = false;
39
    output = copy_from_color(ii, width, height);
40
-   if (!output)
41
+   if (!output) {
42
+       *monochrome = true;
43
        output = copy_from_mask(ii, width, height);
44
+   }
45
 
46
    return output;
47
 }
48
@@ -170,7 +178,8 @@
49
        return false;
50
    }
51
 
52
-   bitmap = cursor_capture_icon_bitmap(&ii, &width, &height);
53
+   bitmap = cursor_capture_icon_bitmap(&ii, &width, &height,
54
+                       &data->monochrome);
55
    if (bitmap) {
56
        if (data->last_cx != width || data->last_cy != height) {
57
            data->texture = get_cached_texture(data, width, height);
58
@@ -228,9 +237,13 @@
59
 
60
    if (data->visible && !!data->texture) {
61
        gs_blend_state_push();
62
-       gs_blend_function_separate(GS_BLEND_SRCALPHA,
63
-                      GS_BLEND_INVSRCALPHA, GS_BLEND_ONE,
64
-                      GS_BLEND_INVSRCALPHA);
65
+       enum gs_blend_type blendMode = data->monochrome
66
+                              ? GS_BLEND_INVDSTCOLOR
67
+                              : GS_BLEND_SRCALPHA;
68
+       gs_blend_function_separate(blendMode, /*src_color*/
69
+                      GS_BLEND_INVSRCALPHA /*dest_color*/,
70
+                      GS_BLEND_ONE /*src_alpha*/,
71
+                      GS_BLEND_INVSRCALPHA /*dest_alpha*/);
72
 
73
        gs_matrix_push();
74
        obs_source_draw(data->texture, x_draw, y_draw, 0, 0, false);
75
obs-studio-28.1.2.tar.xz/plugins/win-capture/cursor-capture.h -> obs-studio-29.0.0.tar.xz/plugins/win-capture/cursor-capture.h Changed
9
 
1
@@ -15,6 +15,7 @@
2
    long x_hotspot;
3
    long y_hotspot;
4
    bool visible;
5
+   bool monochrome;
6
 
7
    uint32_t last_cx;
8
    uint32_t last_cy;
9
obs-studio-28.1.2.tar.xz/plugins/win-capture/duplicator-monitor-capture.c -> obs-studio-29.0.0.tar.xz/plugins/win-capture/duplicator-monitor-capture.c Changed
201
 
1
@@ -68,8 +68,8 @@
2
 struct duplicator_capture {
3
    obs_source_t *source;
4
    pthread_mutex_t update_mutex;
5
-   int monitor;
6
-   int dxgi_index;
7
+   char monitor_id128;
8
+   char monitor_name64;
9
    enum display_capture_method method;
10
    bool reset_wgc;
11
    HMONITOR handle;
12
@@ -93,9 +93,8 @@
13
 };
14
 
15
 struct wgc_monitor_info {
16
-   int cur_id;
17
-   int desired_id;
18
-   int id;
19
+   char device_id128;
20
+   char name64;
21
    RECT rect;
22
    HMONITOR handle;
23
 };
24
@@ -117,49 +116,128 @@
25
    return method_name;
26
 }
27
 
28
+static bool GetMonitorTarget(LPCWSTR device,
29
+                DISPLAYCONFIG_TARGET_DEVICE_NAME *target)
30
+{
31
+   bool found = false;
32
+
33
+   UINT32 numPath, numMode;
34
+   if (GetDisplayConfigBufferSizes(QDC_ONLY_ACTIVE_PATHS, &numPath,
35
+                   &numMode) == ERROR_SUCCESS) {
36
+       DISPLAYCONFIG_PATH_INFO *paths =
37
+           bmalloc(numPath * sizeof(DISPLAYCONFIG_PATH_INFO));
38
+       DISPLAYCONFIG_MODE_INFO *modes =
39
+           bmalloc(numMode * sizeof(DISPLAYCONFIG_MODE_INFO));
40
+       if (QueryDisplayConfig(QDC_ONLY_ACTIVE_PATHS, &numPath, paths,
41
+                      &numMode, modes,
42
+                      NULL) == ERROR_SUCCESS) {
43
+           for (size_t i = 0; i < numPath; ++i) {
44
+               const DISPLAYCONFIG_PATH_INFO *const path =
45
+                   &pathsi;
46
+
47
+               DISPLAYCONFIG_SOURCE_DEVICE_NAME
48
+               source;
49
+               source.header.type =
50
+                   DISPLAYCONFIG_DEVICE_INFO_GET_SOURCE_NAME;
51
+               source.header.size = sizeof(source);
52
+               source.header.adapterId =
53
+                   path->sourceInfo.adapterId;
54
+               source.header.id = path->sourceInfo.id;
55
+               if (DisplayConfigGetDeviceInfo(
56
+                       &source.header) == ERROR_SUCCESS &&
57
+                   wcscmp(device, source.viewGdiDeviceName) ==
58
+                       0) {
59
+                   target->header.type =
60
+                       DISPLAYCONFIG_DEVICE_INFO_GET_TARGET_NAME;
61
+                   target->header.size = sizeof(*target);
62
+                   target->header.adapterId =
63
+                       path->sourceInfo.adapterId;
64
+                   target->header.id = path->targetInfo.id;
65
+                   found = DisplayConfigGetDeviceInfo(
66
+                           &target->header) ==
67
+                       ERROR_SUCCESS;
68
+                   break;
69
+               }
70
+           }
71
+       }
72
+
73
+       bfree(modes);
74
+       bfree(paths);
75
+   }
76
+
77
+   return found;
78
+}
79
+
80
+static void GetMonitorName(HMONITOR handle, char *name, size_t count)
81
+{
82
+   MONITORINFOEXW mi;
83
+   DISPLAYCONFIG_TARGET_DEVICE_NAME target;
84
+
85
+   mi.cbSize = sizeof(mi);
86
+   if (GetMonitorInfoW(handle, (LPMONITORINFO)&mi) &&
87
+       GetMonitorTarget(mi.szDevice, &target)) {
88
+       snprintf(name, count, "%ls", target.monitorFriendlyDeviceName);
89
+   } else {
90
+       strcpy_s(name, count, "OBS: Unknown");
91
+   }
92
+}
93
+
94
 static BOOL CALLBACK enum_monitor(HMONITOR handle, HDC hdc, LPRECT rect,
95
                  LPARAM param)
96
 {
97
+   UNUSED_PARAMETER(hdc);
98
+
99
    struct wgc_monitor_info *monitor = (struct wgc_monitor_info *)param;
100
 
101
-   if (monitor->cur_id == 0 || monitor->desired_id == monitor->cur_id) {
102
-       monitor->id = monitor->cur_id;
103
-       monitor->rect = *rect;
104
-       monitor->handle = handle;
105
+   bool match = false;
106
+
107
+   MONITORINFOEXA mi;
108
+   mi.cbSize = sizeof(mi);
109
+   if (GetMonitorInfoA(handle, (LPMONITORINFO)&mi)) {
110
+       DISPLAY_DEVICEA device;
111
+       device.cb = sizeof(device);
112
+       if (EnumDisplayDevicesA(mi.szDevice, 0, &device,
113
+                   EDD_GET_DEVICE_INTERFACE_NAME)) {
114
+           const bool match = strcmp(monitor->device_id,
115
+                         device.DeviceID) == 0;
116
+           if (match) {
117
+               monitor->rect = *rect;
118
+               monitor->handle = handle;
119
+               GetMonitorName(handle, monitor->name,
120
+                          _countof(monitor->name));
121
+           }
122
+       }
123
    }
124
 
125
-   UNUSED_PARAMETER(hdc);
126
-   return (monitor->desired_id > monitor->cur_id++);
127
+   return !match;
128
 }
129
 
130
-static void log_settings(struct duplicator_capture *capture, int monitor,
131
-            LONG width, LONG height)
132
+static void log_settings(struct duplicator_capture *capture,
133
+            const char *monitor, LONG width, LONG height)
134
 {
135
    info("update settings:\n"
136
-        "\tdisplay: %d (%ldx%ld)\n"
137
+        "\tdisplay: %s (%ldx%ld)\n"
138
         "\tcursor: %s\n"
139
         "\tmethod: %s",
140
-        monitor + 1, width, height,
141
-        capture->capture_cursor ? "true" : "false",
142
+        monitor, width, height, capture->capture_cursor ? "true" : "false",
143
         get_method_name(capture->method));
144
 }
145
 
146
 static enum display_capture_method
147
 choose_method(enum display_capture_method method, bool wgc_supported,
148
-         HMONITOR monitor, int *dxgi_index)
149
+         HMONITOR monitor)
150
 {
151
    if (!wgc_supported)
152
        method = METHOD_DXGI;
153
 
154
-   if (method != METHOD_WGC) {
155
+   if (method == METHOD_AUTO) {
156
+       method = METHOD_DXGI;
157
+
158
        obs_enter_graphics();
159
-       *dxgi_index = gs_duplicator_get_monitor_index(monitor);
160
+       const int dxgi_index = gs_duplicator_get_monitor_index(monitor);
161
        obs_leave_graphics();
162
-   }
163
 
164
-   if (method == METHOD_AUTO) {
165
-       method = METHOD_DXGI;
166
-       if (*dxgi_index == -1) {
167
+       if (dxgi_index == -1) {
168
            method = METHOD_WGC;
169
        } else {
170
            SYSTEM_POWER_STATUS status;
171
@@ -185,14 +263,18 @@
172
    pthread_mutex_lock(&capture->update_mutex);
173
 
174
    struct wgc_monitor_info monitor = {0};
175
-   monitor.desired_id = (int)obs_data_get_int(settings, "monitor");
176
+   strcpy_s(monitor.device_id, _countof(monitor.device_id),
177
+        obs_data_get_string(settings, "monitor_id"));
178
    EnumDisplayMonitors(NULL, NULL, enum_monitor, (LPARAM)&monitor);
179
 
180
-   capture->method = choose_method(
181
-       (int)obs_data_get_int(settings, "method"), wgc_supported,
182
-       monitor.handle, &capture->dxgi_index);
183
+   capture->method =
184
+       choose_method((int)obs_data_get_int(settings, "method"),
185
+                 wgc_supported, monitor.handle);
186
 
187
-   capture->monitor = monitor.id;
188
+   strcpy_s(capture->monitor_id, _countof(capture->monitor_id),
189
+        monitor.device_id);
190
+   strcpy_s(capture->monitor_name, _countof(capture->monitor_name),
191
+        monitor.name);
192
    capture->handle = monitor.handle;
193
 
194
    capture->capture_cursor = obs_data_get_bool(settings, "capture_cursor");
195
@@ -266,7 +348,7 @@
196
 static void duplicator_capture_defaults(obs_data_t *settings)
197
 {
198
    obs_data_set_default_int(settings, "method", METHOD_AUTO);
199
-   obs_data_set_default_int(settings, "monitor", 0);
200
+   obs_data_set_default_string(settings, "monitor_id", "DUMMY");
201
obs-studio-28.1.2.tar.xz/plugins/win-capture/game-capture.c -> obs-studio-29.0.0.tar.xz/plugins/win-capture/game-capture.c Changed
10
 
1
@@ -823,7 +823,7 @@
2
 static inline bool init_pipe(struct game_capture *gc)
3
 {
4
    char name64;
5
-   sprintf(name, "%s%lu", PIPE_NAME, gc->process_id);
6
+   snprintf(name, sizeof(name), "%s%lu", PIPE_NAME, gc->process_id);
7
 
8
    if (!ipc_pipe_server_start(&gc->pipe, name, pipe_log, gc)) {
9
        warn("init_pipe: failed to start pipe");
10
obs-studio-28.1.2.tar.xz/plugins/win-capture/get-graphics-offsets/d3d9-offsets.cpp -> obs-studio-29.0.0.tar.xz/plugins/win-capture/get-graphics-offsets/d3d9-offsets.cpp Changed
201
 
1
@@ -83,15 +83,32 @@
2
 
3
 #define MAX_CMP_SIZE 22
4
 
5
+// clang-format off
6
 static const uint8_t maskMAX_CMP_SIZE = {
7
-   {0xF8, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0x00, 0x00,
8
-    0x00, 0x00, 0x00, 0xFF, 0x00, 0xF8, 0xF8, 0x00, 0x00, 0x00, 0x00},
9
-   {0xF8, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xC0, 0x00, 0x00,
10
-    0x00, 0x00, 0xFF, 0x00, 0xF8, 0xF8, 0x00, 0x00, 0x00, 0x00},
11
-   {0xF8, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xC0, 0x00, 0x00,
12
-    0x00, 0x00, 0xFF, 0x00, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00},
13
-   {0xF8, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xC0, 0x00, 0x00,
14
-    0x00, 0x00, 0x00, 0xFF, 0x00, 0xFF, 0xFF, 0x00, 0x00, 0x00},
15
+   {
16
+       0xF8, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00,
17
+       0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x00,
18
+       0xFF, 0x00,
19
+       0xF8, 0xF8, 0x00, 0x00, 0x00, 0x00
20
+   },
21
+   {
22
+       0xF8, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00,
23
+       0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00,
24
+       0xFF, 0x00,
25
+       0xF8, 0xF8, 0x00, 0x00, 0x00, 0x00
26
+   },
27
+   {
28
+       0xF8, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00,
29
+       0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00,
30
+       0xFF, 0x00,
31
+       0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00
32
+   },
33
+   {
34
+       0xF8, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00,
35
+       0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0x00,
36
+       0xFF, 0x00,
37
+       0xFF, 0xFF, 0x00, 0x00, 0x00
38
+   },
39
 };
40
 
41
 static const uint8_t mask_cmpMAX_CMP_SIZE = {
42
@@ -102,8 +119,12 @@
43
     * 75 12                                   jnz     short loc_7FF7AA90530
44
     * 41 B8 F9 19 00 00                       mov     r8d, 19F9h
45
     */
46
-   {0x48, 0x8B, 0x80, 0x00, 0x00, 0x00, 0x00, 0x44, 0x39, 0x00, 0x00,
47
-    0x00, 0x00, 0x00, 0x75, 0x00, 0x40, 0xB8, 0x00, 0x00, 0x00, 0x00},
48
+   {
49
+       0x48, 0x8B, 0x80, 0x00, 0x00, 0x00, 0x00,
50
+       0x44, 0x39, 0x00, 0x00, 0x00, 0x00, 0x00,
51
+       0x75, 0x00,
52
+       0x40, 0xB8, 0x00, 0x00, 0x00, 0x00
53
+   },
54
    /*
55
     * Windows ???+
56
     * 49 8B 87 78 41 00 00                    mov     rax, r15+4178h
57
@@ -111,8 +132,12 @@
58
     * 75 12                                   jnz     short loc_1800AEC9C
59
     * 41 B9 C3 1A 00 00                       mov     r9d, 1AC3h
60
     */
61
-   {0x48, 0x8B, 0x80, 0x00, 0x00, 0x00, 0x00, 0x39, 0x80, 0x00, 0x00,
62
-    0x00, 0x00, 0x75, 0x00, 0x40, 0xB8, 0x00, 0x00, 0x00, 0x00},
63
+   {
64
+       0x48, 0x8B, 0x80, 0x00, 0x00, 0x00, 0x00,
65
+       0x39, 0x80, 0x00, 0x00, 0x00, 0x00,
66
+       0x75, 0x00,
67
+       0x40, 0xB8, 0x00, 0x00, 0x00, 0x00
68
+   },
69
    /*
70
     * Windows 10 April 2018
71
     * 49 8B 87 58 40 00 00                 mov     rax, r15+4058h
72
@@ -122,8 +147,12 @@
73
     *
74
     * Note: different instructions, last byte skipped due to MAX_CMP_SIZE
75
     */
76
-   {0x48, 0x8B, 0x80, 0x00, 0x00, 0x00, 0x00, 0x39, 0x80, 0x00, 0x00,
77
-    0x00, 0x00, 0x75, 0x00, 0x48, 0x8D, 0x00, 0x00, 0x00, 0x00},
78
+   {
79
+       0x48, 0x8B, 0x80, 0x00, 0x00, 0x00, 0x00,
80
+       0x39, 0x80, 0x00, 0x00, 0x00, 0x00,
81
+       0x75, 0x00,
82
+       0x48, 0x8D, 0x00, 0x00, 0x00, 0x00
83
+   },
84
    /*
85
     * Windows 11 22H2
86
     * 49 8b 86 30 40 00 00     MOV        RAX,qword ptr R14 + 0x4030
87
@@ -133,9 +162,14 @@
88
     *
89
     * Note: different instructions, last byte skipped due to MAX_CMP_SIZE
90
     */
91
-   {0x48, 0x8B, 0x80, 0x00, 0x00, 0x00, 0x00, 0x83, 0x80, 0x00, 0x00,
92
-    0x00, 0x00, 0x00, 0x75, 0x00, 0x48, 0x8D, 0x00, 0x00, 0x00},
93
+   {
94
+       0x48, 0x8B, 0x80, 0x00, 0x00, 0x00, 0x00,
95
+       0x83, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00,
96
+       0x75, 0x00,
97
+       0x48, 0x8D, 0x00, 0x00, 0x00
98
+   },
99
 };
100
+// clang-format on
101
 
102
 // Offset into the code for the numbers we're interested in
103
 static const uint32_t code_offsets2 = {
104
@@ -148,15 +182,32 @@
105
 
106
 #define MAX_CMP_SIZE 20
107
 
108
+// clang-format off
109
 static const uint8_t maskMAX_CMP_SIZE = {
110
-   {0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00,
111
-    0xFF, 0x00, 0xFF, 0x00, 0x00, 0x00, 0x00},
112
-   {0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xC0, 0x00, 0x00,
113
-    0x00, 0x00, 0xFF, 0xFF, 0x00, 0xFF, 0x00, 0x00, 0x00, 0x00},
114
-   {0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xC0, 0x00, 0x00,
115
-    0x00, 0x00, 0xFF, 0xFF, 0x00, 0xFF, 0x00, 0x00, 0x00, 0x00},
116
-   {0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xC0, 0x00, 0x00,
117
-    0x00, 0x00, 0xFF, 0x00, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x00},
118
+   {
119
+       0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00,
120
+       0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00,
121
+       0xFF, 0x00,
122
+       0xFF, 0x00, 0x00, 0x00, 0x00
123
+   },
124
+   {
125
+       0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00,
126
+       0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0xFF,
127
+       0xFF, 0x00,
128
+       0xFF, 0x00, 0x00, 0x00, 0x00
129
+   },
130
+   {
131
+       0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00,
132
+       0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00, 0xFF,
133
+       0xFF, 0x00,
134
+       0xFF, 0x00, 0x00, 0x00, 0x00
135
+   },
136
+   {
137
+       0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00,
138
+       0xFF, 0xC0, 0x00, 0x00, 0x00, 0x00,
139
+       0xFF, 0x00,
140
+       0xFF, 0x00, 0x00, 0x00, 0x00, 0x00
141
+   },
142
 };
143
 
144
 static const uint8_t mask_cmpMAX_CMP_SIZE = {
145
@@ -167,18 +218,24 @@
146
     * 75 14                  jnz     short loc_754CD9E1
147
     * 68 F9 19 00 00         push    19F9h
148
     */
149
-   {0x8B, 0x80, 0x00, 0x00, 0x00, 0x00, 0x39, 0x80, 0x00, 0x00, 0x00, 0x00,
150
-    0x75, 0x00, 0x68, 0x00, 0x00, 0x00, 0x00},
151
-
152
+   {
153
+       0x8B, 0x80, 0x00, 0x00, 0x00, 0x00,
154
+       0x39, 0x80, 0x00, 0x00, 0x00, 0x00,
155
+       0x75, 0x00,
156
+       0x68, 0x00, 0x00, 0x00, 0x00
157
+   },
158
    /* Windows 10 Creator's Update+
159
     * 8B 86 F8 2B 00 00      mov     eax, esi+2BF8h
160
     * 83 B8 00 4D 00 00 00   cmp     dword ptr eax+4D00h, 0
161
     * 75 0F                  jnz     short loc_100D793C
162
     * 68 C3 1A 00 00         push    1AC3h
163
     */
164
-   {0x8B, 0x80, 0x00, 0x00, 0x00, 0x00, 0x83, 0x80, 0x00, 0x00,
165
-    0x00, 0x00, 0x00, 0x75, 0x00, 0x68, 0x00, 0x00, 0x00, 0x00},
166
-
167
+   {
168
+       0x8B, 0x80, 0x00, 0x00, 0x00, 0x00,
169
+       0x83, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00,
170
+       0x75, 0x00,
171
+       0x68, 0x00, 0x00, 0x00, 0x00
172
+   },
173
    /*
174
     * Windows 10 April 2018 Update
175
     * 8B 86 68 2B 00 00         mov     eax, esi+2B68h
176
@@ -186,9 +243,12 @@
177
     * 75 0F                     jnz     short loc_100D9A9C
178
     * BA 08 71 01 10            mov     edx, offset errMsg
179
     */
180
-   {0x8B, 0x80, 0x00, 0x00, 0x00, 0x00, 0x83, 0x80, 0x00, 0x00,
181
-    0x00, 0x00, 0x00, 0x75, 0x00, 0xBA, 0x00, 0x00, 0x00, 0x00},
182
-
183
+   {
184
+       0x8B, 0x80, 0x00, 0x00, 0x00, 0x00,
185
+       0x83, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00,
186
+       0x75, 0x00,
187
+       0xBA, 0x00, 0x00, 0x00, 0x00
188
+   },
189
    /*
190
     * Windows 11 22H2
191
     * 8b 83 3c 2b 00 00        MOV        EAX,dword ptr EBX + 0x2b3c
192
@@ -196,9 +256,14 @@
193
     * 75 0f            JNZ        LAB_100d79a0
194
     * ba d0 c6 00 10       MOV        EDX,s_To_use_...
195
     */
196
-   {0x8B, 0x80, 0x00, 0x00, 0x00, 0x00, 0x39, 0x80, 0x00, 0x00,
197
-    0x00, 0x00, 0x75, 0x00, 0xBA, 0x00, 0x00, 0x00, 0x00, 0x00},
198
+   {
199
+       0x8B, 0x80, 0x00, 0x00, 0x00, 0x00,
200
+       0x39, 0x80, 0x00, 0x00, 0x00, 0x00,
201
obs-studio-28.1.2.tar.xz/plugins/win-capture/graphics-hook/graphics-hook.c -> obs-studio-29.0.0.tar.xz/plugins/win-capture/graphics-hook/graphics-hook.c Changed
11
 
1
@@ -65,7 +65,8 @@
2
 bool init_pipe(void)
3
 {
4
    char new_name64;
5
-   sprintf(new_name, "%s%lu", PIPE_NAME, GetCurrentProcessId());
6
+   snprintf(new_name, sizeof(new_name), "%s%lu", PIPE_NAME,
7
+        GetCurrentProcessId());
8
 
9
    const bool success = ipc_pipe_client_open(&pipe, new_name);
10
    if (!success) {
11
obs-studio-28.1.2.tar.xz/plugins/win-dshow/ffmpeg-decode.c -> obs-studio-29.0.0.tar.xz/plugins/win-dshow/ffmpeg-decode.c Changed
25
 
1
@@ -312,7 +312,7 @@
2
 }
3
 
4
 bool ffmpeg_decode_video(struct ffmpeg_decode *decode, uint8_t *data,
5
-            size_t size, long long *ts,
6
+            size_t size, long long *ts, enum video_colorspace cs,
7
             enum video_range_type range,
8
             struct obs_source_frame2 *frame, bool *got_output)
9
 {
10
@@ -396,9 +396,11 @@
11
                : VIDEO_RANGE_PARTIAL;
12
    }
13
 
14
-   const enum video_colorspace cs = convert_color_space(
15
-       decode->frame->colorspace, decode->frame->color_trc,
16
-       decode->frame->color_primaries);
17
+   if (cs == VIDEO_CS_DEFAULT) {
18
+       cs = convert_color_space(decode->frame->colorspace,
19
+                    decode->frame->color_trc,
20
+                    decode->frame->color_primaries);
21
+   }
22
 
23
    const bool success = video_format_get_parameters_for_format(
24
        cs, range, format, frame->color_matrix, frame->color_range_min,
25
obs-studio-28.1.2.tar.xz/plugins/win-dshow/ffmpeg-decode.h -> obs-studio-29.0.0.tar.xz/plugins/win-dshow/ffmpeg-decode.h Changed
9
 
1
@@ -59,6 +59,7 @@
2
 
3
 extern bool ffmpeg_decode_video(struct ffmpeg_decode *decode, uint8_t *data,
4
                size_t size, long long *ts,
5
+               enum video_colorspace cs,
6
                enum video_range_type range,
7
                struct obs_source_frame2 *frame,
8
                bool *got_output);
9
obs-studio-29.0.0.tar.xz/plugins/win-dshow/libdshowcapture/external/.clang-format Added
5
 
1
@@ -0,0 +1,3 @@
2
+Language: Cpp
3
+SortIncludes: false
4
+DisableFormat: true
5
obs-studio-28.1.2.tar.xz/plugins/win-dshow/libdshowcapture/source/dshow-base.cpp -> obs-studio-29.0.0.tar.xz/plugins/win-dshow/libdshowcapture/source/dshow-base.cpp Changed
32
 
1
@@ -694,6 +694,21 @@
2
        L"3842", /* evga */
3
        L"0B05", /* asus */
4
        L"07CA", /* avermedia */
5
+       L"048D", /* digitnow/pengo */
6
+       L"04B4", /* mokose */
7
+       L"0557", /* aten */
8
+       L"1164", /* startek/kapchr */
9
+       L"1532", /* razer */
10
+       L"1BCF", /* mypin/treaslin/mirabox */
11
+       L"1E4E", /* pengo/cloneralliance */
12
+       L"1E71", /* nzxt */
13
+       L"2040", /* hauppauge */
14
+       L"2935", /* magewell */
15
+       L"298F", /* genki */
16
+       L"2B77", /* epiphan */
17
+       L"32ED", /* ezcap */
18
+       L"534D", /* brand-less/pacoxi/ucec */
19
+       L"EBA4", /* zasluke */
20
    };
21
 
22
    if (MatchingStartToken(path, usbToken)) {
23
@@ -711,6 +726,8 @@
24
    const wstring pciSubsysToken = L"SUBSYS_";
25
    const wstring pciVenIdWhitelist = {
26
        L"1CD7", /* magewell */
27
+       L"8888", /* acasis */
28
+       L"1461", /* avermedia */
29
    };
30
    const wstring pciSubsysIdWhitelist = {
31
        L"1CFA", /* elgato */
32
obs-studio-28.1.2.tar.xz/plugins/win-dshow/virtualcam-module/virtualcam-filter.cpp -> obs-studio-29.0.0.tar.xz/plugins/win-dshow/virtualcam-module/virtualcam-filter.cpp Changed
10
 
1
@@ -146,7 +146,7 @@
2
        return hr;
3
    }
4
 
5
-   os_atomic_set_bool(&active, false);
6
+   os_atomic_set_bool(&active, true);
7
    SetEvent(thread_start);
8
    return S_OK;
9
 }
10
obs-studio-28.1.2.tar.xz/plugins/win-dshow/win-dshow.cpp -> obs-studio-29.0.0.tar.xz/plugins/win-dshow/win-dshow.cpp Changed
40
 
1
@@ -198,6 +198,7 @@
2
    VideoConfig videoConfig;
3
    AudioConfig audioConfig;
4
 
5
+   enum video_colorspace cs;
6
    obs_source_frame2 frame;
7
    obs_source_audio audio;
8
    long lastRotation = 0;
9
@@ -505,7 +506,7 @@
10
    }
11
 
12
    bool got_output;
13
-   bool success = ffmpeg_decode_video(video_decoder, data, size, &ts,
14
+   bool success = ffmpeg_decode_video(video_decoder, data, size, &ts, cs,
15
                       frame.range, &frame, &got_output);
16
    if (!success) {
17
        blog(LOG_WARNING, "Error decoding video");
18
@@ -1155,7 +1156,7 @@
19
    if (device.Start() != Result::Success)
20
        return false;
21
 
22
-   const enum video_colorspace cs = GetColorSpace(settings);
23
+   cs = GetColorSpace(settings);
24
    const enum video_range_type range = GetColorRange(settings);
25
 
26
    enum video_trc trc = VIDEO_TRC_DEFAULT;
27
@@ -1307,6 +1308,12 @@
28
 };
29
 
30
 static const FPSFormat validFPSFormats = {
31
+   {"360", MAKE_DSHOW_FPS(360)},
32
+   {"240", MAKE_DSHOW_FPS(240)},
33
+   {"144", MAKE_DSHOW_FPS(144)},
34
+   {"120", MAKE_DSHOW_FPS(120)},
35
+   {"119.88 NTSC", MAKE_DSHOW_FRACTIONAL_FPS(120000, 1001)},
36
+   {"100", MAKE_DSHOW_FPS(100)},
37
    {"60", MAKE_DSHOW_FPS(60)},
38
    {"59.94 NTSC", MAKE_DSHOW_FRACTIONAL_FPS(60000, 1001)},
39
    {"50", MAKE_DSHOW_FPS(50)},
40