avcpp  2.0
Wrapper for the FFmpeg that simplify usage from C++ projects.
avutils.h
Go to the documentation of this file.
1 #pragma once
2 
3 #include <string>
4 #include <vector>
5 #include <deque>
6 #include <memory>
7 #include <mutex>
8 #include <sstream>
9 #include <algorithm>
10 #include <functional>
11 
12 #include "ffmpeg.h"
13 #include "avtime.h"
14 
15 extern "C" {
16 #include <libavfilter/avfilter.h>
17 #include <libavcodec/avcodec.h>
18 }
19 
20 // WA: codecpar usage need more investigation. Temporary disable it.
21 #define USE_CODECPAR ((LIBAVCODEC_VERSION_MAJOR) >= 59) // FFmpeg 5.0
22 
23 // New Audio Channel Layout API
24 #define API_NEW_CHANNEL_LAYOUT ((LIBAVUTIL_VERSION_MAJOR > 57) || (LIBAVUTIL_VERSION_MAJOR == 57 && (LIBAVUTIL_VERSION_MINOR >= 24)))
25 // `int64_t frame_num` has been added in the 60.2, in the 61.0 it should be removed
26 #define API_FRAME_NUM ((LIBAVCODEC_VERSION_MAJOR > 60) || (LIBAVCODEC_VERSION_MAJOR == 60 && LIBAVCODEC_VERSION_MINOR >= 2))
27 // use AVFormatContext::url
28 #define API_AVFORMAT_URL ((LIBAVFORMAT_VERSION_MAJOR > 58) || (LIBAVFORMAT_VERSION_MAJOR == 58 && LIBAVFORMAT_VERSION_MINOR >= 7))
29 // net key frame flags: AV_FRAME_FLAG_KEY (FFmpeg 6.1)
30 #define API_FRAME_KEY ((LIBAVUTIL_VERSION_MAJOR > 58) || (LIBAVUTIL_VERSION_MAJOR == 58 && LIBAVUTIL_VERSION_MINOR >= 29))
31 // avcodec_close() support
32 #define API_AVCODEC_CLOSE (LIBAVCODEC_VERSION_MAJOR < 61)
33 
34 #if defined(__ICL) || defined (__INTEL_COMPILER)
35 # define FF_DISABLE_DEPRECATION_WARNINGS __pragma(warning(push)) __pragma(warning(disable:1478))
36 # define FF_ENABLE_DEPRECATION_WARNINGS __pragma(warning(pop))
37 #elif defined(_MSC_VER)
38 # define FF_DISABLE_DEPRECATION_WARNINGS __pragma(warning(push)) __pragma(warning(disable:4996))
39 # define FF_ENABLE_DEPRECATION_WARNINGS __pragma(warning(pop))
40 #elif defined(__GNUC__) || defined(__clang__)
41 # define FF_DISABLE_DEPRECATION_WARNINGS _Pragma("GCC diagnostic ignored \"-Wdeprecated-declarations\"")
42 # define FF_ENABLE_DEPRECATION_WARNINGS _Pragma("GCC diagnostic warning \"-Wdeprecated-declarations\"")
43 #else
44 # define FF_DISABLE_DEPRECATION_WARNINGS
45 # define FF_ENABLE_DEPRECATION_WARNINGS
46 #endif
47 
48 // sizeof(AVPacket) is no part of the public ABI, packet must be allocated in heap
49 #define API_AVCODEC_NEW_INIT_PACKET (LIBAVCODEC_VERSION_MAJOR >= 58)
50 // some fields in the AVCodec structure deprecard and replaced by the call of avcodec_get_supported_config()
51 #define API_AVCODEC_GET_SUPPORTED_CONFIG (LIBAVCODEC_VERSION_INT >= AV_VERSION_INT(61, 13, 100))
52 // av_stream_get_codec_timebase() deprecard now without replacement
53 #define API_AVFORMAT_AV_STREAM_GET_CODEC_TIMEBASE (LIBAVFORMAT_VERSION_INT < AV_VERSION_INT(61, 5, 101))
54 
55 // Allow to use spece-ship operator, whem possible
56 #if __has_include(<compare>)
57 # include <compare>
58 # if defined(__cpp_lib_three_way_comparison) && __cpp_lib_three_way_comparison >= 201907
59 # define AVCPP_USE_SPACESHIP_OPERATOR 1
60 # endif
61 #endif
62 
63 //
64 // Functions
65 //
66 namespace av {
67 
68 // Basic FFmpeg constants
69 constexpr auto NoPts = static_cast<int64_t>(AV_NOPTS_VALUE);
70 constexpr auto TimeBase = static_cast<int>(AV_TIME_BASE);
71 constexpr auto TimeBaseQ = AVRational{1, AV_TIME_BASE};
72 
73 
74 template<typename R, typename T>
75 R lexical_cast(const T& v)
76 {
77  R result;
78  std::stringstream ss;
79  ss << v;
80  ss >> result;
81  return result;
82 }
83 
85 {
86 protected:
87  noncopyable() = default;
88  noncopyable( const noncopyable& ) = delete;
89  void operator=( const noncopyable& ) = delete;
90 };
91 
99 void set_logging_level(int32_t level);
100 
101 
111 void set_logging_level(const std::string& level);
112 
113 // Compat code
114 #define setFFmpegLoggingLevel set_logging_level
115 
123 void dumpBinaryBuffer(uint8_t *buffer, int buffer_size, int width = 16);
124 
125 
131 std::string error2string(int error);
132 
133 }
134 
135 
136 
137 
138 //
139 // Classes
140 //
141 namespace av {
142 
144 {
145  void operator()(void *) {}
146 };
147 
153 namespace v1 {
154 struct AvDeleter
155 {
156  bool operator() (struct SwsContext* &swsContext);
157  bool operator() (struct AVCodecContext* &codecContext);
158  bool operator() (struct AVOutputFormat* &format);
159  bool operator() (struct AVFormatContext* &formatContext);
160  bool operator() (struct AVFrame* &frame);
161  bool operator() (struct AVPacket* &packet);
162  bool operator() (struct AVDictionary* &dictionary);
163  bool operator ()(struct AVFilterInOut* &filterInOut);
164 };
165 } // ::v1
166 
167 inline namespace v2 {
169 {
170  template<typename T>
171  bool operator() (T *ptr) {
172  return v1::AvDeleter()(ptr);
173  }
174 };
175 }
176 
177 template<typename T>
178 std::unique_ptr<T, void(*)(void*)> malloc(size_t size)
179 {
180  return {static_cast<T*>(av_malloc(size)), av_free};
181 }
182 
183 template<typename T>
184 std::unique_ptr<T, void(*)(void*)> mallocz(size_t size)
185 {
186  return {static_cast<T*>(av_mallocz(size)), av_free};
187 }
188 
189 template<typename T>
190 std::unique_ptr<T, void(*)(void*)> memdup(const void *p, size_t size)
191 {
192  return {static_cast<T*>(av_memdup(p, size)), av_free};
193 }
194 
198 #if 0
199 struct AvNextElement
200 {
201  AVFilterInOut * operator()(AVFilterInOut * x) const
202  {
203  if (x)
204  return x->next;
205  else
206  return 0;
207  }
208 };
209 #endif
210 
211 
212 
225 template<typename T, typename V = T>
227 {
228 public:
234  ScopedValue(T &var, const V& outValue)
235  : var(var),
236  outValue(outValue)
237  {
238  }
239 
246  ScopedValue(T &var, const V &inValue, const V &outValue)
247  : var(var),
248  outValue(outValue)
249  {
250  this->var = inValue;
251  }
252 
254  {
255  var = outValue;
256  }
257 
258 private:
259  T& var;
260  V outValue;
261 };
262 
263 
292 {
293 public:
294  template<typename Proc>
295  ScopeOutAction(const Proc& proc)
296  : m_proc(proc)
297  {}
298 
299  template<typename Proc>
300  ScopeOutAction(Proc&& proc)
301  : m_proc(std::forward<Proc>(proc))
302  {}
303 
305  {
306  if (m_proc)
307  m_proc();
308  }
309 
310 private:
311  std::function<void()> m_proc;
312 };
313 
314 
316 
317 template<typename T>
319 {
321  : value(value)
322  {}
323 
324  bool operator() (const T& value) const
325  {
326  if (this->value == value)
327  return true;
328 
329  return false;
330  }
331 
332  const T& value;
333 };
334 
350 template<typename T, typename L, typename C>
351 T guessValue(const T& value, const L * list, C endListComparator)
352 {
353  if (!list)
354  return value;
355 
356  // move values to array
357  std::deque<T> values;
358  for (const L * ptr = list; !endListComparator(*ptr); ++ptr)
359  {
360  T v = *ptr;
361  values.push_back(v);
362  }
363 
364  // sort list
365  std::sort(values.begin(), values.end());
366 
367  // Search more appropriate range
368  int begin = 0;
369  int end = values.size() - 1;
370  while ((end - begin) > 1)
371  {
372  int mid = begin + (end - begin) / 2;
373 
374  if (value <= values[mid])
375  {
376  end = mid;
377  }
378  else
379  {
380  begin = mid;
381  }
382  }
383 
384  // distance from VALUE to BEGIN more short or VALUE less then BEGIN
385  if (value <= values[begin] || (value - values[begin]) < (values[end] - value))
386  {
387  return values[begin];
388  }
389 
390  return values[end];
391 }
392 
393 
394 template<typename T, typename Container>
395 void array_to_container(const T* array, std::size_t nelements, Container &container)
396 {
397  if (!array || nelements == 0)
398  return;
399  std::copy_n(array, array + nelements, std::back_inserter(container));
400 }
401 
402 template<typename T, typename Container, typename Callable>
403 void array_to_container(const T* array, std::size_t nelements, Container &container, Callable convert)
404 {
405  if (!array || nelements == 0)
406  return;
407  // TBD: implement in more clean way
408  //std::copy_n(array, array + nelemnts, std::back_inserter(container));
409  for (auto i = 0u; i < nelements; ++i) {
410  container.push_back(convert(array[i]));
411  }
412 }
413 
414 template<typename T, typename Container, typename Compare>
415 void array_to_container(const T* array, Container &container, Compare isEnd)
416 {
417  if (!array)
418  return;
419  T value;
420  while (!isEnd(value = *array++))
421  container.push_back(value);
422 }
423 
424 template<typename T, typename Container, typename Compare, typename Callable>
425 void array_to_container(const T* array, Container &container, Compare isEnd, Callable convert)
426 {
427  if (!array)
428  return;
429  T value;
430  while (!isEnd(value = *array++))
431  container.push_back(convert(value));
432 }
433 
434 } // ::av
435 
av::v1::AvDeleter::operator()
bool operator()(struct SwsContext *&swsContext)
Definition: avutils.cpp:203
av::error2string
string error2string(int error)
C++ verstion of the av_err2str()
Definition: avutils.cpp:195
av::EmptyDeleter::operator()
void operator()(void *)
Definition: avutils.h:145
avtime.h
av::malloc
std::unique_ptr< T, void(*)(void *)> malloc(size_t size)
Definition: avutils.h:178
ffmpeg.h
av::EqualComparator::EqualComparator
EqualComparator(const T &value)
Definition: avutils.h:320
av::v2::SmartDeleter::operator()
bool operator()(T *ptr)
Definition: avutils.h:171
av::set_logging_level
void set_logging_level(int32_t level)
This method can be used to turn up or down FFmpeg's logging level.
Definition: avutils.cpp:19
av::EmptyDeleter
Definition: avutils.h:143
av::memdup
std::unique_ptr< T, void(*)(void *)> memdup(const void *p, size_t size)
Definition: avutils.h:190
av::TimeBaseQ
constexpr auto TimeBaseQ
Definition: avutils.h:71
av::NoPts
constexpr auto NoPts
Definition: avutils.h:69
av::v2::SmartDeleter
Definition: avutils.h:168
av::noncopyable::noncopyable
noncopyable()=default
av::ScopeOutAction
The ScopeOutAction class - guard-type class that allows points callback that will be called at the sc...
Definition: avutils.h:291
av::ScopedValue::ScopedValue
ScopedValue(T &var, const V &inValue, const V &outValue)
Ctor.
Definition: avutils.h:246
av::EqualComparator
Definition: avutils.h:318
av::EqualComparator::value
const T & value
Definition: avutils.h:332
av::ScopeOutAction::ScopeOutAction
ScopeOutAction(const Proc &proc)
Definition: avutils.h:295
av::v1::AvDeleter
Definition: avutils.h:154
av::mallocz
std::unique_ptr< T, void(*)(void *)> mallocz(size_t size)
Definition: avutils.h:184
av::noncopyable::operator=
void operator=(const noncopyable &)=delete
av::array_to_container
void array_to_container(const T *array, std::size_t nelements, Container &container)
Definition: avutils.h:395
av::TimeBase
constexpr auto TimeBase
Definition: avutils.h:70
av::lexical_cast
R lexical_cast(const T &v)
Definition: avutils.h:75
av
Definition: audioresampler.cpp:8
std
Definition: averror.h:228
av::noncopyable
Definition: avutils.h:84
av::dumpBinaryBuffer
void dumpBinaryBuffer(uint8_t *buffer, int buffer_size, int width)
dump_binary_buffer Dump binary buffer to std out in HEX view
Definition: avutils.cpp:70
av::EqualComparator::operator()
bool operator()(const T &value) const
Definition: avutils.h:324
av::ScopedValue
Functor to take next element in list/array.
Definition: avutils.h:226
av::ScopedValue::~ScopedValue
~ScopedValue()
Definition: avutils.h:253
av::guessValue
T guessValue(const T &value, const L *list, C endListComparator)
Select more approptiate value from given value list.
Definition: avutils.h:351
av::ScopeOutAction::~ScopeOutAction
~ScopeOutAction()
Definition: avutils.h:304
av::ScopeOutAction::ScopeOutAction
ScopeOutAction(Proc &&proc)
Definition: avutils.h:300
av::ScopedValue::ScopedValue
ScopedValue(T &var, const V &outValue)
Ctor.
Definition: avutils.h:234