blob: 6eb28de1cd72ff97cb69bc365c6847f6a889c88a [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
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020011#include "modules/video_coding/frame_object.h"
Rasmus Brandt58b72912017-10-25 11:09:23 +020012
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020013#include "common_video/h264/h264_common.h"
14#include "modules/video_coding/packet_buffer.h"
15#include "rtc_base/checks.h"
philipelc707ab72016-04-01 02:01:54 -070016
17namespace webrtc {
18namespace video_coding {
19
philipela1059872016-05-09 11:41:48 +020020FrameObject::FrameObject()
21 : picture_id(0),
22 spatial_layer(0),
philipelbe7a9e52016-05-19 12:19:35 +020023 timestamp(0),
philipela1059872016-05-09 11:41:48 +020024 num_references(0),
25 inter_layer_predicted(false) {}
26
philipelc707ab72016-04-01 02:01:54 -070027RtpFrameObject::RtpFrameObject(PacketBuffer* packet_buffer,
philipel02447bc2016-05-13 06:01:03 -070028 uint16_t first_seq_num,
philipel5ceaaae2016-05-24 10:20:47 +020029 uint16_t last_seq_num,
30 size_t frame_size,
philipelb4d31082016-07-11 08:46:29 -070031 int times_nacked,
32 int64_t received_time)
philipelc707ab72016-04-01 02:01:54 -070033 : packet_buffer_(packet_buffer),
philipel02447bc2016-05-13 06:01:03 -070034 first_seq_num_(first_seq_num),
philipel5ceaaae2016-05-24 10:20:47 +020035 last_seq_num_(last_seq_num),
sprangba050a62017-08-18 02:51:12 -070036 timestamp_(0),
philipelb4d31082016-07-11 08:46:29 -070037 received_time_(received_time),
philipel5ceaaae2016-05-24 10:20:47 +020038 times_nacked_(times_nacked) {
philipel266f0a42016-11-28 08:49:07 -080039 VCMPacket* first_packet = packet_buffer_->GetPacket(first_seq_num);
philipel7d79e632017-05-23 08:19:11 -070040 RTC_CHECK(first_packet);
philipel36928452016-11-07 10:42:36 +010041
philipel266f0a42016-11-28 08:49:07 -080042 // RtpFrameObject members
43 frame_type_ = first_packet->frameType;
44 codec_type_ = first_packet->codec;
philipel36928452016-11-07 10:42:36 +010045
Emircan Uysaler20f21332017-11-28 09:45:25 -080046 // Stereo codec appends CopyCodecSpecific to last packet to avoid copy.
47 VCMPacket* packet_with_codec_specific =
48 codec_type_ == kVideoCodecStereo ? packet_buffer_->GetPacket(last_seq_num)
49 : first_packet;
50
philipel266f0a42016-11-28 08:49:07 -080051 // TODO(philipel): Remove when encoded image is replaced by FrameObject.
52 // VCMEncodedFrame members
Emircan Uysaler20f21332017-11-28 09:45:25 -080053 CopyCodecSpecific(&packet_with_codec_specific->video_header);
philipel266f0a42016-11-28 08:49:07 -080054 _completeFrame = true;
55 _payloadType = first_packet->payloadType;
56 _timeStamp = first_packet->timestamp;
57 ntp_time_ms_ = first_packet->ntp_time_ms_;
Rasmus Brandt88f080a2017-11-02 14:28:06 +010058 _frameType = first_packet->frameType;
philipel36928452016-11-07 10:42:36 +010059
gnishb2a318b2017-05-10 09:21:33 -070060 // Setting frame's playout delays to the same values
61 // as of the first packet's.
62 SetPlayoutDelay(first_packet->video_header.playout_delay);
63
philipel266f0a42016-11-28 08:49:07 -080064 // Since FFmpeg use an optimized bitstream reader that reads in chunks of
65 // 32/64 bits we have to add at least that much padding to the buffer
66 // to make sure the decoder doesn't read out of bounds.
67 // NOTE! EncodedImage::_size is the size of the buffer (think capacity of
68 // an std::vector) and EncodedImage::_length is the actual size of
69 // the bitstream (think size of an std::vector).
70 if (codec_type_ == kVideoCodecH264)
71 _size = frame_size + EncodedImage::kBufferPaddingBytesH264;
72 else
73 _size = frame_size;
philipel552866c2016-07-04 16:18:55 +020074
philipel266f0a42016-11-28 08:49:07 -080075 _buffer = new uint8_t[_size];
76 _length = frame_size;
philipele87c8762017-05-19 06:38:50 -070077
philipel227f8b92017-08-04 06:39:31 -070078 bool bitstream_copied = GetBitstream(_buffer);
79 RTC_DCHECK(bitstream_copied);
philipel59e99b72017-01-12 03:26:04 -080080 _encodedWidth = first_packet->width;
81 _encodedHeight = first_packet->height;
philipel266f0a42016-11-28 08:49:07 -080082
83 // FrameObject members
84 timestamp = first_packet->timestamp;
85
86 VCMPacket* last_packet = packet_buffer_->GetPacket(last_seq_num);
kwibergee89e782017-08-09 17:22:01 -070087 RTC_CHECK(last_packet);
88 RTC_CHECK(last_packet->markerBit);
philipel266f0a42016-11-28 08:49:07 -080089 // http://www.etsi.org/deliver/etsi_ts/126100_126199/126114/12.07.00_60/
90 // ts_126114v120700p.pdf Section 7.4.5.
91 // The MTSI client shall add the payload bytes as defined in this clause
92 // onto the last RTP packet in each group of packets which make up a key
93 // frame (I-frame or IDR frame in H.264 (AVC), or an IRAP picture in H.265
94 // (HEVC)).
95 rotation_ = last_packet->video_header.rotation;
96 _rotation_set = true;
ilnik00d802b2017-04-11 10:34:31 -070097 content_type_ = last_packet->video_header.content_type;
sprangba050a62017-08-18 02:51:12 -070098 if (last_packet->video_header.video_timing.flags !=
99 TimingFrameFlags::kInvalid) {
ilnik04f4d122017-06-19 07:18:55 -0700100 // ntp_time_ms_ may be -1 if not estimated yet. This is not a problem,
101 // as this will be dealt with at the time of reporting.
ilnik04f4d122017-06-19 07:18:55 -0700102 timing_.encode_start_ms =
103 ntp_time_ms_ +
104 last_packet->video_header.video_timing.encode_start_delta_ms;
105 timing_.encode_finish_ms =
106 ntp_time_ms_ +
107 last_packet->video_header.video_timing.encode_finish_delta_ms;
108 timing_.packetization_finish_ms =
109 ntp_time_ms_ +
110 last_packet->video_header.video_timing.packetization_finish_delta_ms;
111 timing_.pacer_exit_ms =
112 ntp_time_ms_ +
113 last_packet->video_header.video_timing.pacer_exit_delta_ms;
114 timing_.network_timestamp_ms =
115 ntp_time_ms_ +
Danil Chapovalov996eb9e2017-10-30 17:14:41 +0100116 last_packet->video_header.video_timing.network_timestamp_delta_ms;
ilnik04f4d122017-06-19 07:18:55 -0700117 timing_.network2_timestamp_ms =
118 ntp_time_ms_ +
Danil Chapovalov996eb9e2017-10-30 17:14:41 +0100119 last_packet->video_header.video_timing.network2_timestamp_delta_ms;
ilnik04f4d122017-06-19 07:18:55 -0700120
121 timing_.receive_start_ms = first_packet->receive_time_ms;
122 timing_.receive_finish_ms = last_packet->receive_time_ms;
ilnik04f4d122017-06-19 07:18:55 -0700123 }
sprangba050a62017-08-18 02:51:12 -0700124 timing_.flags = last_packet->video_header.video_timing.flags;
philipel02447bc2016-05-13 06:01:03 -0700125}
philipelc707ab72016-04-01 02:01:54 -0700126
127RtpFrameObject::~RtpFrameObject() {
128 packet_buffer_->ReturnFrame(this);
129}
130
philipelf4139332016-04-20 10:26:34 +0200131uint16_t RtpFrameObject::first_seq_num() const {
philipel02447bc2016-05-13 06:01:03 -0700132 return first_seq_num_;
philipelc707ab72016-04-01 02:01:54 -0700133}
134
philipelf4139332016-04-20 10:26:34 +0200135uint16_t RtpFrameObject::last_seq_num() const {
philipel02447bc2016-05-13 06:01:03 -0700136 return last_seq_num_;
137}
138
philipel5ceaaae2016-05-24 10:20:47 +0200139int RtpFrameObject::times_nacked() const {
140 return times_nacked_;
141}
142
philipel02447bc2016-05-13 06:01:03 -0700143FrameType RtpFrameObject::frame_type() const {
144 return frame_type_;
145}
146
147VideoCodecType RtpFrameObject::codec_type() const {
148 return codec_type_;
philipelc707ab72016-04-01 02:01:54 -0700149}
150
philipelc707ab72016-04-01 02:01:54 -0700151bool RtpFrameObject::GetBitstream(uint8_t* destination) const {
152 return packet_buffer_->GetBitstream(*this, destination);
153}
154
philipelb4d31082016-07-11 08:46:29 -0700155uint32_t RtpFrameObject::Timestamp() const {
156 return timestamp_;
157}
158
159int64_t RtpFrameObject::ReceivedTime() const {
160 return received_time_;
161}
162
163int64_t RtpFrameObject::RenderTime() const {
164 return _renderTimeMs;
165}
166
philipele0754302017-01-25 08:56:23 -0800167bool RtpFrameObject::delayed_by_retransmission() const {
168 return times_nacked() > 0;
169}
170
philipel88488282016-11-03 08:56:54 -0700171rtc::Optional<RTPVideoTypeHeader> RtpFrameObject::GetCodecHeader() const {
172 rtc::CritScope lock(&packet_buffer_->crit_);
philipel02447bc2016-05-13 06:01:03 -0700173 VCMPacket* packet = packet_buffer_->GetPacket(first_seq_num_);
174 if (!packet)
Oskar Sundbom6bd39022017-11-16 10:54:49 +0100175 return rtc::nullopt;
176 return packet->video_header.codecHeader;
philipel02447bc2016-05-13 06:01:03 -0700177}
178
philipelc707ab72016-04-01 02:01:54 -0700179} // namespace video_coding
180} // namespace webrtc