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

Side by Side Diff: media/base/android/media_codec_bridge.cc

Issue 11973010: AndroidVDA by using Android's MediaCodec API. (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: using CopyTextureCHROMIUMResourceManager Created 7 years, 10 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 unified diff | Download patch
OLDNEW
(Empty)
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 #include "media/base/android/media_codec_bridge.h"
6
7 #include <jni.h>
8
9 #include "base/android/jni_android.h"
10 #include "base/android/jni_array.h"
11 #include "base/android/jni_string.h"
12 #include "base/basictypes.h"
13 #include "base/lazy_instance.h"
14 #include "base/logging.h"
15 #include "base/stringprintf.h"
16
17 #include "jni/MediaCodec_jni.h"
18 #include "jni/MediaFormat_jni.h"
19
20
21 using base::android::AttachCurrentThread;
22 using base::android::ConvertUTF8ToJavaString;
23 using base::android::ScopedJavaLocalRef;
24
25 namespace {
26
27 static jclass g_MediaCodecBufferInfo_clazz = NULL;
28
29 static const char kMediaCodecBufferInfoClassPath[] =
30 "android/media/MediaCodec$BufferInfo";
31
32 static bool MediaCodecBufferInfo_RegisterNativesImpl(JNIEnv* env) {
33 g_MediaCodecBufferInfo_clazz = reinterpret_cast<jclass>(env->NewGlobalRef(
34 base::android::GetUnscopedClass(env, kMediaCodecBufferInfoClassPath)));
35 base::android::CheckException(env);
36 return true;
37 }
38
39 static void GetBufferInfo(JNIEnv* env, jobject buffer_info, int* offset,
40 int* size, int64* presentation_time, int* flags) {
41 static jfieldID offset_id =
42 env->GetFieldID(g_MediaCodecBufferInfo_clazz, "offset", "I");
43 static jfieldID size_id =
44 env->GetFieldID(g_MediaCodecBufferInfo_clazz, "size", "I");
45 static jfieldID presentation_time_id =
46 env->GetFieldID(g_MediaCodecBufferInfo_clazz, "presentationTimeUs", "J");
47 static jfieldID flags_id =
48 env->GetFieldID(g_MediaCodecBufferInfo_clazz, "flags", "I");
49
50 *offset = env->GetIntField(buffer_info, offset_id);
51 *size = env->GetIntField(buffer_info, size_id);
52 *presentation_time = env->GetLongField(buffer_info, presentation_time_id);
53 *flags = env->GetIntField(buffer_info, flags_id);
54 }
55
56 static base::subtle::AtomicWord g_MediaCodecBufferInfo_Constructor = 0;
57 static ScopedJavaLocalRef<jobject> Java_MediaCodecBufferInfo_Constructor(
58 JNIEnv* env) {
59 /* Must call RegisterNativesImpl() */
60 DCHECK(g_MediaCodecBufferInfo_clazz);
61 jmethodID method_id =
62 base::android::MethodID::LazyGet<base::android::MethodID::TYPE_INSTANCE>(
63 env, g_MediaCodecBufferInfo_clazz,
64 "<init>", "()V",
65 &g_MediaCodecBufferInfo_Constructor);
66
67 jobject ret = env->NewObject(g_MediaCodecBufferInfo_clazz, method_id);
68 base::android::CheckException(env);
69 return ScopedJavaLocalRef<jobject>(env, ret);
70 }
71
72 class MediaCodecNativeRegisterer {
73 public:
74 MediaCodecNativeRegisterer() {
75 JNIEnv* env = AttachCurrentThread();
76 jni_initialized_ =
77 MediaCodecBufferInfo_RegisterNativesImpl(env) &&
78 JNI_MediaCodec::RegisterNativesImpl(env) &&
79 JNI_MediaFormat::RegisterNativesImpl(env);
80 }
81
82 bool IsRegistered() {
83 return jni_initialized_;
84 }
85
86 private:
87 bool jni_initialized_;
88 };
89
90 static base::LazyInstance<MediaCodecNativeRegisterer> g_native_registerer =
91 LAZY_INSTANCE_INITIALIZER;
92
93 } // namespace
94
95 namespace media {
96
97 enum { kBufferFlagEndOfStream = 4 };
98
99 static const char* CodecToMimeType(const MediaCodecBridge::Codec codec) {
100 switch (codec) {
101 case MediaCodecBridge::AUDIO_MPEG: return "audio/mpeg";
102 case MediaCodecBridge::VIDEO_H264: return "video/avc";
103 case MediaCodecBridge::VIDEO_VP8: return "video/x-vnd.on2.vp8";
104 default: return NULL;
105 }
106 }
107
108 // static
109 const base::TimeDelta MediaCodecBridge::kTimeOutInfinity =
110 base::TimeDelta::FromMicroseconds(-1);
111
112 // static
113 const base::TimeDelta MediaCodecBridge::kTimeOutNoWait =
114 base::TimeDelta::FromMicroseconds(0);
115
116 MediaCodecBridge::MediaCodecBridge(const Codec codec) {
117 JNIEnv* env = AttachCurrentThread();
118 CHECK(env);
119 CHECK(g_native_registerer.Pointer()->IsRegistered());
120 DCHECK(CodecToMimeType(codec));
121
122 ScopedJavaLocalRef<jstring> j_type =
123 ConvertUTF8ToJavaString(env, CodecToMimeType(codec));
124
125 ScopedJavaLocalRef<jobject> tmp(
126 JNI_MediaCodec::Java_MediaCodec_createDecoderByType(env, j_type.obj()));
127 DCHECK(!tmp.is_null());
128 j_media_codec_.Reset(tmp);
129 }
130
131 MediaCodecBridge::~MediaCodecBridge() {
132 JNIEnv* env = AttachCurrentThread();
133 CHECK(env);
134
135 JNI_MediaCodec::Java_MediaCodec_release(env, j_media_codec_.obj());
136 }
137
138 void MediaCodecBridge::StartAudio(
139 const Codec codec, int sample_rate, int channel_count) {
140 JNIEnv* env = AttachCurrentThread();
141 DCHECK(CodecToMimeType(codec));
142
143 ScopedJavaLocalRef<jstring> j_mime =
144 ConvertUTF8ToJavaString(env, CodecToMimeType(codec));
145 ScopedJavaLocalRef<jobject> j_format(
146 JNI_MediaFormat::Java_MediaFormat_createAudioFormat(
147 env, j_mime.obj(), sample_rate, channel_count));
148 DCHECK(!j_format.is_null());
149
150 JNI_MediaCodec::Java_MediaCodec_configure(
151 env, j_media_codec_.obj(), j_format.obj(), NULL, NULL, 0);
152
153 Start();
154 }
155
156 void MediaCodecBridge::StartVideo(
157 const Codec codec, const gfx::Size& size, jobject surface) {
158 JNIEnv* env = AttachCurrentThread();
159 DCHECK(CodecToMimeType(codec));
160
161 ScopedJavaLocalRef<jstring> j_mime =
162 ConvertUTF8ToJavaString(env, CodecToMimeType(codec));
163 ScopedJavaLocalRef<jobject> j_format(
164 JNI_MediaFormat::Java_MediaFormat_createVideoFormat(
165 env, j_mime.obj(), size.width(), size.height()));
166 DCHECK(!j_format.is_null());
167
168 JNI_MediaCodec::Java_MediaCodec_configure(
169 env, j_media_codec_.obj(), j_format.obj(), surface, NULL, 0);
170
171 Start();
172 }
173
174 void MediaCodecBridge::Start() {
175 JNIEnv* env = AttachCurrentThread();
176 JNI_MediaCodec::Java_MediaCodec_start(env, j_media_codec_.obj());
177 GetInputBuffers();
178 }
179
180 void MediaCodecBridge::Reset() {
181 JNIEnv* env = AttachCurrentThread();
182 JNI_MediaCodec::Java_MediaCodec_flush(env, j_media_codec_.obj());
183 }
184
185 void MediaCodecBridge::Stop() {
186 JNIEnv* env = AttachCurrentThread();
187 JNI_MediaCodec::Java_MediaCodec_stop(env, j_media_codec_.obj());
188 }
189
190 void MediaCodecBridge::GetOutputFormat(int* width, int* height) {
191 JNIEnv* env = AttachCurrentThread();
192
193 ScopedJavaLocalRef<jobject> j_format(
194 JNI_MediaCodec::Java_MediaCodec_getOutputFormat(
195 env, j_media_codec_.obj()));
196 if (!j_format.is_null()) {
197 ScopedJavaLocalRef<jstring> j_key_width =
198 ConvertUTF8ToJavaString(env, "width");
199 *width = JNI_MediaFormat::Java_MediaFormat_getInteger(
200 env, j_format.obj(), j_key_width.obj());
201
202 ScopedJavaLocalRef<jstring> j_key_height =
203 ConvertUTF8ToJavaString(env, "height");
204 *height = JNI_MediaFormat::Java_MediaFormat_getInteger(
205 env, j_format.obj(), j_key_height.obj());
206 }
207 }
208
209 size_t MediaCodecBridge::QueueInputBuffer(
210 int index, const uint8* data, int size,
211 const base::TimeDelta& presentation_time) {
212 JNIEnv* env = AttachCurrentThread();
213
214 ScopedJavaLocalRef<jobject> j_buffer(
215 env, env->GetObjectArrayElement(j_input_buffers_.obj(), index));
216
217 uint8* direct_buffer =
218 static_cast<uint8*>(env->GetDirectBufferAddress(j_buffer.obj()));
219 int64 buffer_capacity = env->GetDirectBufferCapacity(j_buffer.obj());
220
221 size_t size_to_copy = (buffer_capacity < size) ? buffer_capacity : size;
222 if (size_to_copy > 0)
223 memcpy(direct_buffer, data, size_to_copy);
224
225 JNI_MediaCodec::Java_MediaCodec_queueInputBuffer(
226 env, j_media_codec_.obj(),
227 index, 0, size_to_copy, presentation_time.InMicroseconds(), 0);
228 return size_to_copy;
229 }
230
231 void MediaCodecBridge::QueueEOS(int input_buffer_index) {
232 JNIEnv* env = AttachCurrentThread();
233 JNI_MediaCodec::Java_MediaCodec_queueInputBuffer(
234 env, j_media_codec_.obj(),
235 input_buffer_index, 0, 0, 0, kBufferFlagEndOfStream);
236 }
237
238 int MediaCodecBridge::DequeueInputBuffer(base::TimeDelta timeout) {
239 JNIEnv* env = AttachCurrentThread();
240 return JNI_MediaCodec::Java_MediaCodec_dequeueInputBuffer(
241 env, j_media_codec_.obj(), timeout.InMicroseconds());
242 }
243
244 int MediaCodecBridge::DequeueOutputBuffer(
245 base::TimeDelta timeout, int* offset, int* size,
246 base::TimeDelta* presentation_time, bool* end_of_stream) {
247 JNIEnv* env = AttachCurrentThread();
248
249 ScopedJavaLocalRef<jobject> j_info(
250 Java_MediaCodecBufferInfo_Constructor(env));
251 jint j_buffer = JNI_MediaCodec::Java_MediaCodec_dequeueOutputBuffer(
252 env, j_media_codec_.obj(), j_info.obj(), timeout.InMicroseconds());
253
254 if (j_buffer >= 0) {
255 int64 presentation_time_us;
256 int flags;
257 GetBufferInfo(
258 env, j_info.obj(), offset, size, &presentation_time_us, &flags);
259 *presentation_time =
260 base::TimeDelta::FromMicroseconds(presentation_time_us);
261 *end_of_stream = flags & kBufferFlagEndOfStream;
262 }
263 return j_buffer;
264 }
265
266 void MediaCodecBridge::ReleaseOutputBuffer(int index, bool render) {
267 JNIEnv* env = AttachCurrentThread();
268 CHECK(env);
269
270 JNI_MediaCodec::Java_MediaCodec_releaseOutputBuffer(
felipeg 2013/02/20 18:35:26 I thought that before calling releaseOutputBuffer
dwkang1 2013/02/25 23:48:00 Actually, we are assuming that the ByteBuffer is D
271 env, j_media_codec_.obj(), index, render);
272 }
273
274 int MediaCodecBridge::GetInputBuffers() {
275 JNIEnv* env = AttachCurrentThread();
276
277 j_input_buffers_.Reset(
278 JNI_MediaCodec::Java_MediaCodec_getInputBuffers(
279 env, j_media_codec_.obj()));
280
281 return env->GetArrayLength(j_input_buffers_.obj());
282 }
283
284 int MediaCodecBridge::GetOutputBuffers() {
285 JNIEnv* env = AttachCurrentThread();
286
287 j_output_buffers_.Reset(
288 JNI_MediaCodec::Java_MediaCodec_getOutputBuffers(
289 env, j_media_codec_.obj()));
290
291 return env->GetArrayLength(j_output_buffers_.obj());
292 }
293
294 } // namespace media
295
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698