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