1 /* 2 * Copyright (c) 2011 The WebRTC project authors. All Rights Reserved. 3 * 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 6 * tree. An additional intellectual property rights grant can be found 7 * in the file PATENTS. All contributing project authors may 8 * be found in the AUTHORS file in the root of the source tree. 9 */ 10 11 #include "webrtc/modules/rtp_rtcp/source/rtp_format_vp8.h" 12 13 #include <assert.h> // assert 14 #include <string.h> // memcpy 15 16 #include <vector> 17 18 #include "webrtc/modules/rtp_rtcp/source/vp8_partition_aggregator.h" 19 20 namespace webrtc { 21 22 // Define how the VP8PacketizerModes are implemented. 23 // Modes are: kStrict, kAggregate, kEqualSize. 24 const RtpFormatVp8::AggregationMode RtpFormatVp8::aggr_modes_[kNumModes] = 25 { kAggrNone, kAggrPartitions, kAggrFragments }; 26 const bool RtpFormatVp8::balance_modes_[kNumModes] = 27 { true, true, true }; 28 const bool RtpFormatVp8::separate_first_modes_[kNumModes] = 29 { true, false, false }; 30 31 RtpFormatVp8::RtpFormatVp8(const uint8_t* payload_data, 32 uint32_t payload_size, 33 const RTPVideoHeaderVP8& hdr_info, 34 int max_payload_len, 35 const RTPFragmentationHeader& fragmentation, 36 VP8PacketizerMode mode) 37 : payload_data_(payload_data), 38 payload_size_(static_cast<int>(payload_size)), 39 vp8_fixed_payload_descriptor_bytes_(1), 40 aggr_mode_(aggr_modes_[mode]), 41 balance_(balance_modes_[mode]), 42 separate_first_(separate_first_modes_[mode]), 43 hdr_info_(hdr_info), 44 num_partitions_(fragmentation.fragmentationVectorSize), 45 max_payload_len_(max_payload_len), 46 packets_calculated_(false) { 47 part_info_.CopyFrom(fragmentation); 48 } 49 50 RtpFormatVp8::RtpFormatVp8(const uint8_t* payload_data, 51 uint32_t payload_size, 52 const RTPVideoHeaderVP8& hdr_info, 53 int max_payload_len) 54 : payload_data_(payload_data), 55 payload_size_(static_cast<int>(payload_size)), 56 part_info_(), 57 vp8_fixed_payload_descriptor_bytes_(1), 58 aggr_mode_(aggr_modes_[kEqualSize]), 59 balance_(balance_modes_[kEqualSize]), 60 separate_first_(separate_first_modes_[kEqualSize]), 61 hdr_info_(hdr_info), 62 num_partitions_(1), 63 max_payload_len_(max_payload_len), 64 packets_calculated_(false) { 65 part_info_.VerifyAndAllocateFragmentationHeader(1); 66 part_info_.fragmentationLength[0] = payload_size; 67 part_info_.fragmentationOffset[0] = 0; 68 } 69 70 RtpFormatVp8::~RtpFormatVp8() {} 71 72 int RtpFormatVp8::NextPacket(uint8_t* buffer, 73 int* bytes_to_send, 74 bool* last_packet) { 75 if (!packets_calculated_) { 76 int ret = 0; 77 if (aggr_mode_ == kAggrPartitions && balance_) { 78 ret = GeneratePacketsBalancedAggregates(); 79 } else { 80 ret = GeneratePackets(); 81 } 82 if (ret < 0) { 83 return ret; 84 } 85 } 86 if (packets_.empty()) { 87 return -1; 88 } 89 InfoStruct packet_info = packets_.front(); 90 packets_.pop(); 91 92 *bytes_to_send = WriteHeaderAndPayload(packet_info, buffer, max_payload_len_); 93 if (*bytes_to_send < 0) { 94 return -1; 95 } 96 97 *last_packet = packets_.empty(); 98 return packet_info.first_partition_ix; 99 } 100 101 int RtpFormatVp8::CalcNextSize(int max_payload_len, int remaining_bytes, 102 bool split_payload) const { 103 if (max_payload_len == 0 || remaining_bytes == 0) { 104 return 0; 105 } 106 if (!split_payload) { 107 return max_payload_len >= remaining_bytes ? remaining_bytes : 0; 108 } 109 110 if (balance_) { 111 // Balance payload sizes to produce (almost) equal size 112 // fragments. 113 // Number of fragments for remaining_bytes: 114 int num_frags = remaining_bytes / max_payload_len + 1; 115 // Number of bytes in this fragment: 116 return static_cast<int>(static_cast<double>(remaining_bytes) 117 / num_frags + 0.5); 118 } else { 119 return max_payload_len >= remaining_bytes ? remaining_bytes 120 : max_payload_len; 121 } 122 } 123 124 int RtpFormatVp8::GeneratePackets() { 125 if (max_payload_len_ < vp8_fixed_payload_descriptor_bytes_ 126 + PayloadDescriptorExtraLength() + 1) { 127 // The provided payload length is not long enough for the payload 128 // descriptor and one payload byte. Return an error. 129 return -1; 130 } 131 int total_bytes_processed = 0; 132 bool start_on_new_fragment = true; 133 bool beginning = true; 134 int part_ix = 0; 135 while (total_bytes_processed < payload_size_) { 136 int packet_bytes = 0; // How much data to send in this packet. 137 bool split_payload = true; // Splitting of partitions is initially allowed. 138 int remaining_in_partition = part_info_.fragmentationOffset[part_ix] - 139 total_bytes_processed + part_info_.fragmentationLength[part_ix]; 140 int rem_payload_len = max_payload_len_ - 141 (vp8_fixed_payload_descriptor_bytes_ + PayloadDescriptorExtraLength()); 142 int first_partition_in_packet = part_ix; 143 144 while (int next_size = CalcNextSize(rem_payload_len, remaining_in_partition, 145 split_payload)) { 146 packet_bytes += next_size; 147 rem_payload_len -= next_size; 148 remaining_in_partition -= next_size; 149 150 if (remaining_in_partition == 0 && !(beginning && separate_first_)) { 151 // Advance to next partition? 152 // Check that there are more partitions; verify that we are either 153 // allowed to aggregate fragments, or that we are allowed to 154 // aggregate intact partitions and that we started this packet 155 // with an intact partition (indicated by first_fragment_ == true). 156 if (part_ix + 1 < num_partitions_ && 157 ((aggr_mode_ == kAggrFragments) || 158 (aggr_mode_ == kAggrPartitions && start_on_new_fragment))) { 159 assert(part_ix < num_partitions_); 160 remaining_in_partition = part_info_.fragmentationLength[++part_ix]; 161 // Disallow splitting unless kAggrFragments. In kAggrPartitions, 162 // we can only aggregate intact partitions. 163 split_payload = (aggr_mode_ == kAggrFragments); 164 } 165 } else if (balance_ && remaining_in_partition > 0) { 166 break; 167 } 168 } 169 if (remaining_in_partition == 0) { 170 ++part_ix; // Advance to next partition. 171 } 172 assert(packet_bytes > 0); 173 174 QueuePacket(total_bytes_processed, packet_bytes, first_partition_in_packet, 175 start_on_new_fragment); 176 total_bytes_processed += packet_bytes; 177 start_on_new_fragment = (remaining_in_partition == 0); 178 beginning = false; // Next packet cannot be first packet in frame. 179 } 180 packets_calculated_ = true; 181 assert(total_bytes_processed == payload_size_); 182 return 0; 183 } 184 185 int RtpFormatVp8::GeneratePacketsBalancedAggregates() { 186 if (max_payload_len_ < vp8_fixed_payload_descriptor_bytes_ 187 + PayloadDescriptorExtraLength() + 1) { 188 // The provided payload length is not long enough for the payload 189 // descriptor and one payload byte. Return an error. 190 return -1; 191 } 192 std::vector<int> partition_decision; 193 const int overhead = vp8_fixed_payload_descriptor_bytes_ + 194 PayloadDescriptorExtraLength(); 195 const uint32_t max_payload_len = max_payload_len_ - overhead; 196 int min_size, max_size; 197 AggregateSmallPartitions(&partition_decision, &min_size, &max_size); 198 199 int total_bytes_processed = 0; 200 int part_ix = 0; 201 while (part_ix < num_partitions_) { 202 if (partition_decision[part_ix] == -1) { 203 // Split large partitions. 204 int remaining_partition = part_info_.fragmentationLength[part_ix]; 205 int num_fragments = Vp8PartitionAggregator::CalcNumberOfFragments( 206 remaining_partition, max_payload_len, overhead, min_size, max_size); 207 const int packet_bytes = 208 (remaining_partition + num_fragments - 1) / num_fragments; 209 for (int n = 0; n < num_fragments; ++n) { 210 const int this_packet_bytes = packet_bytes < remaining_partition ? 211 packet_bytes : remaining_partition; 212 QueuePacket(total_bytes_processed, this_packet_bytes, part_ix, 213 (n == 0)); 214 remaining_partition -= this_packet_bytes; 215 total_bytes_processed += this_packet_bytes; 216 if (this_packet_bytes < min_size) { 217 min_size = this_packet_bytes; 218 } 219 if (this_packet_bytes > max_size) { 220 max_size = this_packet_bytes; 221 } 222 } 223 assert(remaining_partition == 0); 224 ++part_ix; 225 } else { 226 int this_packet_bytes = 0; 227 const int first_partition_in_packet = part_ix; 228 const int aggregation_index = partition_decision[part_ix]; 229 while (static_cast<size_t>(part_ix) < partition_decision.size() && 230 partition_decision[part_ix] == aggregation_index) { 231 // Collect all partitions that were aggregated into the same packet. 232 this_packet_bytes += part_info_.fragmentationLength[part_ix]; 233 ++part_ix; 234 } 235 QueuePacket(total_bytes_processed, this_packet_bytes, 236 first_partition_in_packet, true); 237 total_bytes_processed += this_packet_bytes; 238 } 239 } 240 packets_calculated_ = true; 241 return 0; 242 } 243 244 void RtpFormatVp8::AggregateSmallPartitions(std::vector<int>* partition_vec, 245 int* min_size, 246 int* max_size) { 247 assert(min_size && max_size); 248 *min_size = -1; 249 *max_size = -1; 250 assert(partition_vec); 251 partition_vec->assign(num_partitions_, -1); 252 const int overhead = vp8_fixed_payload_descriptor_bytes_ + 253 PayloadDescriptorExtraLength(); 254 const uint32_t max_payload_len = max_payload_len_ - overhead; 255 int first_in_set = 0; 256 int last_in_set = 0; 257 int num_aggregate_packets = 0; 258 // Find sets of partitions smaller than max_payload_len_. 259 while (first_in_set < num_partitions_) { 260 if (part_info_.fragmentationLength[first_in_set] < max_payload_len) { 261 // Found start of a set. 262 last_in_set = first_in_set; 263 while (last_in_set + 1 < num_partitions_ && 264 part_info_.fragmentationLength[last_in_set + 1] < max_payload_len) { 265 ++last_in_set; 266 } 267 // Found end of a set. Run optimized aggregator. It is ok if start == end. 268 Vp8PartitionAggregator aggregator(part_info_, first_in_set, 269 last_in_set); 270 if (*min_size >= 0 && *max_size >= 0) { 271 aggregator.SetPriorMinMax(*min_size, *max_size); 272 } 273 Vp8PartitionAggregator::ConfigVec optimal_config = 274 aggregator.FindOptimalConfiguration(max_payload_len, overhead); 275 aggregator.CalcMinMax(optimal_config, min_size, max_size); 276 for (int i = first_in_set, j = 0; i <= last_in_set; ++i, ++j) { 277 // Transfer configuration for this set of partitions to the joint 278 // partition vector representing all partitions in the frame. 279 (*partition_vec)[i] = num_aggregate_packets + optimal_config[j]; 280 } 281 num_aggregate_packets += optimal_config.back() + 1; 282 first_in_set = last_in_set; 283 } 284 ++first_in_set; 285 } 286 } 287 288 void RtpFormatVp8::QueuePacket(int start_pos, 289 int packet_size, 290 int first_partition_in_packet, 291 bool start_on_new_fragment) { 292 // Write info to packet info struct and store in packet info queue. 293 InfoStruct packet_info; 294 packet_info.payload_start_pos = start_pos; 295 packet_info.size = packet_size; 296 packet_info.first_partition_ix = first_partition_in_packet; 297 packet_info.first_fragment = start_on_new_fragment; 298 packets_.push(packet_info); 299 } 300 301 int RtpFormatVp8::WriteHeaderAndPayload(const InfoStruct& packet_info, 302 uint8_t* buffer, 303 int buffer_length) const { 304 // Write the VP8 payload descriptor. 305 // 0 306 // 0 1 2 3 4 5 6 7 8 307 // +-+-+-+-+-+-+-+-+-+ 308 // |X| |N|S| PART_ID | 309 // +-+-+-+-+-+-+-+-+-+ 310 // X: |I|L|T|K| | (mandatory if any of the below are used) 311 // +-+-+-+-+-+-+-+-+-+ 312 // I: |PictureID (8/16b)| (optional) 313 // +-+-+-+-+-+-+-+-+-+ 314 // L: | TL0PIC_IDX | (optional) 315 // +-+-+-+-+-+-+-+-+-+ 316 // T/K: |TID:Y| KEYIDX | (optional) 317 // +-+-+-+-+-+-+-+-+-+ 318 319 assert(packet_info.size > 0); 320 buffer[0] = 0; 321 if (XFieldPresent()) buffer[0] |= kXBit; 322 if (hdr_info_.nonReference) buffer[0] |= kNBit; 323 if (packet_info.first_fragment) buffer[0] |= kSBit; 324 buffer[0] |= (packet_info.first_partition_ix & kPartIdField); 325 326 const int extension_length = WriteExtensionFields(buffer, buffer_length); 327 328 memcpy(&buffer[vp8_fixed_payload_descriptor_bytes_ + extension_length], 329 &payload_data_[packet_info.payload_start_pos], packet_info.size); 330 331 // Return total length of written data. 332 return packet_info.size + vp8_fixed_payload_descriptor_bytes_ 333 + extension_length; 334 } 335 336 int RtpFormatVp8::WriteExtensionFields(uint8_t* buffer, 337 int buffer_length) const { 338 int extension_length = 0; 339 if (XFieldPresent()) { 340 uint8_t* x_field = buffer + vp8_fixed_payload_descriptor_bytes_; 341 *x_field = 0; 342 extension_length = 1; // One octet for the X field. 343 if (PictureIdPresent()) { 344 if (WritePictureIDFields(x_field, buffer, buffer_length, 345 &extension_length) < 0) { 346 return -1; 347 } 348 } 349 if (TL0PicIdxFieldPresent()) { 350 if (WriteTl0PicIdxFields(x_field, buffer, buffer_length, 351 &extension_length) < 0) { 352 return -1; 353 } 354 } 355 if (TIDFieldPresent() || KeyIdxFieldPresent()) { 356 if (WriteTIDAndKeyIdxFields(x_field, buffer, buffer_length, 357 &extension_length) < 0) { 358 return -1; 359 } 360 } 361 assert(extension_length == PayloadDescriptorExtraLength()); 362 } 363 return extension_length; 364 } 365 366 int RtpFormatVp8::WritePictureIDFields(uint8_t* x_field, 367 uint8_t* buffer, 368 int buffer_length, 369 int* extension_length) const { 370 *x_field |= kIBit; 371 const int pic_id_length = WritePictureID( 372 buffer + vp8_fixed_payload_descriptor_bytes_ + *extension_length, 373 buffer_length - vp8_fixed_payload_descriptor_bytes_ 374 - *extension_length); 375 if (pic_id_length < 0) return -1; 376 *extension_length += pic_id_length; 377 return 0; 378 } 379 380 int RtpFormatVp8::WritePictureID(uint8_t* buffer, 381 int buffer_length) const { 382 const uint16_t pic_id = 383 static_cast<uint16_t> (hdr_info_.pictureId); 384 int picture_id_len = PictureIdLength(); 385 if (picture_id_len > buffer_length) return -1; 386 if (picture_id_len == 2) { 387 buffer[0] = 0x80 | ((pic_id >> 8) & 0x7F); 388 buffer[1] = pic_id & 0xFF; 389 } else if (picture_id_len == 1) { 390 buffer[0] = pic_id & 0x7F; 391 } 392 return picture_id_len; 393 } 394 395 int RtpFormatVp8::WriteTl0PicIdxFields(uint8_t* x_field, 396 uint8_t* buffer, 397 int buffer_length, 398 int* extension_length) const { 399 if (buffer_length < vp8_fixed_payload_descriptor_bytes_ + *extension_length 400 + 1) { 401 return -1; 402 } 403 *x_field |= kLBit; 404 buffer[vp8_fixed_payload_descriptor_bytes_ 405 + *extension_length] = hdr_info_.tl0PicIdx; 406 ++*extension_length; 407 return 0; 408 } 409 410 int RtpFormatVp8::WriteTIDAndKeyIdxFields(uint8_t* x_field, 411 uint8_t* buffer, 412 int buffer_length, 413 int* extension_length) const { 414 if (buffer_length < vp8_fixed_payload_descriptor_bytes_ + *extension_length 415 + 1) { 416 return -1; 417 } 418 uint8_t* data_field = 419 &buffer[vp8_fixed_payload_descriptor_bytes_ + *extension_length]; 420 *data_field = 0; 421 if (TIDFieldPresent()) { 422 *x_field |= kTBit; 423 assert(hdr_info_.temporalIdx >= 0 && hdr_info_.temporalIdx <= 3); 424 *data_field |= hdr_info_.temporalIdx << 6; 425 *data_field |= hdr_info_.layerSync ? kYBit : 0; 426 } 427 if (KeyIdxFieldPresent()) { 428 *x_field |= kKBit; 429 *data_field |= (hdr_info_.keyIdx & kKeyIdxField); 430 } 431 ++*extension_length; 432 return 0; 433 } 434 435 int RtpFormatVp8::PayloadDescriptorExtraLength() const { 436 int length_bytes = PictureIdLength(); 437 if (TL0PicIdxFieldPresent()) ++length_bytes; 438 if (TIDFieldPresent() || KeyIdxFieldPresent()) ++length_bytes; 439 if (length_bytes > 0) ++length_bytes; // Include the extension field. 440 return length_bytes; 441 } 442 443 int RtpFormatVp8::PictureIdLength() const { 444 if (hdr_info_.pictureId == kNoPictureId) { 445 return 0; 446 } 447 if (hdr_info_.pictureId <= 0x7F) { 448 return 1; 449 } 450 return 2; 451 } 452 453 bool RtpFormatVp8::XFieldPresent() const { 454 return (TIDFieldPresent() || TL0PicIdxFieldPresent() || PictureIdPresent() 455 || KeyIdxFieldPresent()); 456 } 457 458 bool RtpFormatVp8::TIDFieldPresent() const { 459 assert((hdr_info_.layerSync == false) || 460 (hdr_info_.temporalIdx != kNoTemporalIdx)); 461 return (hdr_info_.temporalIdx != kNoTemporalIdx); 462 } 463 464 bool RtpFormatVp8::KeyIdxFieldPresent() const { 465 return (hdr_info_.keyIdx != kNoKeyIdx); 466 } 467 468 bool RtpFormatVp8::TL0PicIdxFieldPresent() const { 469 return (hdr_info_.tl0PicIdx != kNoTl0PicIdx); 470 } 471 } // namespace webrtc 472