Commit e3fb74f7 authored by Mark Thompson's avatar Mark Thompson

lavfi: Always propagate hw_frames_ctx through links

Also adds a new flag to mark filters which are aware of hwframes and
will perform this task themselves, and marks all appropriate filters
with this flag.

This is required to allow software-mapped hardware frames to work,
because we need to have the frames context available for any later
mapping operation in the filter graph.

The output from the filter graph should only propagate further to an
encoder if the hardware format actually matches the visible format
(mapped frames are valid here and have an hw_frames_ctx, but this
should not be given to the encoder as its hardware context).
parent 7e2561fa
...@@ -38,6 +38,7 @@ ...@@ -38,6 +38,7 @@
#include "libavutil/parseutils.h" #include "libavutil/parseutils.h"
#include "libavutil/samplefmt.h" #include "libavutil/samplefmt.h"
#include "libavutil/fifo.h" #include "libavutil/fifo.h"
#include "libavutil/hwcontext.h"
#include "libavutil/internal.h" #include "libavutil/internal.h"
#include "libavutil/intreadwrite.h" #include "libavutil/intreadwrite.h"
#include "libavutil/dict.h" #include "libavutil/dict.h"
...@@ -2035,7 +2036,9 @@ static int init_output_stream(OutputStream *ost, char *error, int error_len) ...@@ -2035,7 +2036,9 @@ static int init_output_stream(OutputStream *ost, char *error, int error_len)
if (!av_dict_get(ost->encoder_opts, "threads", NULL, 0)) if (!av_dict_get(ost->encoder_opts, "threads", NULL, 0))
av_dict_set(&ost->encoder_opts, "threads", "auto", 0); av_dict_set(&ost->encoder_opts, "threads", "auto", 0);
if (ost->filter && ost->filter->filter->inputs[0]->hw_frames_ctx) { if (ost->filter && ost->filter->filter->inputs[0]->hw_frames_ctx &&
((AVHWFramesContext*)ost->filter->filter->inputs[0]->hw_frames_ctx->data)->format ==
ost->filter->filter->inputs[0]->format) {
ost->enc_ctx->hw_frames_ctx = av_buffer_ref(ost->filter->filter->inputs[0]->hw_frames_ctx); ost->enc_ctx->hw_frames_ctx = av_buffer_ref(ost->filter->filter->inputs[0]->hw_frames_ctx);
if (!ost->enc_ctx->hw_frames_ctx) if (!ost->enc_ctx->hw_frames_ctx)
return AVERROR(ENOMEM); return AVERROR(ENOMEM);
......
...@@ -19,6 +19,7 @@ ...@@ -19,6 +19,7 @@
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/ */
#include "libavutil/avassert.h"
#include "libavutil/avstring.h" #include "libavutil/avstring.h"
#include "libavutil/buffer.h" #include "libavutil/buffer.h"
#include "libavutil/channel_layout.h" #include "libavutil/channel_layout.h"
...@@ -212,14 +213,12 @@ int avfilter_config_links(AVFilterContext *filter) ...@@ -212,14 +213,12 @@ int avfilter_config_links(AVFilterContext *filter)
} }
if (link->src->nb_inputs && link->src->inputs[0]->hw_frames_ctx && if (link->src->nb_inputs && link->src->inputs[0]->hw_frames_ctx &&
!link->hw_frames_ctx) { !(link->src->filter->flags_internal & FF_FILTER_FLAG_HWFRAME_AWARE)) {
AVHWFramesContext *input_ctx = (AVHWFramesContext*)link->src->inputs[0]->hw_frames_ctx->data; av_assert0(!link->hw_frames_ctx &&
"should not be set by non-hwframe-aware filter");
if (input_ctx->format == link->format) { link->hw_frames_ctx = av_buffer_ref(link->src->inputs[0]->hw_frames_ctx);
link->hw_frames_ctx = av_buffer_ref(link->src->inputs[0]->hw_frames_ctx); if (!link->hw_frames_ctx)
if (!link->hw_frames_ctx) return AVERROR(ENOMEM);
return AVERROR(ENOMEM);
}
} }
if ((config_link = link->dstpad->config_props)) if ((config_link = link->dstpad->config_props))
......
...@@ -244,6 +244,8 @@ typedef struct AVFilter { ...@@ -244,6 +244,8 @@ typedef struct AVFilter {
int priv_size; ///< size of private data to allocate for the filter int priv_size; ///< size of private data to allocate for the filter
int flags_internal; ///< Additional flags for avfilter internal use only.
/** /**
* Used by the filter registration system. Must not be touched by any other * Used by the filter registration system. Must not be touched by any other
* code. * code.
......
...@@ -220,4 +220,10 @@ AVFilterContext *ff_filter_alloc(const AVFilter *filter, const char *inst_name); ...@@ -220,4 +220,10 @@ AVFilterContext *ff_filter_alloc(const AVFilter *filter, const char *inst_name);
*/ */
void ff_filter_graph_remove_filter(AVFilterGraph *graph, AVFilterContext *filter); void ff_filter_graph_remove_filter(AVFilterGraph *graph, AVFilterContext *filter);
/**
* The filter is aware of hardware frames, and any hardware frame context
* should not be automatically propagated through it.
*/
#define FF_FILTER_FLAG_HWFRAME_AWARE (1 << 0)
#endif /* AVFILTER_INTERNAL_H */ #endif /* AVFILTER_INTERNAL_H */
...@@ -575,4 +575,6 @@ AVFilter ff_vf_deinterlace_qsv = { ...@@ -575,4 +575,6 @@ AVFilter ff_vf_deinterlace_qsv = {
.inputs = qsvdeint_inputs, .inputs = qsvdeint_inputs,
.outputs = qsvdeint_outputs, .outputs = qsvdeint_outputs,
.flags_internal = FF_FILTER_FLAG_HWFRAME_AWARE,
}; };
...@@ -212,4 +212,5 @@ AVFilter ff_vf_hwdownload = { ...@@ -212,4 +212,5 @@ AVFilter ff_vf_hwdownload = {
.priv_class = &hwdownload_class, .priv_class = &hwdownload_class,
.inputs = hwdownload_inputs, .inputs = hwdownload_inputs,
.outputs = hwdownload_outputs, .outputs = hwdownload_outputs,
.flags_internal = FF_FILTER_FLAG_HWFRAME_AWARE,
}; };
...@@ -233,4 +233,5 @@ AVFilter ff_vf_hwupload = { ...@@ -233,4 +233,5 @@ AVFilter ff_vf_hwupload = {
.priv_class = &hwupload_class, .priv_class = &hwupload_class,
.inputs = hwupload_inputs, .inputs = hwupload_inputs,
.outputs = hwupload_outputs, .outputs = hwupload_outputs,
.flags_internal = FF_FILTER_FLAG_HWFRAME_AWARE,
}; };
...@@ -230,4 +230,6 @@ AVFilter ff_vf_hwupload_cuda = { ...@@ -230,4 +230,6 @@ AVFilter ff_vf_hwupload_cuda = {
.inputs = cudaupload_inputs, .inputs = cudaupload_inputs,
.outputs = cudaupload_outputs, .outputs = cudaupload_outputs,
.flags_internal = FF_FILTER_FLAG_HWFRAME_AWARE,
}; };
...@@ -657,4 +657,6 @@ AVFilter ff_vf_scale_npp = { ...@@ -657,4 +657,6 @@ AVFilter ff_vf_scale_npp = {
.inputs = nppscale_inputs, .inputs = nppscale_inputs,
.outputs = nppscale_outputs, .outputs = nppscale_outputs,
.flags_internal = FF_FILTER_FLAG_HWFRAME_AWARE,
}; };
...@@ -626,4 +626,6 @@ AVFilter ff_vf_scale_qsv = { ...@@ -626,4 +626,6 @@ AVFilter ff_vf_scale_qsv = {
.inputs = qsvscale_inputs, .inputs = qsvscale_inputs,
.outputs = qsvscale_outputs, .outputs = qsvscale_outputs,
.flags_internal = FF_FILTER_FLAG_HWFRAME_AWARE,
}; };
...@@ -463,4 +463,5 @@ AVFilter ff_vf_scale_vaapi = { ...@@ -463,4 +463,5 @@ AVFilter ff_vf_scale_vaapi = {
.inputs = scale_vaapi_inputs, .inputs = scale_vaapi_inputs,
.outputs = scale_vaapi_outputs, .outputs = scale_vaapi_outputs,
.priv_class = &scale_vaapi_class, .priv_class = &scale_vaapi_class,
.flags_internal = FF_FILTER_FLAG_HWFRAME_AWARE,
}; };
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment