Ring Daemon
Loading...
Searching...
No Matches
video_scaler.cpp
Go to the documentation of this file.
1/*
2 * Copyright (C) 2004-2026 Savoir-faire Linux Inc.
3 *
4 * This program is free software: you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation, either version 3 of the License, or
7 * (at your option) any later version.
8 *
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
13 *
14 * You should have received a copy of the GNU General Public License
15 * along with this program. If not, see <https://www.gnu.org/licenses/>.
16 */
17
18#include "libav_deps.h" // MUST BE INCLUDED FIRST
19#include "video_scaler.h"
20#include "media_buffer.h"
21#include "logger.h"
22
23#include <cassert>
24#include <cstddef>
25
26namespace jami {
27namespace video {
28
30 : ctx_(0)
31 , mode_(SWS_FAST_BILINEAR)
32 , tmp_data_()
33{}
34
39
40void
42{
43 scale(input.pointer(), output.pointer());
44}
45
46void
48{
49 ctx_ = sws_getCachedContext(ctx_,
50 input_frame->width,
51 input_frame->height,
52 (AVPixelFormat) input_frame->format,
53 output_frame->width,
54 output_frame->height,
56 mode_,
57 NULL,
58 NULL,
59 NULL);
60 if (!ctx_) {
61 JAMI_ERR("Unable to create a scaler context");
62 return;
63 }
64
65 sws_scale(ctx_,
66 input_frame->data,
67 input_frame->linesize,
68 0,
69 input_frame->height,
70 output_frame->data,
71 output_frame->linesize);
72}
73
74void
76{
77 if (input.width() == output.width() && input.height() == output.height()) {
78 if (input.format() != output.format()) {
80 output.copyFrom(*outPtr);
81 } else {
82 output.copyFrom(input);
83 }
84 } else {
85 auto* output_frame = output.pointer();
86 scale_and_pad(input, output, 0, 0, output_frame->width, output_frame->height, true);
87 }
88}
89
90void
93 unsigned xoff,
94 unsigned yoff,
95 unsigned dest_width,
96 unsigned dest_height,
97 bool keep_aspect)
98{
99 const auto* const input_frame = input.pointer();
100 auto* output_frame = output.pointer();
101
102 /* Correct destination width/height and offset if we need to keep input
103 * frame aspect.
104 */
105 if (keep_aspect) {
106 const float local_ratio = static_cast<float>(dest_width) / static_cast<float>(dest_height);
107 const float input_ratio = static_cast<float>(input_frame->width) / static_cast<float>(input_frame->height);
108
109 if (local_ratio > input_ratio) {
111 dest_width = static_cast<unsigned int>(static_cast<float>(dest_height) * input_ratio);
112 xoff += (old_dest_width - dest_width) / 2;
113 } else {
115 dest_height = static_cast<unsigned int>(static_cast<float>(dest_width) / input_ratio);
117 }
118 }
119
120 // buffer overflow checks
121 if ((xoff + dest_width > (unsigned) output_frame->width) || (yoff + dest_height > (unsigned) output_frame->height)) {
122 JAMI_ERR("Unable to scale video");
123 return;
124 }
125
126 ctx_ = sws_getCachedContext(ctx_,
127 input_frame->width,
128 input_frame->height,
129 (AVPixelFormat) input_frame->format,
130 static_cast<int>(dest_width),
131 static_cast<int>(dest_height),
132 (AVPixelFormat) output_frame->format,
133 mode_,
134 NULL,
135 NULL,
136 NULL);
137 if (!ctx_) {
138 JAMI_ERR("Unable to create a scaler context");
139 return;
140 }
141
142 // Make an offset'ed copy of output data from xoff and yoff
143 const auto* const out_desc = av_pix_fmt_desc_get((AVPixelFormat) output_frame->format);
144 memset(static_cast<void*>(tmp_data_), 0, sizeof(tmp_data_));
145 for (int i = 0; i < 4 && output_frame->linesize[i]; i++) {
146 signed x_shift = static_cast<int>(xoff), y_shift = static_cast<int>(yoff);
147 if (i == 1 || i == 2) {
148 x_shift = -((-x_shift) >> out_desc->log2_chroma_w);
149 y_shift = -((-y_shift) >> out_desc->log2_chroma_h);
150 }
151 auto x_step = out_desc->comp[i].step;
152 tmp_data_[i] = output_frame->data[i] + static_cast<ptrdiff_t>(y_shift * output_frame->linesize[i])
153 + static_cast<ptrdiff_t>(x_shift * x_step);
154 }
155
156 sws_scale(ctx_, input_frame->data, input_frame->linesize, 0, input_frame->height, tmp_data_, output_frame->linesize);
157}
158
159std::unique_ptr<VideoFrame>
161{
162 auto output = std::make_unique<VideoFrame>();
163 output->reserve(pix, input.width(), input.height());
164 scale(input, *output);
165 av_frame_copy_props(output->pointer(), input.pointer());
166 return output;
167}
168
169void
171{
172 if (ctx_) {
173 sws_freeContext(ctx_);
174 ctx_ = nullptr;
175 }
176}
177
178} // namespace video
179} // namespace jami
void scale(const AVFrame *input, AVFrame *output)
void scale_with_aspect(const VideoFrame &input, VideoFrame &output)
void scale_and_pad(const VideoFrame &input, VideoFrame &output, unsigned xoff, unsigned yoff, unsigned dest_width, unsigned dest_height, bool keep_aspect)
std::unique_ptr< VideoFrame > convertFormat(const VideoFrame &input, AVPixelFormat pix)
const AVFrame * pointer() const noexcept
int format() const noexcept
int height() const noexcept
int width() const noexcept
#define JAMI_ERR(...)
Definition logger.h:230
void emitSignal(Args... args)
Definition jami_signal.h:64