blob: 3c677f77e42b38c37591b01c6768ffb946683794 [file] [log] [blame]
niklase@google.com470e71d2011-07-07 08:21:25 +00001/*
pwestin@webrtc.orgddab60b2012-04-23 14:52:15 +00002 * Copyright (c) 2012 The WebRTC project authors. All Rights Reserved.
niklase@google.com470e71d2011-07-07 08:21:25 +00003 *
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/encoded_frame.h"
niklase@google.com470e71d2011-07-07 08:21:25 +000012
Yves Gerey3e707812018-11-28 16:47:49 +010013#include <string.h>
14
15#include "absl/types/variant.h"
16#include "api/video/video_timing.h"
17#include "modules/video_coding/codecs/interface/common_constants.h"
18#include "modules/video_coding/codecs/vp8/include/vp8_globals.h"
19#include "modules/video_coding/codecs/vp9/include/vp9_globals.h"
20
niklase@google.com470e71d2011-07-07 08:21:25 +000021namespace webrtc {
22
23VCMEncodedFrame::VCMEncodedFrame()
guoweis@webrtc.org54d072e2015-03-17 21:54:50 +000024 : webrtc::EncodedImage(),
25 _renderTimeMs(-1),
26 _payloadType(0),
27 _missingFrame(false),
Kári Tristan Helgason84ccb2d2018-08-16 14:35:26 +020028 _codec(kVideoCodecGeneric),
guoweis@webrtc.org54d072e2015-03-17 21:54:50 +000029 _rotation_set(false) {
Kári Tristan Helgason84ccb2d2018-08-16 14:35:26 +020030 _codecSpecificInfo.codecType = kVideoCodecGeneric;
niklase@google.com470e71d2011-07-07 08:21:25 +000031}
32
philipel9d3ab612015-12-21 04:12:39 -080033VCMEncodedFrame::~VCMEncodedFrame() {
34 Free();
niklase@google.com470e71d2011-07-07 08:21:25 +000035}
36
philipel9d3ab612015-12-21 04:12:39 -080037void VCMEncodedFrame::Free() {
38 Reset();
39 if (_buffer != NULL) {
40 delete[] _buffer;
41 _buffer = NULL;
42 }
niklase@google.com470e71d2011-07-07 08:21:25 +000043}
44
philipel9d3ab612015-12-21 04:12:39 -080045void VCMEncodedFrame::Reset() {
Niels Möller23775882018-08-16 10:24:12 +020046 SetTimestamp(0);
Sergey Silkin61832dd2018-12-20 14:32:14 +010047 SetSpatialIndex(absl::nullopt);
philipel9d3ab612015-12-21 04:12:39 -080048 _renderTimeMs = -1;
philipel9d3ab612015-12-21 04:12:39 -080049 _payloadType = 0;
50 _frameType = kVideoFrameDelta;
51 _encodedWidth = 0;
52 _encodedHeight = 0;
53 _completeFrame = false;
54 _missingFrame = false;
Niels Möller77536a22019-01-15 08:50:01 +010055 set_size(0);
Kári Tristan Helgason84ccb2d2018-08-16 14:35:26 +020056 _codecSpecificInfo.codecType = kVideoCodecGeneric;
57 _codec = kVideoCodecGeneric;
perkj414dda12016-07-04 01:45:23 -070058 rotation_ = kVideoRotation_0;
ilnik00d802b2017-04-11 10:34:31 -070059 content_type_ = VideoContentType::UNSPECIFIED;
Ilya Nikolaevskiyb6c462d2018-06-05 15:21:32 +020060 timing_.flags = VideoSendTiming::kInvalid;
philipel9d3ab612015-12-21 04:12:39 -080061 _rotation_set = false;
niklase@google.com470e71d2011-07-07 08:21:25 +000062}
63
philipel9d3ab612015-12-21 04:12:39 -080064void VCMEncodedFrame::CopyCodecSpecific(const RTPVideoHeader* header) {
stefan@webrtc.org2ec56062014-07-31 14:59:24 +000065 if (header) {
66 switch (header->codec) {
Niels Möller520ca4e2018-06-04 11:14:38 +020067 case kVideoCodecVP8: {
Philip Eliassond52a1a62018-09-07 13:03:55 +000068 const auto& vp8_header =
69 absl::get<RTPVideoHeaderVP8>(header->video_type_header);
stefan@webrtc.org2ec56062014-07-31 14:59:24 +000070 if (_codecSpecificInfo.codecType != kVideoCodecVP8) {
71 // This is the first packet for this frame.
stefan@webrtc.org2ec56062014-07-31 14:59:24 +000072 _codecSpecificInfo.codecSpecific.VP8.temporalIdx = 0;
73 _codecSpecificInfo.codecSpecific.VP8.layerSync = false;
74 _codecSpecificInfo.codecSpecific.VP8.keyIdx = -1;
75 _codecSpecificInfo.codecType = kVideoCodecVP8;
henrik.lundin@webrtc.org473bac82011-08-17 09:47:33 +000076 }
stefan@webrtc.org2ec56062014-07-31 14:59:24 +000077 _codecSpecificInfo.codecSpecific.VP8.nonReference =
Philip Eliassond52a1a62018-09-07 13:03:55 +000078 vp8_header.nonReference;
79 if (vp8_header.temporalIdx != kNoTemporalIdx) {
stefan@webrtc.org2ec56062014-07-31 14:59:24 +000080 _codecSpecificInfo.codecSpecific.VP8.temporalIdx =
Philip Eliassond52a1a62018-09-07 13:03:55 +000081 vp8_header.temporalIdx;
82 _codecSpecificInfo.codecSpecific.VP8.layerSync = vp8_header.layerSync;
stefan@webrtc.org2ec56062014-07-31 14:59:24 +000083 }
Philip Eliassond52a1a62018-09-07 13:03:55 +000084 if (vp8_header.keyIdx != kNoKeyIdx) {
85 _codecSpecificInfo.codecSpecific.VP8.keyIdx = vp8_header.keyIdx;
stefan@webrtc.org2ec56062014-07-31 14:59:24 +000086 }
87 break;
88 }
Niels Möller520ca4e2018-06-04 11:14:38 +020089 case kVideoCodecVP9: {
philipel29d88462018-08-08 14:26:00 +020090 const auto& vp9_header =
91 absl::get<RTPVideoHeaderVP9>(header->video_type_header);
asaperssona9455ab2015-07-31 06:10:09 -070092 if (_codecSpecificInfo.codecType != kVideoCodecVP9) {
93 // This is the first packet for this frame.
asaperssona9455ab2015-07-31 06:10:09 -070094 _codecSpecificInfo.codecSpecific.VP9.temporal_idx = 0;
asaperssona9455ab2015-07-31 06:10:09 -070095 _codecSpecificInfo.codecSpecific.VP9.gof_idx = 0;
96 _codecSpecificInfo.codecSpecific.VP9.inter_layer_predicted = false;
asaperssona9455ab2015-07-31 06:10:09 -070097 _codecSpecificInfo.codecType = kVideoCodecVP9;
98 }
99 _codecSpecificInfo.codecSpecific.VP9.inter_pic_predicted =
philipel29d88462018-08-08 14:26:00 +0200100 vp9_header.inter_pic_predicted;
asaperssona9455ab2015-07-31 06:10:09 -0700101 _codecSpecificInfo.codecSpecific.VP9.flexible_mode =
philipel29d88462018-08-08 14:26:00 +0200102 vp9_header.flexible_mode;
philipelcfc319b2015-11-10 07:17:23 -0800103 _codecSpecificInfo.codecSpecific.VP9.num_ref_pics =
philipel29d88462018-08-08 14:26:00 +0200104 vp9_header.num_ref_pics;
105 for (uint8_t r = 0; r < vp9_header.num_ref_pics; ++r) {
philipelcfc319b2015-11-10 07:17:23 -0800106 _codecSpecificInfo.codecSpecific.VP9.p_diff[r] =
philipel29d88462018-08-08 14:26:00 +0200107 vp9_header.pid_diff[r];
philipelcfc319b2015-11-10 07:17:23 -0800108 }
asaperssona9455ab2015-07-31 06:10:09 -0700109 _codecSpecificInfo.codecSpecific.VP9.ss_data_available =
philipel29d88462018-08-08 14:26:00 +0200110 vp9_header.ss_data_available;
111 if (vp9_header.temporal_idx != kNoTemporalIdx) {
asaperssona9455ab2015-07-31 06:10:09 -0700112 _codecSpecificInfo.codecSpecific.VP9.temporal_idx =
philipel29d88462018-08-08 14:26:00 +0200113 vp9_header.temporal_idx;
asaperssona9455ab2015-07-31 06:10:09 -0700114 _codecSpecificInfo.codecSpecific.VP9.temporal_up_switch =
philipel29d88462018-08-08 14:26:00 +0200115 vp9_header.temporal_up_switch;
asaperssona9455ab2015-07-31 06:10:09 -0700116 }
philipel29d88462018-08-08 14:26:00 +0200117 if (vp9_header.spatial_idx != kNoSpatialIdx) {
asaperssona9455ab2015-07-31 06:10:09 -0700118 _codecSpecificInfo.codecSpecific.VP9.inter_layer_predicted =
philipel29d88462018-08-08 14:26:00 +0200119 vp9_header.inter_layer_predicted;
Sergey Silkin61832dd2018-12-20 14:32:14 +0100120 SetSpatialIndex(vp9_header.spatial_idx);
asaperssona9455ab2015-07-31 06:10:09 -0700121 }
philipel29d88462018-08-08 14:26:00 +0200122 if (vp9_header.gof_idx != kNoGofIdx) {
123 _codecSpecificInfo.codecSpecific.VP9.gof_idx = vp9_header.gof_idx;
asaperssona9455ab2015-07-31 06:10:09 -0700124 }
philipel29d88462018-08-08 14:26:00 +0200125 if (vp9_header.ss_data_available) {
asaperssona9455ab2015-07-31 06:10:09 -0700126 _codecSpecificInfo.codecSpecific.VP9.num_spatial_layers =
philipel29d88462018-08-08 14:26:00 +0200127 vp9_header.num_spatial_layers;
asaperssona9455ab2015-07-31 06:10:09 -0700128 _codecSpecificInfo.codecSpecific.VP9
129 .spatial_layer_resolution_present =
philipel29d88462018-08-08 14:26:00 +0200130 vp9_header.spatial_layer_resolution_present;
131 if (vp9_header.spatial_layer_resolution_present) {
132 for (size_t i = 0; i < vp9_header.num_spatial_layers; ++i) {
asaperssona9455ab2015-07-31 06:10:09 -0700133 _codecSpecificInfo.codecSpecific.VP9.width[i] =
philipel29d88462018-08-08 14:26:00 +0200134 vp9_header.width[i];
asaperssona9455ab2015-07-31 06:10:09 -0700135 _codecSpecificInfo.codecSpecific.VP9.height[i] =
philipel29d88462018-08-08 14:26:00 +0200136 vp9_header.height[i];
asaperssona9455ab2015-07-31 06:10:09 -0700137 }
138 }
139 _codecSpecificInfo.codecSpecific.VP9.gof.CopyGofInfoVP9(
philipel29d88462018-08-08 14:26:00 +0200140 vp9_header.gof);
asaperssona9455ab2015-07-31 06:10:09 -0700141 }
142 break;
143 }
Niels Möller520ca4e2018-06-04 11:14:38 +0200144 case kVideoCodecH264: {
stefan@webrtc.org2ec56062014-07-31 14:59:24 +0000145 _codecSpecificInfo.codecType = kVideoCodecH264;
146 break;
147 }
Niels Möller520ca4e2018-06-04 11:14:38 +0200148 default: {
Kári Tristan Helgason84ccb2d2018-08-16 14:35:26 +0200149 _codecSpecificInfo.codecType = kVideoCodecGeneric;
stefan@webrtc.org2ec56062014-07-31 14:59:24 +0000150 break;
151 }
henrik.lundin@webrtc.org473bac82011-08-17 09:47:33 +0000152 }
stefan@webrtc.org2ec56062014-07-31 14:59:24 +0000153 }
henrik.lundin@webrtc.org473bac82011-08-17 09:47:33 +0000154}
155
philipel9d3ab612015-12-21 04:12:39 -0800156void VCMEncodedFrame::VerifyAndAllocate(size_t minimumSize) {
Niels Möller48a79462018-12-07 16:21:18 +0100157 size_t old_capacity = capacity();
158 if (minimumSize > old_capacity) {
philipel9d3ab612015-12-21 04:12:39 -0800159 // create buffer of sufficient size
Niels Möller48a79462018-12-07 16:21:18 +0100160 uint8_t* old_buffer = _buffer;
161
162 set_buffer(new uint8_t[minimumSize], minimumSize);
163 if (old_buffer) {
philipel9d3ab612015-12-21 04:12:39 -0800164 // copy old data
Niels Möller48a79462018-12-07 16:21:18 +0100165 memcpy(_buffer, old_buffer, old_capacity);
166 delete[] old_buffer;
niklase@google.com470e71d2011-07-07 08:21:25 +0000167 }
philipel9d3ab612015-12-21 04:12:39 -0800168 }
niklase@google.com470e71d2011-07-07 08:21:25 +0000169}
170
pbos22993e12015-10-19 02:39:06 -0700171} // namespace webrtc