| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  | // Copyright (c) 2019 GitHub, Inc.
 | 
					
						
							|  |  |  | // Use of this source code is governed by the MIT license that can be
 | 
					
						
							|  |  |  | // found in the LICENSE file.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-19 13:46:59 -07:00
										 |  |  | #include "shell/browser/osr/osr_video_consumer.h"
 | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <utility>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-05 15:53:39 +02:00
										 |  |  | #include "media/base/limits.h"
 | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  | #include "media/base/video_frame_metadata.h"
 | 
					
						
							| 
									
										
										
										
											2021-06-22 12:17:16 -07:00
										 |  |  | #include "media/capture/mojom/video_capture_buffer.mojom.h"
 | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  | #include "media/capture/mojom/video_capture_types.mojom.h"
 | 
					
						
							| 
									
										
										
										
											2022-01-10 17:31:39 -05:00
										 |  |  | #include "services/viz/privileged/mojom/compositing/frame_sink_video_capture.mojom-shared.h"
 | 
					
						
							| 
									
										
										
										
											2019-06-19 13:46:59 -07:00
										 |  |  | #include "shell/browser/osr/osr_render_widget_host_view.h"
 | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  | #include "ui/gfx/skbitmap_operations.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-05 15:53:39 +02:00
										 |  |  | namespace { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool IsValidMinAndMaxFrameSize(gfx::Size min_frame_size, | 
					
						
							|  |  |  |                                gfx::Size max_frame_size) { | 
					
						
							|  |  |  |   // Returns true if
 | 
					
						
							|  |  |  |   // 0 < |min_frame_size| <= |max_frame_size| <= media::limits::kMaxDimension.
 | 
					
						
							|  |  |  |   return 0 < min_frame_size.width() && 0 < min_frame_size.height() && | 
					
						
							|  |  |  |          min_frame_size.width() <= max_frame_size.width() && | 
					
						
							|  |  |  |          min_frame_size.height() <= max_frame_size.height() && | 
					
						
							|  |  |  |          max_frame_size.width() <= media::limits::kMaxDimension && | 
					
						
							|  |  |  |          max_frame_size.height() <= media::limits::kMaxDimension; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }  // namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-19 14:23:04 -07:00
										 |  |  | namespace electron { | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | OffScreenVideoConsumer::OffScreenVideoConsumer( | 
					
						
							|  |  |  |     OffScreenRenderWidgetHostView* view, | 
					
						
							|  |  |  |     OnPaintCallback callback) | 
					
						
							|  |  |  |     : callback_(callback), | 
					
						
							|  |  |  |       view_(view), | 
					
						
							| 
									
										
										
										
											2021-01-26 19:16:21 +01:00
										 |  |  |       video_capturer_(view->CreateVideoCapturer()) { | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  |   video_capturer_->SetAutoThrottlingEnabled(false); | 
					
						
							|  |  |  |   video_capturer_->SetMinSizeChangePeriod(base::TimeDelta()); | 
					
						
							| 
									
										
										
										
											2022-01-10 17:31:39 -05:00
										 |  |  |   video_capturer_->SetFormat(media::PIXEL_FORMAT_ARGB); | 
					
						
							| 
									
										
										
										
											2022-05-05 15:53:39 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   SizeChanged(view_->SizeInPixels()); | 
					
						
							| 
									
										
										
										
											2024-01-29 20:43:28 -06:00
										 |  |  |   SetFrameRate(view_->frame_rate()); | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | OffScreenVideoConsumer::~OffScreenVideoConsumer() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OffScreenVideoConsumer::SetActive(bool active) { | 
					
						
							|  |  |  |   if (active) { | 
					
						
							| 
									
										
										
										
											2022-01-10 17:31:39 -05:00
										 |  |  |     video_capturer_->Start(this, viz::mojom::BufferFormatPreference::kDefault); | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  |   } else { | 
					
						
							|  |  |  |     video_capturer_->Stop(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OffScreenVideoConsumer::SetFrameRate(int frame_rate) { | 
					
						
							| 
									
										
										
										
											2021-11-24 09:45:59 +01:00
										 |  |  |   video_capturer_->SetMinCapturePeriod(base::Seconds(1) / frame_rate); | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-05 15:53:39 +02:00
										 |  |  | void OffScreenVideoConsumer::SizeChanged(const gfx::Size& size_in_pixels) { | 
					
						
							|  |  |  |   DCHECK(IsValidMinAndMaxFrameSize(size_in_pixels, size_in_pixels)); | 
					
						
							|  |  |  |   video_capturer_->SetResolutionConstraints(size_in_pixels, size_in_pixels, | 
					
						
							|  |  |  |                                             true); | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  |   video_capturer_->RequestRefreshFrame(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OffScreenVideoConsumer::OnFrameCaptured( | 
					
						
							| 
									
										
										
										
											2022-01-10 17:31:39 -05:00
										 |  |  |     ::media::mojom::VideoBufferHandlePtr data, | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  |     ::media::mojom::VideoFrameInfoPtr info, | 
					
						
							|  |  |  |     const gfx::Rect& content_rect, | 
					
						
							| 
									
										
										
										
											2020-02-27 11:00:07 -08:00
										 |  |  |     mojo::PendingRemote<viz::mojom::FrameSinkVideoConsumerFrameCallbacks> | 
					
						
							|  |  |  |         callbacks) { | 
					
						
							| 
									
										
										
										
											2022-01-10 17:31:39 -05:00
										 |  |  |   auto& data_region = data->get_read_only_shmem_region(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  |   if (!CheckContentRect(content_rect)) { | 
					
						
							| 
									
										
										
										
											2022-05-05 15:53:39 +02:00
										 |  |  |     SizeChanged(view_->SizeInPixels()); | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-27 11:00:07 -08:00
										 |  |  |   mojo::Remote<viz::mojom::FrameSinkVideoConsumerFrameCallbacks> | 
					
						
							|  |  |  |       callbacks_remote(std::move(callbacks)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-10 17:31:39 -05:00
										 |  |  |   if (!data_region.IsValid()) { | 
					
						
							| 
									
										
										
										
											2020-02-27 11:00:07 -08:00
										 |  |  |     callbacks_remote->Done(); | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-01-10 17:31:39 -05:00
										 |  |  |   base::ReadOnlySharedMemoryMapping mapping = data_region.Map(); | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  |   if (!mapping.IsValid()) { | 
					
						
							|  |  |  |     DLOG(ERROR) << "Shared memory mapping failed."; | 
					
						
							| 
									
										
										
										
											2020-02-27 11:00:07 -08:00
										 |  |  |     callbacks_remote->Done(); | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (mapping.size() < | 
					
						
							|  |  |  |       media::VideoFrame::AllocationSize(info->pixel_format, info->coded_size)) { | 
					
						
							|  |  |  |     DLOG(ERROR) << "Shared memory size was less than expected."; | 
					
						
							| 
									
										
										
										
											2020-02-27 11:00:07 -08:00
										 |  |  |     callbacks_remote->Done(); | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // The SkBitmap's pixels will be marked as immutable, but the installPixels()
 | 
					
						
							|  |  |  |   // API requires a non-const pointer. So, cast away the const.
 | 
					
						
							|  |  |  |   void* const pixels = const_cast<void*>(mapping.memory()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Call installPixels() with a |releaseProc| that: 1) notifies the capturer
 | 
					
						
							|  |  |  |   // that this consumer has finished with the frame, and 2) releases the shared
 | 
					
						
							|  |  |  |   // memory mapping.
 | 
					
						
							|  |  |  |   struct FramePinner { | 
					
						
							|  |  |  |     // Keeps the shared memory that backs |frame_| mapped.
 | 
					
						
							|  |  |  |     base::ReadOnlySharedMemoryMapping mapping; | 
					
						
							|  |  |  |     // Prevents FrameSinkVideoCapturer from recycling the shared memory that
 | 
					
						
							|  |  |  |     // backs |frame_|.
 | 
					
						
							| 
									
										
										
										
											2020-02-27 11:00:07 -08:00
										 |  |  |     mojo::PendingRemote<viz::mojom::FrameSinkVideoConsumerFrameCallbacks> | 
					
						
							|  |  |  |         releaser; | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   SkBitmap bitmap; | 
					
						
							|  |  |  |   bitmap.installPixels( | 
					
						
							|  |  |  |       SkImageInfo::MakeN32(content_rect.width(), content_rect.height(), | 
					
						
							|  |  |  |                            kPremul_SkAlphaType), | 
					
						
							|  |  |  |       pixels, | 
					
						
							| 
									
										
										
										
											2024-06-07 17:18:35 -04:00
										 |  |  |       media::VideoFrame::RowBytes(media::VideoFrame::Plane::kARGB, | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  |                                   info->pixel_format, info->coded_size.width()), | 
					
						
							|  |  |  |       [](void* addr, void* context) { | 
					
						
							|  |  |  |         delete static_cast<FramePinner*>(context); | 
					
						
							|  |  |  |       }, | 
					
						
							| 
									
										
										
										
											2020-02-27 11:00:07 -08:00
										 |  |  |       new FramePinner{std::move(mapping), callbacks_remote.Unbind()}); | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  |   bitmap.setImmutable(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-10 23:23:35 +01:00
										 |  |  |   std::optional<gfx::Rect> update_rect = info->metadata.capture_update_rect; | 
					
						
							| 
									
										
										
										
											2020-06-22 10:35:10 -07:00
										 |  |  |   if (!update_rect.has_value() || update_rect->IsEmpty()) { | 
					
						
							|  |  |  |     update_rect = content_rect; | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-22 10:35:10 -07:00
										 |  |  |   callback_.Run(*update_rect, bitmap); | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-01 10:02:12 -04:00
										 |  |  | void OffScreenVideoConsumer::OnNewSubCaptureTargetVersion( | 
					
						
							|  |  |  |     uint32_t crop_version) {} | 
					
						
							| 
									
										
										
										
											2022-07-13 17:26:16 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-25 13:17:35 -05:00
										 |  |  | void OffScreenVideoConsumer::OnFrameWithEmptyRegionCapture() {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  | void OffScreenVideoConsumer::OnStopped() {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-13 16:39:26 -07:00
										 |  |  | void OffScreenVideoConsumer::OnLog(const std::string& message) {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-17 23:10:04 +02:00
										 |  |  | bool OffScreenVideoConsumer::CheckContentRect(const gfx::Rect& content_rect) { | 
					
						
							|  |  |  |   gfx::Size view_size = view_->SizeInPixels(); | 
					
						
							|  |  |  |   gfx::Size content_size = content_rect.size(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (std::abs(view_size.width() - content_size.width()) > 2) { | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (std::abs(view_size.height() - content_size.height()) > 2) { | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-19 14:23:04 -07:00
										 |  |  | }  // namespace electron
 |