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

Side by Side Diff: webrtc/sdk/android/src/jni/videodecoderwrapper.cc

Issue 3000993002: Implement QP parsing in VideoDecoderWrapper. (Closed)
Patch Set: Add comments Created 3 years, 4 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
« no previous file with comments | « webrtc/sdk/android/src/jni/videodecoderwrapper.h ('k') | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 /* 1 /*
2 * Copyright 2017 The WebRTC project authors. All Rights Reserved. 2 * Copyright 2017 The WebRTC project authors. All Rights Reserved.
3 * 3 *
4 * Use of this source code is governed by a BSD-style license 4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source 5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found 6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may 7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree. 8 * be found in the AUTHORS file in the root of the source tree.
9 */ 9 */
10 10
11 #include "webrtc/sdk/android/src/jni/videodecoderwrapper.h" 11 #include "webrtc/sdk/android/src/jni/videodecoderwrapper.h"
12 12
13 #include "webrtc/api/video/video_frame.h" 13 #include "webrtc/api/video/video_frame.h"
14 #include "webrtc/modules/video_coding/include/video_codec_interface.h" 14 #include "webrtc/modules/video_coding/include/video_codec_interface.h"
15 #include "webrtc/modules/video_coding/utility/vp8_header_parser.h"
16 #include "webrtc/modules/video_coding/utility/vp9_uncompressed_header_parser.h"
15 #include "webrtc/rtc_base/logging.h" 17 #include "webrtc/rtc_base/logging.h"
16 #include "webrtc/sdk/android/src/jni/classreferenceholder.h" 18 #include "webrtc/sdk/android/src/jni/classreferenceholder.h"
17 19
18 namespace webrtc_jni { 20 namespace webrtc_jni {
19 21
20 VideoDecoderWrapper::VideoDecoderWrapper(JNIEnv* jni, jobject decoder) 22 VideoDecoderWrapper::VideoDecoderWrapper(JNIEnv* jni, jobject decoder)
21 : android_video_buffer_factory_(jni), 23 : android_video_buffer_factory_(jni),
22 decoder_(jni, decoder), 24 decoder_(jni, decoder),
23 encoded_image_class_(jni, FindClass(jni, "org/webrtc/EncodedImage")), 25 encoded_image_class_(jni, FindClass(jni, "org/webrtc/EncodedImage")),
24 frame_type_class_(jni, 26 frame_type_class_(jni,
(...skipping 38 matching lines...) Expand 10 before | Expand all | Expand 10 after
63 get_implementation_name_method_ = jni->GetMethodID( 65 get_implementation_name_method_ = jni->GetMethodID(
64 decoder_class, "getImplementationName", "()Ljava/lang/String;"); 66 decoder_class, "getImplementationName", "()Ljava/lang/String;");
65 67
66 get_number_method_ = 68 get_number_method_ =
67 jni->GetMethodID(*video_codec_status_class_, "getNumber", "()I"); 69 jni->GetMethodID(*video_codec_status_class_, "getNumber", "()I");
68 70
69 integer_constructor_ = jni->GetMethodID(*integer_class_, "<init>", "(I)V"); 71 integer_constructor_ = jni->GetMethodID(*integer_class_, "<init>", "(I)V");
70 int_value_method_ = jni->GetMethodID(*integer_class_, "intValue", "()I"); 72 int_value_method_ = jni->GetMethodID(*integer_class_, "intValue", "()I");
71 73
72 initialized_ = false; 74 initialized_ = false;
75 // QP parsing starts enabled and we disable it if the decoder provides frames.
76 qp_parsing_enabled_ = true;
73 } 77 }
74 78
75 int32_t VideoDecoderWrapper::InitDecode( 79 int32_t VideoDecoderWrapper::InitDecode(
76 const webrtc::VideoCodec* codec_settings, 80 const webrtc::VideoCodec* codec_settings,
77 int32_t number_of_cores) { 81 int32_t number_of_cores) {
78 JNIEnv* jni = AttachCurrentThreadIfNeeded(); 82 JNIEnv* jni = AttachCurrentThreadIfNeeded();
79 ScopedLocalRefFrame local_ref_frame(jni); 83 ScopedLocalRefFrame local_ref_frame(jni);
80 84
81 codec_settings_ = *codec_settings; 85 codec_settings_ = *codec_settings;
82 number_of_cores_ = number_of_cores; 86 number_of_cores_ = number_of_cores;
(...skipping 10 matching lines...) Expand all
93 jmethodID callback_constructor = 97 jmethodID callback_constructor =
94 jni->GetMethodID(callback_class, "<init>", "(J)V"); 98 jni->GetMethodID(callback_class, "<init>", "(J)V");
95 jobject callback = jni->NewObject(callback_class, callback_constructor, 99 jobject callback = jni->NewObject(callback_class, callback_constructor,
96 jlongFromPointer(this)); 100 jlongFromPointer(this));
97 101
98 jobject ret = 102 jobject ret =
99 jni->CallObjectMethod(*decoder_, init_decode_method_, settings, callback); 103 jni->CallObjectMethod(*decoder_, init_decode_method_, settings, callback);
100 if (jni->CallIntMethod(ret, get_number_method_) == WEBRTC_VIDEO_CODEC_OK) { 104 if (jni->CallIntMethod(ret, get_number_method_) == WEBRTC_VIDEO_CODEC_OK) {
101 initialized_ = true; 105 initialized_ = true;
102 } 106 }
107
108 // The decoder was reinitialized so re-enable the QP parsing in case it stops
109 // providing QP values.
110 qp_parsing_enabled_ = true;
111
103 return HandleReturnCode(jni, ret); 112 return HandleReturnCode(jni, ret);
104 } 113 }
105 114
106 int32_t VideoDecoderWrapper::Decode( 115 int32_t VideoDecoderWrapper::Decode(
107 const webrtc::EncodedImage& input_image, 116 const webrtc::EncodedImage& input_image,
108 bool missing_frames, 117 bool missing_frames,
109 const webrtc::RTPFragmentationHeader* fragmentation, 118 const webrtc::RTPFragmentationHeader* fragmentation,
110 const webrtc::CodecSpecificInfo* codec_specific_info, 119 const webrtc::CodecSpecificInfo* codec_specific_info,
111 int64_t render_time_ms) { 120 int64_t render_time_ms) {
112 if (!initialized_) { 121 if (!initialized_) {
113 // Most likely initializing the codec failed. 122 // Most likely initializing the codec failed.
114 return WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE; 123 return WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE;
115 } 124 }
116 125
117 JNIEnv* jni = AttachCurrentThreadIfNeeded(); 126 JNIEnv* jni = AttachCurrentThreadIfNeeded();
118 ScopedLocalRefFrame local_ref_frame(jni); 127 ScopedLocalRefFrame local_ref_frame(jni);
119 128
120 FrameExtraInfo frame_extra_info; 129 FrameExtraInfo frame_extra_info;
121 frame_extra_info.capture_time_ms = input_image.capture_time_ms_; 130 frame_extra_info.capture_time_ms = input_image.capture_time_ms_;
122 frame_extra_info.timestamp_rtp = input_image._timeStamp; 131 frame_extra_info.timestamp_rtp = input_image._timeStamp;
132 frame_extra_info.qp =
133 qp_parsing_enabled_ ? ParseQP(input_image) : rtc::Optional<uint8_t>();
123 frame_extra_infos_.push_back(frame_extra_info); 134 frame_extra_infos_.push_back(frame_extra_info);
124 135
125 jobject jinput_image = 136 jobject jinput_image =
126 ConvertEncodedImageToJavaEncodedImage(jni, input_image); 137 ConvertEncodedImageToJavaEncodedImage(jni, input_image);
127 jobject ret = 138 jobject ret =
128 jni->CallObjectMethod(*decoder_, decode_method_, jinput_image, nullptr); 139 jni->CallObjectMethod(*decoder_, decode_method_, jinput_image, nullptr);
129 return HandleReturnCode(jni, ret); 140 return HandleReturnCode(jni, ret);
130 } 141 }
131 142
132 int32_t VideoDecoderWrapper::RegisterDecodeCompleteCallback( 143 int32_t VideoDecoderWrapper::RegisterDecodeCompleteCallback(
(...skipping 49 matching lines...) Expand 10 before | Expand all | Expand 10 after
182 193
183 rtc::Optional<int32_t> decoding_time_ms; 194 rtc::Optional<int32_t> decoding_time_ms;
184 if (jdecode_time_ms != nullptr) { 195 if (jdecode_time_ms != nullptr) {
185 decoding_time_ms = rtc::Optional<int32_t>( 196 decoding_time_ms = rtc::Optional<int32_t>(
186 jni->CallIntMethod(jdecode_time_ms, int_value_method_)); 197 jni->CallIntMethod(jdecode_time_ms, int_value_method_));
187 } 198 }
188 199
189 rtc::Optional<uint8_t> qp; 200 rtc::Optional<uint8_t> qp;
190 if (jqp != nullptr) { 201 if (jqp != nullptr) {
191 qp = rtc::Optional<uint8_t>(jni->CallIntMethod(jqp, int_value_method_)); 202 qp = rtc::Optional<uint8_t>(jni->CallIntMethod(jqp, int_value_method_));
203 // The decoder provides QP values itself, no need to parse the bitstream.
204 qp_parsing_enabled_ = false;
205 } else {
206 qp = frame_extra_info.qp;
207 // The decoder doesn't provide QP values, ensure bitstream parsing is
208 // enabled.
209 qp_parsing_enabled_ = true;
192 } 210 }
193 211
194 callback_->Decoded(frame, decoding_time_ms, rtc::Optional<uint8_t>()); 212 callback_->Decoded(frame, decoding_time_ms, qp);
195 } 213 }
196 214
197 jobject VideoDecoderWrapper::ConvertEncodedImageToJavaEncodedImage( 215 jobject VideoDecoderWrapper::ConvertEncodedImageToJavaEncodedImage(
198 JNIEnv* jni, 216 JNIEnv* jni,
199 const webrtc::EncodedImage& image) { 217 const webrtc::EncodedImage& image) {
200 jobject buffer = jni->NewDirectByteBuffer(image._buffer, image._length); 218 jobject buffer = jni->NewDirectByteBuffer(image._buffer, image._length);
201 jfieldID frame_type_field; 219 jfieldID frame_type_field;
202 switch (image._frameType) { 220 switch (image._frameType) {
203 case webrtc::kEmptyFrame: 221 case webrtc::kEmptyFrame:
204 frame_type_field = empty_frame_field_; 222 frame_type_field = empty_frame_field_;
(...skipping 30 matching lines...) Expand all
235 InitDecodeInternal(jni); 253 InitDecodeInternal(jni);
236 } 254 }
237 255
238 LOG(LS_WARNING) << "Falling back to software decoder."; 256 LOG(LS_WARNING) << "Falling back to software decoder.";
239 return WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE; 257 return WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE;
240 } else { 258 } else {
241 return value; 259 return value;
242 } 260 }
243 } 261 }
244 262
263 rtc::Optional<uint8_t> VideoDecoderWrapper::ParseQP(
264 const webrtc::EncodedImage& input_image) {
265 if (input_image.qp_ != -1) {
266 return rtc::Optional<uint8_t>(input_image.qp_);
267 }
268
269 rtc::Optional<uint8_t> qp;
270 switch (codec_settings_.codecType) {
271 case webrtc::kVideoCodecVP8: {
272 int qp_int;
273 if (webrtc::vp8::GetQp(input_image._buffer, input_image._length,
274 &qp_int)) {
275 qp = rtc::Optional<uint8_t>(qp_int);
276 }
277 break;
278 }
279 case webrtc::kVideoCodecVP9: {
280 int qp_int;
281 if (webrtc::vp9::GetQp(input_image._buffer, input_image._length,
282 &qp_int)) {
283 qp = rtc::Optional<uint8_t>(qp_int);
284 }
285 break;
286 }
287 case webrtc::kVideoCodecH264: {
288 h264_bitstream_parser_.ParseBitstream(input_image._buffer,
289 input_image._length);
290 int qp_int;
291 if (h264_bitstream_parser_.GetLastSliceQp(&qp_int)) {
292 qp = rtc::Optional<uint8_t>(qp_int);
293 }
294 break;
295 }
296 default:
297 break; // Default is to not provide QP.
298 }
299 return qp;
300 }
301
245 JOW(void, VideoDecoderWrapperCallback_nativeOnDecodedFrame) 302 JOW(void, VideoDecoderWrapperCallback_nativeOnDecodedFrame)
246 (JNIEnv* jni, 303 (JNIEnv* jni,
247 jclass, 304 jclass,
248 jlong jnative_decoder, 305 jlong jnative_decoder,
249 jobject jframe, 306 jobject jframe,
250 jobject jdecode_time_ms, 307 jobject jdecode_time_ms,
251 jobject jqp) { 308 jobject jqp) {
252 VideoDecoderWrapper* native_decoder = 309 VideoDecoderWrapper* native_decoder =
253 reinterpret_cast<VideoDecoderWrapper*>(jnative_decoder); 310 reinterpret_cast<VideoDecoderWrapper*>(jnative_decoder);
254 native_decoder->OnDecodedFrame(jni, jframe, jdecode_time_ms, jqp); 311 native_decoder->OnDecodedFrame(jni, jframe, jdecode_time_ms, jqp);
255 } 312 }
256 313
257 } // namespace webrtc_jni 314 } // namespace webrtc_jni
OLDNEW
« no previous file with comments | « webrtc/sdk/android/src/jni/videodecoderwrapper.h ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698