blob: f2a5ab27192ad33db16e2acf3517e1b2dede1219 [file] [log] [blame]
philipelc707ab72016-04-01 02:01:54 -07001/*
2 * Copyright (c) 2016 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
philipel266f0a42016-11-28 08:49:07 -080011#include "webrtc/base/checks.h"
philipele87c8762017-05-19 06:38:50 -070012#include "webrtc/common_video/h264/h264_common.h"
philipelc707ab72016-04-01 02:01:54 -070013#include "webrtc/modules/video_coding/frame_object.h"
philipelc707ab72016-04-01 02:01:54 -070014#include "webrtc/modules/video_coding/packet_buffer.h"
15
16namespace webrtc {
17namespace video_coding {
18
philipela1059872016-05-09 11:41:48 +020019FrameObject::FrameObject()
20 : picture_id(0),
21 spatial_layer(0),
philipelbe7a9e52016-05-19 12:19:35 +020022 timestamp(0),
philipela1059872016-05-09 11:41:48 +020023 num_references(0),
24 inter_layer_predicted(false) {}
25
philipelc707ab72016-04-01 02:01:54 -070026RtpFrameObject::RtpFrameObject(PacketBuffer* packet_buffer,
philipel02447bc2016-05-13 06:01:03 -070027 uint16_t first_seq_num,
philipel5ceaaae2016-05-24 10:20:47 +020028 uint16_t last_seq_num,
29 size_t frame_size,
philipelb4d31082016-07-11 08:46:29 -070030 int times_nacked,
31 int64_t received_time)
philipelc707ab72016-04-01 02:01:54 -070032 : packet_buffer_(packet_buffer),
philipel02447bc2016-05-13 06:01:03 -070033 first_seq_num_(first_seq_num),
philipel5ceaaae2016-05-24 10:20:47 +020034 last_seq_num_(last_seq_num),
philipelb4d31082016-07-11 08:46:29 -070035 received_time_(received_time),
philipel5ceaaae2016-05-24 10:20:47 +020036 times_nacked_(times_nacked) {
philipel266f0a42016-11-28 08:49:07 -080037 VCMPacket* first_packet = packet_buffer_->GetPacket(first_seq_num);
philipel7d79e632017-05-23 08:19:11 -070038 RTC_CHECK(first_packet);
philipel36928452016-11-07 10:42:36 +010039
philipel266f0a42016-11-28 08:49:07 -080040 // RtpFrameObject members
41 frame_type_ = first_packet->frameType;
42 codec_type_ = first_packet->codec;
philipel36928452016-11-07 10:42:36 +010043
philipel266f0a42016-11-28 08:49:07 -080044 // TODO(philipel): Remove when encoded image is replaced by FrameObject.
45 // VCMEncodedFrame members
46 CopyCodecSpecific(&first_packet->video_header);
47 _completeFrame = true;
48 _payloadType = first_packet->payloadType;
49 _timeStamp = first_packet->timestamp;
50 ntp_time_ms_ = first_packet->ntp_time_ms_;
philipel36928452016-11-07 10:42:36 +010051
gnishb2a318b2017-05-10 09:21:33 -070052 // Setting frame's playout delays to the same values
53 // as of the first packet's.
54 SetPlayoutDelay(first_packet->video_header.playout_delay);
55
philipel266f0a42016-11-28 08:49:07 -080056 // Since FFmpeg use an optimized bitstream reader that reads in chunks of
57 // 32/64 bits we have to add at least that much padding to the buffer
58 // to make sure the decoder doesn't read out of bounds.
59 // NOTE! EncodedImage::_size is the size of the buffer (think capacity of
60 // an std::vector) and EncodedImage::_length is the actual size of
61 // the bitstream (think size of an std::vector).
62 if (codec_type_ == kVideoCodecH264)
63 _size = frame_size + EncodedImage::kBufferPaddingBytesH264;
64 else
65 _size = frame_size;
philipel552866c2016-07-04 16:18:55 +020066
philipel266f0a42016-11-28 08:49:07 -080067 _buffer = new uint8_t[_size];
68 _length = frame_size;
philipele87c8762017-05-19 06:38:50 -070069
70 // For H264 frames we can't determine the frame type by just looking at the
71 // first packet. Instead we consider the frame to be a keyframe if it
72 // contains an IDR NALU.
73 if (codec_type_ == kVideoCodecH264) {
74 _frameType = kVideoFrameDelta;
75 frame_type_ = kVideoFrameDelta;
76 for (uint16_t seq_num = first_seq_num;
philipel7d79e632017-05-23 08:19:11 -070077 seq_num != static_cast<uint16_t>(last_seq_num + 1) &&
78 _frameType == kVideoFrameDelta;
philipele87c8762017-05-19 06:38:50 -070079 ++seq_num) {
80 VCMPacket* packet = packet_buffer_->GetPacket(seq_num);
philipel7d79e632017-05-23 08:19:11 -070081 RTC_CHECK(packet);
philipele87c8762017-05-19 06:38:50 -070082 const RTPVideoHeaderH264& header = packet->video_header.codecHeader.H264;
83 for (size_t i = 0; i < header.nalus_length; ++i) {
84 if (header.nalus[i].type == H264::NaluType::kIdr) {
85 _frameType = kVideoFrameKey;
86 frame_type_ = kVideoFrameKey;
87 break;
88 }
89 }
90 }
91 } else {
92 _frameType = first_packet->frameType;
93 frame_type_ = first_packet->frameType;
94 }
95
philipel266f0a42016-11-28 08:49:07 -080096 GetBitstream(_buffer);
philipel59e99b72017-01-12 03:26:04 -080097 _encodedWidth = first_packet->width;
98 _encodedHeight = first_packet->height;
philipel266f0a42016-11-28 08:49:07 -080099
100 // FrameObject members
101 timestamp = first_packet->timestamp;
102
103 VCMPacket* last_packet = packet_buffer_->GetPacket(last_seq_num);
philipel7d79e632017-05-23 08:19:11 -0700104 RTC_CHECK(last_packet && last_packet->markerBit);
philipel266f0a42016-11-28 08:49:07 -0800105 // http://www.etsi.org/deliver/etsi_ts/126100_126199/126114/12.07.00_60/
106 // ts_126114v120700p.pdf Section 7.4.5.
107 // The MTSI client shall add the payload bytes as defined in this clause
108 // onto the last RTP packet in each group of packets which make up a key
109 // frame (I-frame or IDR frame in H.264 (AVC), or an IRAP picture in H.265
110 // (HEVC)).
111 rotation_ = last_packet->video_header.rotation;
112 _rotation_set = true;
ilnik00d802b2017-04-11 10:34:31 -0700113 content_type_ = last_packet->video_header.content_type;
philipel02447bc2016-05-13 06:01:03 -0700114}
philipelc707ab72016-04-01 02:01:54 -0700115
116RtpFrameObject::~RtpFrameObject() {
117 packet_buffer_->ReturnFrame(this);
118}
119
philipelf4139332016-04-20 10:26:34 +0200120uint16_t RtpFrameObject::first_seq_num() const {
philipel02447bc2016-05-13 06:01:03 -0700121 return first_seq_num_;
philipelc707ab72016-04-01 02:01:54 -0700122}
123
philipelf4139332016-04-20 10:26:34 +0200124uint16_t RtpFrameObject::last_seq_num() const {
philipel02447bc2016-05-13 06:01:03 -0700125 return last_seq_num_;
126}
127
philipel5ceaaae2016-05-24 10:20:47 +0200128int RtpFrameObject::times_nacked() const {
129 return times_nacked_;
130}
131
philipel02447bc2016-05-13 06:01:03 -0700132FrameType RtpFrameObject::frame_type() const {
133 return frame_type_;
134}
135
136VideoCodecType RtpFrameObject::codec_type() const {
137 return codec_type_;
philipelc707ab72016-04-01 02:01:54 -0700138}
139
philipelc707ab72016-04-01 02:01:54 -0700140bool RtpFrameObject::GetBitstream(uint8_t* destination) const {
141 return packet_buffer_->GetBitstream(*this, destination);
142}
143
philipelb4d31082016-07-11 08:46:29 -0700144uint32_t RtpFrameObject::Timestamp() const {
145 return timestamp_;
146}
147
148int64_t RtpFrameObject::ReceivedTime() const {
149 return received_time_;
150}
151
152int64_t RtpFrameObject::RenderTime() const {
153 return _renderTimeMs;
154}
155
philipele0754302017-01-25 08:56:23 -0800156bool RtpFrameObject::delayed_by_retransmission() const {
157 return times_nacked() > 0;
158}
159
philipel88488282016-11-03 08:56:54 -0700160rtc::Optional<RTPVideoTypeHeader> RtpFrameObject::GetCodecHeader() const {
161 rtc::CritScope lock(&packet_buffer_->crit_);
philipel02447bc2016-05-13 06:01:03 -0700162 VCMPacket* packet = packet_buffer_->GetPacket(first_seq_num_);
163 if (!packet)
philipel88488282016-11-03 08:56:54 -0700164 return rtc::Optional<RTPVideoTypeHeader>();
165 return rtc::Optional<RTPVideoTypeHeader>(packet->video_header.codecHeader);
philipel02447bc2016-05-13 06:01:03 -0700166}
167
philipelc707ab72016-04-01 02:01:54 -0700168} // namespace video_coding
169} // namespace webrtc