| /* |
| * Copyright (c) 2015 The WebRTC project authors. All Rights Reserved. |
| * |
| * Use of this source code is governed by a BSD-style license |
| * that can be found in the LICENSE file in the root of the source |
| * tree. An additional intellectual property rights grant can be found |
| * in the file PATENTS. All contributing project authors may |
| * be found in the AUTHORS file in the root of the source tree. |
| */ |
| #include "modules/video_coding/utility/vp8_header_parser.h" |
| |
| #include "rtc_base/logging.h" |
| #include "rtc_base/system/arch.h" |
| |
| namespace webrtc { |
| |
| namespace vp8 { |
| namespace { |
| const size_t kCommonPayloadHeaderLength = 3; |
| const size_t kKeyPayloadHeaderLength = 10; |
| const int kMbFeatureTreeProbs = 3; |
| const int kNumMbSegments = 4; |
| const int kNumRefLfDeltas = 4; |
| const int kNumModeLfDeltas = 4; |
| |
| } // namespace |
| |
| // Bitstream parser according to |
| // https://tools.ietf.org/html/rfc6386#section-7.3 |
| void VP8InitBitReader(VP8BitReader* const br, |
| const uint8_t* start, |
| const uint8_t* end) { |
| br->range_ = 255; |
| br->buf_ = start; |
| br->buf_end_ = end; |
| br->value_ = 0; |
| br->bits_ = 0; |
| |
| // Read 2 bytes. |
| int i = 0; |
| while (++i <= 2) { |
| if (br->buf_ != br->buf_end_) { |
| br->value_ = br->value_ << 8 | *br->buf_++; |
| } else { |
| br->value_ = br->value_ << 8; |
| } |
| } |
| } |
| |
| // Bit decoder according to https://tools.ietf.org/html/rfc6386#section-7.3 |
| // Reads one bit from the bitstream, given that it has probability prob/256 to |
| // be 1. |
| int Vp8BitReaderGetBool(VP8BitReader* br, int prob) { |
| uint32_t split = 1 + (((br->range_ - 1) * prob) >> 8); |
| uint32_t split_hi = split << 8; |
| int retval = 0; |
| if (br->value_ >= split_hi) { |
| retval = 1; |
| br->range_ -= split; |
| br->value_ -= split_hi; |
| } else { |
| retval = 0; |
| br->range_ = split; |
| } |
| |
| while (br->range_ < 128) { |
| br->value_ <<= 1; |
| br->range_ <<= 1; |
| if (++br->bits_ == 8) { |
| br->bits_ = 0; |
| if (br->buf_ != br->buf_end_) { |
| br->value_ |= *br->buf_++; |
| } |
| } |
| } |
| return retval; |
| } |
| |
| uint32_t VP8GetValue(VP8BitReader* br, int num_bits) { |
| uint32_t v = 0; |
| while (num_bits--) { |
| // According to https://tools.ietf.org/html/rfc6386 |
| // Probability 128/256 is used to encode header fields. |
| v = (v << 1) | Vp8BitReaderGetBool(br, 128); |
| } |
| return v; |
| } |
| |
| // Not a read_signed_literal() from RFC 6386! |
| // This one is used to read e.g. quantizer_update, which is written as: |
| // L(num_bits), sign-bit. |
| int32_t VP8GetSignedValue(VP8BitReader* br, int num_bits) { |
| int v = VP8GetValue(br, num_bits); |
| int sign = VP8GetValue(br, 1); |
| return sign ? -v : v; |
| } |
| |
| static void ParseSegmentHeader(VP8BitReader* br) { |
| int use_segment = VP8GetValue(br, 1); |
| if (use_segment) { |
| int update_map = VP8GetValue(br, 1); |
| if (VP8GetValue(br, 1)) { // update_segment_feature_data. |
| VP8GetValue(br, 1); // segment_feature_mode. |
| int s; |
| for (s = 0; s < kNumMbSegments; ++s) { |
| bool quantizer_update = VP8GetValue(br, 1); |
| if (quantizer_update) { |
| VP8GetSignedValue(br, 7); |
| } |
| } |
| for (s = 0; s < kNumMbSegments; ++s) { |
| bool loop_filter_update = VP8GetValue(br, 1); |
| if (loop_filter_update) { |
| VP8GetSignedValue(br, 6); |
| } |
| } |
| } |
| if (update_map) { |
| int s; |
| for (s = 0; s < kMbFeatureTreeProbs; ++s) { |
| bool segment_prob_update = VP8GetValue(br, 1); |
| if (segment_prob_update) { |
| VP8GetValue(br, 8); |
| } |
| } |
| } |
| } |
| } |
| |
| static void ParseFilterHeader(VP8BitReader* br) { |
| VP8GetValue(br, 1); // filter_type. |
| VP8GetValue(br, 6); // loop_filter_level. |
| VP8GetValue(br, 3); // sharpness_level. |
| |
| // mb_lf_adjustments. |
| int loop_filter_adj_enable = VP8GetValue(br, 1); |
| if (loop_filter_adj_enable) { |
| int mode_ref_lf_delta_update = VP8GetValue(br, 1); |
| if (mode_ref_lf_delta_update) { |
| int i; |
| for (i = 0; i < kNumRefLfDeltas; ++i) { |
| int ref_frame_delta_update_flag = VP8GetValue(br, 1); |
| if (ref_frame_delta_update_flag) { |
| VP8GetSignedValue(br, 6); // delta_magnitude. |
| } |
| } |
| for (i = 0; i < kNumModeLfDeltas; ++i) { |
| int mb_mode_delta_update_flag = VP8GetValue(br, 1); |
| if (mb_mode_delta_update_flag) { |
| VP8GetSignedValue(br, 6); // delta_magnitude. |
| } |
| } |
| } |
| } |
| } |
| |
| bool GetQp(const uint8_t* buf, size_t length, int* qp) { |
| if (length < kCommonPayloadHeaderLength) { |
| RTC_LOG(LS_WARNING) << "Failed to get QP, invalid length."; |
| return false; |
| } |
| VP8BitReader br; |
| const uint32_t bits = buf[0] | (buf[1] << 8) | (buf[2] << 16); |
| int key_frame = !(bits & 1); |
| // Size of first partition in bytes. |
| uint32_t partition_length = (bits >> 5); |
| size_t header_length = kCommonPayloadHeaderLength; |
| if (key_frame) { |
| header_length = kKeyPayloadHeaderLength; |
| } |
| if (header_length + partition_length > length) { |
| RTC_LOG(LS_WARNING) << "Failed to get QP, invalid length: " << length; |
| return false; |
| } |
| buf += header_length; |
| |
| VP8InitBitReader(&br, buf, buf + partition_length); |
| if (key_frame) { |
| // Color space and pixel type. |
| VP8GetValue(&br, 1); |
| VP8GetValue(&br, 1); |
| } |
| ParseSegmentHeader(&br); |
| ParseFilterHeader(&br); |
| // Parse log2_nbr_of_dct_partitions value. |
| VP8GetValue(&br, 2); |
| // Base QP. |
| const int base_q0 = VP8GetValue(&br, 7); |
| if (br.buf_ == br.buf_end_) { |
| RTC_LOG(LS_WARNING) << "Failed to get QP, bitstream is truncated or" |
| " corrupted."; |
| return false; |
| } |
| *qp = base_q0; |
| return true; |
| } |
| |
| } // namespace vp8 |
| |
| } // namespace webrtc |