// Copyright (c) 2019 GitHub, Inc. // Use of this source code is governed by the MIT license that can be // found in the LICENSE file. #include "shell/browser/osr/osr_host_display_client.h" #include #include "base/memory/shared_memory.h" #include "components/viz/common/resources/resource_format.h" #include "components/viz/common/resources/resource_sizes.h" #include "mojo/public/cpp/system/platform_handle.h" #include "skia/ext/platform_canvas.h" #include "third_party/skia/include/core/SkColor.h" #include "third_party/skia/include/core/SkRect.h" #include "third_party/skia/src/core/SkDevice.h" #include "ui/gfx/skia_util.h" #if defined(OS_WIN) #include "skia/ext/skia_utils_win.h" #endif namespace electron { LayeredWindowUpdater::LayeredWindowUpdater( viz::mojom::LayeredWindowUpdaterRequest request, OnPaintCallback callback) : callback_(callback), binding_(this, std::move(request)) {} LayeredWindowUpdater::~LayeredWindowUpdater() = default; void LayeredWindowUpdater::SetActive(bool active) { active_ = active; } void LayeredWindowUpdater::OnAllocatedSharedMemory( const gfx::Size& pixel_size, mojo::ScopedSharedBufferHandle scoped_buffer_handle) { canvas_.reset(); // Make sure |pixel_size| is sane. size_t expected_bytes; bool size_result = viz::ResourceSizes::MaybeSizeInBytes( pixel_size, viz::ResourceFormat::RGBA_8888, &expected_bytes); if (!size_result) return; #if defined(WIN32) base::SharedMemoryHandle shm_handle; size_t required_bytes; MojoResult unwrap_result = mojo::UnwrapSharedMemoryHandle( std::move(scoped_buffer_handle), &shm_handle, &required_bytes, nullptr); if (unwrap_result != MOJO_RESULT_OK) return; base::SharedMemory shm(shm_handle, false); if (!shm.Map(required_bytes)) { DLOG(ERROR) << "Failed to map " << required_bytes << " bytes"; return; } canvas_ = skia::CreatePlatformCanvasWithSharedSection( pixel_size.width(), pixel_size.height(), false, shm.handle().GetHandle(), skia::CRASH_ON_FAILURE); #else auto shm = mojo::UnwrapWritableSharedMemoryRegion(std::move(scoped_buffer_handle)); if (!shm.IsValid()) { DLOG(ERROR) << "Failed to unwrap shared memory region"; return; } shm_mapping_ = shm.Map(); if (!shm_mapping_.IsValid()) { DLOG(ERROR) << "Failed to map shared memory region"; return; } canvas_ = skia::CreatePlatformCanvasWithPixels( pixel_size.width(), pixel_size.height(), false, static_cast(shm_mapping_.memory()), skia::CRASH_ON_FAILURE); #endif } void LayeredWindowUpdater::Draw(const gfx::Rect& damage_rect, DrawCallback draw_callback) { SkPixmap pixmap; SkBitmap bitmap; if (active_ && canvas_->peekPixels(&pixmap)) { bitmap.installPixels(pixmap); callback_.Run(damage_rect, bitmap); } std::move(draw_callback).Run(); } OffScreenHostDisplayClient::OffScreenHostDisplayClient( gfx::AcceleratedWidget widget, OnPaintCallback callback) : viz::HostDisplayClient(widget), callback_(callback) {} OffScreenHostDisplayClient::~OffScreenHostDisplayClient() {} void OffScreenHostDisplayClient::SetActive(bool active) { active_ = active; if (layered_window_updater_) { layered_window_updater_->SetActive(active_); } } void OffScreenHostDisplayClient::IsOffscreen(IsOffscreenCallback callback) { std::move(callback).Run(true); } void OffScreenHostDisplayClient::CreateLayeredWindowUpdater( viz::mojom::LayeredWindowUpdaterRequest request) { layered_window_updater_ = std::make_unique(std::move(request), callback_); layered_window_updater_->SetActive(active_); } } // namespace electron