| // Copyright 2012 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "content/renderer/pepper/ppb_image_data_impl.h" |
| |
| #include <algorithm> |
| #include <limits> |
| #include <memory> |
| |
| #include "base/check.h" |
| #include "base/notreached.h" |
| #include "content/common/pepper_file_util.h" |
| #include "ppapi/c/pp_errors.h" |
| #include "ppapi/c/pp_instance.h" |
| #include "ppapi/c/pp_resource.h" |
| #include "ppapi/c/ppb_image_data.h" |
| #include "ppapi/thunk/thunk.h" |
| #include "skia/ext/legacy_display_globals.h" |
| #include "skia/ext/platform_canvas.h" |
| #include "third_party/skia/include/core/SkCanvas.h" |
| #include "third_party/skia/include/core/SkPixmap.h" |
| #include "ui/surface/transport_dib.h" |
| |
| using ppapi::thunk::PPB_ImageData_API; |
| |
| namespace content { |
| |
| PPB_ImageData_Impl::PPB_ImageData_Impl(PP_Instance instance, |
| PPB_ImageData_Shared::ImageDataType type) |
| : Resource(ppapi::OBJECT_IS_IMPL, instance), |
| format_(PP_IMAGEDATAFORMAT_BGRA_PREMUL), |
| width_(0), |
| height_(0) { |
| switch (type) { |
| case PPB_ImageData_Shared::PLATFORM: |
| backend_ = std::make_unique<ImageDataPlatformBackend>(); |
| return; |
| case PPB_ImageData_Shared::SIMPLE: |
| backend_ = std::make_unique<ImageDataSimpleBackend>(); |
| return; |
| // No default: so that we get a compiler warning if any types are added. |
| } |
| NOTREACHED(); |
| } |
| |
| PPB_ImageData_Impl::PPB_ImageData_Impl(PP_Instance instance, ForTest) |
| : Resource(ppapi::OBJECT_IS_IMPL, instance), |
| format_(PP_IMAGEDATAFORMAT_BGRA_PREMUL), |
| width_(0), |
| height_(0) { |
| backend_ = std::make_unique<ImageDataPlatformBackend>(); |
| } |
| |
| PPB_ImageData_Impl::~PPB_ImageData_Impl() {} |
| |
| bool PPB_ImageData_Impl::Init(PP_ImageDataFormat format, |
| int width, |
| int height, |
| bool init_to_zero) { |
| // TODO(brettw) this should be called only on the main thread! |
| if (!IsImageDataFormatSupported(format)) |
| return false; // Only support this one format for now. |
| if (width <= 0 || height <= 0) |
| return false; |
| if (static_cast<int64_t>(width) * static_cast<int64_t>(height) >= |
| std::numeric_limits<int32_t>::max() / 4) |
| return false; // Prevent overflow of signed 32-bit ints. |
| |
| format_ = format; |
| width_ = width; |
| height_ = height; |
| return backend_->Init(this, format, width, height, init_to_zero); |
| } |
| |
| // static |
| PP_Resource PPB_ImageData_Impl::Create(PP_Instance instance, |
| PPB_ImageData_Shared::ImageDataType type, |
| PP_ImageDataFormat format, |
| const PP_Size& size, |
| PP_Bool init_to_zero) { |
| scoped_refptr<PPB_ImageData_Impl> data( |
| new PPB_ImageData_Impl(instance, type)); |
| if (!data->Init(format, size.width, size.height, !!init_to_zero)) |
| return 0; |
| return data->GetReference(); |
| } |
| |
| PPB_ImageData_API* PPB_ImageData_Impl::AsPPB_ImageData_API() { return this; } |
| |
| bool PPB_ImageData_Impl::IsMapped() const { return backend_->IsMapped(); } |
| |
| TransportDIB* PPB_ImageData_Impl::GetTransportDIB() const { |
| return backend_->GetTransportDIB(); |
| } |
| |
| PP_Bool PPB_ImageData_Impl::Describe(PP_ImageDataDesc* desc) { |
| desc->format = format_; |
| desc->size.width = width_; |
| desc->size.height = height_; |
| desc->stride = width_ * 4; |
| return PP_TRUE; |
| } |
| |
| void* PPB_ImageData_Impl::Map() { return backend_->Map(); } |
| |
| void PPB_ImageData_Impl::Unmap() { backend_->Unmap(); } |
| |
| int32_t PPB_ImageData_Impl::GetSharedMemoryRegion( |
| base::UnsafeSharedMemoryRegion** region) { |
| return backend_->GetSharedMemoryRegion(region); |
| } |
| |
| SkCanvas* PPB_ImageData_Impl::GetCanvas() { return backend_->GetCanvas(); } |
| |
| void PPB_ImageData_Impl::SetIsCandidateForReuse() { |
| // Nothing to do since we don't support image data re-use in-process. |
| } |
| |
| SkBitmap PPB_ImageData_Impl::GetMappedBitmap() const { |
| return backend_->GetMappedBitmap(); |
| } |
| |
| // ImageDataPlatformBackend ---------------------------------------------------- |
| |
| ImageDataPlatformBackend::ImageDataPlatformBackend() : width_(0), height_(0) { |
| } |
| |
| ImageDataPlatformBackend::~ImageDataPlatformBackend() { |
| } |
| |
| bool ImageDataPlatformBackend::Init(PPB_ImageData_Impl* impl, |
| PP_ImageDataFormat format, |
| int width, |
| int height, |
| bool init_to_zero) { |
| // TODO(brettw): use init_to_zero when we implement caching. |
| width_ = width; |
| height_ = height; |
| uint32_t buffer_size = width_ * height_ * 4; |
| base::UnsafeSharedMemoryRegion region = |
| base::UnsafeSharedMemoryRegion::Create(buffer_size); |
| if (!region.IsValid()) |
| return false; |
| |
| dib_ = TransportDIB::CreateWithHandle(std::move(region)); |
| return !!dib_; |
| } |
| |
| bool ImageDataPlatformBackend::IsMapped() const { |
| return !!mapped_canvas_.get(); |
| } |
| |
| TransportDIB* ImageDataPlatformBackend::GetTransportDIB() const { |
| return dib_.get(); |
| } |
| |
| void* ImageDataPlatformBackend::Map() { |
| if (!mapped_canvas_) { |
| const bool is_opaque = false; |
| mapped_canvas_ = dib_->GetPlatformCanvas(width_, height_, is_opaque); |
| if (!mapped_canvas_) |
| return nullptr; |
| } |
| SkPixmap pixmap; |
| skia::GetWritablePixels(mapped_canvas_.get(), &pixmap); |
| DCHECK(pixmap.addr()); |
| // SkPixmap does not manage the lifetime of this pointer, so it remains |
| // valid after the object goes out of scope. It will become invalid if |
| // the canvas' backing is destroyed or a pending saveLayer() is resolved. |
| return pixmap.writable_addr32(0, 0); |
| } |
| |
| void ImageDataPlatformBackend::Unmap() { |
| // This is currently unimplemented, which is OK. The data will just always |
| // be around once it's mapped. Chrome's TransportDIB isn't currently |
| // unmappable without freeing it, but this may be something we want to support |
| // in the future to save some memory. |
| } |
| |
| int32_t ImageDataPlatformBackend::GetSharedMemoryRegion( |
| base::UnsafeSharedMemoryRegion** region) { |
| *region = dib_->shared_memory_region(); |
| return PP_OK; |
| } |
| |
| SkCanvas* ImageDataPlatformBackend::GetCanvas() { return mapped_canvas_.get(); } |
| |
| SkBitmap ImageDataPlatformBackend::GetMappedBitmap() const { |
| SkBitmap bitmap; |
| if (!mapped_canvas_) |
| return bitmap; |
| |
| SkPixmap pixmap; |
| skia::GetWritablePixels(mapped_canvas_.get(), &pixmap); |
| // SkPixmap does not manage the lifetime of this pointer, so it remains |
| // valid after the object goes out of scope. It will become invalid if |
| // the canvas' backing is destroyed or a pending saveLayer() is resolved. |
| bitmap.installPixels(pixmap); |
| return bitmap; |
| } |
| |
| // ImageDataSimpleBackend ------------------------------------------------------ |
| |
| ImageDataSimpleBackend::ImageDataSimpleBackend() : map_count_(0) {} |
| |
| ImageDataSimpleBackend::~ImageDataSimpleBackend() {} |
| |
| bool ImageDataSimpleBackend::Init(PPB_ImageData_Impl* impl, |
| PP_ImageDataFormat format, |
| int width, |
| int height, |
| bool init_to_zero) { |
| skia_bitmap_.setInfo( |
| SkImageInfo::MakeN32Premul(impl->width(), impl->height())); |
| shm_region_ = |
| base::UnsafeSharedMemoryRegion::Create(skia_bitmap_.computeByteSize()); |
| return shm_region_.IsValid(); |
| } |
| |
| bool ImageDataSimpleBackend::IsMapped() const { |
| return shm_mapping_.IsValid(); |
| } |
| |
| TransportDIB* ImageDataSimpleBackend::GetTransportDIB() const { |
| return nullptr; |
| } |
| |
| void* ImageDataSimpleBackend::Map() { |
| DCHECK(shm_region_.IsValid()); |
| if (map_count_++ == 0) { |
| shm_mapping_ = shm_region_.Map(); |
| if (!shm_mapping_.IsValid()) |
| return nullptr; |
| |
| base::span<uint8_t> mem(shm_mapping_); |
| CHECK_GE(mem.size(), skia_bitmap_.computeByteSize()); |
| skia_bitmap_.setPixels(mem.data()); |
| // Our platform bitmaps are set to opaque by default, which we don't want. |
| skia_bitmap_.setAlphaType(kPremul_SkAlphaType); |
| skia_canvas_ = std::make_unique<SkCanvas>( |
| skia_bitmap_, skia::LegacyDisplayGlobals::GetSkSurfaceProps()); |
| } |
| return skia_bitmap_.isNull() ? nullptr : skia_bitmap_.getAddr32(0, 0); |
| } |
| |
| void ImageDataSimpleBackend::Unmap() { |
| if (--map_count_ == 0) |
| shm_mapping_ = base::WritableSharedMemoryMapping(); |
| } |
| |
| int32_t ImageDataSimpleBackend::GetSharedMemoryRegion( |
| base::UnsafeSharedMemoryRegion** region) { |
| *region = &shm_region_; |
| return PP_OK; |
| } |
| |
| SkCanvas* ImageDataSimpleBackend::GetCanvas() { |
| if (!IsMapped()) |
| return nullptr; |
| return skia_canvas_.get(); |
| } |
| |
| SkBitmap ImageDataSimpleBackend::GetMappedBitmap() const { |
| if (!IsMapped()) |
| return SkBitmap(); |
| return skia_bitmap_; |
| } |
| |
| } // namespace content |