From 8e0524bf609cf57c422abec2bc8fa61b668c748d Mon Sep 17 00:00:00 2001 From: dijunkun Date: Fri, 16 May 2025 18:49:20 +0800 Subject: [PATCH] [feat] display switch supported on MacOSx platform --- .../macosx/screen_capturer_sck.cpp | 33 +- .../macosx/screen_capturer_sck.h | 4 +- .../macosx/screen_capturer_sck_impl.mm | 342 ++++++++++++------ thirdparty/projectx | 2 +- 4 files changed, 249 insertions(+), 132 deletions(-) diff --git a/src/screen_capturer/macosx/screen_capturer_sck.cpp b/src/screen_capturer/macosx/screen_capturer_sck.cpp index 43f2dcc..4133428 100644 --- a/src/screen_capturer/macosx/screen_capturer_sck.cpp +++ b/src/screen_capturer/macosx/screen_capturer_sck.cpp @@ -3,12 +3,7 @@ #include "rd_log.h" ScreenCapturerSck::ScreenCapturerSck() {} -ScreenCapturerSck::~ScreenCapturerSck() { - // if (inited_ && capture_thread_.joinable()) { - // capture_thread_.join(); - // inited_ = false; - // } -} +ScreenCapturerSck::~ScreenCapturerSck() {} int ScreenCapturerSck::Init(const int fps, cb_desktop_data cb) { if (cb) { @@ -27,15 +22,7 @@ int ScreenCapturerSck::Init(const int fps, cb_desktop_data cb) { int ScreenCapturerSck::Destroy() { return 0; } int ScreenCapturerSck::Start() { - // if (running_) { - // return 0; - // } - - // running_ = true; - // capture_thread_ = std::thread([this]() { - // while (running_) { - // } - // }); + screen_capturer_sck_impl_->Start(); return 0; } @@ -46,6 +33,22 @@ int ScreenCapturerSck::Pause(int monitor_index) { return 0; } int ScreenCapturerSck::Resume(int monitor_index) { return 0; } +int ScreenCapturerSck::SwitchTo(int monitor_index) { + if (screen_capturer_sck_impl_) { + return screen_capturer_sck_impl_->SwitchTo(monitor_index); + } + + return -1; +} + +std::vector ScreenCapturerSck::GetDisplayInfoList() { + if (screen_capturer_sck_impl_) { + return screen_capturer_sck_impl_->GetDisplayInfoList(); + } + + return std::vector(); +} + void ScreenCapturerSck::OnFrame() {} void ScreenCapturerSck::CleanUp() {} \ No newline at end of file diff --git a/src/screen_capturer/macosx/screen_capturer_sck.h b/src/screen_capturer/macosx/screen_capturer_sck.h index 7e715aa..75858db 100644 --- a/src/screen_capturer/macosx/screen_capturer_sck.h +++ b/src/screen_capturer/macosx/screen_capturer_sck.h @@ -31,9 +31,9 @@ class ScreenCapturerSck : public ScreenCapturer { int Resume(int monitor_index) override; - int SwitchTo(int monitor_index) override { return 0; } + int SwitchTo(int monitor_index) override; - std::vector GetDisplayInfoList() override { return {}; } + std::vector GetDisplayInfoList() override; void OnFrame(); diff --git a/src/screen_capturer/macosx/screen_capturer_sck_impl.mm b/src/screen_capturer/macosx/screen_capturer_sck_impl.mm index db109f1..c770a2d 100644 --- a/src/screen_capturer/macosx/screen_capturer_sck_impl.mm +++ b/src/screen_capturer/macosx/screen_capturer_sck_impl.mm @@ -10,13 +10,16 @@ #include "screen_capturer_sck.h" -#include "rd_log.h" #include #include #include #include #include +#include +#include "display_info.h" +#include "rd_log.h" +static const int kFullDesktopScreenId = -1; class ScreenCapturerSckImpl; // The ScreenCaptureKit API was available in macOS 12.3, but full-screen capture @@ -38,108 +41,79 @@ API_AVAILABLE(macos(14.0)) @end class API_AVAILABLE(macos(14.0)) ScreenCapturerSckImpl : public ScreenCapturer { -public: + public: explicit ScreenCapturerSckImpl(); ScreenCapturerSckImpl(const ScreenCapturerSckImpl &) = delete; ScreenCapturerSckImpl &operator=(const ScreenCapturerSckImpl &) = delete; ~ScreenCapturerSckImpl(); -public: - int Init(const int fps, cb_desktop_data cb); - void OnReceiveContent(SCShareableContent *content); + public: + int Init(const int fps, cb_desktop_data cb) override; + + int Start() override; + + int SwitchTo(int monitor_index) override; + + int Destroy() override { return 0; } + + int Stop() override { return 0; } + + int Pause(int monitor_index) override { return 0; } + + int Resume(int monitor_index) override { return 0; } + + std::vector GetDisplayInfoList() override { return display_info_list_; } + + private: + std::vector display_info_list_; + std::map display_id_map_; + std::map display_id_map_reverse_; + unsigned char *nv12_frame_ = nullptr; + int width_ = 0; + int height_ = 0; + + public: + // Called by SckHelper when shareable content is returned by ScreenCaptureKit. `content` will be + // nil if an error occurred. May run on an arbitrary thread. + void OnShareableContentCreated(SCShareableContent *content); + // Called by SckHelper to notify of a newly captured frame. May run on an arbitrary thread. void OnNewIOSurface(IOSurfaceRef io_surface, CFDictionaryRef attachment); - virtual int Destroy() { return 0; } - - virtual int Start() { return 0; } - - virtual int Stop() { return 0; } - - virtual int Pause(int monitor_index) { return 0; } - - virtual int Resume(int monitor_index) { return 0; } - - virtual int SwitchTo(int monitor_index) { return 0; } - - virtual std::vector GetDisplayInfoList() { return {}; } - -private: + private: + // Called when starting the capturer or the configuration has changed (either from a + // SwitchTo() call, or the screen-resolution has changed). This tells SCK to fetch new + // shareable content, and the completion-handler will either start a new stream, or reconfigure + // the existing stream. Runs on the caller's thread. + void StartOrReconfigureCapturer(); + // Helper object to receive Objective-C callbacks from ScreenCaptureKit and call into this C++ + // object. The helper may outlive this C++ instance, if a completion-handler is passed to + // ScreenCaptureKit APIs and the C++ object is deleted before the handler executes. SckHelper *__strong helper_; + // Callback for returning captured frames, or errors, to the caller. Only used on the caller's + // thread. + cb_desktop_data _on_data = nullptr; + // Signals that a permanent error occurred. This may be set on any thread, and is read by + // CaptureFrame() which runs on the caller's thread. + std::atomic permanent_error_ = false; + // Guards some variables that may be accessed on different threads. + std::mutex lock_; + // Provides captured desktop frames. SCStream *__strong stream_; - - cb_desktop_data _on_data; - unsigned char *nv12_frame_ = nullptr; - bool permanent_error_ = false; - CGDirectDisplayID current_display_ = -1; - std::mutex mtx_; + // Currently selected display, or 0 if the full desktop is selected. This capturer does not + // support full-desktop capture, and will fall back to the first display. + CGDirectDisplayID current_display_ = 0; }; -@implementation SckHelper { - // This lock is to prevent the capturer being destroyed while an instance - // method is still running on another thread. - std::mutex helper_mtx_; - ScreenCapturerSckImpl *_capturer; -} - -- (instancetype)initWithCapturer:(ScreenCapturerSckImpl *)capturer { - self = [super init]; - if (self) { - _capturer = capturer; - } - return self; -} - -- (void)onShareableContentCreated:(SCShareableContent *)content { - std::lock_guard lock(helper_mtx_); - if (_capturer) { - _capturer->OnReceiveContent(content); - } else { - LOG_ERROR("Invalid capturer"); - } -} - -- (void)stream:(SCStream *)stream - didOutputSampleBuffer:(CMSampleBufferRef)sampleBuffer - ofType:(SCStreamOutputType)type { - CVPixelBufferRef pixelBuffer = CMSampleBufferGetImageBuffer(sampleBuffer); - if (!pixelBuffer) { - return; - } - - IOSurfaceRef ioSurface = CVPixelBufferGetIOSurface(pixelBuffer); - if (!ioSurface) { - return; - } - - CFArrayRef attachmentsArray = CMSampleBufferGetSampleAttachmentsArray( - sampleBuffer, /*createIfNecessary=*/false); - if (!attachmentsArray || CFArrayGetCount(attachmentsArray) <= 0) { - LOG_ERROR("Discarding frame with no attachments"); - return; - } - - CFDictionaryRef attachment = - static_cast(CFArrayGetValueAtIndex(attachmentsArray, 0)); - - std::lock_guard lock(helper_mtx_); - if (_capturer) { - _capturer->OnNewIOSurface(ioSurface, attachment); - } -} - -- (void)releaseCapturer { - std::lock_guard lock(helper_mtx_); - _capturer = nullptr; -} - -@end - ScreenCapturerSckImpl::ScreenCapturerSckImpl() { helper_ = [[SckHelper alloc] initWithCapturer:this]; } ScreenCapturerSckImpl::~ScreenCapturerSckImpl() { + display_info_list_.clear(); + display_id_map_.clear(); + display_id_map_reverse_.clear(); + [stream_ stopCaptureWithCompletionHandler:nil]; [helper_ releaseCapturer]; } @@ -147,17 +121,78 @@ ScreenCapturerSckImpl::~ScreenCapturerSckImpl() { int ScreenCapturerSckImpl::Init(const int fps, cb_desktop_data cb) { _on_data = cb; - SckHelper *local_helper = helper_; - auto handler = ^(SCShareableContent *content, NSError *error) { - [local_helper onShareableContentCreated:content]; - }; + dispatch_semaphore_t sema = dispatch_semaphore_create(0); + __block SCShareableContent *content = nil; - [SCShareableContent getShareableContentWithCompletionHandler:handler]; + [SCShareableContent + getShareableContentWithCompletionHandler:^(SCShareableContent *result, NSError *error) { + if (error) { + NSLog(@"Failed to get shareable content: %@", error); + } else { + content = result; + } + dispatch_semaphore_signal(sema); + }]; + dispatch_semaphore_wait(sema, DISPATCH_TIME_FOREVER); + + if (!content || content.displays.count == 0) { + LOG_ERROR("Failed to get display info") + return 0; + } + + int unnamed_count = 1; + for (SCDisplay *display in content.displays) { + CGDirectDisplayID display_id = display.displayID; + CGRect bounds = CGDisplayBounds(display_id); + bool is_primary = CGDisplayIsMain(display_id); + + std::string name; + if ([display respondsToSelector:@selector(displayName)]) { + NSString *display_name = [display performSelector:@selector(displayName)]; + if (display_name && [display_name length] > 0) { + name = [display_name UTF8String]; + } + } + + if (name.empty()) { + name = "Display " + std::to_string(unnamed_count++); + } + + DisplayInfo info((void *)(uintptr_t)display_id, name, is_primary, + static_cast(bounds.origin.x), static_cast(bounds.origin.y), + static_cast(bounds.origin.x + bounds.size.width), + static_cast(bounds.origin.y + bounds.size.height)); + + display_info_list_.push_back(info); + display_id_map_[display_info_list_.size() - 1] = display_id; + display_id_map_reverse_[display_id] = display_info_list_.size() - 1; + } return 0; } -void ScreenCapturerSckImpl::OnReceiveContent(SCShareableContent *content) { +int ScreenCapturerSckImpl::Start() { + StartOrReconfigureCapturer(); + return 0; +} + +int ScreenCapturerSckImpl::SwitchTo(int monitor_index) { + bool stream_started = false; + { + std::lock_guard lock(lock_); + current_display_ = display_id_map_[monitor_index]; + if (stream_) { + stream_started = true; + } + } + // If the capturer was already started, reconfigure it. Otherwise, wait until Start() gets called. + if (stream_started) { + StartOrReconfigureCapturer(); + } + return 0; +} + +void ScreenCapturerSckImpl::OnShareableContentCreated(SCShareableContent *content) { if (!content) { LOG_ERROR("getShareableContent failed"); permanent_error_ = true; @@ -172,28 +207,27 @@ void ScreenCapturerSckImpl::OnReceiveContent(SCShareableContent *content) { SCDisplay *captured_display; { - std::lock_guard lock(mtx_); + std::lock_guard lock(lock_); for (SCDisplay *display in content.displays) { if (current_display_ == display.displayID) { + LOG_WARN("current_display_: {}", current_display_); captured_display = display; break; } } if (!captured_display) { - if (-1 == current_display_) { + if (kFullDesktopScreenId == current_display_) { LOG_ERROR("Full screen capture is not supported, falling back to first " "display"); } else { - LOG_ERROR("Display [{}] not found, falling back to first display", - current_display_); + LOG_ERROR("Display [{}] not found, falling back to first display", current_display_); } captured_display = content.displays.firstObject; } } - SCContentFilter *filter = - [[SCContentFilter alloc] initWithDisplay:captured_display - excludingWindows:@[]]; + SCContentFilter *filter = [[SCContentFilter alloc] initWithDisplay:captured_display + excludingWindows:@[]]; SCStreamConfiguration *config = [[SCStreamConfiguration alloc] init]; config.pixelFormat = kCVPixelFormatType_420YpCbCr8BiPlanarVideoRange; config.showsCursor = false; @@ -201,25 +235,22 @@ void ScreenCapturerSckImpl::OnReceiveContent(SCShareableContent *content) { config.height = filter.contentRect.size.height * filter.pointPixelScale; config.captureResolution = SCCaptureResolutionNominal; - std::lock_guard lock(mtx_); + std::lock_guard lock(lock_); if (stream_) { LOG_INFO("Updating stream configuration"); [stream_ updateContentFilter:filter completionHandler:nil]; [stream_ updateConfiguration:config completionHandler:nil]; } else { - stream_ = [[SCStream alloc] initWithFilter:filter - configuration:config - delegate:helper_]; + stream_ = [[SCStream alloc] initWithFilter:filter configuration:config delegate:helper_]; // TODO: crbug.com/327458809 - Choose an appropriate sampleHandlerQueue for // best performance. NSError *add_stream_output_error; - bool add_stream_output_result = - [stream_ addStreamOutput:helper_ - type:SCStreamOutputTypeScreen - sampleHandlerQueue:nil - error:&add_stream_output_error]; + bool add_stream_output_result = [stream_ addStreamOutput:helper_ + type:SCStreamOutputTypeScreen + sampleHandlerQueue:nil + error:&add_stream_output_error]; if (!add_stream_output_result) { stream_ = nil; LOG_ERROR("addStreamOutput failed"); @@ -243,22 +274,105 @@ void ScreenCapturerSckImpl::OnReceiveContent(SCShareableContent *content) { } } -void ScreenCapturerSckImpl::OnNewIOSurface(IOSurfaceRef io_surface, - CFDictionaryRef attachment) { +void ScreenCapturerSckImpl::OnNewIOSurface(IOSurfaceRef io_surface, CFDictionaryRef attachment) { size_t width = IOSurfaceGetWidth(io_surface); size_t height = IOSurfaceGetHeight(io_surface); uint32_t aseed; IOSurfaceLock(io_surface, kIOSurfaceLockReadOnly, &aseed); - nv12_frame_ = - static_cast(IOSurfaceGetBaseAddress(io_surface)); + if (!nv12_frame_) { + nv12_frame_ = new unsigned char[width * height * 3 / 2]; + width_ = width; + height_ = height; + } - _on_data(nv12_frame_, width * height * 3 / 2, width, height, 0); + if (nv12_frame_ && width_ * height_ < width * height) { + delete[] nv12_frame_; + nv12_frame_ = new unsigned char[width * height * 3 / 2]; + width_ = width; + height_ = height; + } + + memcpy(nv12_frame_, IOSurfaceGetBaseAddress(io_surface), width * height * 3 / 2); + + _on_data(nv12_frame_, width * height * 3 / 2, width, height, + display_id_map_reverse_[current_display_]); IOSurfaceUnlock(io_surface, kIOSurfaceLockReadOnly, &aseed); } +void ScreenCapturerSckImpl::StartOrReconfigureCapturer() { + // The copy is needed to avoid capturing `this` in the Objective-C block. Accessing `helper_` + // inside the block is equivalent to `this->helper_` and would crash (UAF) if `this` is + // deleted before the block is executed. + SckHelper *local_helper = helper_; + auto handler = ^(SCShareableContent *content, NSError *error) { + [local_helper onShareableContentCreated:content]; + }; + [SCShareableContent getShareableContentWithCompletionHandler:handler]; +} + std::unique_ptr ScreenCapturerSck::CreateScreenCapturerSck() { return std::make_unique(); -} \ No newline at end of file +} + +@implementation SckHelper { + // This lock is to prevent the capturer being destroyed while an instance + // method is still running on another thread. + std::mutex _capturer_lock; + ScreenCapturerSckImpl *_capturer; +} + +- (instancetype)initWithCapturer:(ScreenCapturerSckImpl *)capturer { + self = [super init]; + if (self) { + _capturer = capturer; + } + return self; +} + +- (void)onShareableContentCreated:(SCShareableContent *)content { + std::lock_guard lock(_capturer_lock); + if (_capturer) { + _capturer->OnShareableContentCreated(content); + } else { + LOG_ERROR("Invalid capturer"); + } +} + +- (void)stream:(SCStream *)stream + didOutputSampleBuffer:(CMSampleBufferRef)sampleBuffer + ofType:(SCStreamOutputType)type { + CVPixelBufferRef pixelBuffer = CMSampleBufferGetImageBuffer(sampleBuffer); + if (!pixelBuffer) { + return; + } + + IOSurfaceRef ioSurface = CVPixelBufferGetIOSurface(pixelBuffer); + if (!ioSurface) { + return; + } + + CFArrayRef attachmentsArray = + CMSampleBufferGetSampleAttachmentsArray(sampleBuffer, /*createIfNecessary=*/false); + if (!attachmentsArray || CFArrayGetCount(attachmentsArray) <= 0) { + LOG_ERROR("Discarding frame with no attachments"); + return; + } + + CFDictionaryRef attachment = + static_cast(CFArrayGetValueAtIndex(attachmentsArray, 0)); + + std::lock_guard lock(_capturer_lock); + if (_capturer) { + _capturer->OnNewIOSurface(ioSurface, attachment); + } +} + +- (void)releaseCapturer { + std::lock_guard lock(_capturer_lock); + _capturer = nullptr; +} + +@end \ No newline at end of file diff --git a/thirdparty/projectx b/thirdparty/projectx index 0e6650a..3a691e2 160000 --- a/thirdparty/projectx +++ b/thirdparty/projectx @@ -1 +1 @@ -Subproject commit 0e6650a4607471abe818184a395d3c2a9c21107f +Subproject commit 3a691e266bb312ce188c4aa2bbe678834e274b00