1
0
mirror of https://github.com/mpv-player/mpv synced 2025-01-12 18:02:36 +00:00

vf_vavpp: provide future/past frames to driver

This was missing for extended deinterlacer.

Unfortunately, these deinterlacer still do not work. The provided future
frame (which is all the deinterlacers want) seems to be correct, though.

One minor behavioral change is that this always keeps the previous frame
for PTS computations. This could be avoided (in order to keep exactly
the same behavior as before), but it seems more elegant and should not
do any harm. (Also, if we really cared about reducing hw frame refs,
a more worthy goal is producing the field output incrementally.)
This commit is contained in:
wm4 2015-06-01 01:39:30 +02:00
parent 56c4cdf369
commit ebc5237c36

View File

@ -15,6 +15,8 @@
* with mpv. If not, see <http://www.gnu.org/licenses/>.
*/
#include <assert.h>
#include <va/va.h>
#include <va/va_vpp.h>
@ -35,9 +37,16 @@ static bool check_error(struct vf_instance *vf, VAStatus status, const char *msg
struct surface_refs {
VASurfaceID *surfaces;
int num_required;
int num_surfaces;
};
static void add_surface(void *ta_ctx, struct surface_refs *refs, struct mp_image *s)
{
VASurfaceID id = va_surface_id(s);
if (id != VA_INVALID_ID)
MP_TARRAY_APPEND(ta_ctx, refs->surfaces, refs->num_surfaces, id);
}
struct pipeline {
VABufferID *filters;
int num_filters;
@ -48,7 +57,6 @@ struct pipeline {
};
struct vf_priv_s {
double prev_pts;
int deint_type; // 0: none, 1: discard, 2: double fps
bool do_deint;
VABufferID buffers[VAProcFilterCount];
@ -61,10 +69,20 @@ struct vf_priv_s {
struct pipeline pipe;
struct mp_image_pool *pool;
int current_rt_format;
int needed_future_frames;
int needed_past_frames;
// Queue of input frames, used to determine past/current/future frames.
// queue[0] is the newest frame, queue[num_queue - 1] the oldest.
struct mp_image **queue;
int num_queue;
// queue[current_pos] is the current frame, unless current_pos is not a
// valid index.
int current_pos;
};
static const struct vf_priv_s vf_priv_default = {
.prev_pts = MP_NOPTS_VALUE,
.config = VA_INVALID_ID,
.context = VA_INVALID_ID,
.deint_type = 2,
@ -80,6 +98,15 @@ static const int deint_algorithm[] = {
[5] = VAProcDeinterlacingMotionCompensated,
};
static void flush_frames(struct vf_instance *vf)
{
struct vf_priv_s *p = vf->priv;
for (int n = 0; n < p->num_queue; n++)
talloc_free(p->queue[n]);
p->num_queue = 0;
p->current_pos = -1;
}
static bool update_pipeline(struct vf_instance *vf, bool deint)
{
struct vf_priv_s *p = vf->priv;
@ -91,7 +118,7 @@ static bool update_pipeline(struct vf_instance *vf, bool deint)
}
if (filters == p->pipe.filters && num_filters == p->pipe.num_filters)
return true;
p->pipe.forward.num_required = p->pipe.backward.num_required = 0;
p->pipe.forward.num_surfaces = p->pipe.backward.num_surfaces = 0;
p->pipe.num_input_colors = p->pipe.num_output_colors = 0;
p->pipe.num_filters = 0;
p->pipe.filters = NULL;
@ -110,8 +137,8 @@ static bool update_pipeline(struct vf_instance *vf, bool deint)
p->pipe.num_filters = num_filters;
p->pipe.num_input_colors = caps.num_input_color_standards;
p->pipe.num_output_colors = caps.num_output_color_standards;
MP_TARRAY_GROW(vf, p->pipe.forward.surfaces, caps.num_forward_references);
MP_TARRAY_GROW(vf, p->pipe.backward.surfaces, caps.num_backward_references);
p->needed_future_frames = caps.num_forward_references;
p->needed_past_frames = caps.num_backward_references;
return true;
}
@ -178,10 +205,22 @@ static struct mp_image *render(struct vf_instance *vf, struct mp_image *in,
param->filter_flags = flags;
param->filters = p->pipe.filters;
param->num_filters = p->pipe.num_filters;
for (int n = 0; n < p->needed_future_frames; n++) {
int idx = p->current_pos - 1 - n;
if (idx >= 0 && idx < p->num_queue)
add_surface(p, &p->pipe.forward, p->queue[idx]);
}
param->forward_references = p->pipe.forward.surfaces;
param->num_forward_references = p->pipe.forward.num_surfaces;
for (int n = 0; n < p->needed_past_frames; n++) {
int idx = p->current_pos + 1 + n;
if (idx >= 0 && idx < p->num_queue)
add_surface(p, &p->pipe.backward, p->queue[idx]);
}
param->backward_references = p->pipe.backward.surfaces;
param->num_forward_references = 0;
param->num_backward_references = 0;
param->num_backward_references = p->pipe.backward.num_surfaces;
vaUnmapBuffer(p->display, buffer);
@ -200,9 +239,14 @@ cleanup:
return NULL;
}
static void process(struct vf_instance *vf, struct mp_image *in)
static void output_frames(struct vf_instance *vf)
{
struct vf_priv_s *p = vf->priv;
struct mp_image *in = p->queue[p->current_pos];
double prev_pts = p->current_pos + 1 < p->num_queue
? p->queue[p->current_pos + 1]->pts : MP_NOPTS_VALUE;
bool deint = p->do_deint && p->deint_type > 0;
if (!update_pipeline(vf, deint) || !p->pipe.filters) { // no filtering
vf_add_output_frame(vf, mp_image_new_ref(in));
@ -220,8 +264,8 @@ static void process(struct vf_instance *vf, struct mp_image *in)
// first-field only
if (field == VA_FRAME_PICTURE || (p->do_deint && p->deint_type < 2))
return;
double add = (in->pts - p->prev_pts) * 0.5;
if (p->prev_pts == MP_NOPTS_VALUE || add <= 0.0 || add > 0.5) // no pts, skip it
double add = (in->pts - prev_pts) * 0.5;
if (prev_pts == MP_NOPTS_VALUE || add <= 0.0 || add > 0.5) // no pts, skip it
return;
struct mp_image *out2 = render(vf, in, get_deint_field(p, 1, in) | csp);
if (!out2) // cannot render
@ -249,27 +293,46 @@ static struct mp_image *upload(struct vf_instance *vf, struct mp_image *in)
static int filter_ext(struct vf_instance *vf, struct mp_image *in)
{
struct vf_priv_s *p = vf->priv;
if (!in)
return 0;
int rt_format = in->imgfmt == IMGFMT_VAAPI ? va_surface_rt_format(in)
: VA_RT_FORMAT_YUV420;
if (!p->pool || p->current_rt_format != rt_format) {
talloc_free(p->pool);
p->pool = mp_image_pool_new(20);
va_pool_set_allocator(p->pool, p->va, rt_format);
p->current_rt_format = rt_format;
}
if (in->imgfmt != IMGFMT_VAAPI) {
struct mp_image *tmp = upload(vf, in);
talloc_free(in);
in = tmp;
if (!in)
return -1;
if (in) {
int rt_format = in->imgfmt == IMGFMT_VAAPI ? va_surface_rt_format(in)
: VA_RT_FORMAT_YUV420;
if (!p->pool || p->current_rt_format != rt_format) {
talloc_free(p->pool);
p->pool = mp_image_pool_new(20);
va_pool_set_allocator(p->pool, p->va, rt_format);
p->current_rt_format = rt_format;
}
if (in->imgfmt != IMGFMT_VAAPI) {
struct mp_image *tmp = upload(vf, in);
talloc_free(in);
in = tmp;
if (!in)
return -1;
}
}
process(vf, in);
p->prev_pts = in->pts;
talloc_free(in);
if (in) {
MP_TARRAY_INSERT_AT(p, p->queue, p->num_queue, 0, in);
p->current_pos++;
assert(p->num_queue != 1 || p->current_pos == 0);
}
// Discard unneeded past frames.
// Note that we keep at least 1 past frame (for PTS calculations).
while (p->num_queue - (p->current_pos + 1) > MPMAX(p->needed_past_frames, 1)) {
assert(p->num_queue > 0);
talloc_free(p->queue[p->num_queue - 1]);
p->num_queue--;
}
if (p->current_pos < p->needed_future_frames && in)
return 0; // wait until future frames have been filled
if (p->current_pos >= 0 && p->current_pos < p->num_queue) {
output_frames(vf);
p->current_pos--;
}
return 0;
}
@ -278,10 +341,10 @@ static int reconfig(struct vf_instance *vf, struct mp_image_params *in,
{
struct vf_priv_s *p = vf->priv;
p->prev_pts = MP_NOPTS_VALUE;
p->params = *in;
*out = *in;
out->imgfmt = IMGFMT_VAAPI;
flush_frames(vf);
return 0;
}
@ -295,6 +358,7 @@ static void uninit(struct vf_instance *vf)
if (p->config != VA_INVALID_ID)
vaDestroyConfig(p->display, p->config);
talloc_free(p->pool);
flush_frames(vf);
}
static int query_format(struct vf_instance *vf, unsigned int imgfmt)
@ -315,6 +379,9 @@ static int control(struct vf_instance *vf, int request, void* data)
case VFCTRL_SET_DEINTERLACE:
p->do_deint = *(int*)data;
return true;
case VFCTRL_SEEK_RESET:
flush_frames(vf);
return true;
default:
return CONTROL_UNKNOWN;
}