2016-09-20 05:07:30 -07:00
|
|
|
if(NOT EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/libdshowcapture/dshowcapture.hpp")
|
2022-03-25 18:51:41 +01:00
|
|
|
obs_status(
|
2022-03-16 23:10:42 +01:00
|
|
|
FATAL_ERROR
|
2022-03-25 18:51:41 +01:00
|
|
|
"libdshowcapture submodule not found! Please fetch submodules. win-dshow plugin disabled."
|
2022-03-16 23:10:42 +01:00
|
|
|
)
|
|
|
|
return()
|
|
|
|
endif()
|
|
|
|
|
|
|
|
option(ENABLE_VIRTUALCAM "Enable building with Virtual Camera (Windows)" ON)
|
|
|
|
|
|
|
|
if(NOT ENABLE_VIRTUALCAM)
|
2022-03-25 18:51:41 +01:00
|
|
|
obs_status(DISABLED "Windows Virtual Camera")
|
2022-03-16 23:10:42 +01:00
|
|
|
endif()
|
|
|
|
|
|
|
|
if(ENABLE_VIRTUALCAM AND NOT VIRTUALCAM_GUID)
|
|
|
|
set(VIRTUALCAM_GUID
|
|
|
|
""
|
|
|
|
CACHE STRING "Virtual Camera GUID" FORCE)
|
|
|
|
mark_as_advanced(VIRTUALCAM_GUID)
|
2016-09-20 05:07:30 -07:00
|
|
|
endif()
|
|
|
|
|
2014-05-30 03:29:48 -07:00
|
|
|
project(win-dshow)
|
|
|
|
|
2015-07-18 16:15:01 -07:00
|
|
|
find_package(FFmpeg REQUIRED COMPONENTS avcodec avutil)
|
2014-09-12 19:51:45 -07:00
|
|
|
|
2022-03-16 23:10:42 +01:00
|
|
|
add_library(win-dshow MODULE)
|
|
|
|
add_library(OBS::dshow ALIAS win-dshow)
|
|
|
|
|
|
|
|
target_sources(
|
|
|
|
win-dshow PRIVATE encode-dstr.hpp win-dshow.cpp win-dshow-encoder.cpp
|
|
|
|
dshow-plugin.cpp ffmpeg-decode.c ffmpeg-decode.h)
|
|
|
|
|
|
|
|
add_library(libdshowcapture INTERFACE)
|
|
|
|
add_library(OBS::libdshowcapture ALIAS libdshowcapture)
|
|
|
|
|
|
|
|
target_sources(
|
|
|
|
libdshowcapture
|
|
|
|
INTERFACE libdshowcapture/dshowcapture.hpp
|
|
|
|
libdshowcapture/source/capture-filter.cpp
|
|
|
|
libdshowcapture/source/capture-filter.hpp
|
|
|
|
libdshowcapture/source/output-filter.cpp
|
|
|
|
libdshowcapture/source/output-filter.hpp
|
|
|
|
libdshowcapture/source/dshowcapture.cpp
|
|
|
|
libdshowcapture/source/dshowencode.cpp
|
|
|
|
libdshowcapture/source/device.cpp
|
|
|
|
libdshowcapture/source/device.hpp
|
2022-02-27 20:50:00 -08:00
|
|
|
libdshowcapture/source/device-vendor.cpp
|
2022-03-16 23:10:42 +01:00
|
|
|
libdshowcapture/source/encoder.cpp
|
|
|
|
libdshowcapture/source/encoder.hpp
|
|
|
|
libdshowcapture/source/dshow-base.cpp
|
|
|
|
libdshowcapture/source/dshow-base.hpp
|
|
|
|
libdshowcapture/source/dshow-demux.cpp
|
|
|
|
libdshowcapture/source/dshow-demux.hpp
|
|
|
|
libdshowcapture/source/dshow-device-defs.hpp
|
|
|
|
libdshowcapture/source/dshow-enum.cpp
|
|
|
|
libdshowcapture/source/dshow-enum.hpp
|
|
|
|
libdshowcapture/source/dshow-formats.cpp
|
|
|
|
libdshowcapture/source/dshow-formats.hpp
|
|
|
|
libdshowcapture/source/dshow-media-type.cpp
|
|
|
|
libdshowcapture/source/dshow-encoded-device.cpp
|
|
|
|
libdshowcapture/source/dshow-media-type.hpp
|
|
|
|
libdshowcapture/source/log.cpp
|
|
|
|
libdshowcapture/source/log.hpp
|
|
|
|
libdshowcapture/source/external/IVideoCaptureFilter.h)
|
|
|
|
|
|
|
|
target_include_directories(
|
|
|
|
libdshowcapture INTERFACE ${CMAKE_CURRENT_SOURCE_DIR}/libdshowcapture)
|
2014-09-12 19:51:45 -07:00
|
|
|
|
2019-06-16 21:42:58 +10:00
|
|
|
set(MODULE_DESCRIPTION "OBS DirectShow module")
|
2020-09-06 17:04:06 -07:00
|
|
|
|
2022-03-16 23:10:42 +01:00
|
|
|
configure_file(${CMAKE_SOURCE_DIR}/cmake/bundle/windows/obs-module.rc.in
|
|
|
|
win-dshow.rc)
|
2020-09-06 17:04:06 -07:00
|
|
|
|
2022-03-16 23:10:42 +01:00
|
|
|
target_sources(win-dshow PRIVATE win-dshow.rc)
|
|
|
|
|
|
|
|
target_compile_definitions(
|
|
|
|
win-dshow PRIVATE UNICODE _UNICODE _CRT_SECURE_NO_WARNINGS
|
|
|
|
_CRT_NONSTDC_NO_WARNINGS)
|
|
|
|
|
|
|
|
set(VIRTUALCAM_AVAILABLE OFF)
|
2020-09-06 17:04:06 -07:00
|
|
|
if(VIRTUALCAM_GUID STREQUAL "")
|
2022-03-25 18:51:41 +01:00
|
|
|
obs_status(
|
2022-03-16 23:10:42 +01:00
|
|
|
WARNING
|
2022-03-25 18:51:41 +01:00
|
|
|
"Windows Virtual Camera - GUID not set - specify as 'VIRTUALCAM_GUID' to enable."
|
2022-03-16 23:10:42 +01:00
|
|
|
)
|
2020-09-06 17:04:06 -07:00
|
|
|
else()
|
2022-03-16 23:10:42 +01:00
|
|
|
set(INVALID_GUID ON)
|
|
|
|
|
|
|
|
string(REPLACE "-" ";" GUID_VALS ${VIRTUALCAM_GUID})
|
|
|
|
|
|
|
|
list(LENGTH GUID_VALS GUID_VAL_COUNT)
|
|
|
|
if(GUID_VAL_COUNT EQUAL 5)
|
|
|
|
string(REPLACE ";" "0" GUID_HEX ${GUID_VALS})
|
|
|
|
string(REGEX MATCH "[0-9a-fA-F]+" GUID_ACTUAL_HEX ${GUID_HEX})
|
|
|
|
if(GUID_ACTUAL_HEX STREQUAL GUID_HEX)
|
|
|
|
list(GET GUID_VALS 0 GUID_VALS_DATA1)
|
|
|
|
list(GET GUID_VALS 1 GUID_VALS_DATA2)
|
|
|
|
list(GET GUID_VALS 2 GUID_VALS_DATA3)
|
|
|
|
list(GET GUID_VALS 3 GUID_VALS_DATA4)
|
|
|
|
list(GET GUID_VALS 4 GUID_VALS_DATA5)
|
|
|
|
string(LENGTH ${GUID_VALS_DATA1} GUID_VALS_DATA1_LENGTH)
|
|
|
|
string(LENGTH ${GUID_VALS_DATA2} GUID_VALS_DATA2_LENGTH)
|
|
|
|
string(LENGTH ${GUID_VALS_DATA3} GUID_VALS_DATA3_LENGTH)
|
|
|
|
string(LENGTH ${GUID_VALS_DATA4} GUID_VALS_DATA4_LENGTH)
|
|
|
|
string(LENGTH ${GUID_VALS_DATA5} GUID_VALS_DATA5_LENGTH)
|
|
|
|
if(GUID_VALS_DATA1_LENGTH EQUAL 8
|
|
|
|
AND GUID_VALS_DATA2_LENGTH EQUAL 4
|
|
|
|
AND GUID_VALS_DATA3_LENGTH EQUAL 4
|
|
|
|
AND GUID_VALS_DATA4_LENGTH EQUAL 4
|
|
|
|
AND GUID_VALS_DATA5_LENGTH EQUAL 12)
|
|
|
|
set(GUID_VAL01 ${GUID_VALS_DATA1})
|
|
|
|
set(GUID_VAL02 ${GUID_VALS_DATA2})
|
|
|
|
set(GUID_VAL03 ${GUID_VALS_DATA3})
|
|
|
|
string(SUBSTRING ${GUID_VALS_DATA4} 0 2 GUID_VAL04)
|
|
|
|
string(SUBSTRING ${GUID_VALS_DATA4} 2 2 GUID_VAL05)
|
|
|
|
string(SUBSTRING ${GUID_VALS_DATA5} 0 2 GUID_VAL06)
|
|
|
|
string(SUBSTRING ${GUID_VALS_DATA5} 2 2 GUID_VAL07)
|
|
|
|
string(SUBSTRING ${GUID_VALS_DATA5} 4 2 GUID_VAL08)
|
|
|
|
string(SUBSTRING ${GUID_VALS_DATA5} 6 2 GUID_VAL09)
|
|
|
|
string(SUBSTRING ${GUID_VALS_DATA5} 8 2 GUID_VAL10)
|
|
|
|
string(SUBSTRING ${GUID_VALS_DATA5} 10 2 GUID_VAL11)
|
|
|
|
set(VIRTUALCAM_AVAILABLE ON)
|
|
|
|
set(INVALID_GUID OFF)
|
|
|
|
endif()
|
|
|
|
endif()
|
|
|
|
endif()
|
2020-09-06 17:04:06 -07:00
|
|
|
endif()
|
|
|
|
|
|
|
|
if(INVALID_GUID)
|
2022-03-25 18:51:41 +01:00
|
|
|
obs_status(WARNING "Windows Virtual Camera - invalid GUID supplied")
|
2020-09-06 17:04:06 -07:00
|
|
|
endif()
|
|
|
|
|
2022-03-16 23:10:42 +01:00
|
|
|
target_link_libraries(
|
|
|
|
win-dshow
|
|
|
|
PRIVATE OBS::libobs
|
|
|
|
OBS::w32-pthreads
|
|
|
|
OBS::libdshowcapture
|
|
|
|
setupapi
|
|
|
|
strmiids
|
|
|
|
ksuser
|
|
|
|
winmm
|
|
|
|
wmcodecdspuuid
|
|
|
|
FFmpeg::avcodec
|
|
|
|
FFmpeg::avutil)
|
|
|
|
|
|
|
|
file(GLOB _LIBOBS_SOURCE_FILES ${CMAKE_CURRENT_SOURCE_DIR}/libdshowcapture/*.c
|
|
|
|
${CMAKE_CURRENT_SOURCE_DIR}/libdshowcapture/*.cpp)
|
|
|
|
file(GLOB _LIBOBS_HEADER_FILES ${CMAKE_CURRENT_SOURCE_DIR}/libdshowcapture/*.h
|
|
|
|
${CMAKE_CURRENT_SOURCE_DIR}/libdshowcapture/*.hpp)
|
|
|
|
|
|
|
|
source_group("libdshowcapture\\Source Files" FILES ${_LIBOBS_SOURCE_FILES})
|
|
|
|
source_group("libdshowcapture\\Header Files" FILES ${_LIBOBS_HEADER_FILES})
|
2020-09-06 17:04:06 -07:00
|
|
|
|
win-dshow: Add Virtual Camera (Windows)
The virtual camera adds the ability to use the output of OBS itself as a
camera that can be selected within other Windows applications. This is
very loosely based upon the catxfish virtual camera plugin design.
There is a shared memory queue, but instead of having 10-20 frames in
the queue, there are now only 3 frames in the queue to minimize latency
and reduce memory usage. The third frame is mostly to ensure that
writing does not occur on the same frame being read; the delay is merely
one frame at all times.
The frames of the shared memory queue are NV12 instead of YUYV, which
reduces the memory and data copied, as well as eliminate unnecessary
conversion from NV12. Some programs (such as chrome, which uses webrtc
to capture) do not support NV12 however, so an I420 conversion is
provided, which is far less expensive than YUYV. The CPU cost of NV12
-> I420 is negligible in comparison.
The virtual camera filter itself is based upon the output filter within
the libdshowcapture library, which was originally implemented for other
purposes. This is more ideal than the Microsoft example code because
for one, it's far less convoluted, two, allows us to be able to
customize the filter to our needs a bit more easily, and three, has much
better RAII. The Microsoft CBaseFilter/etc code comprised of about 30
source files, where as the output filter comprises of two or three
required source files which we already had, so it's a huge win to
compile time.
Scaling is avoided whenever possible to minimize CPU usage. When the
virtual camera is activated in OBS, the width, height, and frame
interval are saved, that way if the filter is activated, it will always
remember the last OBS resolution/interval that the virtual camera was
activated with, even if OBS is not active. If for some reason the
filter activates before OBS starts up, and OBS starts up with a
different resolution, it will use simple point scaling intermittently,
and then will remember the new scaling in the future. The scaler could
use some optimization. FFmpeg was not opted for because the FFmpeg DLLs
would have to be provided for both architectures, which would be about
30 megabytes in total, and would make writing the plugin much more
painful. Thus a simple point scaling algorithm is used, and scaling is
avoided whenever possible.
(If another willing participant wants to have a go at improving the
scaling then go for it. But otherwise, it avoids scaling whenever
possible anyway, so it's not a huge deal)
2020-06-20 06:44:19 -07:00
|
|
|
set_target_properties(win-dshow PROPERTIES FOLDER "plugins/win-dshow")
|
|
|
|
|
2022-03-16 23:10:42 +01:00
|
|
|
setup_plugin_target(win-dshow)
|
|
|
|
|
|
|
|
if(ENABLE_VIRTUALCAM AND VIRTUALCAM_AVAILABLE)
|
|
|
|
target_sources(
|
|
|
|
win-dshow PRIVATE tiny-nv12-scale.c tiny-nv12-scale.h shared-memory-queue.c
|
|
|
|
shared-memory-queue.h virtualcam.c)
|
|
|
|
|
|
|
|
target_compile_definitions(win-dshow PRIVATE VIRTUALCAM_AVAILABLE)
|
|
|
|
|
|
|
|
target_include_directories(win-dshow
|
|
|
|
PRIVATE ${CMAKE_CURRENT_BINARY_DIR}/config)
|
|
|
|
|
|
|
|
configure_file(${CMAKE_CURRENT_SOURCE_DIR}/virtualcam-guid.h.in
|
|
|
|
${CMAKE_CURRENT_BINARY_DIR}/config/virtualcam-guid.h)
|
|
|
|
|
|
|
|
target_sources(win-dshow
|
|
|
|
PRIVATE ${CMAKE_CURRENT_BINARY_DIR}/config/virtualcam-guid.h)
|
|
|
|
|
|
|
|
configure_file(virtualcam-install.bat.in
|
|
|
|
"${CMAKE_CURRENT_BINARY_DIR}/virtualcam-install.bat")
|
|
|
|
|
|
|
|
configure_file(virtualcam-uninstall.bat.in
|
|
|
|
"${CMAKE_CURRENT_BINARY_DIR}/virtualcam-uninstall.bat")
|
2014-05-30 03:29:48 -07:00
|
|
|
|
2022-03-16 23:10:42 +01:00
|
|
|
add_target_resource(
|
|
|
|
win-dshow "${CMAKE_CURRENT_BINARY_DIR}/virtualcam-install.bat"
|
|
|
|
"obs-plugins/win-dshow/")
|
|
|
|
add_target_resource(
|
|
|
|
win-dshow "${CMAKE_CURRENT_BINARY_DIR}/virtualcam-uninstall.bat"
|
|
|
|
"obs-plugins/win-dshow/")
|
win-dshow: Add Virtual Camera (Windows)
The virtual camera adds the ability to use the output of OBS itself as a
camera that can be selected within other Windows applications. This is
very loosely based upon the catxfish virtual camera plugin design.
There is a shared memory queue, but instead of having 10-20 frames in
the queue, there are now only 3 frames in the queue to minimize latency
and reduce memory usage. The third frame is mostly to ensure that
writing does not occur on the same frame being read; the delay is merely
one frame at all times.
The frames of the shared memory queue are NV12 instead of YUYV, which
reduces the memory and data copied, as well as eliminate unnecessary
conversion from NV12. Some programs (such as chrome, which uses webrtc
to capture) do not support NV12 however, so an I420 conversion is
provided, which is far less expensive than YUYV. The CPU cost of NV12
-> I420 is negligible in comparison.
The virtual camera filter itself is based upon the output filter within
the libdshowcapture library, which was originally implemented for other
purposes. This is more ideal than the Microsoft example code because
for one, it's far less convoluted, two, allows us to be able to
customize the filter to our needs a bit more easily, and three, has much
better RAII. The Microsoft CBaseFilter/etc code comprised of about 30
source files, where as the output filter comprises of two or three
required source files which we already had, so it's a huge win to
compile time.
Scaling is avoided whenever possible to minimize CPU usage. When the
virtual camera is activated in OBS, the width, height, and frame
interval are saved, that way if the filter is activated, it will always
remember the last OBS resolution/interval that the virtual camera was
activated with, even if OBS is not active. If for some reason the
filter activates before OBS starts up, and OBS starts up with a
different resolution, it will use simple point scaling intermittently,
and then will remember the new scaling in the future. The scaler could
use some optimization. FFmpeg was not opted for because the FFmpeg DLLs
would have to be provided for both architectures, which would be about
30 megabytes in total, and would make writing the plugin much more
painful. Thus a simple point scaling algorithm is used, and scaling is
avoided whenever possible.
(If another willing participant wants to have a go at improving the
scaling then go for it. But otherwise, it avoids scaling whenever
possible anyway, so it's not a huge deal)
2020-06-20 06:44:19 -07:00
|
|
|
|
2022-03-16 23:10:42 +01:00
|
|
|
add_subdirectory(virtualcam-module)
|
2020-09-06 17:04:06 -07:00
|
|
|
endif()
|