| // Copyright 2020 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifndef MEDIA_FUCHSIA_AUDIO_FAKE_AUDIO_CONSUMER_H_ |
| #define MEDIA_FUCHSIA_AUDIO_FAKE_AUDIO_CONSUMER_H_ |
| |
| #include <fuchsia/media/audio/cpp/fidl.h> |
| #include <fuchsia/media/audio/cpp/fidl_test_base.h> |
| #include <fuchsia/media/cpp/fidl.h> |
| #include <fuchsia/media/cpp/fidl_test_base.h> |
| #include <lib/fidl/cpp/binding.h> |
| |
| #include <list> |
| #include <vector> |
| |
| #include "base/fuchsia/scoped_service_binding.h" |
| #include "base/time/time.h" |
| #include "base/timer/timer.h" |
| |
| namespace vfs { |
| class PseudoDir; |
| } // namespace vfs |
| |
| namespace media { |
| |
| // Fake implementation of fuchsia::media::AudioConsumer interface. Used for |
| // tests. |
| class FakeAudioConsumer final |
| : public fuchsia::media::testing::AudioConsumer_TestBase, |
| public fuchsia::media::testing::StreamSink_TestBase, |
| public fuchsia::media::audio::testing::VolumeControl_TestBase { |
| public: |
| // Lead time range returned from WatchStatus(). |
| static const base::TimeDelta kMinLeadTime; |
| static const base::TimeDelta kMaxLeadTime; |
| |
| FakeAudioConsumer( |
| uint64_t session_id, |
| fidl::InterfaceRequest<fuchsia::media::AudioConsumer> request); |
| ~FakeAudioConsumer() override; |
| |
| FakeAudioConsumer(const FakeAudioConsumer&) = delete; |
| FakeAudioConsumer& operator=(const FakeAudioConsumer&) = delete; |
| |
| uint64_t session_id() { return session_id_; } |
| float volume() const { return volume_; } |
| bool is_muted() const { return is_muted_; } |
| |
| base::TimeDelta GetMediaPosition(); |
| |
| private: |
| enum class State { |
| kStopped, |
| kPlaying, |
| kEndOfStream, |
| }; |
| |
| struct Packet { |
| base::TimeDelta pts; |
| bool is_eos = false; |
| }; |
| |
| // fuchsia::media::AudioConsumer interface; |
| void CreateStreamSink( |
| std::vector<zx::vmo> buffers, |
| fuchsia::media::AudioStreamType stream_type, |
| std::unique_ptr<fuchsia::media::Compression> compression, |
| fidl::InterfaceRequest<fuchsia::media::StreamSink> stream_sink_request) |
| final; |
| void Start(fuchsia::media::AudioConsumerStartFlags flags, |
| int64_t reference_time, |
| int64_t media_time) override; |
| void Stop() override; |
| void WatchStatus(WatchStatusCallback callback) override; |
| void SetRate(float rate) override; |
| void BindVolumeControl( |
| fidl::InterfaceRequest<fuchsia::media::audio::VolumeControl> |
| volume_control_request) override; |
| |
| // fuchsia::media::StreamSink interface. |
| void SendPacket(fuchsia::media::StreamPacket packet, |
| SendPacketCallback callback) override; |
| void SendPacketNoReply(fuchsia::media::StreamPacket packet) override; |
| void EndOfStream() override; |
| void DiscardAllPackets(DiscardAllPacketsCallback callback) override; |
| void DiscardAllPacketsNoReply() override; |
| |
| // fuchsia::media::audio::VolumeControl interface. |
| void SetVolume(float volume) override; |
| void SetMute(bool mute) override; |
| |
| // Not-implemented handler for _TestBase parents. |
| void NotImplemented_(const std::string& name) override; |
| |
| void ScheduleNextStreamPosUpdate(); |
| |
| // Updates stream position and drops old packets from the stream. |
| void UpdateStreamPos(); |
| |
| void OnStatusUpdate(); |
| void CallStatusCallback(); |
| |
| const uint64_t session_id_; |
| |
| fidl::Binding<fuchsia::media::AudioConsumer> audio_consumer_binding_; |
| fidl::Binding<fuchsia::media::StreamSink> stream_sink_binding_; |
| fidl::Binding<fuchsia::media::audio::VolumeControl> volume_control_binding_; |
| |
| size_t num_buffers_ = 0; |
| |
| State state_ = State::kStopped; |
| |
| bool have_status_update_ = true; |
| WatchStatusCallback status_callback_; |
| |
| base::TimeTicks reference_time_; |
| |
| // Numerator and denumerator for current playback rate. |
| uint32_t media_delta_ = 1; |
| uint32_t reference_delta_ = 1; |
| |
| // Last known media position. Min value indicates that the stream position |
| // hasn't been set. If stream is playing then value corresponds to |
| // |reference_time_|. |
| base::TimeDelta media_pos_ = base::TimeDelta::Min(); |
| |
| std::list<Packet> pending_packets_; |
| |
| // Timer to call UpdateStreamPos() for the next packet. |
| base::OneShotTimer update_timer_; |
| |
| float volume_ = 1.0; |
| bool is_muted_ = false; |
| }; |
| |
| class FakeAudioConsumerService final |
| : public fuchsia::media::testing::SessionAudioConsumerFactory_TestBase { |
| public: |
| explicit FakeAudioConsumerService(vfs::PseudoDir* pseudo_dir); |
| ~FakeAudioConsumerService() override; |
| |
| FakeAudioConsumerService(const FakeAudioConsumerService&) = delete; |
| FakeAudioConsumerService& operator=(const FakeAudioConsumerService&) = delete; |
| |
| size_t num_instances() { return audio_consumers_.size(); } |
| FakeAudioConsumer* instance(size_t index) { |
| return audio_consumers_[index].get(); |
| } |
| |
| private: |
| // fuchsia::media::SessionAudioConsumerFactory implementation. |
| void CreateAudioConsumer(uint64_t session_id, |
| fidl::InterfaceRequest<fuchsia::media::AudioConsumer> |
| audio_consumer_request) override; |
| |
| // Not-implemented handler for SessionAudioConsumerFactory_TestBase. |
| void NotImplemented_(const std::string& name) override; |
| |
| base::ScopedServiceBinding<fuchsia::media::SessionAudioConsumerFactory> |
| binding_; |
| |
| std::vector<std::unique_ptr<FakeAudioConsumer>> audio_consumers_; |
| }; |
| |
| } // namespace media |
| |
| #endif // MEDIA_FUCHSIA_AUDIO_FAKE_AUDIO_CONSUMER_H_ |