1 /**************************************************************************
2  *
3  * Copyright 2010 Thomas Balling Sørensen.
4  * All Rights Reserved.
5  *
6  * Permission is hereby granted, free of charge, to any person obtaining a
7  * copy of this software and associated documentation files (the
8  * "Software"), to deal in the Software without restriction, including
9  * without limitation the rights to use, copy, modify, merge, publish,
10  * distribute, sub license, and/or sell copies of the Software, and to
11  * permit persons to whom the Software is furnished to do so, subject to
12  * the following conditions:
13  *
14  * The above copyright notice and this permission notice (including the
15  * next paragraph) shall be included in all copies or substantial portions
16  * of the Software.
17  *
18  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
19  * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
20  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT.
21  * IN NO EVENT SHALL VMWARE AND/OR ITS SUPPLIERS BE LIABLE FOR
22  * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
23  * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
24  * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
25  *
26  **************************************************************************/
27 
28 #include <vdpau/vdpau.h>
29 
30 #include "util/u_memory.h"
31 #include "util/u_debug.h"
32 
33 #include "vl/vl_csc.h"
34 
35 #include "vdpau_private.h"
36 
37 /**
38  * Create a VdpVideoMixer.
39  */
40 VdpStatus
vlVdpVideoMixerCreate(VdpDevice device,uint32_t feature_count,VdpVideoMixerFeature const * features,uint32_t parameter_count,VdpVideoMixerParameter const * parameters,void const * const * parameter_values,VdpVideoMixer * mixer)41 vlVdpVideoMixerCreate(VdpDevice device,
42                       uint32_t feature_count,
43                       VdpVideoMixerFeature const *features,
44                       uint32_t parameter_count,
45                       VdpVideoMixerParameter const *parameters,
46                       void const *const *parameter_values,
47                       VdpVideoMixer *mixer)
48 {
49    vlVdpVideoMixer *vmixer = NULL;
50    VdpStatus ret;
51    struct pipe_screen *screen;
52    uint32_t max_2d_texture_level;
53    unsigned max_size, i;
54 
55    vlVdpDevice *dev = vlGetDataHTAB(device);
56    if (!dev)
57       return VDP_STATUS_INVALID_HANDLE;
58    screen = dev->vscreen->pscreen;
59 
60    vmixer = CALLOC(1, sizeof(vlVdpVideoMixer));
61    if (!vmixer)
62       return VDP_STATUS_RESOURCES;
63 
64    DeviceReference(&vmixer->device, dev);
65 
66    mtx_lock(&dev->mutex);
67 
68    if (!vl_compositor_init_state(&vmixer->cstate, dev->context)) {
69       ret = VDP_STATUS_ERROR;
70       goto no_compositor_state;
71    }
72 
73    vl_csc_get_matrix(VL_CSC_COLOR_STANDARD_BT_601, NULL, true, &vmixer->csc);
74    if (!debug_get_bool_option("G3DVL_NO_CSC", FALSE)) {
75       if (!vl_compositor_set_csc_matrix(&vmixer->cstate, (const vl_csc_matrix *)&vmixer->csc, 1.0f, 0.0f)) {
76          ret = VDP_STATUS_ERROR;
77          goto err_csc_matrix;
78       }
79    }
80 
81    *mixer = vlAddDataHTAB(vmixer);
82    if (*mixer == 0) {
83       ret = VDP_STATUS_ERROR;
84       goto no_handle;
85    }
86 
87    ret = VDP_STATUS_INVALID_VIDEO_MIXER_FEATURE;
88    for (i = 0; i < feature_count; ++i) {
89       switch (features[i]) {
90       /* they are valid, but we doesn't support them */
91       case VDP_VIDEO_MIXER_FEATURE_DEINTERLACE_TEMPORAL_SPATIAL:
92       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L2:
93       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L3:
94       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L4:
95       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L5:
96       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L6:
97       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L7:
98       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L8:
99       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L9:
100       case VDP_VIDEO_MIXER_FEATURE_INVERSE_TELECINE:
101          break;
102 
103       case VDP_VIDEO_MIXER_FEATURE_DEINTERLACE_TEMPORAL:
104          vmixer->deint.supported = true;
105          break;
106 
107       case VDP_VIDEO_MIXER_FEATURE_SHARPNESS:
108          vmixer->sharpness.supported = true;
109          break;
110 
111       case VDP_VIDEO_MIXER_FEATURE_NOISE_REDUCTION:
112          vmixer->noise_reduction.supported = true;
113          break;
114 
115       case VDP_VIDEO_MIXER_FEATURE_LUMA_KEY:
116          vmixer->luma_key.supported = true;
117          break;
118 
119       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L1:
120          vmixer->bicubic.supported = true;
121          break;
122       default: goto no_params;
123       }
124    }
125 
126    vmixer->chroma_format = PIPE_VIDEO_CHROMA_FORMAT_420;
127    ret = VDP_STATUS_INVALID_VIDEO_MIXER_PARAMETER;
128    for (i = 0; i < parameter_count; ++i) {
129       switch (parameters[i]) {
130       case VDP_VIDEO_MIXER_PARAMETER_VIDEO_SURFACE_WIDTH:
131          vmixer->video_width = *(uint32_t*)parameter_values[i];
132          break;
133       case VDP_VIDEO_MIXER_PARAMETER_VIDEO_SURFACE_HEIGHT:
134          vmixer->video_height = *(uint32_t*)parameter_values[i];
135          break;
136       case VDP_VIDEO_MIXER_PARAMETER_CHROMA_TYPE:
137          vmixer->chroma_format = ChromaToPipe(*(VdpChromaType*)parameter_values[i]);
138          break;
139       case VDP_VIDEO_MIXER_PARAMETER_LAYERS:
140          vmixer->max_layers = *(uint32_t*)parameter_values[i];
141          break;
142       default: goto no_params;
143       }
144    }
145    ret = VDP_STATUS_INVALID_VALUE;
146    if (vmixer->max_layers > 4) {
147       VDPAU_MSG(VDPAU_WARN, "[VDPAU] Max layers > 4 not supported\n", vmixer->max_layers);
148       goto no_params;
149    }
150 
151    max_2d_texture_level = screen->get_param(screen, PIPE_CAP_MAX_TEXTURE_2D_LEVELS);
152    max_size = pow(2, max_2d_texture_level-1);
153    if (vmixer->video_width < 48 || vmixer->video_width > max_size) {
154       VDPAU_MSG(VDPAU_WARN, "[VDPAU] 48 < %u < %u not valid for width\n",
155                 vmixer->video_width, max_size);
156       goto no_params;
157    }
158    if (vmixer->video_height < 48 || vmixer->video_height > max_size) {
159       VDPAU_MSG(VDPAU_WARN, "[VDPAU] 48 < %u < %u  not valid for height\n",
160                 vmixer->video_height, max_size);
161       goto no_params;
162    }
163    vmixer->luma_key.luma_min = 1.0f;
164    vmixer->luma_key.luma_max = 0.0f;
165    mtx_unlock(&dev->mutex);
166 
167    return VDP_STATUS_OK;
168 
169 no_params:
170    vlRemoveDataHTAB(*mixer);
171 
172 no_handle:
173 err_csc_matrix:
174    vl_compositor_cleanup_state(&vmixer->cstate);
175 no_compositor_state:
176    mtx_unlock(&dev->mutex);
177    DeviceReference(&vmixer->device, NULL);
178    FREE(vmixer);
179    return ret;
180 }
181 
182 /**
183  * Destroy a VdpVideoMixer.
184  */
185 VdpStatus
vlVdpVideoMixerDestroy(VdpVideoMixer mixer)186 vlVdpVideoMixerDestroy(VdpVideoMixer mixer)
187 {
188    vlVdpVideoMixer *vmixer;
189 
190    vmixer = vlGetDataHTAB(mixer);
191    if (!vmixer)
192       return VDP_STATUS_INVALID_HANDLE;
193 
194    mtx_lock(&vmixer->device->mutex);
195 
196    vlRemoveDataHTAB(mixer);
197 
198    vl_compositor_cleanup_state(&vmixer->cstate);
199 
200    if (vmixer->deint.filter) {
201       vl_deint_filter_cleanup(vmixer->deint.filter);
202       FREE(vmixer->deint.filter);
203    }
204 
205    if (vmixer->noise_reduction.filter) {
206       vl_median_filter_cleanup(vmixer->noise_reduction.filter);
207       FREE(vmixer->noise_reduction.filter);
208    }
209 
210    if (vmixer->sharpness.filter) {
211       vl_matrix_filter_cleanup(vmixer->sharpness.filter);
212       FREE(vmixer->sharpness.filter);
213    }
214 
215    if (vmixer->bicubic.filter) {
216       vl_bicubic_filter_cleanup(vmixer->bicubic.filter);
217       FREE(vmixer->bicubic.filter);
218    }
219    mtx_unlock(&vmixer->device->mutex);
220    DeviceReference(&vmixer->device, NULL);
221 
222    FREE(vmixer);
223 
224    return VDP_STATUS_OK;
225 }
226 
227 /**
228  * Perform a video post-processing and compositing operation.
229  */
vlVdpVideoMixerRender(VdpVideoMixer mixer,VdpOutputSurface background_surface,VdpRect const * background_source_rect,VdpVideoMixerPictureStructure current_picture_structure,uint32_t video_surface_past_count,VdpVideoSurface const * video_surface_past,VdpVideoSurface video_surface_current,uint32_t video_surface_future_count,VdpVideoSurface const * video_surface_future,VdpRect const * video_source_rect,VdpOutputSurface destination_surface,VdpRect const * destination_rect,VdpRect const * destination_video_rect,uint32_t layer_count,VdpLayer const * layers)230 VdpStatus vlVdpVideoMixerRender(VdpVideoMixer mixer,
231                                 VdpOutputSurface background_surface,
232                                 VdpRect const *background_source_rect,
233                                 VdpVideoMixerPictureStructure current_picture_structure,
234                                 uint32_t video_surface_past_count,
235                                 VdpVideoSurface const *video_surface_past,
236                                 VdpVideoSurface video_surface_current,
237                                 uint32_t video_surface_future_count,
238                                 VdpVideoSurface const *video_surface_future,
239                                 VdpRect const *video_source_rect,
240                                 VdpOutputSurface destination_surface,
241                                 VdpRect const *destination_rect,
242                                 VdpRect const *destination_video_rect,
243                                 uint32_t layer_count,
244                                 VdpLayer const *layers)
245 {
246    enum vl_compositor_deinterlace deinterlace;
247    struct u_rect rect, clip, *prect, dirty_area;
248    unsigned i, layer = 0;
249    struct pipe_video_buffer *video_buffer;
250    struct pipe_sampler_view *sampler_view, sv_templ;
251    struct pipe_surface *surface, surf_templ;
252    struct pipe_context *pipe = NULL;
253    struct pipe_resource res_tmpl, *res;
254 
255    vlVdpVideoMixer *vmixer;
256    vlVdpSurface *surf;
257    vlVdpOutputSurface *dst, *bg = NULL;
258 
259    struct vl_compositor *compositor;
260 
261    vmixer = vlGetDataHTAB(mixer);
262    if (!vmixer)
263       return VDP_STATUS_INVALID_HANDLE;
264 
265    compositor = &vmixer->device->compositor;
266 
267    surf = vlGetDataHTAB(video_surface_current);
268    if (!surf)
269       return VDP_STATUS_INVALID_HANDLE;
270    video_buffer = surf->video_buffer;
271 
272    if (surf->device != vmixer->device)
273       return VDP_STATUS_HANDLE_DEVICE_MISMATCH;
274 
275    if (vmixer->video_width > video_buffer->width ||
276        vmixer->video_height > video_buffer->height ||
277        vmixer->chroma_format != video_buffer->chroma_format)
278       return VDP_STATUS_INVALID_SIZE;
279 
280    if (layer_count > vmixer->max_layers)
281       return VDP_STATUS_INVALID_VALUE;
282 
283    dst = vlGetDataHTAB(destination_surface);
284    if (!dst)
285       return VDP_STATUS_INVALID_HANDLE;
286 
287    if (background_surface != VDP_INVALID_HANDLE) {
288       bg = vlGetDataHTAB(background_surface);
289       if (!bg)
290          return VDP_STATUS_INVALID_HANDLE;
291    }
292 
293    mtx_lock(&vmixer->device->mutex);
294 
295    vl_compositor_clear_layers(&vmixer->cstate);
296 
297    if (bg)
298       vl_compositor_set_rgba_layer(&vmixer->cstate, compositor, layer++, bg->sampler_view,
299                                    RectToPipe(background_source_rect, &rect), NULL, NULL);
300 
301    switch (current_picture_structure) {
302    case VDP_VIDEO_MIXER_PICTURE_STRUCTURE_TOP_FIELD:
303       deinterlace = VL_COMPOSITOR_BOB_TOP;
304       break;
305 
306    case VDP_VIDEO_MIXER_PICTURE_STRUCTURE_BOTTOM_FIELD:
307       deinterlace = VL_COMPOSITOR_BOB_BOTTOM;
308       break;
309 
310    case VDP_VIDEO_MIXER_PICTURE_STRUCTURE_FRAME:
311       deinterlace = VL_COMPOSITOR_WEAVE;
312       break;
313 
314    default:
315       mtx_unlock(&vmixer->device->mutex);
316       return VDP_STATUS_INVALID_VIDEO_MIXER_PICTURE_STRUCTURE;
317    }
318 
319    if (deinterlace != VL_COMPOSITOR_WEAVE && vmixer->deint.enabled &&
320        video_surface_past_count > 1 && video_surface_future_count > 0) {
321       vlVdpSurface *prevprev = vlGetDataHTAB(video_surface_past[1]);
322       vlVdpSurface *prev = vlGetDataHTAB(video_surface_past[0]);
323       vlVdpSurface *next = vlGetDataHTAB(video_surface_future[0]);
324       if (prevprev && prev && next &&
325           vl_deint_filter_check_buffers(vmixer->deint.filter,
326           prevprev->video_buffer, prev->video_buffer, surf->video_buffer, next->video_buffer)) {
327          vl_deint_filter_render(vmixer->deint.filter, prevprev->video_buffer,
328                                 prev->video_buffer, surf->video_buffer,
329                                 next->video_buffer,
330                                 deinterlace == VL_COMPOSITOR_BOB_BOTTOM);
331          deinterlace = VL_COMPOSITOR_WEAVE;
332          video_buffer = vmixer->deint.filter->video_buffer;
333       }
334    }
335 
336    prect = RectToPipe(video_source_rect, &rect);
337    if (!prect) {
338       rect.x0 = 0;
339       rect.y0 = 0;
340       rect.x1 = surf->templat.width;
341       rect.y1 = surf->templat.height;
342       prect = &rect;
343    }
344    vl_compositor_set_buffer_layer(&vmixer->cstate, compositor, layer, video_buffer, prect, NULL, deinterlace);
345 
346    if (vmixer->bicubic.filter || vmixer->sharpness.filter || vmixer->noise_reduction.filter) {
347       pipe = vmixer->device->context;
348       memset(&res_tmpl, 0, sizeof(res_tmpl));
349 
350       res_tmpl.target = PIPE_TEXTURE_2D;
351       res_tmpl.format = dst->sampler_view->format;
352       res_tmpl.depth0 = 1;
353       res_tmpl.array_size = 1;
354       res_tmpl.bind = PIPE_BIND_SAMPLER_VIEW | PIPE_BIND_RENDER_TARGET;
355       res_tmpl.usage = PIPE_USAGE_DEFAULT;
356 
357       if (!vmixer->bicubic.filter) {
358          res_tmpl.width0 = dst->surface->width;
359          res_tmpl.height0 = dst->surface->height;
360       } else {
361          res_tmpl.width0 = surf->templat.width;
362          res_tmpl.height0 = surf->templat.height;
363       }
364 
365       res = pipe->screen->resource_create(pipe->screen, &res_tmpl);
366 
367       vlVdpDefaultSamplerViewTemplate(&sv_templ, res);
368       sampler_view = pipe->create_sampler_view(pipe, res, &sv_templ);
369 
370       memset(&surf_templ, 0, sizeof(surf_templ));
371       surf_templ.format = res->format;
372       surface = pipe->create_surface(pipe, res, &surf_templ);
373 
374       vl_compositor_reset_dirty_area(&dirty_area);
375       pipe_resource_reference(&res, NULL);
376    } else {
377       surface = dst->surface;
378       sampler_view = dst->sampler_view;
379       dirty_area = dst->dirty_area;
380    }
381 
382    if (!vmixer->bicubic.filter) {
383       vl_compositor_set_layer_dst_area(&vmixer->cstate, layer++, RectToPipe(destination_video_rect, &rect));
384       vl_compositor_set_dst_clip(&vmixer->cstate, RectToPipe(destination_rect, &clip));
385    }
386 
387    for (i = 0; i < layer_count; ++i) {
388       vlVdpOutputSurface *src = vlGetDataHTAB(layers->source_surface);
389       if (!src) {
390          mtx_unlock(&vmixer->device->mutex);
391          return VDP_STATUS_INVALID_HANDLE;
392       }
393 
394       assert(layers->struct_version == VDP_LAYER_VERSION);
395 
396       vl_compositor_set_rgba_layer(&vmixer->cstate, compositor, layer, src->sampler_view,
397                                    RectToPipe(layers->source_rect, &rect), NULL, NULL);
398       vl_compositor_set_layer_dst_area(&vmixer->cstate, layer++, RectToPipe(layers->destination_rect, &rect));
399 
400       ++layers;
401    }
402 
403    vl_compositor_render(&vmixer->cstate, compositor, surface, &dirty_area, true);
404 
405    if (vmixer->noise_reduction.filter) {
406       if (!vmixer->sharpness.filter && !vmixer->bicubic.filter) {
407          vl_median_filter_render(vmixer->noise_reduction.filter,
408                                  sampler_view, dst->surface);
409       } else {
410          res = pipe->screen->resource_create(pipe->screen, &res_tmpl);
411          struct pipe_sampler_view *sampler_view_temp = pipe->create_sampler_view(pipe, res, &sv_templ);
412          struct pipe_surface *surface_temp = pipe->create_surface(pipe, res, &surf_templ);
413          pipe_resource_reference(&res, NULL);
414 
415          vl_median_filter_render(vmixer->noise_reduction.filter,
416                                  sampler_view, surface_temp);
417 
418          pipe_sampler_view_reference(&sampler_view, NULL);
419          pipe_surface_reference(&surface, NULL);
420 
421          sampler_view = sampler_view_temp;
422          surface = surface_temp;
423       }
424    }
425 
426    if (vmixer->sharpness.filter) {
427       if (!vmixer->bicubic.filter) {
428          vl_matrix_filter_render(vmixer->sharpness.filter,
429                                  sampler_view, dst->surface);
430       } else {
431          res = pipe->screen->resource_create(pipe->screen, &res_tmpl);
432          struct pipe_sampler_view *sampler_view_temp = pipe->create_sampler_view(pipe, res, &sv_templ);
433          struct pipe_surface *surface_temp = pipe->create_surface(pipe, res, &surf_templ);
434          pipe_resource_reference(&res, NULL);
435 
436          vl_matrix_filter_render(vmixer->sharpness.filter,
437                                  sampler_view, surface_temp);
438 
439          pipe_sampler_view_reference(&sampler_view, NULL);
440          pipe_surface_reference(&surface, NULL);
441 
442          sampler_view = sampler_view_temp;
443          surface = surface_temp;
444       }
445    }
446 
447    if (vmixer->bicubic.filter)
448       vl_bicubic_filter_render(vmixer->bicubic.filter,
449                                sampler_view, dst->surface,
450                                RectToPipe(destination_video_rect, &rect),
451                                RectToPipe(destination_rect, &clip));
452 
453    if(surface != dst->surface) {
454       pipe_sampler_view_reference(&sampler_view, NULL);
455       pipe_surface_reference(&surface, NULL);
456    }
457    mtx_unlock(&vmixer->device->mutex);
458 
459    return VDP_STATUS_OK;
460 }
461 
462 static void
vlVdpVideoMixerUpdateDeinterlaceFilter(vlVdpVideoMixer * vmixer)463 vlVdpVideoMixerUpdateDeinterlaceFilter(vlVdpVideoMixer *vmixer)
464 {
465    struct pipe_context *pipe = vmixer->device->context;
466    assert(vmixer);
467 
468    /* remove existing filter */
469    if (vmixer->deint.filter) {
470       vl_deint_filter_cleanup(vmixer->deint.filter);
471       FREE(vmixer->deint.filter);
472       vmixer->deint.filter = NULL;
473    }
474 
475    /* create a new filter if requested */
476    if (vmixer->deint.enabled && vmixer->chroma_format == PIPE_VIDEO_CHROMA_FORMAT_420) {
477       vmixer->deint.filter = MALLOC(sizeof(struct vl_deint_filter));
478       vmixer->deint.enabled = vl_deint_filter_init(vmixer->deint.filter, pipe,
479             vmixer->video_width, vmixer->video_height,
480             vmixer->skip_chroma_deint, vmixer->deint.spatial);
481       if (!vmixer->deint.enabled) {
482          FREE(vmixer->deint.filter);
483       }
484    }
485 }
486 
487 /**
488  * Update the noise reduction setting
489  */
490 static void
vlVdpVideoMixerUpdateNoiseReductionFilter(vlVdpVideoMixer * vmixer)491 vlVdpVideoMixerUpdateNoiseReductionFilter(vlVdpVideoMixer *vmixer)
492 {
493    assert(vmixer);
494 
495    /* if present remove the old filter first */
496    if (vmixer->noise_reduction.filter) {
497       vl_median_filter_cleanup(vmixer->noise_reduction.filter);
498       FREE(vmixer->noise_reduction.filter);
499       vmixer->noise_reduction.filter = NULL;
500    }
501 
502    /* and create a new filter as needed */
503    if (vmixer->noise_reduction. enabled && vmixer->noise_reduction.level > 0) {
504       vmixer->noise_reduction.filter = MALLOC(sizeof(struct vl_median_filter));
505       vl_median_filter_init(vmixer->noise_reduction.filter, vmixer->device->context,
506                             vmixer->video_width, vmixer->video_height,
507                             vmixer->noise_reduction.level + 1,
508                             VL_MEDIAN_FILTER_CROSS);
509    }
510 }
511 
512 static void
vlVdpVideoMixerUpdateSharpnessFilter(vlVdpVideoMixer * vmixer)513 vlVdpVideoMixerUpdateSharpnessFilter(vlVdpVideoMixer *vmixer)
514 {
515    assert(vmixer);
516 
517    /* if present remove the old filter first */
518    if (vmixer->sharpness.filter) {
519       vl_matrix_filter_cleanup(vmixer->sharpness.filter);
520       FREE(vmixer->sharpness.filter);
521       vmixer->sharpness.filter = NULL;
522    }
523 
524    /* and create a new filter as needed */
525    if (vmixer->sharpness.enabled && vmixer->sharpness.value != 0.0f) {
526       float matrix[9];
527       unsigned i;
528 
529       if (vmixer->sharpness.value > 0.0f) {
530          matrix[0] = -1.0f; matrix[1] = -1.0f; matrix[2] = -1.0f;
531          matrix[3] = -1.0f; matrix[4] =  8.0f; matrix[5] = -1.0f;
532          matrix[6] = -1.0f; matrix[7] = -1.0f; matrix[8] = -1.0f;
533 
534          for (i = 0; i < 9; ++i)
535             matrix[i] *= vmixer->sharpness.value;
536 
537          matrix[4] += 1.0f;
538 
539       } else {
540          matrix[0] = 1.0f; matrix[1] = 2.0f; matrix[2] = 1.0f;
541          matrix[3] = 2.0f; matrix[4] = 4.0f; matrix[5] = 2.0f;
542          matrix[6] = 1.0f; matrix[7] = 2.0f; matrix[8] = 1.0f;
543 
544          for (i = 0; i < 9; ++i)
545                matrix[i] *= fabsf(vmixer->sharpness.value) / 16.0f;
546 
547          matrix[4] += 1.0f - fabsf(vmixer->sharpness.value);
548       }
549 
550       vmixer->sharpness.filter = MALLOC(sizeof(struct vl_matrix_filter));
551       vl_matrix_filter_init(vmixer->sharpness.filter, vmixer->device->context,
552                             vmixer->video_width, vmixer->video_height,
553                             3, 3, matrix);
554    }
555 }
556 
557 /**
558  * Update the bicubic filter
559  */
560 static void
vlVdpVideoMixerUpdateBicubicFilter(vlVdpVideoMixer * vmixer)561 vlVdpVideoMixerUpdateBicubicFilter(vlVdpVideoMixer *vmixer)
562 {
563    assert(vmixer);
564 
565    /* if present remove the old filter first */
566    if (vmixer->bicubic.filter) {
567       vl_bicubic_filter_cleanup(vmixer->bicubic.filter);
568       FREE(vmixer->bicubic.filter);
569       vmixer->bicubic.filter = NULL;
570    }
571    /* and create a new filter as needed */
572    if (vmixer->bicubic.enabled) {
573       vmixer->bicubic.filter = MALLOC(sizeof(struct vl_bicubic_filter));
574       vl_bicubic_filter_init(vmixer->bicubic.filter, vmixer->device->context,
575                             vmixer->video_width, vmixer->video_height);
576    }
577 }
578 
579 /**
580  * Retrieve whether features were requested at creation time.
581  */
582 VdpStatus
vlVdpVideoMixerGetFeatureSupport(VdpVideoMixer mixer,uint32_t feature_count,VdpVideoMixerFeature const * features,VdpBool * feature_supports)583 vlVdpVideoMixerGetFeatureSupport(VdpVideoMixer mixer,
584                                  uint32_t feature_count,
585                                  VdpVideoMixerFeature const *features,
586                                  VdpBool *feature_supports)
587 {
588    vlVdpVideoMixer *vmixer;
589    unsigned i;
590 
591    if (!(features && feature_supports))
592       return VDP_STATUS_INVALID_POINTER;
593 
594    vmixer = vlGetDataHTAB(mixer);
595    if (!vmixer)
596       return VDP_STATUS_INVALID_HANDLE;
597 
598    for (i = 0; i < feature_count; ++i) {
599       switch (features[i]) {
600       /* they are valid, but we doesn't support them */
601       case VDP_VIDEO_MIXER_FEATURE_DEINTERLACE_TEMPORAL_SPATIAL:
602       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L2:
603       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L3:
604       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L4:
605       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L5:
606       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L6:
607       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L7:
608       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L8:
609       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L9:
610       case VDP_VIDEO_MIXER_FEATURE_INVERSE_TELECINE:
611          feature_supports[i] = false;
612          break;
613 
614       case VDP_VIDEO_MIXER_FEATURE_DEINTERLACE_TEMPORAL:
615          feature_supports[i] = vmixer->deint.supported;
616          break;
617 
618       case VDP_VIDEO_MIXER_FEATURE_SHARPNESS:
619          feature_supports[i] = vmixer->sharpness.supported;
620          break;
621 
622       case VDP_VIDEO_MIXER_FEATURE_NOISE_REDUCTION:
623          feature_supports[i] = vmixer->noise_reduction.supported;
624          break;
625 
626       case VDP_VIDEO_MIXER_FEATURE_LUMA_KEY:
627          feature_supports[i] = vmixer->luma_key.supported;
628          break;
629 
630       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L1:
631          feature_supports[i] = vmixer->bicubic.supported;
632          break;
633 
634       default:
635          return VDP_STATUS_INVALID_VIDEO_MIXER_FEATURE;
636       }
637    }
638 
639    return VDP_STATUS_OK;
640 }
641 
642 /**
643  * Enable or disable features.
644  */
645 VdpStatus
vlVdpVideoMixerSetFeatureEnables(VdpVideoMixer mixer,uint32_t feature_count,VdpVideoMixerFeature const * features,VdpBool const * feature_enables)646 vlVdpVideoMixerSetFeatureEnables(VdpVideoMixer mixer,
647                                  uint32_t feature_count,
648                                  VdpVideoMixerFeature const *features,
649                                  VdpBool const *feature_enables)
650 {
651    vlVdpVideoMixer *vmixer;
652    unsigned i;
653 
654    if (!(features && feature_enables))
655       return VDP_STATUS_INVALID_POINTER;
656 
657    vmixer = vlGetDataHTAB(mixer);
658    if (!vmixer)
659       return VDP_STATUS_INVALID_HANDLE;
660 
661    mtx_lock(&vmixer->device->mutex);
662    for (i = 0; i < feature_count; ++i) {
663       switch (features[i]) {
664       /* they are valid, but we doesn't support them */
665       case VDP_VIDEO_MIXER_FEATURE_DEINTERLACE_TEMPORAL_SPATIAL:
666       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L2:
667       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L3:
668       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L4:
669       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L5:
670       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L6:
671       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L7:
672       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L8:
673       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L9:
674       case VDP_VIDEO_MIXER_FEATURE_INVERSE_TELECINE:
675          break;
676 
677       case VDP_VIDEO_MIXER_FEATURE_DEINTERLACE_TEMPORAL:
678          vmixer->deint.enabled = feature_enables[i];
679          vlVdpVideoMixerUpdateDeinterlaceFilter(vmixer);
680          break;
681 
682       case VDP_VIDEO_MIXER_FEATURE_SHARPNESS:
683          vmixer->sharpness.enabled = feature_enables[i];
684          vlVdpVideoMixerUpdateSharpnessFilter(vmixer);
685          break;
686 
687       case VDP_VIDEO_MIXER_FEATURE_NOISE_REDUCTION:
688          vmixer->noise_reduction.enabled = feature_enables[i];
689          vlVdpVideoMixerUpdateNoiseReductionFilter(vmixer);
690          break;
691 
692       case VDP_VIDEO_MIXER_FEATURE_LUMA_KEY:
693          vmixer->luma_key.enabled = feature_enables[i];
694          if (!debug_get_bool_option("G3DVL_NO_CSC", FALSE))
695             if (!vl_compositor_set_csc_matrix(&vmixer->cstate, (const vl_csc_matrix *)&vmixer->csc,
696                         vmixer->luma_key.luma_min, vmixer->luma_key.luma_max)) {
697                mtx_unlock(&vmixer->device->mutex);
698                return VDP_STATUS_ERROR;
699             }
700          break;
701 
702       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L1:
703          vmixer->bicubic.enabled = feature_enables[i];
704          vlVdpVideoMixerUpdateBicubicFilter(vmixer);
705          break;
706 
707       default:
708          mtx_unlock(&vmixer->device->mutex);
709          return VDP_STATUS_INVALID_VIDEO_MIXER_FEATURE;
710       }
711    }
712    mtx_unlock(&vmixer->device->mutex);
713 
714    return VDP_STATUS_OK;
715 }
716 
717 /**
718  * Retrieve whether features are enabled.
719  */
720 VdpStatus
vlVdpVideoMixerGetFeatureEnables(VdpVideoMixer mixer,uint32_t feature_count,VdpVideoMixerFeature const * features,VdpBool * feature_enables)721 vlVdpVideoMixerGetFeatureEnables(VdpVideoMixer mixer,
722                                  uint32_t feature_count,
723                                  VdpVideoMixerFeature const *features,
724                                  VdpBool *feature_enables)
725 {
726    vlVdpVideoMixer *vmixer;
727    unsigned i;
728 
729    if (!(features && feature_enables))
730       return VDP_STATUS_INVALID_POINTER;
731 
732    vmixer = vlGetDataHTAB(mixer);
733    if (!vmixer)
734       return VDP_STATUS_INVALID_HANDLE;
735 
736    for (i = 0; i < feature_count; ++i) {
737       switch (features[i]) {
738       /* they are valid, but we doesn't support them */
739       case VDP_VIDEO_MIXER_FEATURE_DEINTERLACE_TEMPORAL:
740       case VDP_VIDEO_MIXER_FEATURE_DEINTERLACE_TEMPORAL_SPATIAL:
741       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L2:
742       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L3:
743       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L4:
744       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L5:
745       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L6:
746       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L7:
747       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L8:
748       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L9:
749       case VDP_VIDEO_MIXER_FEATURE_INVERSE_TELECINE:
750          break;
751 
752       case VDP_VIDEO_MIXER_FEATURE_SHARPNESS:
753          feature_enables[i] = vmixer->sharpness.enabled;
754          break;
755 
756       case VDP_VIDEO_MIXER_FEATURE_NOISE_REDUCTION:
757          feature_enables[i] = vmixer->noise_reduction.enabled;
758          break;
759 
760       case VDP_VIDEO_MIXER_FEATURE_LUMA_KEY:
761          feature_enables[i] = vmixer->luma_key.enabled;
762          break;
763 
764       case VDP_VIDEO_MIXER_FEATURE_HIGH_QUALITY_SCALING_L1:
765          feature_enables[i] = vmixer->bicubic.enabled;
766          break;
767 
768       default:
769          return VDP_STATUS_INVALID_VIDEO_MIXER_FEATURE;
770       }
771    }
772 
773    return VDP_STATUS_OK;
774 }
775 
776 /**
777  * Set attribute values.
778  */
779 VdpStatus
vlVdpVideoMixerSetAttributeValues(VdpVideoMixer mixer,uint32_t attribute_count,VdpVideoMixerAttribute const * attributes,void const * const * attribute_values)780 vlVdpVideoMixerSetAttributeValues(VdpVideoMixer mixer,
781                                   uint32_t attribute_count,
782                                   VdpVideoMixerAttribute const *attributes,
783                                   void const *const *attribute_values)
784 {
785    const VdpColor *background_color;
786    union pipe_color_union color;
787    const float *vdp_csc;
788    float val;
789    unsigned i;
790    VdpStatus ret;
791 
792    if (!(attributes && attribute_values))
793       return VDP_STATUS_INVALID_POINTER;
794 
795    vlVdpVideoMixer *vmixer = vlGetDataHTAB(mixer);
796    if (!vmixer)
797       return VDP_STATUS_INVALID_HANDLE;
798 
799    mtx_lock(&vmixer->device->mutex);
800    for (i = 0; i < attribute_count; ++i) {
801       switch (attributes[i]) {
802       case VDP_VIDEO_MIXER_ATTRIBUTE_BACKGROUND_COLOR:
803          background_color = attribute_values[i];
804          color.f[0] = background_color->red;
805          color.f[1] = background_color->green;
806          color.f[2] = background_color->blue;
807          color.f[3] = background_color->alpha;
808          vl_compositor_set_clear_color(&vmixer->cstate, &color);
809          break;
810       case VDP_VIDEO_MIXER_ATTRIBUTE_CSC_MATRIX:
811          vdp_csc = attribute_values[i];
812          vmixer->custom_csc = !!vdp_csc;
813          if (!vdp_csc)
814             vl_csc_get_matrix(VL_CSC_COLOR_STANDARD_BT_601, NULL, 1, &vmixer->csc);
815          else
816             memcpy(vmixer->csc, vdp_csc, sizeof(vl_csc_matrix));
817          if (!debug_get_bool_option("G3DVL_NO_CSC", FALSE))
818             if (!vl_compositor_set_csc_matrix(&vmixer->cstate, (const vl_csc_matrix *)&vmixer->csc,
819                                          vmixer->luma_key.luma_min, vmixer->luma_key.luma_max)) {
820                ret = VDP_STATUS_ERROR;
821                goto fail;
822             }
823          break;
824 
825       case VDP_VIDEO_MIXER_ATTRIBUTE_NOISE_REDUCTION_LEVEL:
826 
827          val = *(float*)attribute_values[i];
828          if (val < 0.0f || val > 1.0f) {
829             ret = VDP_STATUS_INVALID_VALUE;
830             goto fail;
831          }
832 
833          vmixer->noise_reduction.level = val * 10;
834          vlVdpVideoMixerUpdateNoiseReductionFilter(vmixer);
835          break;
836 
837       case VDP_VIDEO_MIXER_ATTRIBUTE_LUMA_KEY_MIN_LUMA:
838          val = *(float*)attribute_values[i];
839          if (val < 0.0f || val > 1.0f) {
840             ret = VDP_STATUS_INVALID_VALUE;
841             goto fail;
842          }
843          vmixer->luma_key.luma_min = val;
844          if (!debug_get_bool_option("G3DVL_NO_CSC", FALSE))
845             if (!vl_compositor_set_csc_matrix(&vmixer->cstate, (const vl_csc_matrix *)&vmixer->csc,
846                         vmixer->luma_key.luma_min, vmixer->luma_key.luma_max)) {
847                ret = VDP_STATUS_ERROR;
848                goto fail;
849             }
850          break;
851 
852       case VDP_VIDEO_MIXER_ATTRIBUTE_LUMA_KEY_MAX_LUMA:
853          val = *(float*)attribute_values[i];
854          if (val < 0.0f || val > 1.0f) {
855             ret = VDP_STATUS_INVALID_VALUE;
856             goto fail;
857          }
858          vmixer->luma_key.luma_max = val;
859          if (!debug_get_bool_option("G3DVL_NO_CSC", FALSE))
860             if (!vl_compositor_set_csc_matrix(&vmixer->cstate, (const vl_csc_matrix *)&vmixer->csc,
861                         vmixer->luma_key.luma_min, vmixer->luma_key.luma_max)) {
862                ret = VDP_STATUS_ERROR;
863                goto fail;
864             }
865          break;
866 
867       case VDP_VIDEO_MIXER_ATTRIBUTE_SHARPNESS_LEVEL:
868 
869          val = *(float*)attribute_values[i];
870          if (val < -1.0f || val > 1.0f) {
871             ret = VDP_STATUS_INVALID_VALUE;
872             goto fail;
873          }
874 
875          vmixer->sharpness.value = val;
876          vlVdpVideoMixerUpdateSharpnessFilter(vmixer);
877          break;
878 
879       case VDP_VIDEO_MIXER_ATTRIBUTE_SKIP_CHROMA_DEINTERLACE:
880          if (*(uint8_t*)attribute_values[i] > 1) {
881             ret = VDP_STATUS_INVALID_VALUE;
882             goto fail;
883          }
884          vmixer->skip_chroma_deint = *(uint8_t*)attribute_values[i];
885          vlVdpVideoMixerUpdateDeinterlaceFilter(vmixer);
886          break;
887       default:
888          ret = VDP_STATUS_INVALID_VIDEO_MIXER_ATTRIBUTE;
889          goto fail;
890       }
891    }
892    mtx_unlock(&vmixer->device->mutex);
893 
894    return VDP_STATUS_OK;
895 fail:
896    mtx_unlock(&vmixer->device->mutex);
897    return ret;
898 }
899 
900 /**
901  * Retrieve parameter values given at creation time.
902  */
903 VdpStatus
vlVdpVideoMixerGetParameterValues(VdpVideoMixer mixer,uint32_t parameter_count,VdpVideoMixerParameter const * parameters,void * const * parameter_values)904 vlVdpVideoMixerGetParameterValues(VdpVideoMixer mixer,
905                                   uint32_t parameter_count,
906                                   VdpVideoMixerParameter const *parameters,
907                                   void *const *parameter_values)
908 {
909    vlVdpVideoMixer *vmixer = vlGetDataHTAB(mixer);
910    unsigned i;
911    if (!vmixer)
912       return VDP_STATUS_INVALID_HANDLE;
913 
914    if (!parameter_count)
915       return VDP_STATUS_OK;
916    if (!(parameters && parameter_values))
917       return VDP_STATUS_INVALID_POINTER;
918    for (i = 0; i < parameter_count; ++i) {
919       switch (parameters[i]) {
920       case VDP_VIDEO_MIXER_PARAMETER_VIDEO_SURFACE_WIDTH:
921          *(uint32_t*)parameter_values[i] = vmixer->video_width;
922          break;
923       case VDP_VIDEO_MIXER_PARAMETER_VIDEO_SURFACE_HEIGHT:
924          *(uint32_t*)parameter_values[i] = vmixer->video_height;
925          break;
926       case VDP_VIDEO_MIXER_PARAMETER_CHROMA_TYPE:
927          *(VdpChromaType*)parameter_values[i] = PipeToChroma(vmixer->chroma_format);
928          break;
929       case VDP_VIDEO_MIXER_PARAMETER_LAYERS:
930          *(uint32_t*)parameter_values[i] = vmixer->max_layers;
931          break;
932       default:
933          return VDP_STATUS_INVALID_VIDEO_MIXER_PARAMETER;
934       }
935    }
936    return VDP_STATUS_OK;
937 }
938 
939 /**
940  * Retrieve current attribute values.
941  */
942 VdpStatus
vlVdpVideoMixerGetAttributeValues(VdpVideoMixer mixer,uint32_t attribute_count,VdpVideoMixerAttribute const * attributes,void * const * attribute_values)943 vlVdpVideoMixerGetAttributeValues(VdpVideoMixer mixer,
944                                   uint32_t attribute_count,
945                                   VdpVideoMixerAttribute const *attributes,
946                                   void *const *attribute_values)
947 {
948    unsigned i;
949    VdpCSCMatrix **vdp_csc;
950 
951    if (!(attributes && attribute_values))
952       return VDP_STATUS_INVALID_POINTER;
953 
954    vlVdpVideoMixer *vmixer = vlGetDataHTAB(mixer);
955    if (!vmixer)
956       return VDP_STATUS_INVALID_HANDLE;
957 
958    mtx_lock(&vmixer->device->mutex);
959    for (i = 0; i < attribute_count; ++i) {
960       switch (attributes[i]) {
961       case VDP_VIDEO_MIXER_ATTRIBUTE_BACKGROUND_COLOR:
962          vl_compositor_get_clear_color(&vmixer->cstate, attribute_values[i]);
963          break;
964       case VDP_VIDEO_MIXER_ATTRIBUTE_CSC_MATRIX:
965          vdp_csc = attribute_values[i];
966          if (!vmixer->custom_csc) {
967              *vdp_csc = NULL;
968             break;
969          }
970          memcpy(*vdp_csc, vmixer->csc, sizeof(float)*12);
971          break;
972 
973       case VDP_VIDEO_MIXER_ATTRIBUTE_NOISE_REDUCTION_LEVEL:
974          *(float*)attribute_values[i] = (float)vmixer->noise_reduction.level / 10.0f;
975          break;
976 
977       case VDP_VIDEO_MIXER_ATTRIBUTE_LUMA_KEY_MIN_LUMA:
978          *(float*)attribute_values[i] = vmixer->luma_key.luma_min;
979          break;
980       case VDP_VIDEO_MIXER_ATTRIBUTE_LUMA_KEY_MAX_LUMA:
981          *(float*)attribute_values[i] = vmixer->luma_key.luma_max;
982          break;
983       case VDP_VIDEO_MIXER_ATTRIBUTE_SHARPNESS_LEVEL:
984          *(float*)attribute_values[i] = vmixer->sharpness.value;
985          break;
986       case VDP_VIDEO_MIXER_ATTRIBUTE_SKIP_CHROMA_DEINTERLACE:
987          *(uint8_t*)attribute_values[i] = vmixer->skip_chroma_deint;
988          break;
989       default:
990          mtx_unlock(&vmixer->device->mutex);
991          return VDP_STATUS_INVALID_VIDEO_MIXER_ATTRIBUTE;
992       }
993    }
994    mtx_unlock(&vmixer->device->mutex);
995    return VDP_STATUS_OK;
996 }
997 
998 /**
999  * Generate a color space conversion matrix.
1000  */
1001 VdpStatus
vlVdpGenerateCSCMatrix(VdpProcamp * procamp,VdpColorStandard standard,VdpCSCMatrix * csc_matrix)1002 vlVdpGenerateCSCMatrix(VdpProcamp *procamp,
1003                        VdpColorStandard standard,
1004                        VdpCSCMatrix *csc_matrix)
1005 {
1006    enum VL_CSC_COLOR_STANDARD vl_std;
1007    struct vl_procamp camp;
1008 
1009    if (!csc_matrix)
1010       return VDP_STATUS_INVALID_POINTER;
1011 
1012    switch (standard) {
1013       case VDP_COLOR_STANDARD_ITUR_BT_601: vl_std = VL_CSC_COLOR_STANDARD_BT_601; break;
1014       case VDP_COLOR_STANDARD_ITUR_BT_709: vl_std = VL_CSC_COLOR_STANDARD_BT_709; break;
1015       case VDP_COLOR_STANDARD_SMPTE_240M:  vl_std = VL_CSC_COLOR_STANDARD_SMPTE_240M; break;
1016       default: return VDP_STATUS_INVALID_COLOR_STANDARD;
1017    }
1018 
1019    if (procamp) {
1020       if (procamp->struct_version > VDP_PROCAMP_VERSION)
1021          return VDP_STATUS_INVALID_STRUCT_VERSION;
1022       camp.brightness = procamp->brightness;
1023       camp.contrast = procamp->contrast;
1024       camp.saturation = procamp->saturation;
1025       camp.hue = procamp->hue;
1026    }
1027 
1028    vl_csc_get_matrix(vl_std, procamp ? &camp : NULL, true, csc_matrix);
1029    return VDP_STATUS_OK;
1030 }
1031