blob: 67df94cff456a49ae3b4949c93ddf8fe6bff0efa [file] [log] [blame]
// Copyright 2016 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "media/mojo/services/mojo_cdm_allocator.h"
#include <limits>
#include <memory>
#include "base/bind.h"
#include "base/callback.h"
#include "base/compiler_specific.h"
#include "base/numerics/safe_conversions.h"
#include "base/numerics/safe_math.h"
#include "media/cdm/api/content_decryption_module.h"
#include "media/cdm/cdm_helpers.h"
#include "media/mojo/common/mojo_shared_buffer_video_frame.h"
#include "mojo/public/cpp/system/buffer.h"
#include "ui/gfx/geometry/rect.h"
#include "ui/gfx/geometry/size.h"
namespace media {
namespace {
typedef base::Callback<void(mojo::ScopedSharedBufferHandle buffer,
size_t capacity)>
MojoSharedBufferDoneCB;
VideoPixelFormat CdmVideoFormatToVideoPixelFormat(cdm::VideoFormat format) {
switch (format) {
case cdm::kYv12:
return PIXEL_FORMAT_YV12;
case cdm::kI420:
return PIXEL_FORMAT_I420;
default:
NOTREACHED();
return PIXEL_FORMAT_UNKNOWN;
}
}
// cdm::Buffer implementation that provides access to mojo shared memory.
// It owns the memory until Destroy() is called.
class MojoCdmBuffer : public cdm::Buffer {
public:
static MojoCdmBuffer* Create(
mojo::ScopedSharedBufferHandle buffer,
size_t capacity,
const MojoSharedBufferDoneCB& mojo_shared_buffer_done_cb) {
DCHECK(buffer.is_valid());
DCHECK(!mojo_shared_buffer_done_cb.is_null());
// cdm::Buffer interface limits capacity to uint32.
DCHECK_LE(capacity, std::numeric_limits<uint32_t>::max());
return new MojoCdmBuffer(std::move(buffer),
base::checked_cast<uint32_t>(capacity),
mojo_shared_buffer_done_cb);
}
// cdm::Buffer implementation.
void Destroy() final {
// Unmap the memory before returning the handle to |allocator_|.
mapping_.reset();
// If nobody has claimed the handle, then return it.
if (buffer_.is_valid())
mojo_shared_buffer_done_cb_.Run(std::move(buffer_), capacity_);
// No need to exist anymore.
delete this;
}
uint32_t Capacity() const final { return capacity_; }
uint8_t* Data() final { return static_cast<uint8_t*>(mapping_.get()); }
void SetSize(uint32_t size) final {
DCHECK_LE(size, Capacity());
size_ = size > Capacity() ? 0 : size;
}
uint32_t Size() const final { return size_; }
const mojo::SharedBufferHandle& Handle() const { return buffer_.get(); }
mojo::ScopedSharedBufferHandle TakeHandle() { return std::move(buffer_); }
private:
MojoCdmBuffer(mojo::ScopedSharedBufferHandle buffer,
uint32_t capacity,
const MojoSharedBufferDoneCB& mojo_shared_buffer_done_cb)
: buffer_(std::move(buffer)),
mojo_shared_buffer_done_cb_(mojo_shared_buffer_done_cb),
capacity_(capacity),
size_(0) {
mapping_ = buffer_->Map(capacity_);
DCHECK(mapping_);
}
~MojoCdmBuffer() final {
// Verify that the buffer has been returned so it can be reused.
DCHECK(!buffer_.is_valid());
}
mojo::ScopedSharedBufferHandle buffer_;
MojoSharedBufferDoneCB mojo_shared_buffer_done_cb_;
mojo::ScopedSharedBufferMapping mapping_;
uint32_t capacity_;
uint32_t size_;
DISALLOW_COPY_AND_ASSIGN(MojoCdmBuffer);
};
// VideoFrameImpl that is able to create a MojoSharedBufferVideoFrame
// out of the data.
class MojoCdmVideoFrame : public VideoFrameImpl {
public:
explicit MojoCdmVideoFrame(
const MojoSharedBufferDoneCB& mojo_shared_buffer_done_cb)
: mojo_shared_buffer_done_cb_(mojo_shared_buffer_done_cb) {}
~MojoCdmVideoFrame() final = default;
// VideoFrameImpl implementation.
scoped_refptr<media::VideoFrame> TransformToVideoFrame(
gfx::Size natural_size) final {
DCHECK(FrameBuffer());
MojoCdmBuffer* buffer = static_cast<MojoCdmBuffer*>(FrameBuffer());
const gfx::Size frame_size(Size().width, Size().height);
// Take ownership of the mojo::ScopedSharedBufferHandle from |buffer|.
uint32_t buffer_size = buffer->Size();
mojo::ScopedSharedBufferHandle handle = buffer->TakeHandle();
DCHECK(handle.is_valid());
// Clear FrameBuffer so that MojoCdmVideoFrame no longer has a reference
// to it (memory will be transferred to MojoSharedBufferVideoFrame).
SetFrameBuffer(nullptr);
// Destroy the MojoCdmBuffer as it is no longer needed.
buffer->Destroy();
scoped_refptr<MojoSharedBufferVideoFrame> frame =
media::MojoSharedBufferVideoFrame::Create(
CdmVideoFormatToVideoPixelFormat(Format()), frame_size,
gfx::Rect(frame_size), natural_size, std::move(handle), buffer_size,
PlaneOffset(kYPlane), PlaneOffset(kUPlane), PlaneOffset(kVPlane),
Stride(kYPlane), Stride(kUPlane), Stride(kVPlane),
base::TimeDelta::FromMicroseconds(Timestamp()));
// |frame| could fail to be created if the memory can't be mapped into
// this address space.
if (frame)
frame->SetMojoSharedBufferDoneCB(mojo_shared_buffer_done_cb_);
return frame;
}
private:
MojoSharedBufferDoneCB mojo_shared_buffer_done_cb_;
DISALLOW_COPY_AND_ASSIGN(MojoCdmVideoFrame);
};
} // namespace
MojoCdmAllocator::MojoCdmAllocator() : weak_ptr_factory_(this) {}
MojoCdmAllocator::~MojoCdmAllocator() = default;
// Creates a cdm::Buffer, reusing an existing buffer if one is available.
// If not, a new buffer is created using AllocateNewBuffer(). The caller is
// responsible for calling Destroy() on the buffer when it is no longer needed.
cdm::Buffer* MojoCdmAllocator::CreateCdmBuffer(size_t capacity) {
DCHECK(thread_checker_.CalledOnValidThread());
if (!capacity)
return nullptr;
// Reuse a buffer in the free map if there is one that fits |capacity|.
// Otherwise, create a new one.
mojo::ScopedSharedBufferHandle buffer;
auto found = available_buffers_.lower_bound(capacity);
if (found == available_buffers_.end()) {
buffer = AllocateNewBuffer(&capacity);
if (!buffer.is_valid())
return nullptr;
} else {
capacity = found->first;
buffer = std::move(found->second);
available_buffers_.erase(found);
}
// Ownership of the SharedBufferHandle is passed to MojoCdmBuffer. When it is
// done with the memory, it must call AddBufferToAvailableMap() to make the
// memory available for another MojoCdmBuffer.
return MojoCdmBuffer::Create(
std::move(buffer), capacity,
base::Bind(&MojoCdmAllocator::AddBufferToAvailableMap,
weak_ptr_factory_.GetWeakPtr()));
}
// Creates a new MojoCdmVideoFrame on every request.
std::unique_ptr<VideoFrameImpl> MojoCdmAllocator::CreateCdmVideoFrame() {
DCHECK(thread_checker_.CalledOnValidThread());
return std::make_unique<MojoCdmVideoFrame>(
base::Bind(&MojoCdmAllocator::AddBufferToAvailableMap,
weak_ptr_factory_.GetWeakPtr()));
}
mojo::ScopedSharedBufferHandle MojoCdmAllocator::AllocateNewBuffer(
size_t* capacity) {
DCHECK(thread_checker_.CalledOnValidThread());
// Always pad new allocated buffer so that we don't need to reallocate
// buffers frequently if requested sizes fluctuate slightly.
static const size_t kBufferPadding = 512;
// Maximum number of free buffers we can keep when allocating new buffers.
static const size_t kFreeLimit = 3;
// Destroy the smallest buffer before allocating a new bigger buffer if the
// number of free buffers exceeds a limit. This mechanism helps avoid ending
// up with too many small buffers, which could happen if the size to be
// allocated keeps increasing.
if (available_buffers_.size() >= kFreeLimit)
available_buffers_.erase(available_buffers_.begin());
// Creation of shared memory may be expensive if it involves synchronous IPC
// calls. That's why we try to avoid AllocateNewBuffer() as much as we can.
base::CheckedNumeric<size_t> requested_capacity(*capacity);
requested_capacity += kBufferPadding;
mojo::ScopedSharedBufferHandle handle =
mojo::SharedBufferHandle::Create(requested_capacity.ValueOrDie());
if (!handle.is_valid())
return handle;
*capacity = requested_capacity.ValueOrDie();
return handle;
}
void MojoCdmAllocator::AddBufferToAvailableMap(
mojo::ScopedSharedBufferHandle buffer,
size_t capacity) {
DCHECK(thread_checker_.CalledOnValidThread());
available_buffers_.insert(std::make_pair(capacity, std::move(buffer)));
}
MojoHandle MojoCdmAllocator::GetHandleForTesting(cdm::Buffer* buffer) {
MojoCdmBuffer* mojo_buffer = static_cast<MojoCdmBuffer*>(buffer);
return mojo_buffer->Handle().value();
}
size_t MojoCdmAllocator::GetAvailableBufferCountForTesting() {
return available_buffers_.size();
}
} // namespace media