// 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/pepper_audio_input_host.h"
#include "base/notreached.h"
#include "build/build_config.h"
#include "content/common/pepper_file_util.h"
#include "content/public/renderer/render_frame.h"
#include "content/renderer/pepper/pepper_media_device_manager.h"
#include "content/renderer/pepper/pepper_platform_audio_input.h"
#include "content/renderer/pepper/pepper_plugin_instance_impl.h"
#include "content/renderer/pepper/renderer_ppapi_host_impl.h"
#include "ipc/ipc_message.h"
#include "ppapi/c/pp_errors.h"
#include "ppapi/host/dispatch_host_message.h"
#include "ppapi/host/ppapi_host.h"
#include "ppapi/proxy/ppapi_messages.h"
#include "ppapi/proxy/serialized_structs.h"
namespace content {
PepperAudioInputHost::PepperAudioInputHost(RendererPpapiHostImpl* host,
PP_Instance instance,
PP_Resource resource)
: ResourceHost(host->GetPpapiHost(), instance, resource),
renderer_ppapi_host_(host),
audio_input_(nullptr),
enumeration_helper_(this,
PepperMediaDeviceManager::GetForRenderFrame(
host->GetRenderFrameForInstance(pp_instance())),
PP_DEVICETYPE_DEV_AUDIOCAPTURE,
host->GetDocumentURL(instance)) {}
PepperAudioInputHost::~PepperAudioInputHost() { Close(); }
int32_t PepperAudioInputHost::OnResourceMessageReceived(
const IPC::Message& msg,
ppapi::host::HostMessageContext* context) {
int32_t result = PP_ERROR_FAILED;
if (enumeration_helper_.HandleResourceMessage(msg, context, &result))
return result;
PPAPI_BEGIN_MESSAGE_MAP(PepperAudioInputHost, msg)
PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_AudioInput_Open, OnOpen)
PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_AudioInput_StartOrStop,
OnStartOrStop)
PPAPI_DISPATCH_HOST_RESOURCE_CALL_0(PpapiHostMsg_AudioInput_Close, OnClose)
PPAPI_END_MESSAGE_MAP()
return PP_ERROR_FAILED;
}
void PepperAudioInputHost::StreamCreated(
base::ReadOnlySharedMemoryRegion shared_memory_region,
base::SyncSocket::ScopedHandle socket) {
OnOpenComplete(PP_OK, std::move(shared_memory_region), std::move(socket));
}
void PepperAudioInputHost::StreamCreationFailed() {
OnOpenComplete(PP_ERROR_FAILED, base::ReadOnlySharedMemoryRegion(),
base::SyncSocket::ScopedHandle());
}
int32_t PepperAudioInputHost::OnOpen(ppapi::host::HostMessageContext* context,
const std::string& device_id,
PP_AudioSampleRate sample_rate,
uint32_t sample_frame_count) {
if (open_context_.is_valid())
return PP_ERROR_INPROGRESS;
if (audio_input_)
return PP_ERROR_FAILED;
GURL document_url = renderer_ppapi_host_->GetDocumentURL(pp_instance());
if (!document_url.is_valid())
return PP_ERROR_FAILED;
// When it is done, we'll get called back on StreamCreated() or
// StreamCreationFailed().
audio_input_ = PepperPlatformAudioInput::Create(
renderer_ppapi_host_->GetRenderFrameForInstance(pp_instance())->
GetRoutingID(),
device_id,
static_cast<int>(sample_rate),
static_cast<int>(sample_frame_count),
this);
if (audio_input_) {
open_context_ = context->MakeReplyMessageContext();
return PP_OK_COMPLETIONPENDING;
} else {
return PP_ERROR_FAILED;
}
}
int32_t PepperAudioInputHost::OnStartOrStop(
ppapi::host::HostMessageContext* /* context */,
bool capture) {
if (!audio_input_)
return PP_ERROR_FAILED;
if (capture)
audio_input_->StartCapture();
else
audio_input_->StopCapture();
return PP_OK;
}
int32_t PepperAudioInputHost::OnClose(
ppapi::host::HostMessageContext* /* context */) {
Close();
return PP_OK;
}
void PepperAudioInputHost::OnOpenComplete(
int32_t result,
base::ReadOnlySharedMemoryRegion shared_memory_region,
base::SyncSocket::ScopedHandle socket_handle) {
// Make sure the handles are cleaned up.
base::SyncSocket scoped_socket(std::move(socket_handle));
if (!open_context_.is_valid()) {
NOTREACHED_IN_MIGRATION();
return;
}
ppapi::proxy::SerializedHandle serialized_socket_handle(
ppapi::proxy::SerializedHandle::SOCKET);
ppapi::proxy::SerializedHandle serialized_shared_memory_handle(
ppapi::proxy::SerializedHandle::SHARED_MEMORY_REGION);
if (result == PP_OK) {
IPC::PlatformFileForTransit temp_socket =
IPC::InvalidPlatformFileForTransit();
base::ReadOnlySharedMemoryRegion temp_shmem;
result = GetRemoteHandles(scoped_socket, shared_memory_region, &temp_socket,
&temp_shmem);
serialized_socket_handle.set_socket(temp_socket);
serialized_shared_memory_handle.set_shmem_region(
base::ReadOnlySharedMemoryRegion::TakeHandleForSerialization(
std::move(temp_shmem)));
}
// Send all the values, even on error. This simplifies some of our cleanup
// code since the handles will be in the other process and could be
// inconvenient to clean up. Our IPC code will automatically handle this for
// us, as long as the remote side always closes the handles it receives, even
// in the failure case.
open_context_.params.AppendHandle(std::move(serialized_socket_handle));
open_context_.params.AppendHandle(std::move(serialized_shared_memory_handle));
SendOpenReply(result);
}
int32_t PepperAudioInputHost::GetRemoteHandles(
const base::SyncSocket& socket,
const base::ReadOnlySharedMemoryRegion& shared_memory_region,
IPC::PlatformFileForTransit* remote_socket_handle,
base::ReadOnlySharedMemoryRegion* remote_shared_memory_region) {
*remote_socket_handle =
renderer_ppapi_host_->ShareHandleWithRemote(socket.handle(), false);
if (*remote_socket_handle == IPC::InvalidPlatformFileForTransit())
return PP_ERROR_FAILED;
*remote_shared_memory_region =
renderer_ppapi_host_->ShareReadOnlySharedMemoryRegionWithRemote(
shared_memory_region);
if (!remote_shared_memory_region->IsValid())
return PP_ERROR_FAILED;
return PP_OK;
}
void PepperAudioInputHost::Close() {
if (!audio_input_)
return;
audio_input_->ShutDown();
audio_input_ = nullptr;
if (open_context_.is_valid())
SendOpenReply(PP_ERROR_ABORTED);
}
void PepperAudioInputHost::SendOpenReply(int32_t result) {
open_context_.params.set_result(result);
host()->SendReply(open_context_, PpapiPluginMsg_AudioInput_OpenReply());
open_context_ = ppapi::host::ReplyMessageContext();
}
} // namespace content