Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(2527)

Unified Diff: content/renderer/pepper/pepper_audio_output_host.cc

Issue 2755613002: Support audio output device enumeration and selection in PPAPI (Closed)
Patch Set: Fix format, Rebase Created 3 years, 8 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: content/renderer/pepper/pepper_audio_output_host.cc
diff --git a/content/renderer/pepper/pepper_audio_output_host.cc b/content/renderer/pepper/pepper_audio_output_host.cc
new file mode 100644
index 0000000000000000000000000000000000000000..6f264b97666e1ae2825f076782c4bcec112ce23a
--- /dev/null
+++ b/content/renderer/pepper/pepper_audio_output_host.cc
@@ -0,0 +1,265 @@
+// Copyright (c) 2017 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 "content/renderer/pepper/pepper_audio_output_host.h"
+
+#include "base/logging.h"
+#include "build/build_config.h"
+#include "content/common/pepper_file_util.h"
+#include "content/renderer/pepper/pepper_audio_controller.h"
+#include "content/renderer/pepper/pepper_media_device_manager.h"
+#include "content/renderer/pepper/pepper_platform_audio_output_dev.h"
+#include "content/renderer/pepper/pepper_plugin_instance_impl.h"
+#include "content/renderer/pepper/plugin_instance_throttler_impl.h"
+#include "content/renderer/pepper/renderer_ppapi_host_impl.h"
+#include "content/renderer/render_frame_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 {
+
+namespace {
+
+base::PlatformFile ConvertSyncSocketHandle(const base::SyncSocket& socket) {
+ return socket.handle();
+}
+
+} // namespace
+
+PepperAudioOutputHost::PepperAudioOutputHost(RendererPpapiHostImpl* host,
+ PP_Instance instance,
+ PP_Resource resource)
+ : ResourceHost(host->GetPpapiHost(), instance, resource),
+ renderer_ppapi_host_(host),
+ audio_output_(NULL),
+ playback_throttled_(false),
+ enumeration_helper_(this,
+ PepperMediaDeviceManager::GetForRenderFrame(
+ host->GetRenderFrameForInstance(pp_instance())),
+ PP_DEVICETYPE_DEV_AUDIOOUTPUT,
+ host->GetDocumentURL(instance)) {
+ PepperPluginInstanceImpl* plugin_instance =
+ static_cast<PepperPluginInstanceImpl*>(
+ PepperPluginInstance::Get(pp_instance()));
+ if (plugin_instance && plugin_instance->throttler())
+ plugin_instance->throttler()->AddObserver(this);
+}
+
+PepperAudioOutputHost::~PepperAudioOutputHost() {
+ PepperPluginInstanceImpl* instance = static_cast<PepperPluginInstanceImpl*>(
+ PepperPluginInstance::Get(pp_instance()));
+ if (instance) {
+ if (instance->throttler()) {
+ instance->throttler()->RemoveObserver(this);
+ }
+ instance->audio_controller().RemoveInstance(this);
+ }
+ Close();
+}
+
+int32_t PepperAudioOutputHost::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(PepperAudioOutputHost, msg)
+ PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_AudioOutput_Open, OnOpen)
+ PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_AudioOutput_StartOrStop,
+ OnStartOrStop)
+ PPAPI_DISPATCH_HOST_RESOURCE_CALL_0(PpapiHostMsg_AudioOutput_Close, OnClose)
+ PPAPI_END_MESSAGE_MAP()
+ return PP_ERROR_FAILED;
+}
+
+void PepperAudioOutputHost::StreamCreated(
+ base::SharedMemoryHandle shared_memory_handle,
+ size_t shared_memory_size,
+ base::SyncSocket::Handle socket) {
+ OnOpenComplete(PP_OK, shared_memory_handle, shared_memory_size, socket);
+}
+
+void PepperAudioOutputHost::StreamCreationFailed() {
+ OnOpenComplete(PP_ERROR_FAILED, base::SharedMemory::NULLHandle(), 0,
+ base::SyncSocket::kInvalidHandle);
+}
+
+void PepperAudioOutputHost::SetVolume(double volume) {
+ if (audio_output_)
+ audio_output_->SetVolume(volume);
+}
+
+int32_t PepperAudioOutputHost::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_output_)
+ 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_output_ = PepperPlatformAudioOutputDev::Create(
+ renderer_ppapi_host_->GetRenderFrameForInstance(pp_instance())
+ ->GetRoutingID(),
+ device_id, document_url, static_cast<int>(sample_rate),
+ static_cast<int>(sample_frame_count), this);
+ if (audio_output_) {
+ open_context_ = context->MakeReplyMessageContext();
+ return PP_OK_COMPLETIONPENDING;
+ } else {
+ return PP_ERROR_FAILED;
+ }
+}
+
+int32_t PepperAudioOutputHost::OnStartOrStop(
+ ppapi::host::HostMessageContext* /* context */,
+ bool playback) {
+ if (!audio_output_)
+ return PP_ERROR_FAILED;
+
+ PepperPluginInstanceImpl* instance = static_cast<PepperPluginInstanceImpl*>(
+ PepperPluginInstance::Get(pp_instance()));
+
+ if (playback) {
+ // If plugin is in power saver mode, defer audio IPC communication.
+ if (instance && instance->throttler() &&
+ instance->throttler()->power_saver_enabled()) {
+ instance->throttler()->NotifyAudioThrottled();
+ playback_throttled_ = true;
+ return PP_TRUE;
+ }
+ if (instance)
+ instance->audio_controller().AddInstance(this);
+
+ audio_output_->StartPlayback();
+ } else {
+ if (instance)
+ instance->audio_controller().RemoveInstance(this);
+
+ audio_output_->StopPlayback();
+ }
+ return PP_OK;
+}
+
+int32_t PepperAudioOutputHost::OnClose(
+ ppapi::host::HostMessageContext* /* context */) {
+ Close();
+ return PP_OK;
+}
+
+void PepperAudioOutputHost::OnOpenComplete(
+ int32_t result,
+ base::SharedMemoryHandle shared_memory_handle,
+ size_t shared_memory_size,
+ base::SyncSocket::Handle socket_handle) {
+ // Make sure the handles are cleaned up.
+ base::SyncSocket scoped_socket(socket_handle);
+ base::SharedMemory scoped_shared_memory(shared_memory_handle, false);
+
+ if (!open_context_.is_valid()) {
+ NOTREACHED();
+ return;
+ }
+
+ ppapi::proxy::SerializedHandle serialized_socket_handle(
+ ppapi::proxy::SerializedHandle::SOCKET);
+ ppapi::proxy::SerializedHandle serialized_shared_memory_handle(
+ ppapi::proxy::SerializedHandle::SHARED_MEMORY);
+
+ if (result == PP_OK) {
+ IPC::PlatformFileForTransit temp_socket =
+ IPC::InvalidPlatformFileForTransit();
+ base::SharedMemoryHandle temp_shmem = base::SharedMemory::NULLHandle();
+ result = GetRemoteHandles(scoped_socket, scoped_shared_memory, &temp_socket,
+ &temp_shmem);
+
+ serialized_socket_handle.set_socket(temp_socket);
+ serialized_shared_memory_handle.set_shmem(temp_shmem, shared_memory_size);
+ }
+
+ // 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(serialized_socket_handle);
+ open_context_.params.AppendHandle(serialized_shared_memory_handle);
+ SendOpenReply(result);
+}
+
+int32_t PepperAudioOutputHost::GetRemoteHandles(
+ const base::SyncSocket& socket,
+ const base::SharedMemory& shared_memory,
+ IPC::PlatformFileForTransit* remote_socket_handle,
+ base::SharedMemoryHandle* remote_shared_memory_handle) {
+ *remote_socket_handle = renderer_ppapi_host_->ShareHandleWithRemote(
+ ConvertSyncSocketHandle(socket), false);
+ if (*remote_socket_handle == IPC::InvalidPlatformFileForTransit())
+ return PP_ERROR_FAILED;
+
+ *remote_shared_memory_handle =
+ renderer_ppapi_host_->ShareSharedMemoryHandleWithRemote(
+ shared_memory.handle());
+ if (!base::SharedMemory::IsHandleValid(*remote_shared_memory_handle))
+ return PP_ERROR_FAILED;
+
+ return PP_OK;
+}
+
+void PepperAudioOutputHost::Close() {
+ if (!audio_output_)
+ return;
+
+ audio_output_->ShutDown();
+ audio_output_ = NULL;
+
+ if (open_context_.is_valid())
+ SendOpenReply(PP_ERROR_ABORTED);
+}
+
+void PepperAudioOutputHost::SendOpenReply(int32_t result) {
+ open_context_.params.set_result(result);
+ host()->SendReply(open_context_, PpapiPluginMsg_AudioOutput_OpenReply());
+ open_context_ = ppapi::host::ReplyMessageContext();
+}
+
+void PepperAudioOutputHost::OnThrottleStateChange() {
+ PepperPluginInstanceImpl* instance = static_cast<PepperPluginInstanceImpl*>(
+ PepperPluginInstance::Get(pp_instance()));
+ if (playback_throttled_ && instance && instance->throttler() &&
+ !instance->throttler()->power_saver_enabled()) {
+ // If we have become unthrottled, and we have a pending playback,
+ // start it.
+ StartDeferredPlayback();
+ }
+}
+
+void PepperAudioOutputHost::StartDeferredPlayback() {
+ if (!audio_output_)
+ return;
+
+ DCHECK(playback_throttled_);
+ playback_throttled_ = false;
+
+ PepperPluginInstanceImpl* instance = static_cast<PepperPluginInstanceImpl*>(
+ PepperPluginInstance::Get(pp_instance()));
+ if (instance)
+ instance->audio_controller().AddInstance(this);
+
+ audio_output_->StartPlayback();
+}
+
+} // namespace content
« no previous file with comments | « content/renderer/pepper/pepper_audio_output_host.h ('k') | content/renderer/pepper/pepper_media_device_manager.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698