• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #ifndef MEDIA_BASE_ANDROID_MEDIA_CODEC_BRIDGE_H_
6 #define MEDIA_BASE_ANDROID_MEDIA_CODEC_BRIDGE_H_
7 
8 #include <jni.h>
9 #include <string>
10 
11 #include "base/android/scoped_java_ref.h"
12 #include "base/time/time.h"
13 #include "media/base/audio_decoder_config.h"
14 #include "media/base/video_decoder_config.h"
15 #include "ui/gfx/size.h"
16 
17 namespace media {
18 
19 struct SubsampleEntry;
20 
21 // These must be in sync with MediaCodecBridge.MEDIA_CODEC_XXX constants in
22 // MediaCodecBridge.java.
23 enum MediaCodecStatus {
24   MEDIA_CODEC_OK,
25   MEDIA_CODEC_DEQUEUE_INPUT_AGAIN_LATER,
26   MEDIA_CODEC_DEQUEUE_OUTPUT_AGAIN_LATER,
27   MEDIA_CODEC_OUTPUT_BUFFERS_CHANGED,
28   MEDIA_CODEC_OUTPUT_FORMAT_CHANGED,
29   MEDIA_CODEC_INPUT_END_OF_STREAM,
30   MEDIA_CODEC_OUTPUT_END_OF_STREAM,
31   MEDIA_CODEC_NO_KEY,
32   MEDIA_CODEC_STOPPED,
33   MEDIA_CODEC_ERROR
34 };
35 
36 // Codec direction.  Keep this in sync with MediaCodecBridge.java.
37 enum MediaCodecDirection {
38   MEDIA_CODEC_DECODER,
39   MEDIA_CODEC_ENCODER,
40 };
41 
42 // This class serves as a bridge for native code to call java functions inside
43 // Android MediaCodec class. For more information on Android MediaCodec, check
44 // http://developer.android.com/reference/android/media/MediaCodec.html
45 // Note: MediaCodec is only available on JB and greater.
46 // Use AudioCodecBridge or VideoCodecBridge to create an instance of this
47 // object.
48 //
49 // TODO(fischman,xhwang): replace this (and the enums that go with it) with
50 // chromium's JNI auto-generation hotness.
51 class MEDIA_EXPORT MediaCodecBridge {
52  public:
53   // Returns true if MediaCodec is available on the device.
54   // All other static methods check IsAvailable() internally. There's no need
55   // to check IsAvailable() explicitly before calling them.
56   static bool IsAvailable();
57 
58   // Returns true if MediaCodec.setParameters() is available on the device.
59   static bool SupportsSetParameters();
60 
61   // Returns whether MediaCodecBridge has a decoder that |is_secure| and can
62   // decode |codec| type.
63   static bool CanDecode(const std::string& codec, bool is_secure);
64 
65   // Represents supported codecs on android.
66   // TODO(qinmin): Currently the codecs string only contains one codec. Do we
67   // need to support codecs separated by comma. (e.g. "vp8" -> "vp8, vp8.0")?
68   struct CodecsInfo {
69     std::string codecs;  // E.g. "vp8" or "avc1".
70     std::string name;    // E.g. "OMX.google.vp8.decoder".
71     MediaCodecDirection direction;
72   };
73 
74   // Get a list of supported codecs.
75   static std::vector<CodecsInfo> GetCodecsInfo();
76 
77   virtual ~MediaCodecBridge();
78 
79   // Resets both input and output, all indices previously returned in calls to
80   // DequeueInputBuffer() and DequeueOutputBuffer() become invalid.
81   // Please note that this clears all the inputs in the media codec. In other
82   // words, there will be no outputs until new input is provided.
83   // Returns MEDIA_CODEC_ERROR if an unexpected error happens, or Media_CODEC_OK
84   // otherwise.
85   MediaCodecStatus Reset();
86 
87   // Finishes the decode/encode session. The instance remains active
88   // and ready to be StartAudio/Video()ed again. HOWEVER, due to the buggy
89   // vendor's implementation , b/8125974, Stop() -> StartAudio/Video() may not
90   // work on some devices. For reliability, Stop() -> delete and recreate new
91   // instance -> StartAudio/Video() is recommended.
92   void Stop();
93 
94   // Used for getting output format. This is valid after DequeueInputBuffer()
95   // returns a format change by returning INFO_OUTPUT_FORMAT_CHANGED
96   void GetOutputFormat(int* width, int* height);
97 
98   // Returns the number of input buffers used by the codec.
99   int GetInputBuffersCount();
100 
101   // Submits a byte array to the given input buffer. Call this after getting an
102   // available buffer from DequeueInputBuffer().  If |data| is NULL, assume the
103   // input buffer has already been populated (but still obey |size|).
104   // |data_size| must be less than kint32max (because Java).
105   MediaCodecStatus QueueInputBuffer(int index,
106                                     const uint8* data,
107                                     size_t data_size,
108                                     const base::TimeDelta& presentation_time);
109 
110   // Similar to the above call, but submits a buffer that is encrypted.  Note:
111   // NULL |subsamples| indicates the whole buffer is encrypted.  If |data| is
112   // NULL, assume the input buffer has already been populated (but still obey
113   // |data_size|).  |data_size| must be less than kint32max (because Java).
114   MediaCodecStatus QueueSecureInputBuffer(
115       int index,
116       const uint8* data,
117       size_t data_size,
118       const uint8* key_id,
119       int key_id_size,
120       const uint8* iv,
121       int iv_size,
122       const SubsampleEntry* subsamples,
123       int subsamples_size,
124       const base::TimeDelta& presentation_time);
125 
126   // Submits an empty buffer with a EOS (END OF STREAM) flag.
127   void QueueEOS(int input_buffer_index);
128 
129   // Returns:
130   // MEDIA_CODEC_OK if an input buffer is ready to be filled with valid data,
131   // MEDIA_CODEC_ENQUEUE_INPUT_AGAIN_LATER if no such buffer is available, or
132   // MEDIA_CODEC_ERROR if unexpected error happens.
133   // Note: Never use infinite timeout as this would block the decoder thread and
134   // prevent the decoder job from being released.
135   MediaCodecStatus DequeueInputBuffer(const base::TimeDelta& timeout,
136                                       int* index);
137 
138   // Dequeues an output buffer, block at most timeout_us microseconds.
139   // Returns the status of this operation. If OK is returned, the output
140   // parameters should be populated. Otherwise, the values of output parameters
141   // should not be used.  Output parameters other than index/offset/size are
142   // optional and only set if not NULL.
143   // Note: Never use infinite timeout as this would block the decoder thread and
144   // prevent the decoder job from being released.
145   // TODO(xhwang): Can we drop |end_of_stream| and return
146   // MEDIA_CODEC_OUTPUT_END_OF_STREAM?
147   MediaCodecStatus DequeueOutputBuffer(const base::TimeDelta& timeout,
148                                        int* index,
149                                        size_t* offset,
150                                        size_t* size,
151                                        base::TimeDelta* presentation_time,
152                                        bool* end_of_stream,
153                                        bool* key_frame);
154 
155   // Returns the buffer to the codec. If you previously specified a surface when
156   // configuring this video decoder you can optionally render the buffer.
157   void ReleaseOutputBuffer(int index, bool render);
158 
159   // Returns the number of output buffers used by the codec.
160   int GetOutputBuffersCount();
161 
162   // Returns the capacity of each output buffer used by the codec.
163   size_t GetOutputBuffersCapacity();
164 
165   // Gets output buffers from media codec and keeps them inside the java class.
166   // To access them, use DequeueOutputBuffer(). Returns whether output buffers
167   // were successfully obtained.
168   bool GetOutputBuffers() WARN_UNUSED_RESULT;
169 
170   // Returns an input buffer's base pointer and capacity.
171   void GetInputBuffer(int input_buffer_index, uint8** data, size_t* capacity);
172 
173   // Copy |dst_size| bytes from output buffer |index|'s |offset| onwards into
174   // |*dst|.
175   bool CopyFromOutputBuffer(int index, size_t offset, void* dst, int dst_size);
176 
177   static bool RegisterMediaCodecBridge(JNIEnv* env);
178 
179  protected:
180   // Returns true if |mime_type| is known to be unaccelerated (i.e. backed by a
181   // software codec instead of a hardware one).
182   static bool IsKnownUnaccelerated(const std::string& mime_type,
183                                    MediaCodecDirection direction);
184 
185   MediaCodecBridge(const std::string& mime,
186                    bool is_secure,
187                    MediaCodecDirection direction);
188 
189   // Calls start() against the media codec instance. Used in StartXXX() after
190   // configuring media codec. Returns whether media codec was successfully
191   // started.
192   bool StartInternal() WARN_UNUSED_RESULT;
193 
media_codec()194   jobject media_codec() { return j_media_codec_.obj(); }
195   MediaCodecDirection direction_;
196 
197  private:
198   // Fills a particular input buffer; returns false if |data_size| exceeds the
199   // input buffer's capacity (and doesn't touch the input buffer in that case).
200   bool FillInputBuffer(int index,
201                        const uint8* data,
202                        size_t data_size) WARN_UNUSED_RESULT;
203 
204   // Java MediaCodec instance.
205   base::android::ScopedJavaGlobalRef<jobject> j_media_codec_;
206 
207   DISALLOW_COPY_AND_ASSIGN(MediaCodecBridge);
208 };
209 
210 class AudioCodecBridge : public MediaCodecBridge {
211  public:
212   // Returns an AudioCodecBridge instance if |codec| is supported, or a NULL
213   // pointer otherwise.
214   static AudioCodecBridge* Create(const AudioCodec& codec);
215 
216   // See MediaCodecBridge::IsKnownUnaccelerated().
217   static bool IsKnownUnaccelerated(const AudioCodec& codec);
218 
219   // Start the audio codec bridge.
220   bool Start(const AudioCodec& codec, int sample_rate, int channel_count,
221              const uint8* extra_data, size_t extra_data_size,
222              bool play_audio, jobject media_crypto) WARN_UNUSED_RESULT;
223 
224   // Play the output buffer. This call must be called after
225   // DequeueOutputBuffer() and before ReleaseOutputBuffer. Returns the playback
226   // head position expressed in frames.
227   int64 PlayOutputBuffer(int index, size_t size);
228 
229   // Set the volume of the audio output.
230   void SetVolume(double volume);
231 
232  private:
233   explicit AudioCodecBridge(const std::string& mime);
234 
235   // Configure the java MediaFormat object with the extra codec data passed in.
236   bool ConfigureMediaFormat(jobject j_format, const AudioCodec& codec,
237                             const uint8* extra_data, size_t extra_data_size);
238 };
239 
240 class MEDIA_EXPORT VideoCodecBridge : public MediaCodecBridge {
241  public:
242   // See MediaCodecBridge::IsKnownUnaccelerated().
243   static bool IsKnownUnaccelerated(const VideoCodec& codec,
244                                    MediaCodecDirection direction);
245 
246   // Create, start, and return a VideoCodecBridge decoder or NULL on failure.
247   static VideoCodecBridge* CreateDecoder(
248       const VideoCodec& codec,  // e.g. media::kCodecVP8
249       bool is_secure,
250       const gfx::Size& size,  // Output frame size.
251       jobject surface,        // Output surface, optional.
252       jobject media_crypto);  // MediaCrypto object, optional.
253 
254   // Create, start, and return a VideoCodecBridge encoder or NULL on failure.
255   static VideoCodecBridge* CreateEncoder(
256       const VideoCodec& codec,  // e.g. media::kCodecVP8
257       const gfx::Size& size,    // input frame size
258       int bit_rate,             // bits/second
259       int frame_rate,           // frames/second
260       int i_frame_interval,     // count
261       int color_format);        // MediaCodecInfo.CodecCapabilities.
262 
263   void SetVideoBitrate(int bps);
264   void RequestKeyFrameSoon();
265 
266   // Returns whether adaptive playback is supported for this object given
267   // the new size.
268   bool IsAdaptivePlaybackSupported(int width, int height);
269 
270   // Test-only method to set the return value of IsAdaptivePlaybackSupported().
271   // Without this function, the return value of that function will be device
272   // dependent. If |adaptive_playback_supported| is equal to 0, the return value
273   // will be false. If |adaptive_playback_supported| is larger than 0, the
274   // return value will be true.
set_adaptive_playback_supported_for_testing(int adaptive_playback_supported)275   void set_adaptive_playback_supported_for_testing(
276       int adaptive_playback_supported) {
277     adaptive_playback_supported_for_testing_ = adaptive_playback_supported;
278   }
279 
280  private:
281   VideoCodecBridge(const std::string& mime,
282                    bool is_secure,
283                    MediaCodecDirection direction);
284 
285   int adaptive_playback_supported_for_testing_;
286 };
287 
288 }  // namespace media
289 
290 #endif  // MEDIA_BASE_ANDROID_MEDIA_CODEC_BRIDGE_H_
291