summaryrefslogtreecommitdiffstats
path: root/media/mojo/services/mojo_renderer_service.h
blob: 085cea2bea1856b439c2149d45ffdb3051cf5a72 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
// Copyright 2014 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 MEDIA_MOJO_SERVICES_MOJO_RENDERER_SERVICE_H_
#define MEDIA_MOJO_SERVICES_MOJO_RENDERER_SERVICE_H_

#include "base/callback.h"
#include "base/compiler_specific.h"
#include "base/macros.h"
#include "base/memory/scoped_ptr.h"
#include "base/memory/weak_ptr.h"
#include "base/timer/timer.h"
#include "media/base/audio_decoder_config.h"
#include "media/base/buffering_state.h"
#include "media/base/media_export.h"
#include "media/base/pipeline_status.h"
#include "media/mojo/interfaces/media_renderer.mojom.h"
#include "third_party/mojo/src/mojo/public/cpp/bindings/strong_binding.h"

namespace mojo {
class ApplicationConnection;
}

namespace media {

class AudioRendererSink;
class DemuxerStreamProviderShim;
class CdmContextProvider;
class MediaLog;
class Renderer;
class RendererFactory;
class VideoRendererSink;

// A mojo::MediaRenderer implementation that uses media::AudioRenderer to
// decode and render audio to a sink obtained from the ApplicationConnection.
class MEDIA_EXPORT MojoRendererService
    : NON_EXPORTED_BASE(mojo::MediaRenderer) {
 public:
  // |cdm_context_provider| can be used to find the CdmContext to support
  // encrypted media. If null, encrypted media is not supported.
  MojoRendererService(CdmContextProvider* cdm_context_provider,
                      RendererFactory* renderer_factory,
                      const scoped_refptr<MediaLog>& media_log,
                      mojo::InterfaceRequest<mojo::MediaRenderer> request);
  ~MojoRendererService() final;

  // mojo::MediaRenderer implementation.
  void Initialize(mojo::MediaRendererClientPtr client,
                  mojo::DemuxerStreamPtr audio,
                  mojo::DemuxerStreamPtr video,
                  const mojo::Closure& callback) final;
  void Flush(const mojo::Closure& callback) final;
  void StartPlayingFrom(int64_t time_delta_usec) final;
  void SetPlaybackRate(double playback_rate) final;
  void SetVolume(float volume) final;
  void SetCdm(int32_t cdm_id, const mojo::Callback<void(bool)>& callback) final;

 private:
  enum State {
    STATE_UNINITIALIZED,
    STATE_INITIALIZING,
    STATE_FLUSHING,
    STATE_PLAYING,
    STATE_ERROR
  };

  // Called when the DemuxerStreamProviderShim is ready to go (has a config,
  // pipe handle, etc) and can be handed off to a renderer for use.
  void OnStreamReady(const mojo::Closure& callback);

  // Called when |audio_renderer_| initialization has completed.
  void OnRendererInitializeDone(const mojo::Closure& callback,
                                PipelineStatus status);

  // Callback executed by filters to update statistics.
  void OnUpdateStatistics(const PipelineStatistics& stats);

  // Periodically polls the media time from the renderer and notifies the client
  // if the media time has changed since the last update.  If |force| is true,
  // the client is notified even if the time is unchanged.
  void UpdateMediaTime(bool force);
  void CancelPeriodicMediaTimeUpdates();
  void SchedulePeriodicMediaTimeUpdates();

  // Callback executed by audio renderer when buffering state changes.
  // TODO(tim): Need old and new.
  void OnBufferingStateChanged(BufferingState new_buffering_state);

  // Callback executed when a renderer has ended.
  void OnRendererEnded();

  // Callback executed when a runtime error happens.
  void OnError(PipelineStatus error);

  // Callback executed once Flush() completes.
  void OnFlushCompleted(const mojo::Closure& callback);

  // Callback executed once SetCdm() completes.
  void OnCdmAttached(const mojo::Callback<void(bool)>& callback, bool success);

  mojo::StrongBinding<mojo::MediaRenderer> binding_;

  CdmContextProvider* cdm_context_provider_;

  State state_;

  // Note: |renderer_| should be destructed before these objects to avoid access
  // violation.
  scoped_ptr<DemuxerStreamProviderShim> stream_provider_;
  scoped_refptr<AudioRendererSink> audio_renderer_sink_;
  scoped_ptr<VideoRendererSink> video_renderer_sink_;

  scoped_ptr<Renderer> renderer_;

  base::RepeatingTimer<MojoRendererService> time_update_timer_;
  uint64_t last_media_time_usec_;

  mojo::MediaRendererClientPtr client_;

  base::WeakPtr<MojoRendererService> weak_this_;
  base::WeakPtrFactory<MojoRendererService> weak_factory_;

  DISALLOW_COPY_AND_ASSIGN(MojoRendererService);
};

}  // namespace media

#endif  // MEDIA_MOJO_SERVICES_MOJO_RENDERER_SERVICE_H_