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

Unified Diff: content/renderer/media/audio_renderer_sink_cache_impl.h

Issue 1942803002: Caching AudioOutputDevice instances in mixer manager (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: First round of comments addressed Created 4 years, 7 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/media/audio_renderer_sink_cache_impl.h
diff --git a/content/renderer/media/audio_renderer_mixer_manager.h b/content/renderer/media/audio_renderer_sink_cache_impl.h
similarity index 22%
copy from content/renderer/media/audio_renderer_mixer_manager.h
copy to content/renderer/media/audio_renderer_sink_cache_impl.h
index 20a1aa59eaf3db3d8e65c0ef805dfe98c16e5e9e..1c6eabf139346925f86968d01b98c06de227a1eb 100644
--- a/content/renderer/media/audio_renderer_mixer_manager.h
+++ b/content/renderer/media/audio_renderer_sink_cache_impl.h
@@ -1,110 +1,73 @@
-// Copyright (c) 2012 The Chromium Authors. All rights reserved.
+// 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.
-#ifndef CONTENT_RENDERER_MEDIA_AUDIO_RENDERER_MIXER_MANAGER_H_
-#define CONTENT_RENDERER_MEDIA_AUDIO_RENDERER_MIXER_MANAGER_H_
+#include "content/renderer/media/audio_renderer_sink_cache.h"
#include <map>
-#include <string>
-#include <utility>
-#include "base/macros.h"
+#include "base/single_thread_task_runner.h"
#include "base/synchronization/lock.h"
#include "content/common/content_export.h"
#include "media/audio/audio_device_description.h"
-#include "media/base/audio_parameters.h"
-#include "media/base/output_device_info.h"
-#include "url/origin.h"
-
-namespace media {
-class AudioHardwareConfig;
-class AudioRendererMixer;
-class AudioRendererMixerInput;
-class AudioRendererSink;
-}
+#include "media/base/audio_renderer_sink.h"
namespace content {
-// Manages sharing of an AudioRendererMixer among AudioRendererMixerInputs based
-// on their AudioParameters configuration. Inputs with the same AudioParameters
-// configuration will share a mixer while a new AudioRendererMixer will be
-// lazily created if one with the exact AudioParameters does not exist.
-//
-// There should only be one instance of AudioRendererMixerManager per render
-// thread.
-//
-// TODO(dalecurtis): Right now we require AudioParameters to be an exact match
-// when we should be able to ignore bits per channel since we're only dealing
-// with floats. However, bits per channel is currently used to interleave the
-// audio data by AudioOutputDevice::AudioThreadCallback::Process for consumption
-// via the shared memory. See http://crbug.com/114700.
-class CONTENT_EXPORT AudioRendererMixerManager {
+// AudioRendererSinkCache implementation.
+class CONTENT_EXPORT AudioRendererSinkCacheImpl
+ : public AudioRendererSinkCache {
public:
- AudioRendererMixerManager();
- ~AudioRendererMixerManager();
-
- // Creates an AudioRendererMixerInput with the proper callbacks necessary to
- // retrieve an AudioRendererMixer instance from AudioRendererMixerManager.
- // |source_render_frame_id| refers to the RenderFrame containing the entity
- // rendering the audio. Caller must ensure AudioRendererMixerManager outlives
- // the returned input. |device_id|, |session_id| and |security_origin|
- // identify the output device to use. If |device_id| is empty and |session_id|
- // is nonzero, output device associated with the opened input device
- // designated by |session_id| is used. Otherwise, |session_id| is ignored.
- media::AudioRendererMixerInput* CreateInput(
+ // Callback to be used for AudioRendererSink creation
+ using CreateSinkCallback =
+ base::Callback<scoped_refptr<media::AudioRendererSink>(
+ int render_frame_id,
+ int session_id,
+ const std::string& device_id,
+ const url::Origin& security_origin)>;
+
+ AudioRendererSinkCacheImpl(
+ scoped_refptr<base::SingleThreadTaskRunner> task_runner,
+ const CreateSinkCallback& create_sink_callback,
+ int delete_timeout_ms);
+
+ ~AudioRendererSinkCacheImpl() final;
+
+ media::OutputDeviceInfo GetSinkInfo(int source_render_frame_id,
+ int session_id,
+ const std::string& device_id,
+ const url::Origin& security_origin) final;
+
+ scoped_refptr<media::AudioRendererSink> GetSink(
int source_render_frame_id,
- int session_id,
const std::string& device_id,
- const url::Origin& security_origin);
+ const url::Origin& security_origin) final;
- // Returns a mixer instance based on AudioParameters; an existing one if one
- // with the provided AudioParameters exists or a new one if not.
- media::AudioRendererMixer* GetMixer(int source_render_frame_id,
- const media::AudioParameters& params,
- const std::string& device_id,
- const url::Origin& security_origin,
- media::OutputDeviceStatus* device_status);
-
- // Remove a mixer instance given a mixer if the only other reference is held
- // by AudioRendererMixerManager. Every AudioRendererMixer owner must call
- // this method when it's done with a mixer.
- void RemoveMixer(int source_render_frame_id,
- const media::AudioParameters& params,
+ void ReleaseSink(int source_render_frame_id,
const std::string& device_id,
- const url::Origin& security_origin);
+ const url::Origin& security_origin,
+ const media::AudioRendererSink* sink) final;
private:
- friend class AudioRendererMixerManagerTest;
-
- // Define a key so that only those AudioRendererMixerInputs from the same
- // RenderView, AudioParameters and output device can be mixed together.
- struct MixerKey {
- MixerKey(int source_render_frame_id,
- const media::AudioParameters& params,
- const std::string& device_id,
- const url::Origin& security_origin);
- MixerKey(const MixerKey& other);
+ friend class AudioRendererSinkCacheTest;
+
+ // The key to be used to access a cached sink (not unique,there can be
+ // multiple sinks per device.
+ struct SinkKey {
+ SinkKey(int source_render_frame_id,
+ const std::string& device_id,
+ const url::Origin& security_origin);
+ SinkKey(const SinkKey& other);
int source_render_frame_id;
- media::AudioParameters params;
std::string device_id;
url::Origin security_origin;
};
- // Custom compare operator for the AudioRendererMixerMap. Allows reuse of
- // mixers where only irrelevant keys mismatch; e.g., effects, bits per sample.
- struct MixerKeyCompare {
- bool operator()(const MixerKey& a, const MixerKey& b) const {
+ // Strict weak ordering of the keys.
+ struct SinkKeyCompare {
+ bool operator()(const SinkKey& a, const SinkKey& b) const {
if (a.source_render_frame_id != b.source_render_frame_id)
return a.source_render_frame_id < b.source_render_frame_id;
- if (a.params.channels() != b.params.channels())
- return a.params.channels() < b.params.channels();
-
- // Ignore effects(), bits_per_sample(), format(), and frames_per_buffer(),
- // these parameters do not affect mixer reuse. All AudioRendererMixer
- // units disable FIFO, so frames_per_buffer() can be safely ignored.
- if (a.params.channel_layout() != b.params.channel_layout())
- return a.params.channel_layout() < b.params.channel_layout();
if (media::AudioDeviceDescription::IsDefaultDevice(a.device_id) &&
media::AudioDeviceDescription::IsDefaultDevice(b.device_id)) {
@@ -121,23 +84,58 @@ class CONTENT_EXPORT AudioRendererMixerManager {
}
};
- // Map of MixerKey to <AudioRendererMixer, Count>. Count allows
- // AudioRendererMixerManager to keep track explicitly (v.s. RefCounted which
- // is implicit) of the number of outstanding AudioRendererMixers.
- struct AudioRendererMixerReference {
- media::AudioRendererMixer* mixer;
- int ref_count;
+ // Cached sink data.
+ struct AudioRendererSinkReference {
+ AudioRendererSinkReference(scoped_refptr<media::AudioRendererSink> sink,
+ bool used);
+ AudioRendererSinkReference(const AudioRendererSinkReference& other);
+ ~AudioRendererSinkReference();
+ scoped_refptr<media::AudioRendererSink> sink; // Sink instance
+ bool used; // True if in used by a client.
};
- typedef std::map<MixerKey, AudioRendererMixerReference, MixerKeyCompare>
- AudioRendererMixerMap;
- // Active mixers.
- AudioRendererMixerMap mixers_;
- base::Lock mixers_lock_;
+ using AudioRendererSinkMap =
+ std::multimap<SinkKey, AudioRendererSinkReference, SinkKeyCompare>;
+
+ // Schedules a sink for deletion. Deletion will be performed on the same
+ // thread the cache is created on.
+ void DeleteLaterIfUnused(const SinkKey& key,
+ const media::AudioRendererSink* sink);
+
+ // Deletes a sink from the cache. If |force_delete_used| is set, a sink being
+ // deleted can (and should) be in use at the moment of deletion; otherwise the
+ // sink is deleted only if unused.
+ void DeleteSink(const SinkKey& key,
+ const media::AudioRendererSink* sink,
+ bool force_delete_used);
- DISALLOW_COPY_AND_ASSIGN(AudioRendererMixerManager);
+ // Creates a sink with a specified |key| and |used| state and inserts it into
+ // the cache.
+ scoped_refptr<media::AudioRendererSink> InsertNewSinkWhileLockHeld(
+ const SinkKey& key,
+ bool used);
+
+ // Cached sink deletion timeout.
+ // For example: (1) sink was create and cached in GetSinkInfo(), and then (2)
+ // the same sink is requested in GetSink(), if time interval between (1) and
+ // (2) is less than |kDeleteTimeoutMs|, then sink cached in (1) is reused in
+ // (2). On the other hand, if after (1) nobody is interested in the sink
+ // within |kDeleteTimeoutMs|, it is garbage-collected.
+ const int delete_timeout_ms_;
+
+ // Task runner for scheduled sink garbage collection.
+ const scoped_refptr<base::SingleThreadTaskRunner> task_runner_;
+
+ // Callback used for sink creation.
+ const CreateSinkCallback create_sink_cb_;
+
+ // Cached sinks, protected by lock.
+ base::Lock sinks_lock_;
+ AudioRendererSinkMap sinks_;
+
+ base::WeakPtrFactory<AudioRendererSinkCacheImpl> weak_ptr_factory_;
+
+ DISALLOW_COPY_AND_ASSIGN(AudioRendererSinkCacheImpl);
};
} // namespace content
-
-#endif // CONTENT_RENDERER_MEDIA_AUDIO_RENDERER_MIXER_MANAGER_H_

Powered by Google App Engine
This is Rietveld 408576698