mirror of
				https://github.com/kunkundi/crossdesk.git
				synced 2025-10-26 12:15:34 +08:00 
			
		
		
		
	[feat] display switch supported on MacOSx platform
This commit is contained in:
		| @@ -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<DisplayInfo> ScreenCapturerSck::GetDisplayInfoList() { | ||||
|   if (screen_capturer_sck_impl_) { | ||||
|     return screen_capturer_sck_impl_->GetDisplayInfoList(); | ||||
|   } | ||||
|  | ||||
|   return std::vector<DisplayInfo>(); | ||||
| } | ||||
|  | ||||
| void ScreenCapturerSck::OnFrame() {} | ||||
|  | ||||
| void ScreenCapturerSck::CleanUp() {} | ||||
| @@ -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<DisplayInfo> GetDisplayInfoList() override { return {}; } | ||||
|   std::vector<DisplayInfo> GetDisplayInfoList() override; | ||||
|  | ||||
|   void OnFrame(); | ||||
|  | ||||
|   | ||||
| @@ -10,13 +10,16 @@ | ||||
|  | ||||
| #include "screen_capturer_sck.h" | ||||
|  | ||||
| #include "rd_log.h" | ||||
| #include <CoreGraphics/CoreGraphics.h> | ||||
| #include <IOSurface/IOSurface.h> | ||||
| #include <ScreenCaptureKit/ScreenCaptureKit.h> | ||||
| #include <atomic> | ||||
| #include <mutex> | ||||
| #include <vector> | ||||
| #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<DisplayInfo> GetDisplayInfoList() override { return display_info_list_; } | ||||
|  | ||||
|  private: | ||||
|   std::vector<DisplayInfo> display_info_list_; | ||||
|   std::map<int, CGDirectDisplayID> display_id_map_; | ||||
|   std::map<CGDirectDisplayID, int> 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<DisplayInfo> 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<bool> 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<std::mutex> 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<CFDictionaryRef>(CFArrayGetValueAtIndex(attachmentsArray, 0)); | ||||
|  | ||||
|   std::lock_guard<std::mutex> lock(helper_mtx_); | ||||
|   if (_capturer) { | ||||
|     _capturer->OnNewIOSurface(ioSurface, attachment); | ||||
|   } | ||||
| } | ||||
|  | ||||
| - (void)releaseCapturer { | ||||
|   std::lock_guard<std::mutex> 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<int>(bounds.origin.x), static_cast<int>(bounds.origin.y), | ||||
|                      static_cast<int>(bounds.origin.x + bounds.size.width), | ||||
|                      static_cast<int>(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<std::mutex> 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<std::mutex> lock(mtx_); | ||||
|     std::lock_guard<std::mutex> 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<std::mutex> lock(mtx_); | ||||
|   std::lock_guard<std::mutex> 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<unsigned char *>(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<ScreenCapturer> ScreenCapturerSck::CreateScreenCapturerSck() { | ||||
|   return std::make_unique<ScreenCapturerSckImpl>(); | ||||
| } | ||||
| } | ||||
|  | ||||
| @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<std::mutex> 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<CFDictionaryRef>(CFArrayGetValueAtIndex(attachmentsArray, 0)); | ||||
|  | ||||
|   std::lock_guard<std::mutex> lock(_capturer_lock); | ||||
|   if (_capturer) { | ||||
|     _capturer->OnNewIOSurface(ioSurface, attachment); | ||||
|   } | ||||
| } | ||||
|  | ||||
| - (void)releaseCapturer { | ||||
|   std::lock_guard<std::mutex> lock(_capturer_lock); | ||||
|   _capturer = nullptr; | ||||
| } | ||||
|  | ||||
| @end | ||||
							
								
								
									
										2
									
								
								thirdparty/projectx
									
									
									
									
										vendored
									
									
								
							
							
								
								
								
								
								
							
						
						
									
										2
									
								
								thirdparty/projectx
									
									
									
									
										vendored
									
									
								
							 Submodule thirdparty/projectx updated: 0e6650a460...3a691e266b
									
								
							
		Reference in New Issue
	
	Block a user