/* -*- Mode: C++; tab-width: 20; indent-tabs-mode: nil; c-basic-offset: 2 -*- * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "SharedPlanarYCbCrImage.h" #include "ShmemYCbCrImage.h" #include "ISurfaceAllocator.h" #include "mozilla/layers/LayersSurfaces.h" namespace mozilla { namespace layers { using namespace mozilla::ipc; SharedPlanarYCbCrImage::~SharedPlanarYCbCrImage() { MOZ_COUNT_DTOR(SharedPlanarYCbCrImage); if (mAllocated) { SurfaceDescriptor desc; DropToSurfaceDescriptor(desc); mSurfaceAllocator->DestroySharedSurface(&desc); } } void SharedPlanarYCbCrImage::SetData(const PlanarYCbCrImage::Data& aData) { // If mShmem has not been allocated (through Allocate(aData)), allocate it. // This code path is slower than the one used when Allocate has been called // since it will trigger a full copy. if (!mAllocated) { Data data = aData; if (!Allocate(data)) { return; } } // do not set mBuffer like in PlanarYCbCrImage because the later // will try to manage this memory without knowing it belongs to a // shmem. mBufferSize = ShmemYCbCrImage::ComputeMinBufferSize(mData.mYSize, mData.mCbCrSize); mSize = mData.mPicSize; ShmemYCbCrImage shmImg(mShmem); MOZ_ASSERT(aData.mCbSkip == aData.mCrSkip); if (!shmImg.CopyData(aData.mYChannel, aData.mCbChannel, aData.mCrChannel, aData.mYSize, aData.mYStride, aData.mCbCrSize, aData.mCbCrStride, aData.mYSkip, aData.mCbSkip)) { NS_WARNING("Failed to copy image data!"); } mData.mYChannel = shmImg.GetYData(); mData.mCbChannel = shmImg.GetCbData(); mData.mCrChannel = shmImg.GetCrData(); } // needs to be overriden because the parent class sets mBuffer which we // do not want to happen. uint8_t* SharedPlanarYCbCrImage::AllocateAndGetNewBuffer(uint32_t aSize) { NS_ABORT_IF_FALSE(!mAllocated, "This image already has allocated data"); size_t size = ShmemYCbCrImage::ComputeMinBufferSize(aSize); // update buffer size mBufferSize = size; // get new buffer _without_ setting mBuffer. AllocateBuffer(mBufferSize); ShmemYCbCrImage shmImg(mShmem); return shmImg.GetData(); } void SharedPlanarYCbCrImage::SetDataNoCopy(const Data &aData) { mData = aData; mSize = aData.mPicSize; ShmemYCbCrImage::InitializeBufferInfo(mShmem.get(), aData.mYSize, aData.mCbCrSize); } uint8_t* SharedPlanarYCbCrImage::AllocateBuffer(uint32_t aSize) { NS_ABORT_IF_FALSE(!mAllocated, "This image already has allocated data"); SharedMemory::SharedMemoryType shmType = OptimalShmemType(); if (!mSurfaceAllocator->AllocUnsafeShmem(aSize, shmType, &mShmem)) { return nullptr; } mAllocated = true; return mShmem.get(); } bool SharedPlanarYCbCrImage::Allocate(PlanarYCbCrImage::Data& aData) { NS_ABORT_IF_FALSE(!mAllocated, "This image already has allocated data"); size_t size = ShmemYCbCrImage::ComputeMinBufferSize(aData.mYSize, aData.mCbCrSize); if (AllocateBuffer(static_cast(size)) == nullptr) { return false; } ShmemYCbCrImage::InitializeBufferInfo(mShmem.get(), aData.mYSize, aData.mCbCrSize); ShmemYCbCrImage shmImg(mShmem); if (!shmImg.IsValid() || mShmem.Size() < size) { mSurfaceAllocator->DeallocShmem(mShmem); return false; } aData.mYChannel = shmImg.GetYData(); aData.mCbChannel = shmImg.GetCbData(); aData.mCrChannel = shmImg.GetCrData(); // copy some of aData's values in mData (most of them) mData.mYChannel = aData.mYChannel; mData.mCbChannel = aData.mCbChannel; mData.mCrChannel = aData.mCrChannel; mData.mYSize = aData.mYSize; mData.mCbCrSize = aData.mCbCrSize; mData.mPicX = aData.mPicX; mData.mPicY = aData.mPicY; mData.mPicSize = aData.mPicSize; mData.mStereoMode = aData.mStereoMode; // those members are not always equal to aData's, due to potentially different // packing. mData.mYSkip = 0; mData.mCbSkip = 0; mData.mCrSkip = 0; mData.mYStride = mData.mYSize.width; mData.mCbCrStride = mData.mCbCrSize.width; mAllocated = true; return true; } bool SharedPlanarYCbCrImage::ToSurfaceDescriptor(SurfaceDescriptor& aDesc) { if (!mAllocated) { return false; } aDesc = YCbCrImage(mShmem, 0, reinterpret_cast(this)); this->AddRef(); return true; } bool SharedPlanarYCbCrImage::DropToSurfaceDescriptor(SurfaceDescriptor& aDesc) { if (!mAllocated) { return false; } aDesc = YCbCrImage(mShmem, 0, 0); mShmem = Shmem(); mAllocated = false; return true; } SharedPlanarYCbCrImage* SharedPlanarYCbCrImage::FromSurfaceDescriptor(const SurfaceDescriptor& aDescriptor) { if (aDescriptor.type() != SurfaceDescriptor::TYCbCrImage) { return nullptr; } const YCbCrImage& ycbcr = aDescriptor.get_YCbCrImage(); if (ycbcr.owner() == 0) { return nullptr; } return reinterpret_cast(ycbcr.owner()); } } // namespace } // namespace