libavfilter/formats.c
39135465
 /*
  * Filter layer - format negotiation
3fa77bde
  * Copyright (c) 2007 Bobby Bingham
39135465
  *
  * This file is part of FFmpeg.
  *
  * FFmpeg is free software; you can redistribute it and/or
  * modify it under the terms of the GNU Lesser General Public
  * License as published by the Free Software Foundation; either
  * version 2.1 of the License, or (at your option) any later version.
  *
  * FFmpeg is distributed in the hope that it will be useful,
  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
  * Lesser General Public License for more details.
  *
  * You should have received a copy of the GNU Lesser General Public
  * License along with FFmpeg; if not, write to the Free Software
  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  */
 
3f07d40e
 #include "libavutil/eval.h"
5ff84183
 #include "libavutil/pixdesc.h"
fd2c0a5d
 #include "libavutil/audioconvert.h"
39135465
 #include "avfilter.h"
7464a53a
 #include "internal.h"
39135465
 
4c4de9ca
 /**
  * Add all refs from a to ret and destroy a.
  */
37e0b997
 static void merge_ref(AVFilterFormats *ret, AVFilterFormats *a)
 {
     int i;
93faa9fa
 
f4228097
     for (i = 0; i < a->refcount; i++) {
37e0b997
         ret->refs[ret->refcount] = a->refs[i];
         *ret->refs[ret->refcount++] = ret;
     }
93faa9fa
 
     av_free(a->refs);
     av_free(a->formats);
     av_free(a);
37e0b997
 }
 
39135465
 AVFilterFormats *avfilter_merge_formats(AVFilterFormats *a, AVFilterFormats *b)
 {
     AVFilterFormats *ret;
     unsigned i, j, k = 0;
 
79a0ec1a
     if (a == b) return a;
 
e5f4e249
     if (a == b)
         return a;
 
39135465
     ret = av_mallocz(sizeof(AVFilterFormats));
 
     /* merge list of formats */
243370cb
     ret->formats = av_malloc(sizeof(*ret->formats) * FFMIN(a->format_count,
                                                            b->format_count));
f4228097
     for (i = 0; i < a->format_count; i++)
         for (j = 0; j < b->format_count; j++)
8cda755c
             if (a->formats[i] == b->formats[j]){
248bb81e
                 if(k >= FFMIN(a->format_count, b->format_count)){
                     av_log(0, AV_LOG_ERROR, "Duplicate formats in avfilter_merge_formats() detected\n");
                     av_free(ret->formats);
                     av_free(ret);
                     return NULL;
                 }
39135465
                 ret->formats[k++] = a->formats[i];
8cda755c
             }
39135465
 
9189411b
     ret->format_count = k;
39135465
     /* check that there was at least one common format */
f4228097
     if (!ret->format_count) {
39135465
         av_free(ret->formats);
         av_free(ret);
         return NULL;
     }
 
     ret->refs = av_malloc(sizeof(AVFilterFormats**)*(a->refcount+b->refcount));
37e0b997
 
     merge_ref(ret, a);
     merge_ref(ret, b);
39135465
 
     return ret;
 }
 
7464a53a
 int ff_fmt_is_in(int fmt, const int *fmts)
 {
     const int *p;
 
     for (p = fmts; *p != -1; p++) {
         if (fmt == *p)
             return 1;
     }
     return 0;
 }
 
386aee68
 #define COPY_INT_LIST(list_copy, list, type) {                          \
     int count = 0;                                                      \
     if (list)                                                           \
         for (count = 0; list[count] != -1; count++)                     \
             ;                                                           \
     list_copy = av_calloc(count+1, sizeof(type));                       \
     if (list_copy) {                                                    \
         memcpy(list_copy, list, sizeof(type) * count);                  \
         list_copy[count] = -1;                                          \
     }                                                                   \
 }
 
 int *ff_copy_int_list(const int * const list)
 {
     int *ret = NULL;
     COPY_INT_LIST(ret, list, int);
     return ret;
 }
 
 int64_t *ff_copy_int64_list(const int64_t * const list)
 {
     int64_t *ret = NULL;
     COPY_INT_LIST(ret, list, int64_t);
     return ret;
 }
 
527ca398
 #define MAKE_FORMAT_LIST()                                              \
     AVFilterFormats *formats;                                           \
     int count = 0;                                                      \
     if (fmts)                                                           \
         for (count = 0; fmts[count] != -1; count++)                     \
             ;                                                           \
     formats = av_mallocz(sizeof(AVFilterFormats));                      \
     if (!formats) return NULL;                                          \
     formats->format_count = count;                                      \
     if (count) {                                                        \
f4228097
         formats->formats = av_malloc(sizeof(*formats->formats)*count);  \
527ca398
         if (!formats->formats) {                                        \
             av_free(formats);                                           \
             return NULL;                                                \
         }                                                               \
     }
 
bdab614b
 AVFilterFormats *avfilter_make_format_list(const int *fmts)
f6a1fa85
 {
527ca398
     MAKE_FORMAT_LIST();
     while (count--)
         formats->formats[count] = fmts[count];
f6a1fa85
 
527ca398
     return formats;
 }
f6a1fa85
 
527ca398
 AVFilterFormats *avfilter_make_format64_list(const int64_t *fmts)
 {
     MAKE_FORMAT_LIST();
     if (count)
47d2ca32
         memcpy(formats->formats, fmts, sizeof(*formats->formats) * count);
f6a1fa85
 
     return formats;
 }
 
527ca398
 int avfilter_add_format(AVFilterFormats **avff, int64_t fmt)
4fd1f187
 {
527ca398
     int64_t *fmts;
4fd1f187
 
c1d662fd
     if (!(*avff) && !(*avff = av_mallocz(sizeof(AVFilterFormats))))
         return AVERROR(ENOMEM);
 
bdab614b
     fmts = av_realloc((*avff)->formats,
8f349b64
                       sizeof(*(*avff)->formats) * ((*avff)->format_count+1));
bdab614b
     if (!fmts)
4fd1f187
         return AVERROR(ENOMEM);
 
bdab614b
     (*avff)->formats = fmts;
     (*avff)->formats[(*avff)->format_count++] = fmt;
4fd1f187
     return 0;
 }
 
9899037d
 #if FF_API_OLD_ALL_FORMATS_API
bdab614b
 AVFilterFormats *avfilter_all_formats(enum AVMediaType type)
39135465
 {
9899037d
     return avfilter_make_all_formats(type);
 }
 #endif
 
 AVFilterFormats *avfilter_make_all_formats(enum AVMediaType type)
 {
c1d662fd
     AVFilterFormats *ret = NULL;
bdab614b
     int fmt;
ad2c9501
     int num_formats = type == AVMEDIA_TYPE_VIDEO ? PIX_FMT_NB    :
5d6e4c16
                       type == AVMEDIA_TYPE_AUDIO ? AV_SAMPLE_FMT_NB : 0;
d3a4e41c
 
bdab614b
     for (fmt = 0; fmt < num_formats; fmt++)
         if ((type != AVMEDIA_TYPE_VIDEO) ||
             (type == AVMEDIA_TYPE_VIDEO && !(av_pix_fmt_descriptors[fmt].flags & PIX_FMT_HWACCEL)))
             avfilter_add_format(&ret, fmt);
d3a4e41c
 
     return ret;
39135465
 }
 
ea8de109
 const int64_t avfilter_all_channel_layouts[] = {
 #include "all_channel_layouts.h"
     -1
 };
 
9899037d
 AVFilterFormats *avfilter_make_all_channel_layouts(void)
fd2c0a5d
 {
ea8de109
     return avfilter_make_format64_list(avfilter_all_channel_layouts);
fd2c0a5d
 }
 
9899037d
 AVFilterFormats *avfilter_make_all_packing_formats(void)
b57df29f
 {
b052601b
     static const int packing[] = {
b57df29f
         AVFILTER_PACKED,
         AVFILTER_PLANAR,
         -1,
     };
 
     return avfilter_make_format_list(packing);
 }
 
39135465
 void avfilter_formats_ref(AVFilterFormats *f, AVFilterFormats **ref)
 {
     *ref = f;
     f->refs = av_realloc(f->refs, sizeof(AVFilterFormats**) * ++f->refcount);
     f->refs[f->refcount-1] = ref;
 }
 
30f4baeb
 static int find_ref_index(AVFilterFormats **ref)
88cfb804
 {
     int i;
f4228097
     for (i = 0; i < (*ref)->refcount; i++)
         if ((*ref)->refs[i] == ref)
88cfb804
             return i;
     return -1;
 }
 
39135465
 void avfilter_formats_unref(AVFilterFormats **ref)
 {
d3c01751
     int idx;
 
063e7692
     if (!*ref)
         return;
 
d3c01751
     idx = find_ref_index(ref);
88cfb804
 
f4228097
     if (idx >= 0)
88cfb804
         memmove((*ref)->refs + idx, (*ref)->refs + idx+1,
             sizeof(AVFilterFormats**) * ((*ref)->refcount-idx-1));
 
f4228097
     if (!--(*ref)->refcount) {
39135465
         av_free((*ref)->formats);
         av_free((*ref)->refs);
         av_free(*ref);
     }
     *ref = NULL;
 }
 
eac24950
 void avfilter_formats_changeref(AVFilterFormats **oldref,
                                 AVFilterFormats **newref)
 {
9189411b
     int idx = find_ref_index(oldref);
eac24950
 
f4228097
     if (idx >= 0) {
eac24950
         (*oldref)->refs[idx] = newref;
         *newref = *oldref;
         *oldref = NULL;
     }
 }
 
3f07d40e
 /* internal functions for parsing audio format arguments */
 
e26782a9
 int ff_parse_pixel_format(enum PixelFormat *ret, const char *arg, void *log_ctx)
 {
     char *tail;
     int pix_fmt = av_get_pix_fmt(arg);
     if (pix_fmt == PIX_FMT_NONE) {
         pix_fmt = strtol(arg, &tail, 0);
         if (*tail || (unsigned)pix_fmt >= PIX_FMT_NB) {
             av_log(log_ctx, AV_LOG_ERROR, "Invalid pixel format '%s'\n", arg);
             return AVERROR(EINVAL);
         }
     }
     *ret = pix_fmt;
     return 0;
 }
 
3f07d40e
 int ff_parse_sample_format(int *ret, const char *arg, void *log_ctx)
 {
     char *tail;
     int sfmt = av_get_sample_fmt(arg);
     if (sfmt == AV_SAMPLE_FMT_NONE) {
         sfmt = strtol(arg, &tail, 0);
         if (*tail || (unsigned)sfmt >= AV_SAMPLE_FMT_NB) {
             av_log(log_ctx, AV_LOG_ERROR, "Invalid sample format '%s'\n", arg);
             return AVERROR(EINVAL);
         }
     }
     *ret = sfmt;
     return 0;
 }
 
4381bddc
 int ff_parse_sample_rate(int *ret, const char *arg, void *log_ctx)
3f07d40e
 {
     char *tail;
     double srate = av_strtod(arg, &tail);
4381bddc
     if (*tail || srate < 1 || (int)srate != srate || srate > INT_MAX) {
3f07d40e
         av_log(log_ctx, AV_LOG_ERROR, "Invalid sample rate '%s'\n", arg);
         return AVERROR(EINVAL);
     }
     *ret = srate;
     return 0;
 }
 
 int ff_parse_channel_layout(int64_t *ret, const char *arg, void *log_ctx)
 {
     char *tail;
     int64_t chlayout = av_get_channel_layout(arg);
     if (chlayout == 0) {
         chlayout = strtol(arg, &tail, 10);
         if (*tail || chlayout == 0) {
             av_log(log_ctx, AV_LOG_ERROR, "Invalid channel layout '%s'\n", arg);
             return AVERROR(EINVAL);
         }
     }
     *ret = chlayout;
     return 0;
 }
 
 int ff_parse_packing_format(int *ret, const char *arg, void *log_ctx)
 {
     char *tail;
     int planar = strtol(arg, &tail, 10);
     if (*tail) {
a7196795
         planar = !strcmp(arg, "packed") ? 0:
                  !strcmp(arg, "planar") ? 1: -1;
     }
 
     if (planar != 0 && planar != 1) {
3f07d40e
         av_log(log_ctx, AV_LOG_ERROR, "Invalid packing format '%s'\n", arg);
         return AVERROR(EINVAL);
     }
     *ret = planar;
     return 0;
 }
 
ea8de109
 #ifdef TEST
 
 #undef printf
 
 int main(void)
 {
     const int64_t *cl;
     char buf[512];
 
41215fde
     for (cl = avfilter_all_channel_layouts; *cl != -1; cl++) {
ea8de109
         av_get_channel_layout_string(buf, sizeof(buf), -1, *cl);
         printf("%s\n", buf);
     }
 
     return 0;
 }
 
 #endif