1cabdff1aSopenharmony_ci/*
2cabdff1aSopenharmony_ci * This file is part of FFmpeg.
3cabdff1aSopenharmony_ci *
4cabdff1aSopenharmony_ci * FFmpeg is free software; you can redistribute it and/or
5cabdff1aSopenharmony_ci * modify it under the terms of the GNU Lesser General Public
6cabdff1aSopenharmony_ci * License as published by the Free Software Foundation; either
7cabdff1aSopenharmony_ci * version 2.1 of the License, or (at your option) any later version.
8cabdff1aSopenharmony_ci *
9cabdff1aSopenharmony_ci * FFmpeg is distributed in the hope that it will be useful,
10cabdff1aSopenharmony_ci * but WITHOUT ANY WARRANTY; without even the implied warranty of
11cabdff1aSopenharmony_ci * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12cabdff1aSopenharmony_ci * Lesser General Public License for more details.
13cabdff1aSopenharmony_ci *
14cabdff1aSopenharmony_ci * You should have received a copy of the GNU Lesser General Public
15cabdff1aSopenharmony_ci * License along with FFmpeg; if not, write to the Free Software
16cabdff1aSopenharmony_ci * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
17cabdff1aSopenharmony_ci */
18cabdff1aSopenharmony_ci
19cabdff1aSopenharmony_ci#include "libavutil/opt.h"
20cabdff1aSopenharmony_ci#include "libavcodec/codec.h"
21cabdff1aSopenharmony_ci#include "libavcodec/codec_desc.h"
22cabdff1aSopenharmony_ci#include "libavcodec/codec_internal.h"
23cabdff1aSopenharmony_ci#include "libavcodec/internal.h"
24cabdff1aSopenharmony_ci
25cabdff1aSopenharmony_cistatic const char *get_type_string(enum AVMediaType type)
26cabdff1aSopenharmony_ci{
27cabdff1aSopenharmony_ci    const char *ret = av_get_media_type_string(type);
28cabdff1aSopenharmony_ci    return ret ? ret : "unknown";
29cabdff1aSopenharmony_ci}
30cabdff1aSopenharmony_ci
31cabdff1aSopenharmony_ci#define AV_LOG(...) av_log(NULL, AV_LOG_FATAL, __VA_ARGS__)
32cabdff1aSopenharmony_ci#define ERR_INTERNAL(msg, ...)                                  \
33cabdff1aSopenharmony_cido {                                                            \
34cabdff1aSopenharmony_ci    AV_LOG(msg, codec->name __VA_ARGS__);                       \
35cabdff1aSopenharmony_ci    ret = 1;                                                    \
36cabdff1aSopenharmony_ci} while (0)
37cabdff1aSopenharmony_ci#define ERR(msg)           ERR_INTERNAL(msg, )
38cabdff1aSopenharmony_ci#define ERR_EXT(msg, ...)  ERR_INTERNAL(msg, , __VA_ARGS__)
39cabdff1aSopenharmony_ci
40cabdff1aSopenharmony_cistatic int priv_data_size_wrong(const FFCodec *codec)
41cabdff1aSopenharmony_ci{
42cabdff1aSopenharmony_ci    if (codec->priv_data_size < 0 ||
43cabdff1aSopenharmony_ci        codec->p.priv_class && codec->priv_data_size < sizeof(AVClass*))
44cabdff1aSopenharmony_ci        return 1;
45cabdff1aSopenharmony_ci    if (!codec->p.priv_class || !codec->p.priv_class->option)
46cabdff1aSopenharmony_ci        return 0;
47cabdff1aSopenharmony_ci    for (const AVOption *opt = codec->p.priv_class->option; opt->name; opt++) {
48cabdff1aSopenharmony_ci        if (opt->offset >= codec->priv_data_size ||
49cabdff1aSopenharmony_ci            opt->type == AV_OPT_TYPE_CONST && opt->offset != 0 ||
50cabdff1aSopenharmony_ci            opt->type != AV_OPT_TYPE_CONST && (opt->offset < sizeof(AVClass*) || opt->offset < 0)) {
51cabdff1aSopenharmony_ci            AV_LOG("Option %s offset %d nonsensical\n",
52cabdff1aSopenharmony_ci                   opt->name, opt->offset);
53cabdff1aSopenharmony_ci            return 1;
54cabdff1aSopenharmony_ci        }
55cabdff1aSopenharmony_ci    }
56cabdff1aSopenharmony_ci    return 0;
57cabdff1aSopenharmony_ci}
58cabdff1aSopenharmony_ci
59cabdff1aSopenharmony_ciint main(void){
60cabdff1aSopenharmony_ci    void *iter = NULL;
61cabdff1aSopenharmony_ci    const AVCodec *codec = NULL;
62cabdff1aSopenharmony_ci    int ret = 0;
63cabdff1aSopenharmony_ci
64cabdff1aSopenharmony_ci    while (codec = av_codec_iterate(&iter)) {
65cabdff1aSopenharmony_ci        const FFCodec *const codec2 = ffcodec(codec);
66cabdff1aSopenharmony_ci        const AVCodecDescriptor *desc;
67cabdff1aSopenharmony_ci        int is_decoder = 0, is_encoder = 0;
68cabdff1aSopenharmony_ci
69cabdff1aSopenharmony_ci        if (!codec->name) {
70cabdff1aSopenharmony_ci            AV_LOG("Codec for format %s has no name\n",
71cabdff1aSopenharmony_ci                   avcodec_get_name(codec->id));
72cabdff1aSopenharmony_ci            ret = 1;
73cabdff1aSopenharmony_ci            continue;
74cabdff1aSopenharmony_ci        }
75cabdff1aSopenharmony_ci        if (codec->type != AVMEDIA_TYPE_VIDEO &&
76cabdff1aSopenharmony_ci            codec->type != AVMEDIA_TYPE_AUDIO &&
77cabdff1aSopenharmony_ci            codec->type != AVMEDIA_TYPE_SUBTITLE)
78cabdff1aSopenharmony_ci            ERR_EXT("Codec %s has unsupported type %s\n",
79cabdff1aSopenharmony_ci                    get_type_string(codec->type));
80cabdff1aSopenharmony_ci        if (codec->type != AVMEDIA_TYPE_AUDIO) {
81cabdff1aSopenharmony_ci            if (codec->ch_layouts || codec->sample_fmts ||
82cabdff1aSopenharmony_ci                codec->supported_samplerates)
83cabdff1aSopenharmony_ci                ERR("Non-audio codec %s has audio-only fields set\n");
84cabdff1aSopenharmony_ci            if (codec->capabilities & (AV_CODEC_CAP_SMALL_LAST_FRAME |
85cabdff1aSopenharmony_ci                                       AV_CODEC_CAP_CHANNEL_CONF     |
86cabdff1aSopenharmony_ci                                       AV_CODEC_CAP_VARIABLE_FRAME_SIZE))
87cabdff1aSopenharmony_ci                ERR("Non-audio codec %s has audio-only capabilities set\n");
88cabdff1aSopenharmony_ci        }
89cabdff1aSopenharmony_ci        if (codec->type != AVMEDIA_TYPE_VIDEO) {
90cabdff1aSopenharmony_ci            if (codec->pix_fmts || codec->supported_framerates)
91cabdff1aSopenharmony_ci                ERR("Non-video codec %s has video-only fields set\n");
92cabdff1aSopenharmony_ci            if (codec2->caps_internal & FF_CODEC_CAP_EXPORTS_CROPPING)
93cabdff1aSopenharmony_ci                ERR("Non-video codec %s exports cropping\n");
94cabdff1aSopenharmony_ci        }
95cabdff1aSopenharmony_ci        if (codec2->caps_internal  & FF_CODEC_CAP_SLICE_THREAD_HAS_MF &&
96cabdff1aSopenharmony_ci            !(codec->capabilities & AV_CODEC_CAP_SLICE_THREADS))
97cabdff1aSopenharmony_ci            ERR("Codec %s wants mainfunction despite not being "
98cabdff1aSopenharmony_ci                "slice-threading capable");
99cabdff1aSopenharmony_ci        if (codec2->caps_internal  & FF_CODEC_CAP_AUTO_THREADS &&
100cabdff1aSopenharmony_ci            !(codec->capabilities & (AV_CODEC_CAP_FRAME_THREADS |
101cabdff1aSopenharmony_ci                                     AV_CODEC_CAP_SLICE_THREADS |
102cabdff1aSopenharmony_ci                                     AV_CODEC_CAP_OTHER_THREADS)))
103cabdff1aSopenharmony_ci            ERR("Codec %s has private-only threading support\n");
104cabdff1aSopenharmony_ci
105cabdff1aSopenharmony_ci        switch (codec2->cb_type) {
106cabdff1aSopenharmony_ci        case FF_CODEC_CB_TYPE_DECODE:
107cabdff1aSopenharmony_ci        case FF_CODEC_CB_TYPE_DECODE_SUB:
108cabdff1aSopenharmony_ci        case FF_CODEC_CB_TYPE_RECEIVE_FRAME:
109cabdff1aSopenharmony_ci            is_decoder = 1;
110cabdff1aSopenharmony_ci            break;
111cabdff1aSopenharmony_ci        case FF_CODEC_CB_TYPE_ENCODE:
112cabdff1aSopenharmony_ci        case FF_CODEC_CB_TYPE_ENCODE_SUB:
113cabdff1aSopenharmony_ci        case FF_CODEC_CB_TYPE_RECEIVE_PACKET:
114cabdff1aSopenharmony_ci            is_encoder = 1;
115cabdff1aSopenharmony_ci            break;
116cabdff1aSopenharmony_ci        default:
117cabdff1aSopenharmony_ci            ERR("Codec %s has unknown cb_type\n");
118cabdff1aSopenharmony_ci            continue;
119cabdff1aSopenharmony_ci        }
120cabdff1aSopenharmony_ci        if (is_decoder != av_codec_is_decoder(codec) ||
121cabdff1aSopenharmony_ci            is_encoder != av_codec_is_encoder(codec)) {
122cabdff1aSopenharmony_ci            ERR("Codec %s cb_type and av_codec_is_(de|en)coder inconsistent.\n");
123cabdff1aSopenharmony_ci            continue;
124cabdff1aSopenharmony_ci        }
125cabdff1aSopenharmony_ci#define CHECK(TYPE, type) (codec2->cb_type == FF_CODEC_CB_TYPE_ ## TYPE && !codec2->cb.type)
126cabdff1aSopenharmony_ci        if (CHECK(DECODE, decode) || CHECK(DECODE_SUB, decode_sub) ||
127cabdff1aSopenharmony_ci            CHECK(RECEIVE_PACKET, receive_packet) ||
128cabdff1aSopenharmony_ci            CHECK(ENCODE, encode) || CHECK(ENCODE_SUB, encode_sub) ||
129cabdff1aSopenharmony_ci            CHECK(RECEIVE_FRAME, receive_frame)) {
130cabdff1aSopenharmony_ci            ERR_EXT("Codec %s does not implement its %s callback.\n",
131cabdff1aSopenharmony_ci                    is_decoder ? "decoding" : "encoding");
132cabdff1aSopenharmony_ci        }
133cabdff1aSopenharmony_ci#undef CHECK
134cabdff1aSopenharmony_ci        if (is_encoder) {
135cabdff1aSopenharmony_ci            if ((codec->type == AVMEDIA_TYPE_SUBTITLE) != (codec2->cb_type == FF_CODEC_CB_TYPE_ENCODE_SUB))
136cabdff1aSopenharmony_ci                ERR("Encoder %s is both subtitle encoder and not subtitle encoder.");
137cabdff1aSopenharmony_ci            if (codec2->update_thread_context || codec2->update_thread_context_for_user || codec2->bsfs)
138cabdff1aSopenharmony_ci                ERR("Encoder %s has decoder-only thread functions or bsf.\n");
139cabdff1aSopenharmony_ci            if (codec->type == AVMEDIA_TYPE_AUDIO) {
140cabdff1aSopenharmony_ci                if (!codec->sample_fmts) {
141cabdff1aSopenharmony_ci                    av_log(NULL, AV_LOG_FATAL, "Encoder %s is missing the sample_fmts field\n", codec->name);
142cabdff1aSopenharmony_ci                    ret = 1;
143cabdff1aSopenharmony_ci                }
144cabdff1aSopenharmony_ci            }
145cabdff1aSopenharmony_ci            if (codec2->caps_internal & (FF_CODEC_CAP_ALLOCATE_PROGRESS |
146cabdff1aSopenharmony_ci                                        FF_CODEC_CAP_SETS_PKT_DTS |
147cabdff1aSopenharmony_ci                                        FF_CODEC_CAP_SKIP_FRAME_FILL_PARAM |
148cabdff1aSopenharmony_ci                                        FF_CODEC_CAP_EXPORTS_CROPPING |
149cabdff1aSopenharmony_ci                                        FF_CODEC_CAP_SETS_FRAME_PROPS) ||
150cabdff1aSopenharmony_ci                codec->capabilities  & (AV_CODEC_CAP_AVOID_PROBING |
151cabdff1aSopenharmony_ci                                        AV_CODEC_CAP_CHANNEL_CONF  |
152cabdff1aSopenharmony_ci                                        AV_CODEC_CAP_DRAW_HORIZ_BAND |
153cabdff1aSopenharmony_ci                                        AV_CODEC_CAP_SUBFRAMES))
154cabdff1aSopenharmony_ci                ERR("Encoder %s has decoder-only capabilities set\n");
155cabdff1aSopenharmony_ci            if (codec->capabilities & AV_CODEC_CAP_FRAME_THREADS &&
156cabdff1aSopenharmony_ci                codec->capabilities & AV_CODEC_CAP_ENCODER_FLUSH)
157cabdff1aSopenharmony_ci                ERR("Frame-threaded encoder %s claims to support flushing\n");
158cabdff1aSopenharmony_ci        } else {
159cabdff1aSopenharmony_ci            if ((codec->type == AVMEDIA_TYPE_SUBTITLE) != (codec2->cb_type == FF_CODEC_CB_TYPE_DECODE_SUB))
160cabdff1aSopenharmony_ci                ERR("Subtitle decoder %s does not implement decode_sub callback\n");
161cabdff1aSopenharmony_ci            if (codec->type == AVMEDIA_TYPE_SUBTITLE && codec2->bsfs)
162cabdff1aSopenharmony_ci                ERR("Automatic bitstream filtering unsupported for subtitles; "
163cabdff1aSopenharmony_ci                    "yet decoder %s has it set\n");
164cabdff1aSopenharmony_ci            if (codec->capabilities & (AV_CODEC_CAP_SMALL_LAST_FRAME    |
165cabdff1aSopenharmony_ci                                       AV_CODEC_CAP_VARIABLE_FRAME_SIZE |
166cabdff1aSopenharmony_ci                                       AV_CODEC_CAP_ENCODER_REORDERED_OPAQUE |
167cabdff1aSopenharmony_ci                                       AV_CODEC_CAP_ENCODER_FLUSH))
168cabdff1aSopenharmony_ci                ERR("Decoder %s has encoder-only capabilities\n");
169cabdff1aSopenharmony_ci            if (codec2->caps_internal & FF_CODEC_CAP_ALLOCATE_PROGRESS &&
170cabdff1aSopenharmony_ci                !(codec->capabilities & AV_CODEC_CAP_FRAME_THREADS))
171cabdff1aSopenharmony_ci                ERR("Decoder %s wants allocated progress without supporting"
172cabdff1aSopenharmony_ci                    "frame threads\n");
173cabdff1aSopenharmony_ci        }
174cabdff1aSopenharmony_ci        if (priv_data_size_wrong(codec2))
175cabdff1aSopenharmony_ci            ERR_EXT("Private context of codec %s is impossibly-sized (size %d).",
176cabdff1aSopenharmony_ci                    codec2->priv_data_size);
177cabdff1aSopenharmony_ci        if (!(desc = avcodec_descriptor_get(codec->id))) {
178cabdff1aSopenharmony_ci            ERR("Codec %s lacks a corresponding descriptor\n");
179cabdff1aSopenharmony_ci        } else if (desc->type != codec->type)
180cabdff1aSopenharmony_ci            ERR_EXT("The type of AVCodec %s and its AVCodecDescriptor differ: "
181cabdff1aSopenharmony_ci                    "%s vs %s\n",
182cabdff1aSopenharmony_ci                    get_type_string(codec->type), get_type_string(desc->type));
183cabdff1aSopenharmony_ci    }
184cabdff1aSopenharmony_ci    return ret;
185cabdff1aSopenharmony_ci}
186