891 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			891 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * 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
 | 
						|
 */
 | 
						|
 | 
						|
#ifndef FFTOOLS_FFMPEG_H
 | 
						|
#define FFTOOLS_FFMPEG_H
 | 
						|
 | 
						|
#include "config.h"
 | 
						|
 | 
						|
#include <stdatomic.h>
 | 
						|
#include <stdint.h>
 | 
						|
#include <stdio.h>
 | 
						|
#include <signal.h>
 | 
						|
 | 
						|
#include "cmdutils.h"
 | 
						|
#include "ffmpeg_sched.h"
 | 
						|
#include "sync_queue.h"
 | 
						|
 | 
						|
#include "libavformat/avformat.h"
 | 
						|
#include "libavformat/avio.h"
 | 
						|
 | 
						|
#include "libavcodec/avcodec.h"
 | 
						|
#include "libavcodec/bsf.h"
 | 
						|
 | 
						|
#include "libavfilter/avfilter.h"
 | 
						|
 | 
						|
#include "libavutil/avutil.h"
 | 
						|
#include "libavutil/dict.h"
 | 
						|
#include "libavutil/eval.h"
 | 
						|
#include "libavutil/fifo.h"
 | 
						|
#include "libavutil/hwcontext.h"
 | 
						|
#include "libavutil/pixfmt.h"
 | 
						|
#include "libavutil/rational.h"
 | 
						|
#include "libavutil/thread.h"
 | 
						|
#include "libavutil/threadmessage.h"
 | 
						|
 | 
						|
#include "libswresample/swresample.h"
 | 
						|
 | 
						|
// deprecated features
 | 
						|
#define FFMPEG_OPT_QPHIST 1
 | 
						|
#define FFMPEG_OPT_ADRIFT_THRESHOLD 1
 | 
						|
#define FFMPEG_OPT_ENC_TIME_BASE_NUM 1
 | 
						|
#define FFMPEG_OPT_TOP 1
 | 
						|
#define FFMPEG_OPT_FORCE_KF_SOURCE_NO_DROP 1
 | 
						|
#define FFMPEG_OPT_VSYNC_DROP 1
 | 
						|
#define FFMPEG_OPT_VSYNC 1
 | 
						|
#define FFMPEG_OPT_FILTER_SCRIPT 1
 | 
						|
 | 
						|
#define FFMPEG_ERROR_RATE_EXCEEDED FFERRTAG('E', 'R', 'E', 'D')
 | 
						|
 | 
						|
enum VideoSyncMethod {
 | 
						|
    VSYNC_AUTO = -1,
 | 
						|
    VSYNC_PASSTHROUGH,
 | 
						|
    VSYNC_CFR,
 | 
						|
    VSYNC_VFR,
 | 
						|
    VSYNC_VSCFR,
 | 
						|
#if FFMPEG_OPT_VSYNC_DROP
 | 
						|
    VSYNC_DROP,
 | 
						|
#endif
 | 
						|
};
 | 
						|
 | 
						|
enum EncTimeBase {
 | 
						|
    ENC_TIME_BASE_DEMUX  = -1,
 | 
						|
    ENC_TIME_BASE_FILTER = -2,
 | 
						|
};
 | 
						|
 | 
						|
enum HWAccelID {
 | 
						|
    HWACCEL_NONE = 0,
 | 
						|
    HWACCEL_AUTO,
 | 
						|
    HWACCEL_GENERIC,
 | 
						|
};
 | 
						|
 | 
						|
enum FrameOpaque {
 | 
						|
    FRAME_OPAQUE_SUB_HEARTBEAT = 1,
 | 
						|
    FRAME_OPAQUE_EOF,
 | 
						|
    FRAME_OPAQUE_SEND_COMMAND,
 | 
						|
};
 | 
						|
 | 
						|
enum PacketOpaque {
 | 
						|
    PKT_OPAQUE_SUB_HEARTBEAT = 1,
 | 
						|
    PKT_OPAQUE_FIX_SUB_DURATION,
 | 
						|
};
 | 
						|
 | 
						|
enum LatencyProbe {
 | 
						|
    LATENCY_PROBE_DEMUX,
 | 
						|
    LATENCY_PROBE_DEC_PRE,
 | 
						|
    LATENCY_PROBE_DEC_POST,
 | 
						|
    LATENCY_PROBE_FILTER_PRE,
 | 
						|
    LATENCY_PROBE_FILTER_POST,
 | 
						|
    LATENCY_PROBE_ENC_PRE,
 | 
						|
    LATENCY_PROBE_ENC_POST,
 | 
						|
    LATENCY_PROBE_NB,
 | 
						|
};
 | 
						|
 | 
						|
typedef struct HWDevice {
 | 
						|
    const char *name;
 | 
						|
    enum AVHWDeviceType type;
 | 
						|
    AVBufferRef *device_ref;
 | 
						|
} HWDevice;
 | 
						|
 | 
						|
/* select an input stream for an output stream */
 | 
						|
typedef struct StreamMap {
 | 
						|
    int disabled;           /* 1 is this mapping is disabled by a negative map */
 | 
						|
    int file_index;
 | 
						|
    int stream_index;
 | 
						|
    char *linklabel;       /* name of an output link, for mapping lavfi outputs */
 | 
						|
} StreamMap;
 | 
						|
 | 
						|
typedef struct OptionsContext {
 | 
						|
    OptionGroup *g;
 | 
						|
 | 
						|
    /* input/output options */
 | 
						|
    int64_t start_time;
 | 
						|
    int64_t start_time_eof;
 | 
						|
    int seek_timestamp;
 | 
						|
    const char *format;
 | 
						|
 | 
						|
    SpecifierOptList codec_names;
 | 
						|
    SpecifierOptList audio_ch_layouts;
 | 
						|
    SpecifierOptList audio_channels;
 | 
						|
    SpecifierOptList audio_sample_rate;
 | 
						|
    SpecifierOptList frame_rates;
 | 
						|
    SpecifierOptList max_frame_rates;
 | 
						|
    SpecifierOptList frame_sizes;
 | 
						|
    SpecifierOptList frame_pix_fmts;
 | 
						|
 | 
						|
    /* input options */
 | 
						|
    int64_t input_ts_offset;
 | 
						|
    int loop;
 | 
						|
    int rate_emu;
 | 
						|
    float readrate;
 | 
						|
    double readrate_initial_burst;
 | 
						|
    int accurate_seek;
 | 
						|
    int thread_queue_size;
 | 
						|
    int input_sync_ref;
 | 
						|
    int find_stream_info;
 | 
						|
 | 
						|
    SpecifierOptList ts_scale;
 | 
						|
    SpecifierOptList dump_attachment;
 | 
						|
    SpecifierOptList hwaccels;
 | 
						|
    SpecifierOptList hwaccel_devices;
 | 
						|
    SpecifierOptList hwaccel_output_formats;
 | 
						|
    SpecifierOptList autorotate;
 | 
						|
 | 
						|
    /* output options */
 | 
						|
    StreamMap *stream_maps;
 | 
						|
    int     nb_stream_maps;
 | 
						|
    const char **attachments;
 | 
						|
    int       nb_attachments;
 | 
						|
 | 
						|
    int chapters_input_file;
 | 
						|
 | 
						|
    int64_t recording_time;
 | 
						|
    int64_t stop_time;
 | 
						|
    int64_t limit_filesize;
 | 
						|
    float mux_preload;
 | 
						|
    float mux_max_delay;
 | 
						|
    float shortest_buf_duration;
 | 
						|
    int shortest;
 | 
						|
    int bitexact;
 | 
						|
 | 
						|
    int video_disable;
 | 
						|
    int audio_disable;
 | 
						|
    int subtitle_disable;
 | 
						|
    int data_disable;
 | 
						|
 | 
						|
    // keys are stream indices
 | 
						|
    AVDictionary *streamid;
 | 
						|
 | 
						|
    SpecifierOptList metadata;
 | 
						|
    SpecifierOptList max_frames;
 | 
						|
    SpecifierOptList bitstream_filters;
 | 
						|
    SpecifierOptList codec_tags;
 | 
						|
    SpecifierOptList sample_fmts;
 | 
						|
    SpecifierOptList qscale;
 | 
						|
    SpecifierOptList forced_key_frames;
 | 
						|
    SpecifierOptList fps_mode;
 | 
						|
    SpecifierOptList force_fps;
 | 
						|
    SpecifierOptList frame_aspect_ratios;
 | 
						|
    SpecifierOptList display_rotations;
 | 
						|
    SpecifierOptList display_hflips;
 | 
						|
    SpecifierOptList display_vflips;
 | 
						|
    SpecifierOptList rc_overrides;
 | 
						|
    SpecifierOptList intra_matrices;
 | 
						|
    SpecifierOptList inter_matrices;
 | 
						|
    SpecifierOptList chroma_intra_matrices;
 | 
						|
#if FFMPEG_OPT_TOP
 | 
						|
    SpecifierOptList top_field_first;
 | 
						|
#endif
 | 
						|
    SpecifierOptList metadata_map;
 | 
						|
    SpecifierOptList presets;
 | 
						|
    SpecifierOptList copy_initial_nonkeyframes;
 | 
						|
    SpecifierOptList copy_prior_start;
 | 
						|
    SpecifierOptList filters;
 | 
						|
#if FFMPEG_OPT_FILTER_SCRIPT
 | 
						|
    SpecifierOptList filter_scripts;
 | 
						|
#endif
 | 
						|
    SpecifierOptList reinit_filters;
 | 
						|
    SpecifierOptList fix_sub_duration;
 | 
						|
    SpecifierOptList fix_sub_duration_heartbeat;
 | 
						|
    SpecifierOptList canvas_sizes;
 | 
						|
    SpecifierOptList pass;
 | 
						|
    SpecifierOptList passlogfiles;
 | 
						|
    SpecifierOptList max_muxing_queue_size;
 | 
						|
    SpecifierOptList muxing_queue_data_threshold;
 | 
						|
    SpecifierOptList guess_layout_max;
 | 
						|
    SpecifierOptList apad;
 | 
						|
    SpecifierOptList discard;
 | 
						|
    SpecifierOptList disposition;
 | 
						|
    SpecifierOptList program;
 | 
						|
    SpecifierOptList stream_groups;
 | 
						|
    SpecifierOptList time_bases;
 | 
						|
    SpecifierOptList enc_time_bases;
 | 
						|
    SpecifierOptList autoscale;
 | 
						|
    SpecifierOptList bits_per_raw_sample;
 | 
						|
    SpecifierOptList enc_stats_pre;
 | 
						|
    SpecifierOptList enc_stats_post;
 | 
						|
    SpecifierOptList mux_stats;
 | 
						|
    SpecifierOptList enc_stats_pre_fmt;
 | 
						|
    SpecifierOptList enc_stats_post_fmt;
 | 
						|
    SpecifierOptList mux_stats_fmt;
 | 
						|
} OptionsContext;
 | 
						|
 | 
						|
enum IFilterFlags {
 | 
						|
    IFILTER_FLAG_AUTOROTATE     = (1 << 0),
 | 
						|
    IFILTER_FLAG_REINIT         = (1 << 1),
 | 
						|
    IFILTER_FLAG_CFR            = (1 << 2),
 | 
						|
};
 | 
						|
 | 
						|
typedef struct InputFilterOptions {
 | 
						|
    int64_t             trim_start_us;
 | 
						|
    int64_t             trim_end_us;
 | 
						|
 | 
						|
    uint8_t            *name;
 | 
						|
 | 
						|
    /* When IFILTER_FLAG_CFR is set, the stream is guaranteed to be CFR with
 | 
						|
     * this framerate.
 | 
						|
     *
 | 
						|
     * Otherwise, this is an estimate that should not be relied upon to be
 | 
						|
     * accurate */
 | 
						|
    AVRational          framerate;
 | 
						|
 | 
						|
    int                 sub2video_width;
 | 
						|
    int                 sub2video_height;
 | 
						|
 | 
						|
    // a combination of IFILTER_FLAG_*
 | 
						|
    unsigned            flags;
 | 
						|
 | 
						|
    AVFrame            *fallback;
 | 
						|
} InputFilterOptions;
 | 
						|
 | 
						|
enum OFilterFlags {
 | 
						|
    OFILTER_FLAG_DISABLE_CONVERT    = (1 << 0),
 | 
						|
    // produce 24-bit audio
 | 
						|
    OFILTER_FLAG_AUDIO_24BIT        = (1 << 1),
 | 
						|
    OFILTER_FLAG_AUTOSCALE          = (1 << 2),
 | 
						|
};
 | 
						|
 | 
						|
typedef struct OutputFilterOptions {
 | 
						|
    // Caller-provided name for this output
 | 
						|
    char               *name;
 | 
						|
 | 
						|
    // Codec used for encoding, may be NULL
 | 
						|
    const AVCodec      *enc;
 | 
						|
    // Overrides encoder pixel formats when set.
 | 
						|
    const enum AVPixelFormat *pix_fmts;
 | 
						|
 | 
						|
    int64_t             trim_start_us;
 | 
						|
    int64_t             trim_duration_us;
 | 
						|
    int64_t             ts_offset;
 | 
						|
 | 
						|
    /* Desired output timebase.
 | 
						|
     * Numerator can be one of EncTimeBase values, or 0 when no preference.
 | 
						|
     */
 | 
						|
    AVRational          output_tb;
 | 
						|
 | 
						|
    AVDictionary       *sws_opts;
 | 
						|
    AVDictionary       *swr_opts;
 | 
						|
 | 
						|
    const char         *nb_threads;
 | 
						|
 | 
						|
    // A combination of OFilterFlags.
 | 
						|
    unsigned            flags;
 | 
						|
 | 
						|
    int                 format;
 | 
						|
    int                 width;
 | 
						|
    int                 height;
 | 
						|
 | 
						|
    enum VideoSyncMethod vsync_method;
 | 
						|
 | 
						|
    int                 sample_rate;
 | 
						|
    AVChannelLayout     ch_layout;
 | 
						|
} OutputFilterOptions;
 | 
						|
 | 
						|
typedef struct InputFilter {
 | 
						|
    struct FilterGraph *graph;
 | 
						|
    uint8_t            *name;
 | 
						|
} InputFilter;
 | 
						|
 | 
						|
typedef struct OutputFilter {
 | 
						|
    const AVClass       *class;
 | 
						|
 | 
						|
    struct FilterGraph  *graph;
 | 
						|
    uint8_t             *name;
 | 
						|
 | 
						|
    /* for filters that are not yet bound to an output stream,
 | 
						|
     * this stores the output linklabel, if any */
 | 
						|
    int                  bound;
 | 
						|
    uint8_t             *linklabel;
 | 
						|
 | 
						|
    char                *apad;
 | 
						|
 | 
						|
    enum AVMediaType     type;
 | 
						|
 | 
						|
    atomic_uint_least64_t nb_frames_dup;
 | 
						|
    atomic_uint_least64_t nb_frames_drop;
 | 
						|
} OutputFilter;
 | 
						|
 | 
						|
typedef struct FilterGraph {
 | 
						|
    const AVClass *class;
 | 
						|
    int            index;
 | 
						|
 | 
						|
    InputFilter   **inputs;
 | 
						|
    int          nb_inputs;
 | 
						|
    OutputFilter **outputs;
 | 
						|
    int         nb_outputs;
 | 
						|
} FilterGraph;
 | 
						|
 | 
						|
enum DecoderFlags {
 | 
						|
    DECODER_FLAG_FIX_SUB_DURATION = (1 << 0),
 | 
						|
    // input timestamps are unreliable (guessed by demuxer)
 | 
						|
    DECODER_FLAG_TS_UNRELIABLE    = (1 << 1),
 | 
						|
    // decoder should override timestamps by fixed framerate
 | 
						|
    // from DecoderOpts.framerate
 | 
						|
    DECODER_FLAG_FRAMERATE_FORCED = (1 << 2),
 | 
						|
#if FFMPEG_OPT_TOP
 | 
						|
    DECODER_FLAG_TOP_FIELD_FIRST  = (1 << 3),
 | 
						|
#endif
 | 
						|
    DECODER_FLAG_SEND_END_TS      = (1 << 4),
 | 
						|
    // force bitexact decoding
 | 
						|
    DECODER_FLAG_BITEXACT         = (1 << 5),
 | 
						|
};
 | 
						|
 | 
						|
typedef struct DecoderOpts {
 | 
						|
    int                         flags;
 | 
						|
 | 
						|
    char                       *name;
 | 
						|
    void                       *log_parent;
 | 
						|
 | 
						|
    const AVCodec              *codec;
 | 
						|
    const AVCodecParameters    *par;
 | 
						|
 | 
						|
    /* hwaccel options */
 | 
						|
    enum HWAccelID              hwaccel_id;
 | 
						|
    enum AVHWDeviceType         hwaccel_device_type;
 | 
						|
    char                       *hwaccel_device;
 | 
						|
    enum AVPixelFormat          hwaccel_output_format;
 | 
						|
 | 
						|
    AVRational                  time_base;
 | 
						|
 | 
						|
    // Either forced (when DECODER_FLAG_FRAMERATE_FORCED is set) or
 | 
						|
    // estimated (otherwise) video framerate.
 | 
						|
    AVRational                  framerate;
 | 
						|
} DecoderOpts;
 | 
						|
 | 
						|
typedef struct Decoder {
 | 
						|
    const AVClass   *class;
 | 
						|
 | 
						|
    enum AVMediaType type;
 | 
						|
 | 
						|
    const uint8_t   *subtitle_header;
 | 
						|
    int              subtitle_header_size;
 | 
						|
 | 
						|
    // number of frames/samples retrieved from the decoder
 | 
						|
    uint64_t         frames_decoded;
 | 
						|
    uint64_t         samples_decoded;
 | 
						|
    uint64_t         decode_errors;
 | 
						|
} Decoder;
 | 
						|
 | 
						|
typedef struct InputStream {
 | 
						|
    const AVClass        *class;
 | 
						|
 | 
						|
    /* parent source */
 | 
						|
    struct InputFile     *file;
 | 
						|
 | 
						|
    int                   index;
 | 
						|
 | 
						|
    AVStream             *st;
 | 
						|
    int                   user_set_discard;
 | 
						|
 | 
						|
    /**
 | 
						|
     * Codec parameters - to be used by the decoding/streamcopy code.
 | 
						|
     * st->codecpar should not be accessed, because it may be modified
 | 
						|
     * concurrently by the demuxing thread.
 | 
						|
     */
 | 
						|
    AVCodecParameters    *par;
 | 
						|
    Decoder              *decoder;
 | 
						|
    const AVCodec        *dec;
 | 
						|
 | 
						|
    /* framerate forced with -r */
 | 
						|
    AVRational            framerate;
 | 
						|
#if FFMPEG_OPT_TOP
 | 
						|
    int                   top_field_first;
 | 
						|
#endif
 | 
						|
 | 
						|
    int                   fix_sub_duration;
 | 
						|
 | 
						|
    /* decoded data from this stream goes into all those filters
 | 
						|
     * currently video and audio only */
 | 
						|
    InputFilter         **filters;
 | 
						|
    int                nb_filters;
 | 
						|
 | 
						|
    /*
 | 
						|
     * Output targets that do not go through lavfi, i.e. subtitles or
 | 
						|
     * streamcopy. Those two cases are distinguished by the OutputStream
 | 
						|
     * having an encoder or not.
 | 
						|
     */
 | 
						|
    struct OutputStream **outputs;
 | 
						|
    int                nb_outputs;
 | 
						|
} InputStream;
 | 
						|
 | 
						|
typedef struct InputFile {
 | 
						|
    const AVClass   *class;
 | 
						|
 | 
						|
    int              index;
 | 
						|
 | 
						|
    AVFormatContext *ctx;
 | 
						|
    int64_t          input_ts_offset;
 | 
						|
    int              input_sync_ref;
 | 
						|
    /**
 | 
						|
     * Effective format start time based on enabled streams.
 | 
						|
     */
 | 
						|
    int64_t          start_time_effective;
 | 
						|
    int64_t          ts_offset;
 | 
						|
    /* user-specified start time in AV_TIME_BASE or AV_NOPTS_VALUE */
 | 
						|
    int64_t          start_time;
 | 
						|
 | 
						|
    /* streams that ffmpeg is aware of;
 | 
						|
     * there may be extra streams in ctx that are not mapped to an InputStream
 | 
						|
     * if new streams appear dynamically during demuxing */
 | 
						|
    InputStream    **streams;
 | 
						|
    int           nb_streams;
 | 
						|
} InputFile;
 | 
						|
 | 
						|
enum forced_keyframes_const {
 | 
						|
    FKF_N,
 | 
						|
    FKF_N_FORCED,
 | 
						|
    FKF_PREV_FORCED_N,
 | 
						|
    FKF_PREV_FORCED_T,
 | 
						|
    FKF_T,
 | 
						|
    FKF_NB
 | 
						|
};
 | 
						|
 | 
						|
#define ABORT_ON_FLAG_EMPTY_OUTPUT        (1 <<  0)
 | 
						|
#define ABORT_ON_FLAG_EMPTY_OUTPUT_STREAM (1 <<  1)
 | 
						|
 | 
						|
enum EncStatsType {
 | 
						|
    ENC_STATS_LITERAL = 0,
 | 
						|
    ENC_STATS_FILE_IDX,
 | 
						|
    ENC_STATS_STREAM_IDX,
 | 
						|
    ENC_STATS_FRAME_NUM,
 | 
						|
    ENC_STATS_FRAME_NUM_IN,
 | 
						|
    ENC_STATS_TIMEBASE,
 | 
						|
    ENC_STATS_TIMEBASE_IN,
 | 
						|
    ENC_STATS_PTS,
 | 
						|
    ENC_STATS_PTS_TIME,
 | 
						|
    ENC_STATS_PTS_IN,
 | 
						|
    ENC_STATS_PTS_TIME_IN,
 | 
						|
    ENC_STATS_DTS,
 | 
						|
    ENC_STATS_DTS_TIME,
 | 
						|
    ENC_STATS_SAMPLE_NUM,
 | 
						|
    ENC_STATS_NB_SAMPLES,
 | 
						|
    ENC_STATS_PKT_SIZE,
 | 
						|
    ENC_STATS_BITRATE,
 | 
						|
    ENC_STATS_AVG_BITRATE,
 | 
						|
    ENC_STATS_KEYFRAME,
 | 
						|
};
 | 
						|
 | 
						|
typedef struct EncStatsComponent {
 | 
						|
    enum EncStatsType type;
 | 
						|
 | 
						|
    uint8_t *str;
 | 
						|
    size_t   str_len;
 | 
						|
} EncStatsComponent;
 | 
						|
 | 
						|
typedef struct EncStats {
 | 
						|
    EncStatsComponent  *components;
 | 
						|
    int              nb_components;
 | 
						|
 | 
						|
    AVIOContext        *io;
 | 
						|
 | 
						|
    pthread_mutex_t     lock;
 | 
						|
    int                 lock_initialized;
 | 
						|
} EncStats;
 | 
						|
 | 
						|
extern const char *const forced_keyframes_const_names[];
 | 
						|
 | 
						|
typedef enum {
 | 
						|
    ENCODER_FINISHED = 1,
 | 
						|
    MUXER_FINISHED = 2,
 | 
						|
} OSTFinished ;
 | 
						|
 | 
						|
enum {
 | 
						|
    KF_FORCE_SOURCE         = 1,
 | 
						|
#if FFMPEG_OPT_FORCE_KF_SOURCE_NO_DROP
 | 
						|
    KF_FORCE_SOURCE_NO_DROP = 2,
 | 
						|
#endif
 | 
						|
};
 | 
						|
 | 
						|
typedef struct KeyframeForceCtx {
 | 
						|
    int          type;
 | 
						|
 | 
						|
    int64_t      ref_pts;
 | 
						|
 | 
						|
    // timestamps of the forced keyframes, in AV_TIME_BASE_Q
 | 
						|
    int64_t     *pts;
 | 
						|
    int       nb_pts;
 | 
						|
    int          index;
 | 
						|
 | 
						|
    AVExpr      *pexpr;
 | 
						|
    double       expr_const_values[FKF_NB];
 | 
						|
 | 
						|
    int          dropped_keyframe;
 | 
						|
} KeyframeForceCtx;
 | 
						|
 | 
						|
typedef struct Encoder Encoder;
 | 
						|
 | 
						|
typedef struct OutputStream {
 | 
						|
    const AVClass *class;
 | 
						|
 | 
						|
    enum AVMediaType type;
 | 
						|
 | 
						|
    /* parent muxer */
 | 
						|
    struct OutputFile *file;
 | 
						|
 | 
						|
    int index;               /* stream index in the output file */
 | 
						|
 | 
						|
    /**
 | 
						|
     * Codec parameters for packets submitted to the muxer (i.e. before
 | 
						|
     * bitstream filtering, if any).
 | 
						|
     */
 | 
						|
    AVCodecParameters *par_in;
 | 
						|
 | 
						|
    /* input stream that is the source for this output stream;
 | 
						|
     * may be NULL for streams with no well-defined source, e.g.
 | 
						|
     * attachments or outputs from complex filtergraphs */
 | 
						|
    InputStream *ist;
 | 
						|
 | 
						|
    AVStream *st;            /* stream in the output file */
 | 
						|
 | 
						|
    Encoder *enc;
 | 
						|
    AVCodecContext *enc_ctx;
 | 
						|
 | 
						|
    /* video only */
 | 
						|
    AVRational frame_rate;
 | 
						|
    AVRational max_frame_rate;
 | 
						|
    int force_fps;
 | 
						|
#if FFMPEG_OPT_TOP
 | 
						|
    int top_field_first;
 | 
						|
#endif
 | 
						|
    int bitexact;
 | 
						|
    int bits_per_raw_sample;
 | 
						|
 | 
						|
    AVRational frame_aspect_ratio;
 | 
						|
 | 
						|
    KeyframeForceCtx kf;
 | 
						|
 | 
						|
    char *logfile_prefix;
 | 
						|
    FILE *logfile;
 | 
						|
 | 
						|
    // simple filtergraph feeding this stream, if any
 | 
						|
    FilterGraph  *fg_simple;
 | 
						|
    OutputFilter *filter;
 | 
						|
 | 
						|
    AVDictionary *encoder_opts;
 | 
						|
 | 
						|
    char *attachment_filename;
 | 
						|
 | 
						|
    /* stats */
 | 
						|
    // number of packets send to the muxer
 | 
						|
    atomic_uint_least64_t packets_written;
 | 
						|
    // number of frames/samples sent to the encoder
 | 
						|
    uint64_t frames_encoded;
 | 
						|
    uint64_t samples_encoded;
 | 
						|
 | 
						|
    /* packet quality factor */
 | 
						|
    atomic_int quality;
 | 
						|
 | 
						|
    EncStats enc_stats_pre;
 | 
						|
    EncStats enc_stats_post;
 | 
						|
 | 
						|
    /*
 | 
						|
     * bool on whether this stream should be utilized for splitting
 | 
						|
     * subtitles utilizing fix_sub_duration at random access points.
 | 
						|
     */
 | 
						|
    unsigned int fix_sub_duration_heartbeat;
 | 
						|
} OutputStream;
 | 
						|
 | 
						|
typedef struct OutputFile {
 | 
						|
    const AVClass *class;
 | 
						|
 | 
						|
    int index;
 | 
						|
 | 
						|
    const char           *url;
 | 
						|
 | 
						|
    OutputStream **streams;
 | 
						|
    int         nb_streams;
 | 
						|
 | 
						|
    int64_t recording_time;  ///< desired length of the resulting file in microseconds == AV_TIME_BASE units
 | 
						|
    int64_t start_time;      ///< start time in microseconds == AV_TIME_BASE units
 | 
						|
 | 
						|
    int bitexact;
 | 
						|
} OutputFile;
 | 
						|
 | 
						|
// optionally attached as opaque_ref to decoded AVFrames
 | 
						|
typedef struct FrameData {
 | 
						|
    // demuxer-estimated dts in AV_TIME_BASE_Q,
 | 
						|
    // to be used when real dts is missing
 | 
						|
    int64_t dts_est;
 | 
						|
 | 
						|
    // properties that come from the decoder
 | 
						|
    struct {
 | 
						|
        uint64_t   frame_num;
 | 
						|
 | 
						|
        int64_t    pts;
 | 
						|
        AVRational tb;
 | 
						|
    } dec;
 | 
						|
 | 
						|
    AVRational frame_rate_filter;
 | 
						|
 | 
						|
    int        bits_per_raw_sample;
 | 
						|
 | 
						|
    int64_t wallclock[LATENCY_PROBE_NB];
 | 
						|
 | 
						|
    AVCodecParameters *par_enc;
 | 
						|
} FrameData;
 | 
						|
 | 
						|
extern InputFile   **input_files;
 | 
						|
extern int        nb_input_files;
 | 
						|
 | 
						|
extern OutputFile   **output_files;
 | 
						|
extern int         nb_output_files;
 | 
						|
 | 
						|
// complex filtergraphs
 | 
						|
extern FilterGraph **filtergraphs;
 | 
						|
extern int        nb_filtergraphs;
 | 
						|
 | 
						|
// standalone decoders (not tied to demuxed streams)
 | 
						|
extern Decoder     **decoders;
 | 
						|
extern int        nb_decoders;
 | 
						|
 | 
						|
extern char *vstats_filename;
 | 
						|
 | 
						|
extern float dts_delta_threshold;
 | 
						|
extern float dts_error_threshold;
 | 
						|
 | 
						|
extern enum VideoSyncMethod video_sync_method;
 | 
						|
extern float frame_drop_threshold;
 | 
						|
extern int do_benchmark;
 | 
						|
extern int do_benchmark_all;
 | 
						|
extern int do_hex_dump;
 | 
						|
extern int do_pkt_dump;
 | 
						|
extern int copy_ts;
 | 
						|
extern int start_at_zero;
 | 
						|
extern int copy_tb;
 | 
						|
extern int debug_ts;
 | 
						|
extern int exit_on_error;
 | 
						|
extern int abort_on_flags;
 | 
						|
extern int print_stats;
 | 
						|
extern int64_t stats_period;
 | 
						|
extern int stdin_interaction;
 | 
						|
extern AVIOContext *progress_avio;
 | 
						|
extern float max_error_rate;
 | 
						|
 | 
						|
extern char *filter_nbthreads;
 | 
						|
extern int filter_complex_nbthreads;
 | 
						|
extern int vstats_version;
 | 
						|
extern int auto_conversion_filters;
 | 
						|
 | 
						|
extern const AVIOInterruptCB int_cb;
 | 
						|
 | 
						|
extern const OptionDef options[];
 | 
						|
extern HWDevice *filter_hw_device;
 | 
						|
 | 
						|
extern atomic_uint nb_output_dumped;
 | 
						|
 | 
						|
extern int ignore_unknown_streams;
 | 
						|
extern int copy_unknown_streams;
 | 
						|
 | 
						|
extern int recast_media;
 | 
						|
 | 
						|
extern FILE *vstats_file;
 | 
						|
 | 
						|
void term_init(void);
 | 
						|
void term_exit(void);
 | 
						|
 | 
						|
void show_usage(void);
 | 
						|
 | 
						|
int assert_file_overwrite(const char *filename);
 | 
						|
AVDictionary *strip_specifiers(const AVDictionary *dict);
 | 
						|
int find_codec(void *logctx, const char *name,
 | 
						|
               enum AVMediaType type, int encoder, const AVCodec **codec);
 | 
						|
int parse_and_set_vsync(const char *arg, int *vsync_var, int file_idx, int st_idx, int is_global);
 | 
						|
 | 
						|
int filtergraph_is_simple(const FilterGraph *fg);
 | 
						|
int init_simple_filtergraph(InputStream *ist, OutputStream *ost,
 | 
						|
                            char *graph_desc,
 | 
						|
                            Scheduler *sch, unsigned sch_idx_enc,
 | 
						|
                            const OutputFilterOptions *opts);
 | 
						|
int fg_finalise_bindings(void);
 | 
						|
 | 
						|
/**
 | 
						|
 * Get our axiliary frame data attached to the frame, allocating it
 | 
						|
 * if needed.
 | 
						|
 */
 | 
						|
FrameData *frame_data(AVFrame *frame);
 | 
						|
 | 
						|
const FrameData *frame_data_c(AVFrame *frame);
 | 
						|
 | 
						|
FrameData       *packet_data  (AVPacket *pkt);
 | 
						|
const FrameData *packet_data_c(AVPacket *pkt);
 | 
						|
 | 
						|
int ofilter_bind_ost(OutputFilter *ofilter, OutputStream *ost,
 | 
						|
                     unsigned sched_idx_enc,
 | 
						|
                     const OutputFilterOptions *opts);
 | 
						|
 | 
						|
/**
 | 
						|
 * Create a new filtergraph in the global filtergraph list.
 | 
						|
 *
 | 
						|
 * @param graph_desc Graph description; an av_malloc()ed string, filtergraph
 | 
						|
 *                   takes ownership of it.
 | 
						|
 */
 | 
						|
int fg_create(FilterGraph **pfg, char *graph_desc, Scheduler *sch);
 | 
						|
 | 
						|
void fg_free(FilterGraph **pfg);
 | 
						|
 | 
						|
void fg_send_command(FilterGraph *fg, double time, const char *target,
 | 
						|
                     const char *command, const char *arg, int all_filters);
 | 
						|
 | 
						|
int ffmpeg_parse_options(int argc, char **argv, Scheduler *sch);
 | 
						|
 | 
						|
void enc_stats_write(OutputStream *ost, EncStats *es,
 | 
						|
                     const AVFrame *frame, const AVPacket *pkt,
 | 
						|
                     uint64_t frame_num);
 | 
						|
 | 
						|
HWDevice *hw_device_get_by_name(const char *name);
 | 
						|
HWDevice *hw_device_get_by_type(enum AVHWDeviceType type);
 | 
						|
int hw_device_init_from_string(const char *arg, HWDevice **dev);
 | 
						|
int hw_device_init_from_type(enum AVHWDeviceType type,
 | 
						|
                             const char *device,
 | 
						|
                             HWDevice **dev_out);
 | 
						|
void hw_device_free_all(void);
 | 
						|
 | 
						|
/**
 | 
						|
 * Get a hardware device to be used with this filtergraph.
 | 
						|
 * The returned reference is owned by the callee, the caller
 | 
						|
 * must ref it explicitly for long-term use.
 | 
						|
 */
 | 
						|
AVBufferRef *hw_device_for_filter(void);
 | 
						|
 | 
						|
/**
 | 
						|
 * Create a standalone decoder.
 | 
						|
 */
 | 
						|
int dec_create(const OptionsContext *o, const char *arg, Scheduler *sch);
 | 
						|
 | 
						|
/**
 | 
						|
 * @param dec_opts Dictionary filled with decoder options. Its ownership
 | 
						|
 *                 is transferred to the decoder.
 | 
						|
 * @param param_out If non-NULL, media properties after opening the decoder
 | 
						|
 *                  are written here.
 | 
						|
 *
 | 
						|
 * @retval ">=0" non-negative scheduler index on success
 | 
						|
 * @retval "<0"  an error code on failure
 | 
						|
 */
 | 
						|
int dec_init(Decoder **pdec, Scheduler *sch,
 | 
						|
             AVDictionary **dec_opts, const DecoderOpts *o,
 | 
						|
             AVFrame *param_out);
 | 
						|
void dec_free(Decoder **pdec);
 | 
						|
 | 
						|
/*
 | 
						|
 * Called by filters to connect decoder's output to given filtergraph input.
 | 
						|
 *
 | 
						|
 * @param opts filtergraph input options, to be filled by this function
 | 
						|
 */
 | 
						|
int dec_filter_add(Decoder *dec, InputFilter *ifilter, InputFilterOptions *opts);
 | 
						|
 | 
						|
int enc_alloc(Encoder **penc, const AVCodec *codec,
 | 
						|
              Scheduler *sch, unsigned sch_idx);
 | 
						|
void enc_free(Encoder **penc);
 | 
						|
 | 
						|
int enc_open(void *opaque, const AVFrame *frame);
 | 
						|
 | 
						|
int enc_loopback(Encoder *enc);
 | 
						|
 | 
						|
/*
 | 
						|
 * Initialize muxing state for the given stream, should be called
 | 
						|
 * after the codec/streamcopy setup has been done.
 | 
						|
 *
 | 
						|
 * Open the muxer once all the streams have been initialized.
 | 
						|
 */
 | 
						|
int of_stream_init(OutputFile *of, OutputStream *ost);
 | 
						|
int of_write_trailer(OutputFile *of);
 | 
						|
int of_open(const OptionsContext *o, const char *filename, Scheduler *sch);
 | 
						|
void of_free(OutputFile **pof);
 | 
						|
 | 
						|
void of_enc_stats_close(void);
 | 
						|
 | 
						|
int64_t of_filesize(OutputFile *of);
 | 
						|
 | 
						|
int ifile_open(const OptionsContext *o, const char *filename, Scheduler *sch);
 | 
						|
void ifile_close(InputFile **f);
 | 
						|
 | 
						|
int ist_output_add(InputStream *ist, OutputStream *ost);
 | 
						|
int ist_filter_add(InputStream *ist, InputFilter *ifilter, int is_simple,
 | 
						|
                   InputFilterOptions *opts);
 | 
						|
 | 
						|
/**
 | 
						|
 * Find an unused input stream of given type.
 | 
						|
 */
 | 
						|
InputStream *ist_find_unused(enum AVMediaType type);
 | 
						|
 | 
						|
/* iterate over all input streams in all input files;
 | 
						|
 * pass NULL to start iteration */
 | 
						|
InputStream *ist_iter(InputStream *prev);
 | 
						|
 | 
						|
/* iterate over all output streams in all output files;
 | 
						|
 * pass NULL to start iteration */
 | 
						|
OutputStream *ost_iter(OutputStream *prev);
 | 
						|
 | 
						|
void update_benchmark(const char *fmt, ...);
 | 
						|
 | 
						|
#define SPECIFIER_OPT_FMT_str  "%s"
 | 
						|
#define SPECIFIER_OPT_FMT_i    "%i"
 | 
						|
#define SPECIFIER_OPT_FMT_i64  "%"PRId64
 | 
						|
#define SPECIFIER_OPT_FMT_ui64 "%"PRIu64
 | 
						|
#define SPECIFIER_OPT_FMT_f    "%f"
 | 
						|
#define SPECIFIER_OPT_FMT_dbl  "%lf"
 | 
						|
 | 
						|
#define WARN_MULTIPLE_OPT_USAGE(optname, type, idx, st)\
 | 
						|
{\
 | 
						|
    char namestr[128] = "";\
 | 
						|
    const SpecifierOpt *so = &o->optname.opt[idx];\
 | 
						|
    const char *spec = so->specifier && so->specifier[0] ? so->specifier : "";\
 | 
						|
    snprintf(namestr, sizeof(namestr), "-%s", o->optname.opt_canon->name);\
 | 
						|
    if (o->optname.opt_canon->flags & OPT_HAS_ALT) {\
 | 
						|
        const char * const *names_alt = o->optname.opt_canon->u1.names_alt;\
 | 
						|
        for (int _i = 0; names_alt[_i]; _i++)\
 | 
						|
            av_strlcatf(namestr, sizeof(namestr), "/-%s", names_alt[_i]);\
 | 
						|
    }\
 | 
						|
    av_log(NULL, AV_LOG_WARNING, "Multiple %s options specified for stream %d, only the last option '-%s%s%s "SPECIFIER_OPT_FMT_##type"' will be used.\n",\
 | 
						|
           namestr, st->index, o->optname.opt_canon->name, spec[0] ? ":" : "", spec, so->u.type);\
 | 
						|
}
 | 
						|
 | 
						|
#define MATCH_PER_STREAM_OPT(name, type, outvar, fmtctx, st)\
 | 
						|
{\
 | 
						|
    int _ret, _matches = 0, _match_idx;\
 | 
						|
    for (int _i = 0; _i < o->name.nb_opt; _i++) {\
 | 
						|
        char *spec = o->name.opt[_i].specifier;\
 | 
						|
        if ((_ret = check_stream_specifier(fmtctx, st, spec)) > 0) {\
 | 
						|
            outvar = o->name.opt[_i].u.type;\
 | 
						|
            _match_idx = _i;\
 | 
						|
            _matches++;\
 | 
						|
        } else if (_ret < 0)\
 | 
						|
            return _ret;\
 | 
						|
    }\
 | 
						|
    if (_matches > 1 && o->name.opt_canon)\
 | 
						|
       WARN_MULTIPLE_OPT_USAGE(name, type, _match_idx, st);\
 | 
						|
}
 | 
						|
 | 
						|
const char *opt_match_per_type_str(const SpecifierOptList *sol,
 | 
						|
                                   char mediatype);
 | 
						|
 | 
						|
int muxer_thread(void *arg);
 | 
						|
int encoder_thread(void *arg);
 | 
						|
 | 
						|
#endif /* FFTOOLS_FFMPEG_H */
 |