| 
									
										
										
										
											2015-09-18 15:57:43 +08:00
										 |  |  | // Copyright (c) 2015 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/api/frame_subscriber.h"
 | 
					
						
							| 
									
										
										
										
											2015-09-18 15:57:43 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-08 17:58:54 +01:00
										 |  |  | #include <utility>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "content/public/browser/render_view_host.h"
 | 
					
						
							|  |  |  | #include "content/public/browser/render_widget_host.h"
 | 
					
						
							|  |  |  | #include "content/public/browser/render_widget_host_view.h"
 | 
					
						
							| 
									
										
										
										
											2021-06-22 12:17:16 -07:00
										 |  |  | #include "media/capture/mojom/video_capture_buffer.mojom.h"
 | 
					
						
							| 
									
										
										
										
											2019-02-21 23:19:58 +05:30
										 |  |  | #include "media/capture/mojom/video_capture_types.mojom.h"
 | 
					
						
							| 
									
										
										
										
											2019-10-28 18:12:35 -04:00
										 |  |  | #include "mojo/public/cpp/bindings/remote.h"
 | 
					
						
							| 
									
										
										
										
											2022-01-10 17:31:39 -05:00
										 |  |  | #include "services/viz/privileged/mojom/compositing/frame_sink_video_capture.mojom-shared.h"
 | 
					
						
							| 
									
										
										
										
											2019-04-08 11:35:33 +09:00
										 |  |  | #include "ui/gfx/geometry/size_conversions.h"
 | 
					
						
							| 
									
										
										
										
											2019-04-05 03:36:12 +09:00
										 |  |  | #include "ui/gfx/image/image.h"
 | 
					
						
							| 
									
										
										
										
											2018-05-14 19:09:05 +02:00
										 |  |  | #include "ui/gfx/skbitmap_operations.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-06 17:24:37 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-29 12:55:47 -07:00
										 |  |  | namespace electron::api { | 
					
						
							| 
									
										
										
										
											2015-09-18 15:57:43 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-08 17:58:54 +01:00
										 |  |  | constexpr static int kMaxFrameRate = 30; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-05 03:36:12 +09:00
										 |  |  | FrameSubscriber::FrameSubscriber(content::WebContents* web_contents, | 
					
						
							| 
									
										
										
										
											2018-05-14 19:09:05 +02:00
										 |  |  |                                  const FrameCaptureCallback& callback, | 
					
						
							|  |  |  |                                  bool only_dirty) | 
					
						
							| 
									
										
										
										
											2018-05-14 17:55:39 +02:00
										 |  |  |     : content::WebContentsObserver(web_contents), | 
					
						
							| 
									
										
										
										
											2018-05-14 19:09:05 +02:00
										 |  |  |       callback_(callback), | 
					
						
							| 
									
										
										
										
											2021-01-26 19:16:21 +01:00
										 |  |  |       only_dirty_(only_dirty) { | 
					
						
							| 
									
										
										
										
											2023-02-03 12:43:42 +01:00
										 |  |  |   AttachToHost(web_contents->GetPrimaryMainFrame()->GetRenderWidgetHost()); | 
					
						
							| 
									
										
										
										
											2018-11-08 17:58:54 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2017-04-13 19:21:30 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-18 02:45:15 +02:00
										 |  |  | FrameSubscriber::~FrameSubscriber() = default; | 
					
						
							| 
									
										
										
										
											2015-09-18 15:57:43 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-08 17:58:54 +01:00
										 |  |  | void FrameSubscriber::AttachToHost(content::RenderWidgetHost* host) { | 
					
						
							|  |  |  |   host_ = host; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // The view can be null if the renderer process has crashed.
 | 
					
						
							|  |  |  |   // (https://crbug.com/847363)
 | 
					
						
							|  |  |  |   if (!host_->GetView()) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Create and configure the video capturer.
 | 
					
						
							| 
									
										
										
										
											2019-04-08 11:35:33 +09:00
										 |  |  |   gfx::Size size = GetRenderViewSize(); | 
					
						
							| 
									
										
										
										
											2024-04-08 10:30:23 -07:00
										 |  |  |   DCHECK(!size.IsEmpty()); | 
					
						
							| 
									
										
										
										
											2018-11-08 17:58:54 +01:00
										 |  |  |   video_capturer_ = host_->GetView()->CreateVideoCapturer(); | 
					
						
							| 
									
										
										
										
											2019-04-08 11:35:33 +09:00
										 |  |  |   video_capturer_->SetResolutionConstraints(size, size, true); | 
					
						
							| 
									
										
										
										
											2018-11-08 17:58:54 +01: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); | 
					
						
							| 
									
										
										
										
											2021-11-24 09:45:59 +01:00
										 |  |  |   video_capturer_->SetMinCapturePeriod(base::Seconds(1) / kMaxFrameRate); | 
					
						
							| 
									
										
										
										
											2022-01-10 17:31:39 -05:00
										 |  |  |   video_capturer_->Start(this, viz::mojom::BufferFormatPreference::kDefault); | 
					
						
							| 
									
										
										
										
											2018-11-08 17:58:54 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void FrameSubscriber::DetachFromHost() { | 
					
						
							|  |  |  |   if (!host_) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   video_capturer_.reset(); | 
					
						
							|  |  |  |   host_ = nullptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-04 09:27:05 -08:00
										 |  |  | void FrameSubscriber::RenderFrameCreated( | 
					
						
							|  |  |  |     content::RenderFrameHost* render_frame_host) { | 
					
						
							| 
									
										
										
										
											2018-11-08 17:58:54 +01:00
										 |  |  |   if (!host_) | 
					
						
							| 
									
										
										
										
											2021-03-04 09:27:05 -08:00
										 |  |  |     AttachToHost(render_frame_host->GetRenderWidgetHost()); | 
					
						
							| 
									
										
										
										
											2018-11-08 17:58:54 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void FrameSubscriber::RenderViewDeleted(content::RenderViewHost* host) { | 
					
						
							|  |  |  |   if (host->GetWidget() == host_) { | 
					
						
							|  |  |  |     DetachFromHost(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-03 12:43:42 +01:00
										 |  |  | void FrameSubscriber::PrimaryPageChanged(content::Page& page) { | 
					
						
							|  |  |  |   if (auto* host = page.GetMainDocument().GetMainFrame()->GetRenderWidgetHost(); | 
					
						
							|  |  |  |       host_ != host) { | 
					
						
							| 
									
										
										
										
											2018-11-08 17:58:54 +01:00
										 |  |  |     DetachFromHost(); | 
					
						
							| 
									
										
										
										
											2023-02-03 12:43:42 +01:00
										 |  |  |     AttachToHost(host); | 
					
						
							| 
									
										
										
										
											2018-11-08 17:58:54 +01:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void FrameSubscriber::OnFrameCaptured( | 
					
						
							| 
									
										
										
										
											2022-01-10 17:31:39 -05:00
										 |  |  |     ::media::mojom::VideoBufferHandlePtr data, | 
					
						
							| 
									
										
										
										
											2018-11-08 17:58:54 +01:00
										 |  |  |     ::media::mojom::VideoFrameInfoPtr info, | 
					
						
							|  |  |  |     const gfx::Rect& content_rect, | 
					
						
							| 
									
										
										
										
											2019-10-28 18:12:35 -04: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-08 11:35:33 +09:00
										 |  |  |   gfx::Size size = GetRenderViewSize(); | 
					
						
							|  |  |  |   if (size != content_rect.size()) { | 
					
						
							|  |  |  |     video_capturer_->SetResolutionConstraints(size, size, true); | 
					
						
							| 
									
										
										
										
											2018-11-08 17:58:54 +01:00
										 |  |  |     video_capturer_->RequestRefreshFrame(); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-28 18:12:35 -04:00
										 |  |  |   mojo::Remote<viz::mojom::FrameSinkVideoConsumerFrameCallbacks> | 
					
						
							|  |  |  |       callbacks_remote(std::move(callbacks)); | 
					
						
							| 
									
										
										
										
											2022-01-10 17:31:39 -05:00
										 |  |  |   if (!data_region.IsValid()) { | 
					
						
							| 
									
										
										
										
											2019-10-28 18:12:35 -04:00
										 |  |  |     callbacks_remote->Done(); | 
					
						
							| 
									
										
										
										
											2018-11-08 17:58:54 +01:00
										 |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-01-10 17:31:39 -05:00
										 |  |  |   base::ReadOnlySharedMemoryMapping mapping = data_region.Map(); | 
					
						
							| 
									
										
										
										
											2018-11-08 17:58:54 +01:00
										 |  |  |   if (!mapping.IsValid()) { | 
					
						
							|  |  |  |     DLOG(ERROR) << "Shared memory mapping failed."; | 
					
						
							|  |  |  |     return; | 
					
						
							| 
									
										
										
										
											2018-04-18 02:45:15 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-11-08 17:58:54 +01:00
										 |  |  |   if (mapping.size() < | 
					
						
							|  |  |  |       media::VideoFrame::AllocationSize(info->pixel_format, info->coded_size)) { | 
					
						
							|  |  |  |     DLOG(ERROR) << "Shared memory size was less than expected."; | 
					
						
							|  |  |  |     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_|.
 | 
					
						
							| 
									
										
										
										
											2019-10-28 18:12:35 -04:00
										 |  |  |     mojo::Remote<viz::mojom::FrameSinkVideoConsumerFrameCallbacks> releaser; | 
					
						
							| 
									
										
										
										
											2018-11-08 17:58:54 +01: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, | 
					
						
							| 
									
										
										
										
											2018-11-08 17:58:54 +01:00
										 |  |  |                                   info->pixel_format, info->coded_size.width()), | 
					
						
							|  |  |  |       [](void* addr, void* context) { | 
					
						
							|  |  |  |         delete static_cast<FramePinner*>(context); | 
					
						
							|  |  |  |       }, | 
					
						
							| 
									
										
										
										
											2019-10-28 18:12:35 -04:00
										 |  |  |       new FramePinner{std::move(mapping), std::move(callbacks_remote)}); | 
					
						
							| 
									
										
										
										
											2018-11-08 17:58:54 +01:00
										 |  |  |   bitmap.setImmutable(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   Done(content_rect, bitmap); | 
					
						
							| 
									
										
										
										
											2018-05-14 17:55:39 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2017-07-28 11:57:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-01 10:02:12 -04:00
										 |  |  | void FrameSubscriber::OnNewSubCaptureTargetVersion(uint32_t crop_version) {} | 
					
						
							| 
									
										
										
										
											2022-07-13 17:26:16 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-25 13:17:35 -05:00
										 |  |  | void FrameSubscriber::OnFrameWithEmptyRegionCapture() {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-08 17:58:54 +01:00
										 |  |  | void FrameSubscriber::OnStopped() {} | 
					
						
							| 
									
										
										
										
											2018-04-18 02:45:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-13 16:39:26 -07:00
										 |  |  | void FrameSubscriber::OnLog(const std::string& message) {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-14 17:55:39 +02:00
										 |  |  | void FrameSubscriber::Done(const gfx::Rect& damage, const SkBitmap& frame) { | 
					
						
							|  |  |  |   if (frame.drawsNothing()) | 
					
						
							|  |  |  |     return; | 
					
						
							| 
									
										
										
										
											2018-04-18 02:45:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-19 19:49:44 +02:00
										 |  |  |   const SkBitmap& bitmap = only_dirty_ ? SkBitmapOperations::CreateTiledBitmap( | 
					
						
							|  |  |  |                                              frame, damage.x(), damage.y(), | 
					
						
							|  |  |  |                                              damage.width(), damage.height()) | 
					
						
							|  |  |  |                                        : frame; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-05 03:36:12 +09:00
										 |  |  |   // Copying SkBitmap does not copy the internal pixels, we have to manually
 | 
					
						
							|  |  |  |   // allocate and write pixels otherwise crash may happen when the original
 | 
					
						
							|  |  |  |   // frame is modified.
 | 
					
						
							|  |  |  |   SkBitmap copy; | 
					
						
							|  |  |  |   copy.allocPixels(SkImageInfo::Make(bitmap.width(), bitmap.height(), | 
					
						
							| 
									
										
										
										
											2020-12-22 14:14:44 -08:00
										 |  |  |                                      kN32_SkColorType, kPremul_SkAlphaType)); | 
					
						
							| 
									
										
										
										
											2019-04-05 03:36:12 +09:00
										 |  |  |   SkPixmap pixmap; | 
					
						
							|  |  |  |   bool success = bitmap.peekPixels(&pixmap) && copy.writePixels(pixmap, 0, 0); | 
					
						
							|  |  |  |   CHECK(success); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   callback_.Run(gfx::Image::CreateFrom1xBitmap(copy), damage); | 
					
						
							| 
									
										
										
										
											2015-09-18 15:57:43 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-08 11:35:33 +09:00
										 |  |  | gfx::Size FrameSubscriber::GetRenderViewSize() const { | 
					
						
							|  |  |  |   content::RenderWidgetHostView* view = host_->GetView(); | 
					
						
							|  |  |  |   gfx::Size size = view->GetViewBounds().size(); | 
					
						
							|  |  |  |   return gfx::ToRoundedSize( | 
					
						
							|  |  |  |       gfx::ScaleSize(gfx::SizeF(size), view->GetDeviceScaleFactor())); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-29 12:55:47 -07:00
										 |  |  | }  // namespace electron::api
 |