| Index: content/renderer/pepper/pepper_video_source_host.cc
|
| diff --git a/content/renderer/pepper/pepper_video_source_host.cc b/content/renderer/pepper/pepper_video_source_host.cc
|
| index 462c8aca537a2ba9e2749600bf9bad3514b63ee2..0852655d442e3f7b3f7fc2c69b12b8a678de5204 100644
|
| --- a/content/renderer/pepper/pepper_video_source_host.cc
|
| +++ b/content/renderer/pepper/pepper_video_source_host.cc
|
| @@ -26,6 +26,39 @@ using ppapi::host::ReplyMessageContext;
|
|
|
| namespace content {
|
|
|
| +PepperVideoSourceHost::FrameReceiver::FrameReceiver(
|
| + const base::WeakPtr<PepperVideoSourceHost>& host)
|
| + : host_(host),
|
| + main_message_loop_proxy_(base::MessageLoopProxy::current()) {
|
| +}
|
| +
|
| +PepperVideoSourceHost::FrameReceiver::~FrameReceiver() {
|
| +}
|
| +
|
| +bool PepperVideoSourceHost::FrameReceiver::GotFrame(
|
| + cricket::VideoFrame* frame) {
|
| + // It's not safe to access the host from this thread, so post a task to our
|
| + // main thread to transfer the new frame.
|
| + main_message_loop_proxy_->PostTask(
|
| + FROM_HERE,
|
| + base::Bind(&FrameReceiver::OnGotFrame,
|
| + this,
|
| + base::Passed(scoped_ptr<cricket::VideoFrame>(frame))));
|
| +
|
| + return true;
|
| +}
|
| +
|
| +void PepperVideoSourceHost::FrameReceiver::OnGotFrame(
|
| + scoped_ptr<cricket::VideoFrame> frame) {
|
| + if (host_) {
|
| + // Take ownership of the new frame, and possibly delete any unsent one.
|
| + host_->last_frame_.swap(frame);
|
| +
|
| + if (host_->get_frame_pending_)
|
| + host_->SendGetFrameReply();
|
| + }
|
| +}
|
| +
|
| PepperVideoSourceHost::PepperVideoSourceHost(
|
| RendererPpapiHost* host,
|
| PP_Instance instance,
|
| @@ -33,8 +66,8 @@ PepperVideoSourceHost::PepperVideoSourceHost(
|
| : ResourceHost(host->GetPpapiHost(), instance, resource),
|
| renderer_ppapi_host_(host),
|
| weak_factory_(this),
|
| - main_message_loop_proxy_(base::MessageLoopProxy::current()),
|
| source_handler_(new content::VideoSourceHandler(NULL)),
|
| + frame_receiver_(new FrameReceiver(weak_factory_.GetWeakPtr())),
|
| get_frame_pending_(false) {
|
| }
|
|
|
| @@ -56,37 +89,13 @@ int32_t PepperVideoSourceHost::OnResourceMessageReceived(
|
| return PP_ERROR_FAILED;
|
| }
|
|
|
| -bool PepperVideoSourceHost::GotFrame(cricket::VideoFrame* frame) {
|
| - // It's not safe to access this on another thread, so post a task to our
|
| - // main thread to transfer the new frame.
|
| - main_message_loop_proxy_->PostTask(
|
| - FROM_HERE,
|
| - base::Bind(&PepperVideoSourceHost::OnGotFrame,
|
| - weak_factory_.GetWeakPtr(),
|
| - base::Passed(scoped_ptr<cricket::VideoFrame>(frame))));
|
| -
|
| - return true;
|
| -}
|
| -
|
| -void PepperVideoSourceHost::OnGotFrame(scoped_ptr<cricket::VideoFrame> frame) {
|
| - // Take ownership of the new frame, and possibly delete any unsent one.
|
| - last_frame_.swap(frame);
|
| -
|
| - if (get_frame_pending_) {
|
| - ppapi::HostResource image_data_resource;
|
| - PP_TimeTicks timestamp = 0;
|
| - int32_t result = ConvertFrame(&image_data_resource, ×tamp);
|
| - SendFrame(image_data_resource, timestamp, result);
|
| - }
|
| -}
|
| -
|
| int32_t PepperVideoSourceHost::OnHostMsgOpen(HostMessageContext* context,
|
| const std::string& stream_url) {
|
| GURL gurl(stream_url);
|
| if (!gurl.is_valid())
|
| return PP_ERROR_BADARGUMENT;
|
|
|
| - if (!source_handler_->Open(gurl.spec(), this))
|
| + if (!source_handler_->Open(gurl.spec(), frame_receiver_.get()))
|
| return PP_ERROR_BADARGUMENT;
|
|
|
| stream_url_ = gurl.spec();
|
| @@ -107,18 +116,9 @@ int32_t PepperVideoSourceHost::OnHostMsgGetFrame(
|
| reply_context_ = context->MakeReplyMessageContext();
|
| get_frame_pending_ = true;
|
|
|
| - // If a frame is ready, try to convert it and reply.
|
| - if (last_frame_.get()) {
|
| - ppapi::HostResource image_data_resource;
|
| - PP_TimeTicks timestamp = 0;
|
| - int32_t result = ConvertFrame(&image_data_resource, ×tamp);
|
| - if (result == PP_OK) {
|
| - SendFrame(image_data_resource, timestamp, result);
|
| - } else {
|
| - reply_context_ = ppapi::host::ReplyMessageContext();
|
| - return result;
|
| - }
|
| - }
|
| + // If a frame is ready, try to convert it and send the reply.
|
| + if (last_frame_.get())
|
| + SendGetFrameReply();
|
|
|
| return PP_OK_COMPLETIONPENDING;
|
| }
|
| @@ -128,16 +128,17 @@ int32_t PepperVideoSourceHost::OnHostMsgClose(HostMessageContext* context) {
|
| return PP_OK;
|
| }
|
|
|
| -int32_t PepperVideoSourceHost::ConvertFrame(
|
| - ppapi::HostResource* image_data_resource,
|
| - PP_TimeTicks* timestamp) {
|
| +void PepperVideoSourceHost::SendGetFrameReply() {
|
| + DCHECK(get_frame_pending_);
|
| + get_frame_pending_ = false;
|
| +
|
| DCHECK(last_frame_.get());
|
| scoped_ptr<cricket::VideoFrame> frame(last_frame_.release());
|
|
|
| int32_t width = base::checked_numeric_cast<int32_t>(frame->GetWidth());
|
| int32_t height = base::checked_numeric_cast<int32_t>(frame->GetHeight());
|
| // Create an image data resource to hold the frame pixels.
|
| - PP_ImageDataDesc desc;
|
| + PP_ImageDataDesc image_desc;
|
| IPC::PlatformFileForTransit image_handle;
|
| uint32_t byte_count;
|
| ppapi::ScopedPPResource resource(
|
| @@ -148,27 +149,37 @@ int32_t PepperVideoSourceHost::ConvertFrame(
|
| PP_MakeSize(width, height),
|
| false /* init_to_zero */,
|
| false /* is_nacl_plugin */,
|
| - &desc, &image_handle, &byte_count));
|
| - if (!resource.get())
|
| - return PP_ERROR_FAILED;
|
| + &image_desc, &image_handle, &byte_count));
|
| + if (!resource.get()) {
|
| + SendGetFrameErrorReply(PP_ERROR_FAILED);
|
| + return;
|
| + }
|
|
|
| ppapi::thunk::EnterResourceNoLock<ppapi::thunk::PPB_ImageData_API>
|
| enter_resource(resource, false);
|
| - if (enter_resource.failed())
|
| - return PP_ERROR_FAILED;
|
| + if (enter_resource.failed()) {
|
| + SendGetFrameErrorReply(PP_ERROR_FAILED);
|
| + return;
|
| + }
|
|
|
| webkit::ppapi::PPB_ImageData_Impl* image_data =
|
| static_cast<webkit::ppapi::PPB_ImageData_Impl*>(enter_resource.object());
|
| webkit::ppapi::ImageDataAutoMapper mapper(image_data);
|
| - if (!mapper.is_valid())
|
| - return PP_ERROR_FAILED;
|
| + if (!mapper.is_valid()) {
|
| + SendGetFrameErrorReply(PP_ERROR_FAILED);
|
| + return;
|
| + }
|
|
|
| const SkBitmap* bitmap = image_data->GetMappedBitmap();
|
| - if (!bitmap)
|
| - return PP_ERROR_FAILED;
|
| + if (!bitmap) {
|
| + SendGetFrameErrorReply(PP_ERROR_FAILED);
|
| + return;
|
| + }
|
| uint8_t* bitmap_pixels = static_cast<uint8_t*>(bitmap->getPixels());
|
| - if (!bitmap_pixels)
|
| - return PP_ERROR_FAILED;
|
| + if (!bitmap_pixels) {
|
| + SendGetFrameErrorReply(PP_ERROR_FAILED);
|
| + return;
|
| + }
|
|
|
| size_t bitmap_size = bitmap->getSize();
|
| frame->ConvertToRgbBuffer(cricket::FOURCC_BGRA,
|
| @@ -176,36 +187,68 @@ int32_t PepperVideoSourceHost::ConvertFrame(
|
| bitmap_size,
|
| bitmap->rowBytes());
|
|
|
| - image_data_resource->SetHostResource(pp_instance(), resource.get());
|
| + ppapi::HostResource host_resource;
|
| + host_resource.SetHostResource(pp_instance(), resource.get());
|
|
|
| // Convert a video timestamp (int64, in nanoseconds) to a time delta (int64,
|
| // microseconds) and then to a PP_TimeTicks (a double, in seconds). All times
|
| // are relative to the Unix Epoch.
|
| base::TimeDelta time_delta = base::TimeDelta::FromMicroseconds(
|
| frame->GetTimeStamp() / base::Time::kNanosecondsPerMicrosecond);
|
| - *timestamp = time_delta.InSecondsF();
|
| - return PP_OK;
|
| + PP_TimeTicks timestamp = time_delta.InSecondsF();
|
| +
|
| + reply_context_.params.set_result(PP_OK);
|
| +
|
| + // TODO(bbudge) Change the PDF Host's image creation code to match.
|
| +#if defined(OS_WIN) || defined(OS_MACOSX) || defined(OS_ANDROID)
|
| + ppapi::proxy::SerializedHandle serialized_handle;
|
| + PpapiPluginMsg_VideoSource_GetFrameReply reply_msg(host_resource,
|
| + image_desc,
|
| + 0,
|
| + timestamp);
|
| + serialized_handle.set_shmem(image_handle, byte_count);
|
| + reply_context_.params.AppendHandle(serialized_handle);
|
| +#elif defined(OS_LINUX)
|
| + // For Linux, we pass the SysV shared memory key in the message.
|
| + PpapiPluginMsg_VideoSource_GetFrameReply reply_msg(host_resource,
|
| + image_desc,
|
| + image_handle.fd,
|
| + timestamp);
|
| +#else
|
| + // Not supported on other platforms.
|
| + // This is a stub reply_msg to not break the build.
|
| + PpapiPluginMsg_VideoSource_GetFrameReply reply_msg(host_resource,
|
| + image_desc,
|
| + 0,
|
| + timestamp);
|
| + NOTIMPLEMENTED();
|
| + SendGetFrameErrorReply(PP_ERROR_NOTSUPPORTED);
|
| + return;
|
| +#endif
|
| +
|
| + host()->SendReply(reply_context_, reply_msg);
|
| +
|
| + reply_context_ = ppapi::host::ReplyMessageContext();
|
| +
|
| + // Keep a reference once we know this method succeeds.
|
| + resource.Release();
|
| }
|
|
|
| -void PepperVideoSourceHost::SendFrame(
|
| - const ppapi::HostResource& image_data_resource,
|
| - PP_TimeTicks timestamp,
|
| - int32_t result) {
|
| - DCHECK(get_frame_pending_);
|
| - reply_context_.params.set_result(result);
|
| +void PepperVideoSourceHost::SendGetFrameErrorReply(int32_t error) {
|
| + reply_context_.params.set_result(error);
|
| host()->SendReply(
|
| reply_context_,
|
| - PpapiPluginMsg_VideoSource_GetFrameReply(image_data_resource, timestamp));
|
| -
|
| + PpapiPluginMsg_VideoSource_GetFrameReply(
|
| + ppapi::HostResource(), PP_ImageDataDesc(), -1, 0.0));
|
| reply_context_ = ppapi::host::ReplyMessageContext();
|
| - get_frame_pending_ = false;
|
| }
|
|
|
| void PepperVideoSourceHost::Close() {
|
| - if (source_handler_.get()) {
|
| - source_handler_->Close(stream_url_, this);
|
| - source_handler_.reset(NULL);
|
| - }
|
| + if (source_handler_.get() && !stream_url_.empty())
|
| + source_handler_->Close(stream_url_, frame_receiver_.get());
|
| +
|
| + source_handler_.reset(NULL);
|
| + stream_url_.clear();
|
| }
|
|
|
| } // namespace content
|
|
|