/* * Copyright (c) 2011 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/rtp_rtcp/source/rtp_format_vp8.h" #include // memcpy #include // assert namespace webrtc { // Define how the VP8PacketizerModes are implemented. // Modes are: kStrict, kAggregate, kSloppy. const RtpFormatVp8::AggregationMode RtpFormatVp8::aggr_modes_[kNumModes] = { kAggrNone, kAggrPartitions, kAggrFragments }; const bool RtpFormatVp8::balance_modes_[kNumModes] = { true, true, false }; const bool RtpFormatVp8::separate_first_modes_[kNumModes] = { true, false, false }; RtpFormatVp8::RtpFormatVp8(const WebRtc_UWord8* payload_data, WebRtc_UWord32 payload_size, const RTPVideoHeaderVP8& hdr_info, const RTPFragmentationHeader& fragmentation, VP8PacketizerMode mode) : payload_data_(payload_data), payload_size_(static_cast(payload_size)), payload_bytes_sent_(0), part_ix_(0), beginning_(true), first_fragment_(true), vp8_fixed_payload_descriptor_bytes_(1), aggr_mode_(aggr_modes_[mode]), balance_(balance_modes_[mode]), separate_first_(separate_first_modes_[mode]), hdr_info_(hdr_info), first_partition_in_packet_(0) { part_info_ = fragmentation; } RtpFormatVp8::RtpFormatVp8(const WebRtc_UWord8* payload_data, WebRtc_UWord32 payload_size, const RTPVideoHeaderVP8& hdr_info) : payload_data_(payload_data), payload_size_(static_cast(payload_size)), part_info_(), payload_bytes_sent_(0), part_ix_(0), beginning_(true), first_fragment_(true), vp8_fixed_payload_descriptor_bytes_(1), aggr_mode_(aggr_modes_[kSloppy]), balance_(balance_modes_[kSloppy]), separate_first_(separate_first_modes_[kSloppy]), hdr_info_(hdr_info), first_partition_in_packet_(0) { part_info_.VerifyAndAllocateFragmentationHeader(1); part_info_.fragmentationLength[0] = payload_size; part_info_.fragmentationOffset[0] = 0; } int RtpFormatVp8::CalcNextSize(int max_payload_len, int remaining_bytes, bool split_payload) const { if (max_payload_len == 0 || remaining_bytes == 0) { return 0; } if (!split_payload) { return max_payload_len >= remaining_bytes ? remaining_bytes : 0; } if (balance_) { // Balance payload sizes to produce (almost) equal size // fragments. // Number of fragments for remaining_bytes: int num_frags = remaining_bytes / max_payload_len + 1; // Number of bytes in this fragment: return static_cast(static_cast(remaining_bytes) / num_frags + 0.5); } else { return max_payload_len >= remaining_bytes ? remaining_bytes : max_payload_len; } } int RtpFormatVp8::NextPacket(int max_payload_len, WebRtc_UWord8* buffer, int* bytes_to_send, bool* last_packet) { if (max_payload_len < vp8_fixed_payload_descriptor_bytes_ + PayloadDescriptorExtraLength() + 1) { // The provided payload length is not long enough for the payload // descriptor and one payload byte. Return an error. return -1; } const int num_partitions = part_info_.fragmentationVectorSize; int send_bytes = 0; // How much data to send in this packet. bool split_payload = true; // Splitting of partitions is initially allowed. int remaining_in_partition = part_info_.fragmentationOffset[part_ix_] - payload_bytes_sent_ + part_info_.fragmentationLength[part_ix_] + PayloadDescriptorExtraLength(); int rem_payload_len = max_payload_len - vp8_fixed_payload_descriptor_bytes_; first_partition_in_packet_ = part_ix_; if (first_partition_in_packet_ > 8) return -1; while (int next_size = CalcNextSize(rem_payload_len, remaining_in_partition, split_payload)) { send_bytes += next_size; rem_payload_len -= next_size; remaining_in_partition -= next_size; if (remaining_in_partition == 0 && !(beginning_ && separate_first_)) { // Advance to next partition? // Check that there are more partitions; verify that we are either // allowed to aggregate fragments, or that we are allowed to // aggregate intact partitions and that we started this packet // with an intact partition (indicated by first_fragment_ == true). if (part_ix_ + 1 < num_partitions && ((aggr_mode_ == kAggrFragments) || (aggr_mode_ == kAggrPartitions && first_fragment_))) { remaining_in_partition = part_info_.fragmentationLength[++part_ix_]; // Disallow splitting unless kAggrFragments. In kAggrPartitions, // we can only aggregate intact partitions. split_payload = (aggr_mode_ == kAggrFragments); } } else if (balance_ && remaining_in_partition > 0) { break; } } if (remaining_in_partition == 0) { ++part_ix_; // Advance to next partition. } send_bytes -= PayloadDescriptorExtraLength(); // Remove extra length again. assert(send_bytes > 0); // Write the payload header and the payload to buffer. *bytes_to_send = WriteHeaderAndPayload(send_bytes, buffer, max_payload_len); if (*bytes_to_send < 0) { return -1; } beginning_ = false; // Next packet cannot be first packet in frame. // Next packet starts new fragment if this ended one. first_fragment_ = (remaining_in_partition == 0); *last_packet = (payload_bytes_sent_ >= payload_size_); assert(!*last_packet || (payload_bytes_sent_ == payload_size_)); return first_partition_in_packet_; } int RtpFormatVp8::WriteHeaderAndPayload(int payload_bytes, WebRtc_UWord8* buffer, int buffer_length) { // Write the VP8 payload descriptor. // 0 // 0 1 2 3 4 5 6 7 8 // +-+-+-+-+-+-+-+-+-+ // |X| |N|S| PART_ID | // +-+-+-+-+-+-+-+-+-+ // X: |I|L|T|K| | (mandatory if any of the below are used) // +-+-+-+-+-+-+-+-+-+ // I: |PictureID (8/16b)| (optional) // +-+-+-+-+-+-+-+-+-+ // L: | TL0PIC_IDX | (optional) // +-+-+-+-+-+-+-+-+-+ // T/K: |TID:Y| KEYIDX | (optional) // +-+-+-+-+-+-+-+-+-+ assert(payload_bytes > 0); assert(payload_bytes_sent_ + payload_bytes <= payload_size_); assert(vp8_fixed_payload_descriptor_bytes_ + PayloadDescriptorExtraLength() + payload_bytes <= buffer_length); buffer[0] = 0; if (XFieldPresent()) buffer[0] |= kXBit; if (hdr_info_.nonReference) buffer[0] |= kNBit; if (first_fragment_) buffer[0] |= kSBit; buffer[0] |= (first_partition_in_packet_ & kPartIdField); const int extension_length = WriteExtensionFields(buffer, buffer_length); memcpy(&buffer[vp8_fixed_payload_descriptor_bytes_ + extension_length], &payload_data_[payload_bytes_sent_], payload_bytes); payload_bytes_sent_ += payload_bytes; // Return total length of written data. return payload_bytes + vp8_fixed_payload_descriptor_bytes_ + extension_length; } int RtpFormatVp8::WriteExtensionFields(WebRtc_UWord8* buffer, int buffer_length) const { int extension_length = 0; if (XFieldPresent()) { WebRtc_UWord8* x_field = buffer + vp8_fixed_payload_descriptor_bytes_; *x_field = 0; extension_length = 1; // One octet for the X field. if (PictureIdPresent()) { if (WritePictureIDFields(x_field, buffer, buffer_length, &extension_length) < 0) { return -1; } } if (TL0PicIdxFieldPresent()) { if (WriteTl0PicIdxFields(x_field, buffer, buffer_length, &extension_length) < 0) { return -1; } } if (TIDFieldPresent() || KeyIdxFieldPresent()) { if (WriteTIDAndKeyIdxFields(x_field, buffer, buffer_length, &extension_length) < 0) { return -1; } } assert(extension_length == PayloadDescriptorExtraLength()); } return extension_length; } int RtpFormatVp8::WritePictureIDFields(WebRtc_UWord8* x_field, WebRtc_UWord8* buffer, int buffer_length, int* extension_length) const { *x_field |= kIBit; const int pic_id_length = WritePictureID( buffer + vp8_fixed_payload_descriptor_bytes_ + *extension_length, buffer_length - vp8_fixed_payload_descriptor_bytes_ - *extension_length); if (pic_id_length < 0) return -1; *extension_length += pic_id_length; return 0; } int RtpFormatVp8::WritePictureID(WebRtc_UWord8* buffer, int buffer_length) const { const WebRtc_UWord16 pic_id = static_cast (hdr_info_.pictureId); int picture_id_len = PictureIdLength(); if (picture_id_len > buffer_length) return -1; if (picture_id_len == 2) { buffer[0] = 0x80 | ((pic_id >> 8) & 0x7F); buffer[1] = pic_id & 0xFF; } else if (picture_id_len == 1) { buffer[0] = pic_id & 0x7F; } return picture_id_len; } int RtpFormatVp8::WriteTl0PicIdxFields(WebRtc_UWord8* x_field, WebRtc_UWord8* buffer, int buffer_length, int* extension_length) const { if (buffer_length < vp8_fixed_payload_descriptor_bytes_ + *extension_length + 1) { return -1; } *x_field |= kLBit; buffer[vp8_fixed_payload_descriptor_bytes_ + *extension_length] = hdr_info_.tl0PicIdx; ++*extension_length; return 0; } int RtpFormatVp8::WriteTIDAndKeyIdxFields(WebRtc_UWord8* x_field, WebRtc_UWord8* buffer, int buffer_length, int* extension_length) const { if (buffer_length < vp8_fixed_payload_descriptor_bytes_ + *extension_length + 1) { return -1; } WebRtc_UWord8* data_field = &buffer[vp8_fixed_payload_descriptor_bytes_ + *extension_length]; *data_field = 0; if (TIDFieldPresent()) { *x_field |= kTBit; assert(hdr_info_.temporalIdx >= 0 && hdr_info_.temporalIdx <= 3); *data_field |= hdr_info_.temporalIdx << 6; *data_field |= hdr_info_.layerSync ? kYBit : 0; } if (KeyIdxFieldPresent()) { *x_field |= kKBit; *data_field |= (hdr_info_.keyIdx & kKeyIdxField); } ++*extension_length; return 0; } int RtpFormatVp8::PayloadDescriptorExtraLength() const { int length_bytes = PictureIdLength(); if (TL0PicIdxFieldPresent()) ++length_bytes; if (TIDFieldPresent() || KeyIdxFieldPresent()) ++length_bytes; if (length_bytes > 0) ++length_bytes; // Include the extension field. return length_bytes; } int RtpFormatVp8::PictureIdLength() const { if (hdr_info_.pictureId == kNoPictureId) { return 0; } if (hdr_info_.pictureId <= 0x7F) { return 1; } return 2; } bool RtpFormatVp8::XFieldPresent() const { return (TIDFieldPresent() || TL0PicIdxFieldPresent() || PictureIdPresent() || KeyIdxFieldPresent()); } bool RtpFormatVp8::TIDFieldPresent() const { assert((hdr_info_.layerSync == false) || (hdr_info_.temporalIdx != kNoTemporalIdx)); return (hdr_info_.temporalIdx != kNoTemporalIdx); } bool RtpFormatVp8::KeyIdxFieldPresent() const { return (hdr_info_.keyIdx != kNoKeyIdx); } bool RtpFormatVp8::TL0PicIdxFieldPresent() const { return (hdr_info_.tl0PicIdx != kNoTl0PicIdx); } } // namespace webrtc