This source file includes following definitions.
- enumeration_helper_
- Init
- OnResourceMessageReceived
- OnInitialized
- OnStarted
- OnStopped
- OnPaused
- OnError
- PostErrorReply
- OnRemoved
- OnFrameReady
- AllocBuffers
- OnOpen
- OnStartCapture
- OnReuseBuffer
- OnStopCapture
- OnClose
- StopCapture
- Close
- ReleaseBuffers
- SendStatus
- SetRequestedInfo
- DetachPlatformVideoCapture
- SetStatus
- buffer
#include "content/renderer/pepper/pepper_video_capture_host.h"
#include "content/renderer/pepper/host_globals.h"
#include "content/renderer/pepper/pepper_media_device_manager.h"
#include "content/renderer/pepper/pepper_platform_video_capture.h"
#include "content/renderer/pepper/pepper_plugin_instance_impl.h"
#include "content/renderer/pepper/renderer_ppapi_host_impl.h"
#include "content/renderer/render_view_impl.h"
#include "media/base/limits.h"
#include "media/base/video_frame.h"
#include "ppapi/host/dispatch_host_message.h"
#include "ppapi/host/ppapi_host.h"
#include "ppapi/proxy/host_dispatcher.h"
#include "ppapi/proxy/ppapi_messages.h"
#include "ppapi/shared_impl/host_resource.h"
#include "ppapi/thunk/enter.h"
#include "ppapi/thunk/ppb_buffer_api.h"
using ppapi::HostResource;
using ppapi::TrackedCallback;
using ppapi::thunk::EnterResourceNoLock;
using ppapi::thunk::PPB_Buffer_API;
namespace {
const uint32_t kMaxBuffers = 20;
}
namespace content {
PepperVideoCaptureHost::PepperVideoCaptureHost(RendererPpapiHostImpl* host,
PP_Instance instance,
PP_Resource resource)
: ResourceHost(host->GetPpapiHost(), instance, resource),
renderer_ppapi_host_(host),
buffer_count_hint_(0),
status_(PP_VIDEO_CAPTURE_STATUS_STOPPED),
enumeration_helper_(
this,
PepperMediaDeviceManager::GetForRenderView(
host->GetRenderViewForInstance(pp_instance())),
PP_DEVICETYPE_DEV_VIDEOCAPTURE,
host->GetDocumentURL(instance)) {
}
PepperVideoCaptureHost::~PepperVideoCaptureHost() {
Close();
}
bool PepperVideoCaptureHost::Init() {
return !!renderer_ppapi_host_->GetPluginInstance(pp_instance());
}
int32_t PepperVideoCaptureHost::OnResourceMessageReceived(
const IPC::Message& msg,
ppapi::host::HostMessageContext* context) {
int32_t result = PP_ERROR_FAILED;
if (enumeration_helper_.HandleResourceMessage(msg, context, &result))
return result;
IPC_BEGIN_MESSAGE_MAP(PepperVideoCaptureHost, msg)
PPAPI_DISPATCH_HOST_RESOURCE_CALL(
PpapiHostMsg_VideoCapture_Open,
OnOpen)
PPAPI_DISPATCH_HOST_RESOURCE_CALL_0(
PpapiHostMsg_VideoCapture_StartCapture,
OnStartCapture)
PPAPI_DISPATCH_HOST_RESOURCE_CALL(
PpapiHostMsg_VideoCapture_ReuseBuffer,
OnReuseBuffer)
PPAPI_DISPATCH_HOST_RESOURCE_CALL_0(
PpapiHostMsg_VideoCapture_StopCapture,
OnStopCapture)
PPAPI_DISPATCH_HOST_RESOURCE_CALL_0(
PpapiHostMsg_VideoCapture_Close,
OnClose)
IPC_END_MESSAGE_MAP()
return PP_ERROR_FAILED;
}
void PepperVideoCaptureHost::OnInitialized(media::VideoCapture* capture,
bool succeeded) {
DCHECK(capture == platform_video_capture_.get());
if (succeeded) {
open_reply_context_.params.set_result(PP_OK);
} else {
DetachPlatformVideoCapture();
open_reply_context_.params.set_result(PP_ERROR_FAILED);
}
host()->SendReply(open_reply_context_,
PpapiPluginMsg_VideoCapture_OpenReply());
}
void PepperVideoCaptureHost::OnStarted(media::VideoCapture* capture) {
if (SetStatus(PP_VIDEO_CAPTURE_STATUS_STARTED, false))
SendStatus();
}
void PepperVideoCaptureHost::OnStopped(media::VideoCapture* capture) {
if (SetStatus(PP_VIDEO_CAPTURE_STATUS_STOPPED, false))
SendStatus();
}
void PepperVideoCaptureHost::OnPaused(media::VideoCapture* capture) {
if (SetStatus(PP_VIDEO_CAPTURE_STATUS_PAUSED, false))
SendStatus();
}
void PepperVideoCaptureHost::OnError(media::VideoCapture* capture,
int error_code) {
DCHECK(error_code == 1);
PostErrorReply();
}
void PepperVideoCaptureHost::PostErrorReply() {
SetStatus(PP_VIDEO_CAPTURE_STATUS_STOPPED, true);
host()->SendUnsolicitedReply(pp_resource(),
PpapiPluginMsg_VideoCapture_OnError(PP_ERROR_FAILED));
}
void PepperVideoCaptureHost::OnRemoved(media::VideoCapture* capture) {
}
void PepperVideoCaptureHost::OnFrameReady(
media::VideoCapture* capture,
const scoped_refptr<media::VideoFrame>& frame) {
DCHECK(frame.get());
if (alloc_size_ != frame->coded_size()) {
AllocBuffers(frame->coded_size(), capture->CaptureFrameRate());
alloc_size_ = frame->coded_size();
}
for (uint32_t i = 0; i < buffers_.size(); ++i) {
if (!buffers_[i].in_use) {
DCHECK_EQ(frame->format(), media::VideoFrame::I420);
if (buffers_[i].buffer->size() <
media::VideoFrame::AllocationSize(frame->format(),
frame->coded_size())) {
return;
}
uint8* dst = reinterpret_cast<uint8*>(buffers_[i].data);
COMPILE_ASSERT(media::VideoFrame::kYPlane == 0, y_plane_should_be_0);
COMPILE_ASSERT(media::VideoFrame::kUPlane == 1, u_plane_should_be_1);
COMPILE_ASSERT(media::VideoFrame::kVPlane == 2, v_plane_should_be_2);
for (size_t j = 0; j < media::VideoFrame::NumPlanes(frame->format());
++j) {
const uint8* src = frame->data(j);
const size_t row_bytes = frame->row_bytes(j);
const size_t src_stride = frame->stride(j);
for (int k = 0; k < frame->rows(j); ++k) {
memcpy(dst, src, row_bytes);
dst += row_bytes;
src += src_stride;
}
}
buffers_[i].in_use = true;
host()->SendUnsolicitedReply(pp_resource(),
PpapiPluginMsg_VideoCapture_OnBufferReady(i));
return;
}
}
}
void PepperVideoCaptureHost::AllocBuffers(
const gfx::Size& resolution,
int frame_rate) {
PP_VideoCaptureDeviceInfo_Dev info = {
static_cast<uint32_t>(resolution.width()),
static_cast<uint32_t>(resolution.height()),
static_cast<uint32_t>(frame_rate)
};
ReleaseBuffers();
const size_t size = media::VideoFrame::AllocationSize(
media::VideoFrame::I420, gfx::Size(info.width, info.height));
ppapi::proxy::ResourceMessageReplyParams params(pp_resource(), 0);
std::vector<HostResource> buffer_host_resources;
buffers_.reserve(buffer_count_hint_);
ppapi::ResourceTracker* tracker =
HostGlobals::Get()->GetResourceTracker();
ppapi::proxy::HostDispatcher* dispatcher =
ppapi::proxy::HostDispatcher::GetForInstance(pp_instance());
for (size_t i = 0; i < buffer_count_hint_; ++i) {
PP_Resource res = PPB_Buffer_Impl::Create(pp_instance(), size);
if (!res)
break;
EnterResourceNoLock<PPB_Buffer_API> enter(res, true);
DCHECK(enter.succeeded());
BufferInfo buf;
buf.buffer = static_cast<PPB_Buffer_Impl*>(enter.object());
buf.data = buf.buffer->Map();
if (!buf.data) {
tracker->ReleaseResource(res);
break;
}
buffers_.push_back(buf);
{
HostResource host_resource;
host_resource.SetHostResource(pp_instance(), res);
buffer_host_resources.push_back(host_resource);
tracker->AddRefResource(res);
}
{
EnterResourceNoLock<PPB_Buffer_API> enter(res, true);
DCHECK(enter.succeeded());
int handle;
int32_t result = enter.object()->GetSharedMemory(&handle);
DCHECK(result == PP_OK);
base::PlatformFile platform_file =
#if defined(OS_WIN)
reinterpret_cast<HANDLE>(static_cast<intptr_t>(handle));
#elif defined(OS_POSIX)
handle;
#else
#error Not implemented.
#endif
params.AppendHandle(
ppapi::proxy::SerializedHandle(
dispatcher->ShareHandleWithRemote(platform_file, false),
size));
}
}
if (buffers_.empty()) {
SetStatus(PP_VIDEO_CAPTURE_STATUS_STOPPING, true);
platform_video_capture_->StopCapture(this);
PostErrorReply();
return;
}
host()->Send(new PpapiPluginMsg_ResourceReply(
params, PpapiPluginMsg_VideoCapture_OnDeviceInfo(
info, buffer_host_resources, size)));
}
int32_t PepperVideoCaptureHost::OnOpen(
ppapi::host::HostMessageContext* context,
const std::string& device_id,
const PP_VideoCaptureDeviceInfo_Dev& requested_info,
uint32_t buffer_count) {
if (platform_video_capture_.get())
return PP_ERROR_FAILED;
SetRequestedInfo(requested_info, buffer_count);
GURL document_url = renderer_ppapi_host_->GetDocumentURL(pp_instance());
if (!document_url.is_valid())
return PP_ERROR_FAILED;
RenderViewImpl* render_view = static_cast<RenderViewImpl*>(
renderer_ppapi_host_->GetRenderViewForInstance(pp_instance()));
platform_video_capture_ = new PepperPlatformVideoCapture(
render_view->AsWeakPtr(), device_id,
document_url, this);
open_reply_context_ = context->MakeReplyMessageContext();
return PP_OK_COMPLETIONPENDING;
}
int32_t PepperVideoCaptureHost::OnStartCapture(
ppapi::host::HostMessageContext* context) {
if (!SetStatus(PP_VIDEO_CAPTURE_STATUS_STARTING, false) ||
!platform_video_capture_.get())
return PP_ERROR_FAILED;
DCHECK(buffers_.empty());
platform_video_capture_->StartCapture(this, video_capture_params_);
return PP_OK;
}
int32_t PepperVideoCaptureHost::OnReuseBuffer(
ppapi::host::HostMessageContext* context,
uint32_t buffer) {
if (buffer >= buffers_.size() || !buffers_[buffer].in_use)
return PP_ERROR_BADARGUMENT;
buffers_[buffer].in_use = false;
return PP_OK;
}
int32_t PepperVideoCaptureHost::OnStopCapture(
ppapi::host::HostMessageContext* context) {
return StopCapture();
}
int32_t PepperVideoCaptureHost::OnClose(
ppapi::host::HostMessageContext* context) {
return Close();
}
int32_t PepperVideoCaptureHost::StopCapture() {
if (!SetStatus(PP_VIDEO_CAPTURE_STATUS_STOPPING, false))
return PP_ERROR_FAILED;
DCHECK(platform_video_capture_.get());
ReleaseBuffers();
platform_video_capture_->StopCapture(this);
return PP_OK;
}
int32_t PepperVideoCaptureHost::Close() {
if (!platform_video_capture_.get())
return PP_OK;
StopCapture();
DCHECK(buffers_.empty());
DetachPlatformVideoCapture();
return PP_OK;
}
void PepperVideoCaptureHost::ReleaseBuffers() {
ppapi::ResourceTracker* tracker = HostGlobals::Get()->GetResourceTracker();
for (size_t i = 0; i < buffers_.size(); ++i) {
buffers_[i].buffer->Unmap();
tracker->ReleaseResource(buffers_[i].buffer->pp_resource());
}
buffers_.clear();
}
void PepperVideoCaptureHost::SendStatus() {
host()->SendUnsolicitedReply(pp_resource(),
PpapiPluginMsg_VideoCapture_OnStatus(status_));
}
void PepperVideoCaptureHost::SetRequestedInfo(
const PP_VideoCaptureDeviceInfo_Dev& device_info,
uint32_t buffer_count) {
buffer_count_hint_ = std::min(std::max(buffer_count, 1U), kMaxBuffers);
int frames_per_second =
std::min(std::max(device_info.frames_per_second, 1U),
static_cast<uint32_t>(media::limits::kMaxFramesPerSecond - 1));
video_capture_params_.requested_format = media::VideoCaptureFormat(
gfx::Size(device_info.width, device_info.height),
frames_per_second,
media::PIXEL_FORMAT_I420);
video_capture_params_.allow_resolution_change = false;
}
void PepperVideoCaptureHost::DetachPlatformVideoCapture() {
if (platform_video_capture_.get()) {
platform_video_capture_->DetachEventHandler();
platform_video_capture_ = NULL;
}
}
bool PepperVideoCaptureHost::SetStatus(PP_VideoCaptureStatus_Dev status,
bool forced) {
if (!forced) {
switch (status) {
case PP_VIDEO_CAPTURE_STATUS_STOPPED:
if (status_ != PP_VIDEO_CAPTURE_STATUS_STOPPING)
return false;
break;
case PP_VIDEO_CAPTURE_STATUS_STARTING:
if (status_ != PP_VIDEO_CAPTURE_STATUS_STOPPED)
return false;
break;
case PP_VIDEO_CAPTURE_STATUS_STARTED:
switch (status_) {
case PP_VIDEO_CAPTURE_STATUS_STARTING:
case PP_VIDEO_CAPTURE_STATUS_PAUSED:
break;
default:
return false;
}
break;
case PP_VIDEO_CAPTURE_STATUS_PAUSED:
switch (status_) {
case PP_VIDEO_CAPTURE_STATUS_STARTING:
case PP_VIDEO_CAPTURE_STATUS_STARTED:
break;
default:
return false;
}
break;
case PP_VIDEO_CAPTURE_STATUS_STOPPING:
switch (status_) {
case PP_VIDEO_CAPTURE_STATUS_STARTING:
case PP_VIDEO_CAPTURE_STATUS_STARTED:
case PP_VIDEO_CAPTURE_STATUS_PAUSED:
break;
default:
return false;
}
break;
}
}
status_ = status;
return true;
}
PepperVideoCaptureHost::BufferInfo::BufferInfo()
: in_use(false),
data(NULL),
buffer() {
}
PepperVideoCaptureHost::BufferInfo::~BufferInfo() {
}
}