xref: /aosp_15_r20/external/mesa3d/src/gallium/frontends/va/picture.c (revision 6104692788411f58d303aa86923a9ff6ecaded22)
1*61046927SAndroid Build Coastguard Worker /**************************************************************************
2*61046927SAndroid Build Coastguard Worker  *
3*61046927SAndroid Build Coastguard Worker  * Copyright 2010 Thomas Balling Sørensen & Orasanu Lucian.
4*61046927SAndroid Build Coastguard Worker  * Copyright 2014 Advanced Micro Devices, Inc.
5*61046927SAndroid Build Coastguard Worker  * All Rights Reserved.
6*61046927SAndroid Build Coastguard Worker  *
7*61046927SAndroid Build Coastguard Worker  * Permission is hereby granted, free of charge, to any person obtaining a
8*61046927SAndroid Build Coastguard Worker  * copy of this software and associated documentation files (the
9*61046927SAndroid Build Coastguard Worker  * "Software"), to deal in the Software without restriction, including
10*61046927SAndroid Build Coastguard Worker  * without limitation the rights to use, copy, modify, merge, publish,
11*61046927SAndroid Build Coastguard Worker  * distribute, sub license, and/or sell copies of the Software, and to
12*61046927SAndroid Build Coastguard Worker  * permit persons to whom the Software is furnished to do so, subject to
13*61046927SAndroid Build Coastguard Worker  * the following conditions:
14*61046927SAndroid Build Coastguard Worker  *
15*61046927SAndroid Build Coastguard Worker  * The above copyright notice and this permission notice (including the
16*61046927SAndroid Build Coastguard Worker  * next paragraph) shall be included in all copies or substantial portions
17*61046927SAndroid Build Coastguard Worker  * of the Software.
18*61046927SAndroid Build Coastguard Worker  *
19*61046927SAndroid Build Coastguard Worker  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
20*61046927SAndroid Build Coastguard Worker  * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
21*61046927SAndroid Build Coastguard Worker  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT.
22*61046927SAndroid Build Coastguard Worker  * IN NO EVENT SHALL THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR
23*61046927SAndroid Build Coastguard Worker  * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
24*61046927SAndroid Build Coastguard Worker  * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
25*61046927SAndroid Build Coastguard Worker  * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
26*61046927SAndroid Build Coastguard Worker  *
27*61046927SAndroid Build Coastguard Worker  **************************************************************************/
28*61046927SAndroid Build Coastguard Worker 
29*61046927SAndroid Build Coastguard Worker #include "pipe/p_video_codec.h"
30*61046927SAndroid Build Coastguard Worker 
31*61046927SAndroid Build Coastguard Worker #include "util/u_handle_table.h"
32*61046927SAndroid Build Coastguard Worker #include "util/u_video.h"
33*61046927SAndroid Build Coastguard Worker #include "util/u_memory.h"
34*61046927SAndroid Build Coastguard Worker #include "util/set.h"
35*61046927SAndroid Build Coastguard Worker 
36*61046927SAndroid Build Coastguard Worker #include "util/vl_vlc.h"
37*61046927SAndroid Build Coastguard Worker #include "vl/vl_winsys.h"
38*61046927SAndroid Build Coastguard Worker 
39*61046927SAndroid Build Coastguard Worker #include "va_private.h"
40*61046927SAndroid Build Coastguard Worker 
41*61046927SAndroid Build Coastguard Worker void
vlVaSetSurfaceContext(vlVaDriver * drv,vlVaSurface * surf,vlVaContext * context)42*61046927SAndroid Build Coastguard Worker vlVaSetSurfaceContext(vlVaDriver *drv, vlVaSurface *surf, vlVaContext *context)
43*61046927SAndroid Build Coastguard Worker {
44*61046927SAndroid Build Coastguard Worker    if (surf->ctx == context)
45*61046927SAndroid Build Coastguard Worker       return;
46*61046927SAndroid Build Coastguard Worker 
47*61046927SAndroid Build Coastguard Worker    if (surf->ctx) {
48*61046927SAndroid Build Coastguard Worker       assert(_mesa_set_search(surf->ctx->surfaces, surf));
49*61046927SAndroid Build Coastguard Worker       _mesa_set_remove_key(surf->ctx->surfaces, surf);
50*61046927SAndroid Build Coastguard Worker 
51*61046927SAndroid Build Coastguard Worker       /* Only drivers supporting PIPE_VIDEO_ENTRYPOINT_PROCESSING will create
52*61046927SAndroid Build Coastguard Worker        * decoder for postproc context and thus be able to wait on and destroy
53*61046927SAndroid Build Coastguard Worker        * the surface fence. On other drivers we need to destroy the fence here
54*61046927SAndroid Build Coastguard Worker        * otherwise vaQuerySurfaceStatus/vaSyncSurface will fail and we'll also
55*61046927SAndroid Build Coastguard Worker        * potentially leak the fence.
56*61046927SAndroid Build Coastguard Worker        */
57*61046927SAndroid Build Coastguard Worker       if (surf->fence && !context->decoder &&
58*61046927SAndroid Build Coastguard Worker           context->templat.entrypoint == PIPE_VIDEO_ENTRYPOINT_PROCESSING &&
59*61046927SAndroid Build Coastguard Worker           surf->ctx->decoder && surf->ctx->decoder->destroy_fence &&
60*61046927SAndroid Build Coastguard Worker           !drv->pipe->screen->get_video_param(drv->pipe->screen,
61*61046927SAndroid Build Coastguard Worker                                               PIPE_VIDEO_PROFILE_UNKNOWN,
62*61046927SAndroid Build Coastguard Worker                                               PIPE_VIDEO_ENTRYPOINT_PROCESSING,
63*61046927SAndroid Build Coastguard Worker                                               PIPE_VIDEO_CAP_SUPPORTED)) {
64*61046927SAndroid Build Coastguard Worker          surf->ctx->decoder->destroy_fence(surf->ctx->decoder, surf->fence);
65*61046927SAndroid Build Coastguard Worker          surf->fence = NULL;
66*61046927SAndroid Build Coastguard Worker       }
67*61046927SAndroid Build Coastguard Worker    }
68*61046927SAndroid Build Coastguard Worker 
69*61046927SAndroid Build Coastguard Worker    surf->ctx = context;
70*61046927SAndroid Build Coastguard Worker    _mesa_set_add(surf->ctx->surfaces, surf);
71*61046927SAndroid Build Coastguard Worker }
72*61046927SAndroid Build Coastguard Worker 
73*61046927SAndroid Build Coastguard Worker VAStatus
vlVaBeginPicture(VADriverContextP ctx,VAContextID context_id,VASurfaceID render_target)74*61046927SAndroid Build Coastguard Worker vlVaBeginPicture(VADriverContextP ctx, VAContextID context_id, VASurfaceID render_target)
75*61046927SAndroid Build Coastguard Worker {
76*61046927SAndroid Build Coastguard Worker    vlVaDriver *drv;
77*61046927SAndroid Build Coastguard Worker    vlVaContext *context;
78*61046927SAndroid Build Coastguard Worker    vlVaSurface *surf;
79*61046927SAndroid Build Coastguard Worker 
80*61046927SAndroid Build Coastguard Worker    if (!ctx)
81*61046927SAndroid Build Coastguard Worker       return VA_STATUS_ERROR_INVALID_CONTEXT;
82*61046927SAndroid Build Coastguard Worker 
83*61046927SAndroid Build Coastguard Worker    drv = VL_VA_DRIVER(ctx);
84*61046927SAndroid Build Coastguard Worker    if (!drv)
85*61046927SAndroid Build Coastguard Worker       return VA_STATUS_ERROR_INVALID_CONTEXT;
86*61046927SAndroid Build Coastguard Worker 
87*61046927SAndroid Build Coastguard Worker    mtx_lock(&drv->mutex);
88*61046927SAndroid Build Coastguard Worker    context = handle_table_get(drv->htab, context_id);
89*61046927SAndroid Build Coastguard Worker    if (!context) {
90*61046927SAndroid Build Coastguard Worker       mtx_unlock(&drv->mutex);
91*61046927SAndroid Build Coastguard Worker       return VA_STATUS_ERROR_INVALID_CONTEXT;
92*61046927SAndroid Build Coastguard Worker    }
93*61046927SAndroid Build Coastguard Worker 
94*61046927SAndroid Build Coastguard Worker    if (u_reduce_video_profile(context->templat.profile) == PIPE_VIDEO_FORMAT_MPEG12) {
95*61046927SAndroid Build Coastguard Worker       context->desc.mpeg12.intra_matrix = NULL;
96*61046927SAndroid Build Coastguard Worker       context->desc.mpeg12.non_intra_matrix = NULL;
97*61046927SAndroid Build Coastguard Worker    }
98*61046927SAndroid Build Coastguard Worker 
99*61046927SAndroid Build Coastguard Worker    surf = handle_table_get(drv->htab, render_target);
100*61046927SAndroid Build Coastguard Worker    vlVaGetSurfaceBuffer(drv, surf);
101*61046927SAndroid Build Coastguard Worker    if (!surf || !surf->buffer) {
102*61046927SAndroid Build Coastguard Worker       mtx_unlock(&drv->mutex);
103*61046927SAndroid Build Coastguard Worker       return VA_STATUS_ERROR_INVALID_SURFACE;
104*61046927SAndroid Build Coastguard Worker    }
105*61046927SAndroid Build Coastguard Worker 
106*61046927SAndroid Build Coastguard Worker    context->target_id = render_target;
107*61046927SAndroid Build Coastguard Worker    vlVaSetSurfaceContext(drv, surf, context);
108*61046927SAndroid Build Coastguard Worker    context->target = surf->buffer;
109*61046927SAndroid Build Coastguard Worker    context->mjpeg.sampling_factor = 0;
110*61046927SAndroid Build Coastguard Worker 
111*61046927SAndroid Build Coastguard Worker    if (!context->decoder) {
112*61046927SAndroid Build Coastguard Worker 
113*61046927SAndroid Build Coastguard Worker       /* VPP */
114*61046927SAndroid Build Coastguard Worker       if (context->templat.profile == PIPE_VIDEO_PROFILE_UNKNOWN &&
115*61046927SAndroid Build Coastguard Worker           context->target->buffer_format != PIPE_FORMAT_B8G8R8A8_UNORM &&
116*61046927SAndroid Build Coastguard Worker           context->target->buffer_format != PIPE_FORMAT_R8G8B8A8_UNORM &&
117*61046927SAndroid Build Coastguard Worker           context->target->buffer_format != PIPE_FORMAT_B8G8R8X8_UNORM &&
118*61046927SAndroid Build Coastguard Worker           context->target->buffer_format != PIPE_FORMAT_R8G8B8X8_UNORM &&
119*61046927SAndroid Build Coastguard Worker           context->target->buffer_format != PIPE_FORMAT_B10G10R10A2_UNORM &&
120*61046927SAndroid Build Coastguard Worker           context->target->buffer_format != PIPE_FORMAT_R10G10B10A2_UNORM &&
121*61046927SAndroid Build Coastguard Worker           context->target->buffer_format != PIPE_FORMAT_B10G10R10X2_UNORM &&
122*61046927SAndroid Build Coastguard Worker           context->target->buffer_format != PIPE_FORMAT_R10G10B10X2_UNORM &&
123*61046927SAndroid Build Coastguard Worker           context->target->buffer_format != PIPE_FORMAT_NV12 &&
124*61046927SAndroid Build Coastguard Worker           context->target->buffer_format != PIPE_FORMAT_P010 &&
125*61046927SAndroid Build Coastguard Worker           context->target->buffer_format != PIPE_FORMAT_P016) {
126*61046927SAndroid Build Coastguard Worker          mtx_unlock(&drv->mutex);
127*61046927SAndroid Build Coastguard Worker          return VA_STATUS_ERROR_UNIMPLEMENTED;
128*61046927SAndroid Build Coastguard Worker       }
129*61046927SAndroid Build Coastguard Worker 
130*61046927SAndroid Build Coastguard Worker       if (drv->pipe->screen->get_video_param(drv->pipe->screen,
131*61046927SAndroid Build Coastguard Worker                               PIPE_VIDEO_PROFILE_UNKNOWN,
132*61046927SAndroid Build Coastguard Worker                               PIPE_VIDEO_ENTRYPOINT_PROCESSING,
133*61046927SAndroid Build Coastguard Worker                               PIPE_VIDEO_CAP_SUPPORTED)) {
134*61046927SAndroid Build Coastguard Worker          context->needs_begin_frame = true;
135*61046927SAndroid Build Coastguard Worker       }
136*61046927SAndroid Build Coastguard Worker 
137*61046927SAndroid Build Coastguard Worker       mtx_unlock(&drv->mutex);
138*61046927SAndroid Build Coastguard Worker       return VA_STATUS_SUCCESS;
139*61046927SAndroid Build Coastguard Worker    }
140*61046927SAndroid Build Coastguard Worker 
141*61046927SAndroid Build Coastguard Worker    if (context->decoder->entrypoint != PIPE_VIDEO_ENTRYPOINT_ENCODE)
142*61046927SAndroid Build Coastguard Worker       context->needs_begin_frame = true;
143*61046927SAndroid Build Coastguard Worker 
144*61046927SAndroid Build Coastguard Worker    /* meta data and seis are per picture basis, it needs to be
145*61046927SAndroid Build Coastguard Worker     * cleared before rendering the picture. */
146*61046927SAndroid Build Coastguard Worker    if (context->decoder->entrypoint == PIPE_VIDEO_ENTRYPOINT_ENCODE) {
147*61046927SAndroid Build Coastguard Worker       switch (u_reduce_video_profile(context->templat.profile)) {
148*61046927SAndroid Build Coastguard Worker          case PIPE_VIDEO_FORMAT_AV1:
149*61046927SAndroid Build Coastguard Worker             context->desc.av1enc.metadata_flags.value = 0;
150*61046927SAndroid Build Coastguard Worker             context->desc.av1enc.roi.num = 0;
151*61046927SAndroid Build Coastguard Worker             context->desc.av1enc.intra_refresh.mode = INTRA_REFRESH_MODE_NONE;
152*61046927SAndroid Build Coastguard Worker             break;
153*61046927SAndroid Build Coastguard Worker          case PIPE_VIDEO_FORMAT_HEVC:
154*61046927SAndroid Build Coastguard Worker             context->desc.h265enc.roi.num = 0;
155*61046927SAndroid Build Coastguard Worker             context->desc.h265enc.intra_refresh.mode = INTRA_REFRESH_MODE_NONE;
156*61046927SAndroid Build Coastguard Worker             break;
157*61046927SAndroid Build Coastguard Worker          case PIPE_VIDEO_FORMAT_MPEG4_AVC:
158*61046927SAndroid Build Coastguard Worker             context->desc.h264enc.roi.num = 0;
159*61046927SAndroid Build Coastguard Worker             context->desc.h264enc.intra_refresh.mode = INTRA_REFRESH_MODE_NONE;
160*61046927SAndroid Build Coastguard Worker             break;
161*61046927SAndroid Build Coastguard Worker          default:
162*61046927SAndroid Build Coastguard Worker             break;
163*61046927SAndroid Build Coastguard Worker       }
164*61046927SAndroid Build Coastguard Worker    }
165*61046927SAndroid Build Coastguard Worker 
166*61046927SAndroid Build Coastguard Worker    context->slice_data_offset = 0;
167*61046927SAndroid Build Coastguard Worker    context->have_slice_params = false;
168*61046927SAndroid Build Coastguard Worker 
169*61046927SAndroid Build Coastguard Worker    mtx_unlock(&drv->mutex);
170*61046927SAndroid Build Coastguard Worker    return VA_STATUS_SUCCESS;
171*61046927SAndroid Build Coastguard Worker }
172*61046927SAndroid Build Coastguard Worker 
173*61046927SAndroid Build Coastguard Worker void
vlVaGetReferenceFrame(vlVaDriver * drv,VASurfaceID surface_id,struct pipe_video_buffer ** ref_frame)174*61046927SAndroid Build Coastguard Worker vlVaGetReferenceFrame(vlVaDriver *drv, VASurfaceID surface_id,
175*61046927SAndroid Build Coastguard Worker                       struct pipe_video_buffer **ref_frame)
176*61046927SAndroid Build Coastguard Worker {
177*61046927SAndroid Build Coastguard Worker    vlVaSurface *surf = handle_table_get(drv->htab, surface_id);
178*61046927SAndroid Build Coastguard Worker    if (surf)
179*61046927SAndroid Build Coastguard Worker       *ref_frame = vlVaGetSurfaceBuffer(drv, surf);
180*61046927SAndroid Build Coastguard Worker    else
181*61046927SAndroid Build Coastguard Worker       *ref_frame = NULL;
182*61046927SAndroid Build Coastguard Worker }
183*61046927SAndroid Build Coastguard Worker /*
184*61046927SAndroid Build Coastguard Worker  * in->quality = 0; without any settings, it is using speed preset
185*61046927SAndroid Build Coastguard Worker  *                  and no preencode and no vbaq. It is the fastest setting.
186*61046927SAndroid Build Coastguard Worker  * in->quality = 1; suggested setting, with balanced preset, and
187*61046927SAndroid Build Coastguard Worker  *                  preencode and vbaq
188*61046927SAndroid Build Coastguard Worker  * in->quality = others; it is the customized setting
189*61046927SAndroid Build Coastguard Worker  *                  with valid bit (bit #0) set to "1"
190*61046927SAndroid Build Coastguard Worker  *                  for example:
191*61046927SAndroid Build Coastguard Worker  *
192*61046927SAndroid Build Coastguard Worker  *                  0x3  (balance preset, no pre-encoding, no vbaq)
193*61046927SAndroid Build Coastguard Worker  *                  0x13 (balanced preset, no pre-encoding, vbaq)
194*61046927SAndroid Build Coastguard Worker  *                  0x13 (balanced preset, no pre-encoding, vbaq)
195*61046927SAndroid Build Coastguard Worker  *                  0x9  (speed preset, pre-encoding, no vbaq)
196*61046927SAndroid Build Coastguard Worker  *                  0x19 (speed preset, pre-encoding, vbaq)
197*61046927SAndroid Build Coastguard Worker  *
198*61046927SAndroid Build Coastguard Worker  *                  The quality value has to be treated as a combination
199*61046927SAndroid Build Coastguard Worker  *                  of preset mode, pre-encoding and vbaq settings.
200*61046927SAndroid Build Coastguard Worker  *                  The quality and speed could be vary according to
201*61046927SAndroid Build Coastguard Worker  *                  different settings,
202*61046927SAndroid Build Coastguard Worker  */
203*61046927SAndroid Build Coastguard Worker void
vlVaHandleVAEncMiscParameterTypeQualityLevel(struct pipe_enc_quality_modes * p,vlVaQualityBits * in)204*61046927SAndroid Build Coastguard Worker vlVaHandleVAEncMiscParameterTypeQualityLevel(struct pipe_enc_quality_modes *p, vlVaQualityBits *in)
205*61046927SAndroid Build Coastguard Worker {
206*61046927SAndroid Build Coastguard Worker    if (!in->quality) {
207*61046927SAndroid Build Coastguard Worker       p->level = 0;
208*61046927SAndroid Build Coastguard Worker       p->preset_mode = PRESET_MODE_SPEED;
209*61046927SAndroid Build Coastguard Worker       p->pre_encode_mode = PREENCODING_MODE_DISABLE;
210*61046927SAndroid Build Coastguard Worker       p->vbaq_mode = VBAQ_DISABLE;
211*61046927SAndroid Build Coastguard Worker 
212*61046927SAndroid Build Coastguard Worker       return;
213*61046927SAndroid Build Coastguard Worker    }
214*61046927SAndroid Build Coastguard Worker 
215*61046927SAndroid Build Coastguard Worker    if (p->level != in->quality) {
216*61046927SAndroid Build Coastguard Worker       if (in->quality == 1) {
217*61046927SAndroid Build Coastguard Worker          p->preset_mode = PRESET_MODE_BALANCE;
218*61046927SAndroid Build Coastguard Worker          p->pre_encode_mode = PREENCODING_MODE_DEFAULT;
219*61046927SAndroid Build Coastguard Worker          p->vbaq_mode = VBAQ_AUTO;
220*61046927SAndroid Build Coastguard Worker       } else {
221*61046927SAndroid Build Coastguard Worker          p->preset_mode = in->preset_mode > PRESET_MODE_HIGH_QUALITY
222*61046927SAndroid Build Coastguard Worker             ? PRESET_MODE_HIGH_QUALITY : in->preset_mode;
223*61046927SAndroid Build Coastguard Worker          p->pre_encode_mode = in->pre_encode_mode;
224*61046927SAndroid Build Coastguard Worker          p->vbaq_mode = in->vbaq_mode;
225*61046927SAndroid Build Coastguard Worker       }
226*61046927SAndroid Build Coastguard Worker    }
227*61046927SAndroid Build Coastguard Worker    p->level = in->quality;
228*61046927SAndroid Build Coastguard Worker }
229*61046927SAndroid Build Coastguard Worker 
230*61046927SAndroid Build Coastguard Worker static VAStatus
handlePictureParameterBuffer(vlVaDriver * drv,vlVaContext * context,vlVaBuffer * buf)231*61046927SAndroid Build Coastguard Worker handlePictureParameterBuffer(vlVaDriver *drv, vlVaContext *context, vlVaBuffer *buf)
232*61046927SAndroid Build Coastguard Worker {
233*61046927SAndroid Build Coastguard Worker    VAStatus vaStatus = VA_STATUS_SUCCESS;
234*61046927SAndroid Build Coastguard Worker    enum pipe_video_format format =
235*61046927SAndroid Build Coastguard Worker       u_reduce_video_profile(context->templat.profile);
236*61046927SAndroid Build Coastguard Worker 
237*61046927SAndroid Build Coastguard Worker    switch (format) {
238*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_MPEG12:
239*61046927SAndroid Build Coastguard Worker       vlVaHandlePictureParameterBufferMPEG12(drv, context, buf);
240*61046927SAndroid Build Coastguard Worker       break;
241*61046927SAndroid Build Coastguard Worker 
242*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_MPEG4_AVC:
243*61046927SAndroid Build Coastguard Worker       vlVaHandlePictureParameterBufferH264(drv, context, buf);
244*61046927SAndroid Build Coastguard Worker       break;
245*61046927SAndroid Build Coastguard Worker 
246*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_VC1:
247*61046927SAndroid Build Coastguard Worker       vlVaHandlePictureParameterBufferVC1(drv, context, buf);
248*61046927SAndroid Build Coastguard Worker       break;
249*61046927SAndroid Build Coastguard Worker 
250*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_MPEG4:
251*61046927SAndroid Build Coastguard Worker       vlVaHandlePictureParameterBufferMPEG4(drv, context, buf);
252*61046927SAndroid Build Coastguard Worker       break;
253*61046927SAndroid Build Coastguard Worker 
254*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_HEVC:
255*61046927SAndroid Build Coastguard Worker       vlVaHandlePictureParameterBufferHEVC(drv, context, buf);
256*61046927SAndroid Build Coastguard Worker       break;
257*61046927SAndroid Build Coastguard Worker 
258*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_JPEG:
259*61046927SAndroid Build Coastguard Worker       vlVaHandlePictureParameterBufferMJPEG(drv, context, buf);
260*61046927SAndroid Build Coastguard Worker       break;
261*61046927SAndroid Build Coastguard Worker 
262*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_VP9:
263*61046927SAndroid Build Coastguard Worker       vlVaHandlePictureParameterBufferVP9(drv, context, buf);
264*61046927SAndroid Build Coastguard Worker       break;
265*61046927SAndroid Build Coastguard Worker 
266*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_AV1:
267*61046927SAndroid Build Coastguard Worker       vlVaHandlePictureParameterBufferAV1(drv, context, buf);
268*61046927SAndroid Build Coastguard Worker       break;
269*61046927SAndroid Build Coastguard Worker 
270*61046927SAndroid Build Coastguard Worker    default:
271*61046927SAndroid Build Coastguard Worker       break;
272*61046927SAndroid Build Coastguard Worker    }
273*61046927SAndroid Build Coastguard Worker 
274*61046927SAndroid Build Coastguard Worker    /* Create the decoder once max_references is known. */
275*61046927SAndroid Build Coastguard Worker    if (!context->decoder) {
276*61046927SAndroid Build Coastguard Worker       if (!context->target)
277*61046927SAndroid Build Coastguard Worker          return VA_STATUS_ERROR_INVALID_CONTEXT;
278*61046927SAndroid Build Coastguard Worker 
279*61046927SAndroid Build Coastguard Worker       if (format == PIPE_VIDEO_FORMAT_MPEG4_AVC)
280*61046927SAndroid Build Coastguard Worker          context->templat.level = u_get_h264_level(context->templat.width,
281*61046927SAndroid Build Coastguard Worker             context->templat.height, &context->templat.max_references);
282*61046927SAndroid Build Coastguard Worker 
283*61046927SAndroid Build Coastguard Worker       context->decoder = drv->pipe->create_video_codec(drv->pipe,
284*61046927SAndroid Build Coastguard Worker          &context->templat);
285*61046927SAndroid Build Coastguard Worker 
286*61046927SAndroid Build Coastguard Worker       if (!context->decoder)
287*61046927SAndroid Build Coastguard Worker          return VA_STATUS_ERROR_ALLOCATION_FAILED;
288*61046927SAndroid Build Coastguard Worker 
289*61046927SAndroid Build Coastguard Worker       context->needs_begin_frame = true;
290*61046927SAndroid Build Coastguard Worker    }
291*61046927SAndroid Build Coastguard Worker 
292*61046927SAndroid Build Coastguard Worker    if (format == PIPE_VIDEO_FORMAT_VP9) {
293*61046927SAndroid Build Coastguard Worker       context->decoder->width =
294*61046927SAndroid Build Coastguard Worker          context->desc.vp9.picture_parameter.frame_width;
295*61046927SAndroid Build Coastguard Worker       context->decoder->height =
296*61046927SAndroid Build Coastguard Worker          context->desc.vp9.picture_parameter.frame_height;
297*61046927SAndroid Build Coastguard Worker    }
298*61046927SAndroid Build Coastguard Worker 
299*61046927SAndroid Build Coastguard Worker    return vaStatus;
300*61046927SAndroid Build Coastguard Worker }
301*61046927SAndroid Build Coastguard Worker 
302*61046927SAndroid Build Coastguard Worker static void
handleIQMatrixBuffer(vlVaContext * context,vlVaBuffer * buf)303*61046927SAndroid Build Coastguard Worker handleIQMatrixBuffer(vlVaContext *context, vlVaBuffer *buf)
304*61046927SAndroid Build Coastguard Worker {
305*61046927SAndroid Build Coastguard Worker    switch (u_reduce_video_profile(context->templat.profile)) {
306*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_MPEG12:
307*61046927SAndroid Build Coastguard Worker       vlVaHandleIQMatrixBufferMPEG12(context, buf);
308*61046927SAndroid Build Coastguard Worker       break;
309*61046927SAndroid Build Coastguard Worker 
310*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_MPEG4_AVC:
311*61046927SAndroid Build Coastguard Worker       vlVaHandleIQMatrixBufferH264(context, buf);
312*61046927SAndroid Build Coastguard Worker       break;
313*61046927SAndroid Build Coastguard Worker 
314*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_MPEG4:
315*61046927SAndroid Build Coastguard Worker       vlVaHandleIQMatrixBufferMPEG4(context, buf);
316*61046927SAndroid Build Coastguard Worker       break;
317*61046927SAndroid Build Coastguard Worker 
318*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_HEVC:
319*61046927SAndroid Build Coastguard Worker       vlVaHandleIQMatrixBufferHEVC(context, buf);
320*61046927SAndroid Build Coastguard Worker       break;
321*61046927SAndroid Build Coastguard Worker 
322*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_JPEG:
323*61046927SAndroid Build Coastguard Worker       vlVaHandleIQMatrixBufferMJPEG(context, buf);
324*61046927SAndroid Build Coastguard Worker       break;
325*61046927SAndroid Build Coastguard Worker 
326*61046927SAndroid Build Coastguard Worker    default:
327*61046927SAndroid Build Coastguard Worker       break;
328*61046927SAndroid Build Coastguard Worker    }
329*61046927SAndroid Build Coastguard Worker }
330*61046927SAndroid Build Coastguard Worker 
331*61046927SAndroid Build Coastguard Worker static void
handleSliceParameterBuffer(vlVaContext * context,vlVaBuffer * buf)332*61046927SAndroid Build Coastguard Worker handleSliceParameterBuffer(vlVaContext *context, vlVaBuffer *buf)
333*61046927SAndroid Build Coastguard Worker {
334*61046927SAndroid Build Coastguard Worker    switch (u_reduce_video_profile(context->templat.profile)) {
335*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_MPEG12:
336*61046927SAndroid Build Coastguard Worker       vlVaHandleSliceParameterBufferMPEG12(context, buf);
337*61046927SAndroid Build Coastguard Worker       break;
338*61046927SAndroid Build Coastguard Worker 
339*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_VC1:
340*61046927SAndroid Build Coastguard Worker       vlVaHandleSliceParameterBufferVC1(context, buf);
341*61046927SAndroid Build Coastguard Worker       break;
342*61046927SAndroid Build Coastguard Worker 
343*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_MPEG4_AVC:
344*61046927SAndroid Build Coastguard Worker       vlVaHandleSliceParameterBufferH264(context, buf);
345*61046927SAndroid Build Coastguard Worker       break;
346*61046927SAndroid Build Coastguard Worker 
347*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_MPEG4:
348*61046927SAndroid Build Coastguard Worker       vlVaHandleSliceParameterBufferMPEG4(context, buf);
349*61046927SAndroid Build Coastguard Worker       break;
350*61046927SAndroid Build Coastguard Worker 
351*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_HEVC:
352*61046927SAndroid Build Coastguard Worker       vlVaHandleSliceParameterBufferHEVC(context, buf);
353*61046927SAndroid Build Coastguard Worker       break;
354*61046927SAndroid Build Coastguard Worker 
355*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_JPEG:
356*61046927SAndroid Build Coastguard Worker       vlVaHandleSliceParameterBufferMJPEG(context, buf);
357*61046927SAndroid Build Coastguard Worker       break;
358*61046927SAndroid Build Coastguard Worker 
359*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_VP9:
360*61046927SAndroid Build Coastguard Worker       vlVaHandleSliceParameterBufferVP9(context, buf);
361*61046927SAndroid Build Coastguard Worker       break;
362*61046927SAndroid Build Coastguard Worker 
363*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_AV1:
364*61046927SAndroid Build Coastguard Worker       vlVaHandleSliceParameterBufferAV1(context, buf);
365*61046927SAndroid Build Coastguard Worker       break;
366*61046927SAndroid Build Coastguard Worker 
367*61046927SAndroid Build Coastguard Worker    default:
368*61046927SAndroid Build Coastguard Worker       break;
369*61046927SAndroid Build Coastguard Worker    }
370*61046927SAndroid Build Coastguard Worker }
371*61046927SAndroid Build Coastguard Worker 
372*61046927SAndroid Build Coastguard Worker static unsigned int
bufHasStartcode(vlVaBuffer * buf,unsigned int code,unsigned int bits)373*61046927SAndroid Build Coastguard Worker bufHasStartcode(vlVaBuffer *buf, unsigned int code, unsigned int bits)
374*61046927SAndroid Build Coastguard Worker {
375*61046927SAndroid Build Coastguard Worker    struct vl_vlc vlc = {0};
376*61046927SAndroid Build Coastguard Worker    int i;
377*61046927SAndroid Build Coastguard Worker 
378*61046927SAndroid Build Coastguard Worker    /* search the first 64 bytes for a startcode */
379*61046927SAndroid Build Coastguard Worker    vl_vlc_init(&vlc, 1, (const void * const*)&buf->data, &buf->size);
380*61046927SAndroid Build Coastguard Worker    for (i = 0; i < 64 && vl_vlc_bits_left(&vlc) >= bits; ++i) {
381*61046927SAndroid Build Coastguard Worker       if (vl_vlc_peekbits(&vlc, bits) == code)
382*61046927SAndroid Build Coastguard Worker          return 1;
383*61046927SAndroid Build Coastguard Worker       vl_vlc_eatbits(&vlc, 8);
384*61046927SAndroid Build Coastguard Worker       vl_vlc_fillbits(&vlc);
385*61046927SAndroid Build Coastguard Worker    }
386*61046927SAndroid Build Coastguard Worker 
387*61046927SAndroid Build Coastguard Worker    return 0;
388*61046927SAndroid Build Coastguard Worker }
389*61046927SAndroid Build Coastguard Worker 
390*61046927SAndroid Build Coastguard Worker static void
handleVAProtectedSliceDataBufferType(vlVaContext * context,vlVaBuffer * buf)391*61046927SAndroid Build Coastguard Worker handleVAProtectedSliceDataBufferType(vlVaContext *context, vlVaBuffer *buf)
392*61046927SAndroid Build Coastguard Worker {
393*61046927SAndroid Build Coastguard Worker 	uint8_t* encrypted_data = (uint8_t*) buf->data;
394*61046927SAndroid Build Coastguard Worker         uint8_t* drm_key;
395*61046927SAndroid Build Coastguard Worker 
396*61046927SAndroid Build Coastguard Worker 	unsigned int drm_key_size = buf->size;
397*61046927SAndroid Build Coastguard Worker 
398*61046927SAndroid Build Coastguard Worker         drm_key = REALLOC(context->desc.base.decrypt_key,
399*61046927SAndroid Build Coastguard Worker                           context->desc.base.key_size, drm_key_size);
400*61046927SAndroid Build Coastguard Worker         if (!drm_key)
401*61046927SAndroid Build Coastguard Worker             return;
402*61046927SAndroid Build Coastguard Worker         context->desc.base.decrypt_key = drm_key;
403*61046927SAndroid Build Coastguard Worker 	memcpy(context->desc.base.decrypt_key, encrypted_data, drm_key_size);
404*61046927SAndroid Build Coastguard Worker 	context->desc.base.key_size = drm_key_size;
405*61046927SAndroid Build Coastguard Worker 	context->desc.base.protected_playback = true;
406*61046927SAndroid Build Coastguard Worker }
407*61046927SAndroid Build Coastguard Worker 
408*61046927SAndroid Build Coastguard Worker static VAStatus
handleVASliceDataBufferType(vlVaContext * context,vlVaBuffer * buf)409*61046927SAndroid Build Coastguard Worker handleVASliceDataBufferType(vlVaContext *context, vlVaBuffer *buf)
410*61046927SAndroid Build Coastguard Worker {
411*61046927SAndroid Build Coastguard Worker    enum pipe_video_format format = u_reduce_video_profile(context->templat.profile);
412*61046927SAndroid Build Coastguard Worker    static const uint8_t start_code_h264[] = { 0x00, 0x00, 0x01 };
413*61046927SAndroid Build Coastguard Worker    static const uint8_t start_code_h265[] = { 0x00, 0x00, 0x01 };
414*61046927SAndroid Build Coastguard Worker    static const uint8_t start_code_vc1[] = { 0x00, 0x00, 0x01, 0x0d };
415*61046927SAndroid Build Coastguard Worker    static const uint8_t eoi_jpeg[] = { 0xff, 0xd9 };
416*61046927SAndroid Build Coastguard Worker 
417*61046927SAndroid Build Coastguard Worker    if (!context->decoder)
418*61046927SAndroid Build Coastguard Worker       return VA_STATUS_ERROR_INVALID_CONTEXT;
419*61046927SAndroid Build Coastguard Worker 
420*61046927SAndroid Build Coastguard Worker    if (context->bs.allocated_size - context->bs.num_buffers < 3) {
421*61046927SAndroid Build Coastguard Worker       context->bs.buffers = REALLOC(context->bs.buffers,
422*61046927SAndroid Build Coastguard Worker                                     context->bs.allocated_size * sizeof(*context->bs.buffers),
423*61046927SAndroid Build Coastguard Worker                                     (context->bs.allocated_size + 3) * sizeof(*context->bs.buffers));
424*61046927SAndroid Build Coastguard Worker       context->bs.sizes = REALLOC(context->bs.sizes,
425*61046927SAndroid Build Coastguard Worker                                   context->bs.allocated_size * sizeof(*context->bs.sizes),
426*61046927SAndroid Build Coastguard Worker                                   (context->bs.allocated_size + 3) * sizeof(*context->bs.sizes));
427*61046927SAndroid Build Coastguard Worker       context->bs.allocated_size += 3;
428*61046927SAndroid Build Coastguard Worker    }
429*61046927SAndroid Build Coastguard Worker 
430*61046927SAndroid Build Coastguard Worker    format = u_reduce_video_profile(context->templat.profile);
431*61046927SAndroid Build Coastguard Worker    if (!context->desc.base.protected_playback) {
432*61046927SAndroid Build Coastguard Worker       switch (format) {
433*61046927SAndroid Build Coastguard Worker       case PIPE_VIDEO_FORMAT_MPEG4_AVC:
434*61046927SAndroid Build Coastguard Worker          if (bufHasStartcode(buf, 0x000001, 24))
435*61046927SAndroid Build Coastguard Worker             break;
436*61046927SAndroid Build Coastguard Worker 
437*61046927SAndroid Build Coastguard Worker          context->bs.buffers[context->bs.num_buffers] = (void *const)&start_code_h264;
438*61046927SAndroid Build Coastguard Worker          context->bs.sizes[context->bs.num_buffers++] = sizeof(start_code_h264);
439*61046927SAndroid Build Coastguard Worker          break;
440*61046927SAndroid Build Coastguard Worker       case PIPE_VIDEO_FORMAT_HEVC:
441*61046927SAndroid Build Coastguard Worker          if (bufHasStartcode(buf, 0x000001, 24))
442*61046927SAndroid Build Coastguard Worker             break;
443*61046927SAndroid Build Coastguard Worker 
444*61046927SAndroid Build Coastguard Worker          context->bs.buffers[context->bs.num_buffers] = (void *const)&start_code_h265;
445*61046927SAndroid Build Coastguard Worker          context->bs.sizes[context->bs.num_buffers++] = sizeof(start_code_h265);
446*61046927SAndroid Build Coastguard Worker          break;
447*61046927SAndroid Build Coastguard Worker       case PIPE_VIDEO_FORMAT_VC1:
448*61046927SAndroid Build Coastguard Worker          if (bufHasStartcode(buf, 0x0000010d, 32) ||
449*61046927SAndroid Build Coastguard Worker              bufHasStartcode(buf, 0x0000010c, 32) ||
450*61046927SAndroid Build Coastguard Worker              bufHasStartcode(buf, 0x0000010b, 32))
451*61046927SAndroid Build Coastguard Worker             break;
452*61046927SAndroid Build Coastguard Worker 
453*61046927SAndroid Build Coastguard Worker          if (context->decoder->profile == PIPE_VIDEO_PROFILE_VC1_ADVANCED) {
454*61046927SAndroid Build Coastguard Worker             context->bs.buffers[context->bs.num_buffers] = (void *const)&start_code_vc1;
455*61046927SAndroid Build Coastguard Worker             context->bs.sizes[context->bs.num_buffers++] = sizeof(start_code_vc1);
456*61046927SAndroid Build Coastguard Worker          }
457*61046927SAndroid Build Coastguard Worker          break;
458*61046927SAndroid Build Coastguard Worker       case PIPE_VIDEO_FORMAT_MPEG4:
459*61046927SAndroid Build Coastguard Worker          if (bufHasStartcode(buf, 0x000001, 24))
460*61046927SAndroid Build Coastguard Worker             break;
461*61046927SAndroid Build Coastguard Worker 
462*61046927SAndroid Build Coastguard Worker          vlVaDecoderFixMPEG4Startcode(context);
463*61046927SAndroid Build Coastguard Worker          context->bs.buffers[context->bs.num_buffers] = (void *)context->mpeg4.start_code;
464*61046927SAndroid Build Coastguard Worker          context->bs.sizes[context->bs.num_buffers++] = context->mpeg4.start_code_size;
465*61046927SAndroid Build Coastguard Worker          break;
466*61046927SAndroid Build Coastguard Worker       case PIPE_VIDEO_FORMAT_JPEG:
467*61046927SAndroid Build Coastguard Worker          if (bufHasStartcode(buf, 0xffd8ffdb, 32))
468*61046927SAndroid Build Coastguard Worker             break;
469*61046927SAndroid Build Coastguard Worker 
470*61046927SAndroid Build Coastguard Worker          vlVaGetJpegSliceHeader(context);
471*61046927SAndroid Build Coastguard Worker          context->bs.buffers[context->bs.num_buffers] = (void *)context->mjpeg.slice_header;
472*61046927SAndroid Build Coastguard Worker          context->bs.sizes[context->bs.num_buffers++] = context->mjpeg.slice_header_size;
473*61046927SAndroid Build Coastguard Worker          break;
474*61046927SAndroid Build Coastguard Worker       case PIPE_VIDEO_FORMAT_VP9:
475*61046927SAndroid Build Coastguard Worker          if (false == context->desc.base.protected_playback)
476*61046927SAndroid Build Coastguard Worker             vlVaDecoderVP9BitstreamHeader(context, buf);
477*61046927SAndroid Build Coastguard Worker          break;
478*61046927SAndroid Build Coastguard Worker       case PIPE_VIDEO_FORMAT_AV1:
479*61046927SAndroid Build Coastguard Worker          break;
480*61046927SAndroid Build Coastguard Worker       default:
481*61046927SAndroid Build Coastguard Worker          break;
482*61046927SAndroid Build Coastguard Worker       }
483*61046927SAndroid Build Coastguard Worker    }
484*61046927SAndroid Build Coastguard Worker 
485*61046927SAndroid Build Coastguard Worker    context->bs.buffers[context->bs.num_buffers] = buf->data;
486*61046927SAndroid Build Coastguard Worker    context->bs.sizes[context->bs.num_buffers++] = buf->size;
487*61046927SAndroid Build Coastguard Worker 
488*61046927SAndroid Build Coastguard Worker    if (format == PIPE_VIDEO_FORMAT_JPEG) {
489*61046927SAndroid Build Coastguard Worker       context->bs.buffers[context->bs.num_buffers] = (void *const)&eoi_jpeg;
490*61046927SAndroid Build Coastguard Worker       context->bs.sizes[context->bs.num_buffers++] = sizeof(eoi_jpeg);
491*61046927SAndroid Build Coastguard Worker    }
492*61046927SAndroid Build Coastguard Worker 
493*61046927SAndroid Build Coastguard Worker    if (context->needs_begin_frame) {
494*61046927SAndroid Build Coastguard Worker       context->decoder->begin_frame(context->decoder, context->target,
495*61046927SAndroid Build Coastguard Worker          &context->desc.base);
496*61046927SAndroid Build Coastguard Worker       context->needs_begin_frame = false;
497*61046927SAndroid Build Coastguard Worker    }
498*61046927SAndroid Build Coastguard Worker    return VA_STATUS_SUCCESS;
499*61046927SAndroid Build Coastguard Worker }
500*61046927SAndroid Build Coastguard Worker 
501*61046927SAndroid Build Coastguard Worker static VAStatus
handleVAEncMiscParameterTypeRateControl(vlVaContext * context,VAEncMiscParameterBuffer * misc)502*61046927SAndroid Build Coastguard Worker handleVAEncMiscParameterTypeRateControl(vlVaContext *context, VAEncMiscParameterBuffer *misc)
503*61046927SAndroid Build Coastguard Worker {
504*61046927SAndroid Build Coastguard Worker    VAStatus status = VA_STATUS_SUCCESS;
505*61046927SAndroid Build Coastguard Worker 
506*61046927SAndroid Build Coastguard Worker    switch (u_reduce_video_profile(context->templat.profile)) {
507*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_MPEG4_AVC:
508*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncMiscParameterTypeRateControlH264(context, misc);
509*61046927SAndroid Build Coastguard Worker       break;
510*61046927SAndroid Build Coastguard Worker 
511*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_HEVC:
512*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncMiscParameterTypeRateControlHEVC(context, misc);
513*61046927SAndroid Build Coastguard Worker       break;
514*61046927SAndroid Build Coastguard Worker 
515*61046927SAndroid Build Coastguard Worker #if VA_CHECK_VERSION(1, 16, 0)
516*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_AV1:
517*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncMiscParameterTypeRateControlAV1(context, misc);
518*61046927SAndroid Build Coastguard Worker       break;
519*61046927SAndroid Build Coastguard Worker #endif
520*61046927SAndroid Build Coastguard Worker    default:
521*61046927SAndroid Build Coastguard Worker       break;
522*61046927SAndroid Build Coastguard Worker    }
523*61046927SAndroid Build Coastguard Worker 
524*61046927SAndroid Build Coastguard Worker    return status;
525*61046927SAndroid Build Coastguard Worker }
526*61046927SAndroid Build Coastguard Worker 
527*61046927SAndroid Build Coastguard Worker static VAStatus
handleVAEncMiscParameterTypeFrameRate(vlVaContext * context,VAEncMiscParameterBuffer * misc)528*61046927SAndroid Build Coastguard Worker handleVAEncMiscParameterTypeFrameRate(vlVaContext *context, VAEncMiscParameterBuffer *misc)
529*61046927SAndroid Build Coastguard Worker {
530*61046927SAndroid Build Coastguard Worker    VAStatus status = VA_STATUS_SUCCESS;
531*61046927SAndroid Build Coastguard Worker 
532*61046927SAndroid Build Coastguard Worker    switch (u_reduce_video_profile(context->templat.profile)) {
533*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_MPEG4_AVC:
534*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncMiscParameterTypeFrameRateH264(context, misc);
535*61046927SAndroid Build Coastguard Worker       break;
536*61046927SAndroid Build Coastguard Worker 
537*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_HEVC:
538*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncMiscParameterTypeFrameRateHEVC(context, misc);
539*61046927SAndroid Build Coastguard Worker       break;
540*61046927SAndroid Build Coastguard Worker 
541*61046927SAndroid Build Coastguard Worker #if VA_CHECK_VERSION(1, 16, 0)
542*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_AV1:
543*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncMiscParameterTypeFrameRateAV1(context, misc);
544*61046927SAndroid Build Coastguard Worker       break;
545*61046927SAndroid Build Coastguard Worker #endif
546*61046927SAndroid Build Coastguard Worker    default:
547*61046927SAndroid Build Coastguard Worker       break;
548*61046927SAndroid Build Coastguard Worker    }
549*61046927SAndroid Build Coastguard Worker 
550*61046927SAndroid Build Coastguard Worker    return status;
551*61046927SAndroid Build Coastguard Worker }
552*61046927SAndroid Build Coastguard Worker 
553*61046927SAndroid Build Coastguard Worker static VAStatus
handleVAEncMiscParameterTypeTemporalLayer(vlVaContext * context,VAEncMiscParameterBuffer * misc)554*61046927SAndroid Build Coastguard Worker handleVAEncMiscParameterTypeTemporalLayer(vlVaContext *context, VAEncMiscParameterBuffer *misc)
555*61046927SAndroid Build Coastguard Worker {
556*61046927SAndroid Build Coastguard Worker    VAStatus status = VA_STATUS_SUCCESS;
557*61046927SAndroid Build Coastguard Worker 
558*61046927SAndroid Build Coastguard Worker    switch (u_reduce_video_profile(context->templat.profile)) {
559*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_MPEG4_AVC:
560*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncMiscParameterTypeTemporalLayerH264(context, misc);
561*61046927SAndroid Build Coastguard Worker       break;
562*61046927SAndroid Build Coastguard Worker 
563*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_HEVC:
564*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncMiscParameterTypeTemporalLayerHEVC(context, misc);
565*61046927SAndroid Build Coastguard Worker       break;
566*61046927SAndroid Build Coastguard Worker 
567*61046927SAndroid Build Coastguard Worker    default:
568*61046927SAndroid Build Coastguard Worker       break;
569*61046927SAndroid Build Coastguard Worker    }
570*61046927SAndroid Build Coastguard Worker 
571*61046927SAndroid Build Coastguard Worker    return status;
572*61046927SAndroid Build Coastguard Worker }
573*61046927SAndroid Build Coastguard Worker 
574*61046927SAndroid Build Coastguard Worker static VAStatus
handleVAEncSequenceParameterBufferType(vlVaDriver * drv,vlVaContext * context,vlVaBuffer * buf)575*61046927SAndroid Build Coastguard Worker handleVAEncSequenceParameterBufferType(vlVaDriver *drv, vlVaContext *context, vlVaBuffer *buf)
576*61046927SAndroid Build Coastguard Worker {
577*61046927SAndroid Build Coastguard Worker    VAStatus status = VA_STATUS_SUCCESS;
578*61046927SAndroid Build Coastguard Worker 
579*61046927SAndroid Build Coastguard Worker    switch (u_reduce_video_profile(context->templat.profile)) {
580*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_MPEG4_AVC:
581*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncSequenceParameterBufferTypeH264(drv, context, buf);
582*61046927SAndroid Build Coastguard Worker       break;
583*61046927SAndroid Build Coastguard Worker 
584*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_HEVC:
585*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncSequenceParameterBufferTypeHEVC(drv, context, buf);
586*61046927SAndroid Build Coastguard Worker       break;
587*61046927SAndroid Build Coastguard Worker 
588*61046927SAndroid Build Coastguard Worker #if VA_CHECK_VERSION(1, 16, 0)
589*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_AV1:
590*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncSequenceParameterBufferTypeAV1(drv, context, buf);
591*61046927SAndroid Build Coastguard Worker       break;
592*61046927SAndroid Build Coastguard Worker #endif
593*61046927SAndroid Build Coastguard Worker 
594*61046927SAndroid Build Coastguard Worker    default:
595*61046927SAndroid Build Coastguard Worker       break;
596*61046927SAndroid Build Coastguard Worker    }
597*61046927SAndroid Build Coastguard Worker 
598*61046927SAndroid Build Coastguard Worker    return status;
599*61046927SAndroid Build Coastguard Worker }
600*61046927SAndroid Build Coastguard Worker 
601*61046927SAndroid Build Coastguard Worker static VAStatus
handleVAEncMiscParameterTypeQualityLevel(vlVaContext * context,VAEncMiscParameterBuffer * misc)602*61046927SAndroid Build Coastguard Worker handleVAEncMiscParameterTypeQualityLevel(vlVaContext *context, VAEncMiscParameterBuffer *misc)
603*61046927SAndroid Build Coastguard Worker {
604*61046927SAndroid Build Coastguard Worker    VAStatus status = VA_STATUS_SUCCESS;
605*61046927SAndroid Build Coastguard Worker 
606*61046927SAndroid Build Coastguard Worker    switch (u_reduce_video_profile(context->templat.profile)) {
607*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_MPEG4_AVC:
608*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncMiscParameterTypeQualityLevelH264(context, misc);
609*61046927SAndroid Build Coastguard Worker       break;
610*61046927SAndroid Build Coastguard Worker 
611*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_HEVC:
612*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncMiscParameterTypeQualityLevelHEVC(context, misc);
613*61046927SAndroid Build Coastguard Worker       break;
614*61046927SAndroid Build Coastguard Worker 
615*61046927SAndroid Build Coastguard Worker #if VA_CHECK_VERSION(1, 16, 0)
616*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_AV1:
617*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncMiscParameterTypeQualityLevelAV1(context, misc);
618*61046927SAndroid Build Coastguard Worker       break;
619*61046927SAndroid Build Coastguard Worker #endif
620*61046927SAndroid Build Coastguard Worker 
621*61046927SAndroid Build Coastguard Worker    default:
622*61046927SAndroid Build Coastguard Worker       break;
623*61046927SAndroid Build Coastguard Worker    }
624*61046927SAndroid Build Coastguard Worker 
625*61046927SAndroid Build Coastguard Worker    return status;
626*61046927SAndroid Build Coastguard Worker }
627*61046927SAndroid Build Coastguard Worker 
628*61046927SAndroid Build Coastguard Worker static VAStatus
handleVAEncMiscParameterTypeMaxFrameSize(vlVaContext * context,VAEncMiscParameterBuffer * misc)629*61046927SAndroid Build Coastguard Worker handleVAEncMiscParameterTypeMaxFrameSize(vlVaContext *context, VAEncMiscParameterBuffer *misc)
630*61046927SAndroid Build Coastguard Worker {
631*61046927SAndroid Build Coastguard Worker    VAStatus status = VA_STATUS_SUCCESS;
632*61046927SAndroid Build Coastguard Worker 
633*61046927SAndroid Build Coastguard Worker    switch (u_reduce_video_profile(context->templat.profile)) {
634*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_MPEG4_AVC:
635*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncMiscParameterTypeMaxFrameSizeH264(context, misc);
636*61046927SAndroid Build Coastguard Worker       break;
637*61046927SAndroid Build Coastguard Worker 
638*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_HEVC:
639*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncMiscParameterTypeMaxFrameSizeHEVC(context, misc);
640*61046927SAndroid Build Coastguard Worker       break;
641*61046927SAndroid Build Coastguard Worker 
642*61046927SAndroid Build Coastguard Worker #if VA_CHECK_VERSION(1, 16, 0)
643*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_AV1:
644*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncMiscParameterTypeMaxFrameSizeAV1(context, misc);
645*61046927SAndroid Build Coastguard Worker       break;
646*61046927SAndroid Build Coastguard Worker #endif
647*61046927SAndroid Build Coastguard Worker 
648*61046927SAndroid Build Coastguard Worker    default:
649*61046927SAndroid Build Coastguard Worker       break;
650*61046927SAndroid Build Coastguard Worker    }
651*61046927SAndroid Build Coastguard Worker 
652*61046927SAndroid Build Coastguard Worker    return status;
653*61046927SAndroid Build Coastguard Worker }
654*61046927SAndroid Build Coastguard Worker static VAStatus
handleVAEncMiscParameterTypeHRD(vlVaContext * context,VAEncMiscParameterBuffer * misc)655*61046927SAndroid Build Coastguard Worker handleVAEncMiscParameterTypeHRD(vlVaContext *context, VAEncMiscParameterBuffer *misc)
656*61046927SAndroid Build Coastguard Worker {
657*61046927SAndroid Build Coastguard Worker    VAStatus status = VA_STATUS_SUCCESS;
658*61046927SAndroid Build Coastguard Worker 
659*61046927SAndroid Build Coastguard Worker    switch (u_reduce_video_profile(context->templat.profile)) {
660*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_MPEG4_AVC:
661*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncMiscParameterTypeHRDH264(context, misc);
662*61046927SAndroid Build Coastguard Worker       break;
663*61046927SAndroid Build Coastguard Worker 
664*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_HEVC:
665*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncMiscParameterTypeHRDHEVC(context, misc);
666*61046927SAndroid Build Coastguard Worker       break;
667*61046927SAndroid Build Coastguard Worker 
668*61046927SAndroid Build Coastguard Worker #if VA_CHECK_VERSION(1, 16, 0)
669*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_AV1:
670*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncMiscParameterTypeHRDAV1(context, misc);
671*61046927SAndroid Build Coastguard Worker       break;
672*61046927SAndroid Build Coastguard Worker #endif
673*61046927SAndroid Build Coastguard Worker 
674*61046927SAndroid Build Coastguard Worker    default:
675*61046927SAndroid Build Coastguard Worker       break;
676*61046927SAndroid Build Coastguard Worker    }
677*61046927SAndroid Build Coastguard Worker 
678*61046927SAndroid Build Coastguard Worker    return status;
679*61046927SAndroid Build Coastguard Worker }
680*61046927SAndroid Build Coastguard Worker 
681*61046927SAndroid Build Coastguard Worker static VAStatus
handleVAEncMiscParameterTypeMaxSliceSize(vlVaContext * context,VAEncMiscParameterBuffer * misc)682*61046927SAndroid Build Coastguard Worker handleVAEncMiscParameterTypeMaxSliceSize(vlVaContext *context, VAEncMiscParameterBuffer *misc)
683*61046927SAndroid Build Coastguard Worker {
684*61046927SAndroid Build Coastguard Worker    VAStatus status = VA_STATUS_SUCCESS;
685*61046927SAndroid Build Coastguard Worker    VAEncMiscParameterMaxSliceSize *max_slice_size_buffer = (VAEncMiscParameterMaxSliceSize *)misc->data;
686*61046927SAndroid Build Coastguard Worker    switch (u_reduce_video_profile(context->templat.profile)) {
687*61046927SAndroid Build Coastguard Worker       case PIPE_VIDEO_FORMAT_MPEG4_AVC:
688*61046927SAndroid Build Coastguard Worker       {
689*61046927SAndroid Build Coastguard Worker          context->desc.h264enc.slice_mode = PIPE_VIDEO_SLICE_MODE_MAX_SLICE_SIZE;
690*61046927SAndroid Build Coastguard Worker          context->desc.h264enc.max_slice_bytes = max_slice_size_buffer->max_slice_size;
691*61046927SAndroid Build Coastguard Worker       } break;
692*61046927SAndroid Build Coastguard Worker       case PIPE_VIDEO_FORMAT_HEVC:
693*61046927SAndroid Build Coastguard Worker       {
694*61046927SAndroid Build Coastguard Worker          context->desc.h265enc.slice_mode = PIPE_VIDEO_SLICE_MODE_MAX_SLICE_SIZE;
695*61046927SAndroid Build Coastguard Worker          context->desc.h265enc.max_slice_bytes = max_slice_size_buffer->max_slice_size;
696*61046927SAndroid Build Coastguard Worker       } break;
697*61046927SAndroid Build Coastguard Worker       default:
698*61046927SAndroid Build Coastguard Worker          break;
699*61046927SAndroid Build Coastguard Worker    }
700*61046927SAndroid Build Coastguard Worker    return status;
701*61046927SAndroid Build Coastguard Worker }
702*61046927SAndroid Build Coastguard Worker 
703*61046927SAndroid Build Coastguard Worker static VAStatus
handleVAEncMiscParameterTypeRIR(vlVaContext * context,VAEncMiscParameterBuffer * misc)704*61046927SAndroid Build Coastguard Worker handleVAEncMiscParameterTypeRIR(vlVaContext *context, VAEncMiscParameterBuffer *misc)
705*61046927SAndroid Build Coastguard Worker {
706*61046927SAndroid Build Coastguard Worker    VAStatus status = VA_STATUS_SUCCESS;
707*61046927SAndroid Build Coastguard Worker    struct pipe_enc_intra_refresh *p_intra_refresh = NULL;
708*61046927SAndroid Build Coastguard Worker 
709*61046927SAndroid Build Coastguard Worker    switch (u_reduce_video_profile(context->templat.profile)) {
710*61046927SAndroid Build Coastguard Worker       case PIPE_VIDEO_FORMAT_MPEG4_AVC:
711*61046927SAndroid Build Coastguard Worker          p_intra_refresh = &context->desc.h264enc.intra_refresh;
712*61046927SAndroid Build Coastguard Worker          break;
713*61046927SAndroid Build Coastguard Worker       case PIPE_VIDEO_FORMAT_HEVC:
714*61046927SAndroid Build Coastguard Worker          p_intra_refresh = &context->desc.h265enc.intra_refresh;
715*61046927SAndroid Build Coastguard Worker          break;
716*61046927SAndroid Build Coastguard Worker #if VA_CHECK_VERSION(1, 16, 0)
717*61046927SAndroid Build Coastguard Worker       case PIPE_VIDEO_FORMAT_AV1:
718*61046927SAndroid Build Coastguard Worker          p_intra_refresh = &context->desc.av1enc.intra_refresh;
719*61046927SAndroid Build Coastguard Worker          break;
720*61046927SAndroid Build Coastguard Worker #endif
721*61046927SAndroid Build Coastguard Worker       default:
722*61046927SAndroid Build Coastguard Worker          p_intra_refresh = NULL;
723*61046927SAndroid Build Coastguard Worker          break;
724*61046927SAndroid Build Coastguard Worker    };
725*61046927SAndroid Build Coastguard Worker 
726*61046927SAndroid Build Coastguard Worker    if (p_intra_refresh) {
727*61046927SAndroid Build Coastguard Worker       VAEncMiscParameterRIR *ir = (VAEncMiscParameterRIR *)misc->data;
728*61046927SAndroid Build Coastguard Worker 
729*61046927SAndroid Build Coastguard Worker       if (ir->rir_flags.value == VA_ENC_INTRA_REFRESH_ROLLING_ROW)
730*61046927SAndroid Build Coastguard Worker          p_intra_refresh->mode = INTRA_REFRESH_MODE_UNIT_ROWS;
731*61046927SAndroid Build Coastguard Worker       else if (ir->rir_flags.value == VA_ENC_INTRA_REFRESH_ROLLING_COLUMN)
732*61046927SAndroid Build Coastguard Worker          p_intra_refresh->mode = INTRA_REFRESH_MODE_UNIT_COLUMNS;
733*61046927SAndroid Build Coastguard Worker       else if (ir->rir_flags.value) /* if any other values to use the default one*/
734*61046927SAndroid Build Coastguard Worker          p_intra_refresh->mode = INTRA_REFRESH_MODE_UNIT_COLUMNS;
735*61046927SAndroid Build Coastguard Worker       else /* if no mode specified then no intra-refresh */
736*61046927SAndroid Build Coastguard Worker          p_intra_refresh->mode = INTRA_REFRESH_MODE_NONE;
737*61046927SAndroid Build Coastguard Worker 
738*61046927SAndroid Build Coastguard Worker       /* intra refresh should be started with sequence level headers */
739*61046927SAndroid Build Coastguard Worker       p_intra_refresh->need_sequence_header = 0;
740*61046927SAndroid Build Coastguard Worker       if (p_intra_refresh->mode) {
741*61046927SAndroid Build Coastguard Worker          p_intra_refresh->region_size = ir->intra_insert_size;
742*61046927SAndroid Build Coastguard Worker          p_intra_refresh->offset = ir->intra_insertion_location;
743*61046927SAndroid Build Coastguard Worker          if (p_intra_refresh->offset == 0)
744*61046927SAndroid Build Coastguard Worker             p_intra_refresh->need_sequence_header = 1;
745*61046927SAndroid Build Coastguard Worker       }
746*61046927SAndroid Build Coastguard Worker    } else {
747*61046927SAndroid Build Coastguard Worker       p_intra_refresh->mode = INTRA_REFRESH_MODE_NONE;
748*61046927SAndroid Build Coastguard Worker       p_intra_refresh->region_size = 0;
749*61046927SAndroid Build Coastguard Worker       p_intra_refresh->offset = 0;
750*61046927SAndroid Build Coastguard Worker       p_intra_refresh->need_sequence_header = 0;
751*61046927SAndroid Build Coastguard Worker    }
752*61046927SAndroid Build Coastguard Worker 
753*61046927SAndroid Build Coastguard Worker    return status;
754*61046927SAndroid Build Coastguard Worker }
755*61046927SAndroid Build Coastguard Worker 
756*61046927SAndroid Build Coastguard Worker static VAStatus
handleVAEncMiscParameterTypeROI(vlVaContext * context,VAEncMiscParameterBuffer * misc)757*61046927SAndroid Build Coastguard Worker handleVAEncMiscParameterTypeROI(vlVaContext *context, VAEncMiscParameterBuffer *misc)
758*61046927SAndroid Build Coastguard Worker {
759*61046927SAndroid Build Coastguard Worker    VAStatus status = VA_STATUS_SUCCESS;
760*61046927SAndroid Build Coastguard Worker    struct pipe_enc_roi *proi= NULL;
761*61046927SAndroid Build Coastguard Worker    switch (u_reduce_video_profile(context->templat.profile)) {
762*61046927SAndroid Build Coastguard Worker       case PIPE_VIDEO_FORMAT_MPEG4_AVC:
763*61046927SAndroid Build Coastguard Worker          proi = &context->desc.h264enc.roi;
764*61046927SAndroid Build Coastguard Worker          break;
765*61046927SAndroid Build Coastguard Worker       case PIPE_VIDEO_FORMAT_HEVC:
766*61046927SAndroid Build Coastguard Worker          proi = &context->desc.h265enc.roi;
767*61046927SAndroid Build Coastguard Worker          break;
768*61046927SAndroid Build Coastguard Worker #if VA_CHECK_VERSION(1, 16, 0)
769*61046927SAndroid Build Coastguard Worker       case PIPE_VIDEO_FORMAT_AV1:
770*61046927SAndroid Build Coastguard Worker          proi = &context->desc.av1enc.roi;
771*61046927SAndroid Build Coastguard Worker          break;
772*61046927SAndroid Build Coastguard Worker #endif
773*61046927SAndroid Build Coastguard Worker       default:
774*61046927SAndroid Build Coastguard Worker          break;
775*61046927SAndroid Build Coastguard Worker    };
776*61046927SAndroid Build Coastguard Worker 
777*61046927SAndroid Build Coastguard Worker    if (proi) {
778*61046927SAndroid Build Coastguard Worker       VAEncMiscParameterBufferROI *roi = (VAEncMiscParameterBufferROI *)misc->data;
779*61046927SAndroid Build Coastguard Worker       /* do not support priority type, and the maximum region is 32  */
780*61046927SAndroid Build Coastguard Worker       if ((roi->num_roi > 0 && roi->roi_flags.bits.roi_value_is_qp_delta == 0)
781*61046927SAndroid Build Coastguard Worker            || roi->num_roi > PIPE_ENC_ROI_REGION_NUM_MAX)
782*61046927SAndroid Build Coastguard Worker          status = VA_STATUS_ERROR_FLAG_NOT_SUPPORTED;
783*61046927SAndroid Build Coastguard Worker       else {
784*61046927SAndroid Build Coastguard Worker          uint32_t i;
785*61046927SAndroid Build Coastguard Worker          VAEncROI *src = roi->roi;
786*61046927SAndroid Build Coastguard Worker 
787*61046927SAndroid Build Coastguard Worker          proi->num = roi->num_roi;
788*61046927SAndroid Build Coastguard Worker          for (i = 0; i < roi->num_roi; i++) {
789*61046927SAndroid Build Coastguard Worker             proi->region[i].valid = true;
790*61046927SAndroid Build Coastguard Worker             proi->region[i].x = src->roi_rectangle.x;
791*61046927SAndroid Build Coastguard Worker             proi->region[i].y = src->roi_rectangle.y;
792*61046927SAndroid Build Coastguard Worker             proi->region[i].width = src->roi_rectangle.width;
793*61046927SAndroid Build Coastguard Worker             proi->region[i].height = src->roi_rectangle.height;
794*61046927SAndroid Build Coastguard Worker             proi->region[i].qp_value = (int32_t)CLAMP(src->roi_value, roi->min_delta_qp, roi->max_delta_qp);
795*61046927SAndroid Build Coastguard Worker             src++;
796*61046927SAndroid Build Coastguard Worker          }
797*61046927SAndroid Build Coastguard Worker 
798*61046927SAndroid Build Coastguard Worker          for (; i < PIPE_ENC_ROI_REGION_NUM_MAX; i++)
799*61046927SAndroid Build Coastguard Worker             proi->region[i].valid = false;
800*61046927SAndroid Build Coastguard Worker       }
801*61046927SAndroid Build Coastguard Worker    }
802*61046927SAndroid Build Coastguard Worker 
803*61046927SAndroid Build Coastguard Worker    return status;
804*61046927SAndroid Build Coastguard Worker }
805*61046927SAndroid Build Coastguard Worker 
806*61046927SAndroid Build Coastguard Worker static VAStatus
handleVAEncMiscParameterBufferType(vlVaContext * context,vlVaBuffer * buf)807*61046927SAndroid Build Coastguard Worker handleVAEncMiscParameterBufferType(vlVaContext *context, vlVaBuffer *buf)
808*61046927SAndroid Build Coastguard Worker {
809*61046927SAndroid Build Coastguard Worker    VAStatus vaStatus = VA_STATUS_SUCCESS;
810*61046927SAndroid Build Coastguard Worker    VAEncMiscParameterBuffer *misc;
811*61046927SAndroid Build Coastguard Worker    misc = buf->data;
812*61046927SAndroid Build Coastguard Worker 
813*61046927SAndroid Build Coastguard Worker    switch (misc->type) {
814*61046927SAndroid Build Coastguard Worker    case VAEncMiscParameterTypeRateControl:
815*61046927SAndroid Build Coastguard Worker       vaStatus = handleVAEncMiscParameterTypeRateControl(context, misc);
816*61046927SAndroid Build Coastguard Worker       break;
817*61046927SAndroid Build Coastguard Worker 
818*61046927SAndroid Build Coastguard Worker    case VAEncMiscParameterTypeFrameRate:
819*61046927SAndroid Build Coastguard Worker       vaStatus = handleVAEncMiscParameterTypeFrameRate(context, misc);
820*61046927SAndroid Build Coastguard Worker       break;
821*61046927SAndroid Build Coastguard Worker 
822*61046927SAndroid Build Coastguard Worker    case VAEncMiscParameterTypeTemporalLayerStructure:
823*61046927SAndroid Build Coastguard Worker       vaStatus = handleVAEncMiscParameterTypeTemporalLayer(context, misc);
824*61046927SAndroid Build Coastguard Worker       break;
825*61046927SAndroid Build Coastguard Worker 
826*61046927SAndroid Build Coastguard Worker    case VAEncMiscParameterTypeQualityLevel:
827*61046927SAndroid Build Coastguard Worker       vaStatus = handleVAEncMiscParameterTypeQualityLevel(context, misc);
828*61046927SAndroid Build Coastguard Worker       break;
829*61046927SAndroid Build Coastguard Worker 
830*61046927SAndroid Build Coastguard Worker    case VAEncMiscParameterTypeMaxFrameSize:
831*61046927SAndroid Build Coastguard Worker       vaStatus = handleVAEncMiscParameterTypeMaxFrameSize(context, misc);
832*61046927SAndroid Build Coastguard Worker       break;
833*61046927SAndroid Build Coastguard Worker 
834*61046927SAndroid Build Coastguard Worker    case VAEncMiscParameterTypeHRD:
835*61046927SAndroid Build Coastguard Worker       vaStatus = handleVAEncMiscParameterTypeHRD(context, misc);
836*61046927SAndroid Build Coastguard Worker       break;
837*61046927SAndroid Build Coastguard Worker 
838*61046927SAndroid Build Coastguard Worker    case VAEncMiscParameterTypeRIR:
839*61046927SAndroid Build Coastguard Worker       vaStatus = handleVAEncMiscParameterTypeRIR(context, misc);
840*61046927SAndroid Build Coastguard Worker       break;
841*61046927SAndroid Build Coastguard Worker 
842*61046927SAndroid Build Coastguard Worker    case VAEncMiscParameterTypeMaxSliceSize:
843*61046927SAndroid Build Coastguard Worker       vaStatus = handleVAEncMiscParameterTypeMaxSliceSize(context, misc);
844*61046927SAndroid Build Coastguard Worker       break;
845*61046927SAndroid Build Coastguard Worker 
846*61046927SAndroid Build Coastguard Worker    case VAEncMiscParameterTypeROI:
847*61046927SAndroid Build Coastguard Worker       vaStatus = handleVAEncMiscParameterTypeROI(context, misc);
848*61046927SAndroid Build Coastguard Worker       break;
849*61046927SAndroid Build Coastguard Worker 
850*61046927SAndroid Build Coastguard Worker    default:
851*61046927SAndroid Build Coastguard Worker       break;
852*61046927SAndroid Build Coastguard Worker    }
853*61046927SAndroid Build Coastguard Worker 
854*61046927SAndroid Build Coastguard Worker    return vaStatus;
855*61046927SAndroid Build Coastguard Worker }
856*61046927SAndroid Build Coastguard Worker 
857*61046927SAndroid Build Coastguard Worker static VAStatus
handleVAEncPictureParameterBufferType(vlVaDriver * drv,vlVaContext * context,vlVaBuffer * buf)858*61046927SAndroid Build Coastguard Worker handleVAEncPictureParameterBufferType(vlVaDriver *drv, vlVaContext *context, vlVaBuffer *buf)
859*61046927SAndroid Build Coastguard Worker {
860*61046927SAndroid Build Coastguard Worker    VAStatus status = VA_STATUS_SUCCESS;
861*61046927SAndroid Build Coastguard Worker 
862*61046927SAndroid Build Coastguard Worker    switch (u_reduce_video_profile(context->templat.profile)) {
863*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_MPEG4_AVC:
864*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncPictureParameterBufferTypeH264(drv, context, buf);
865*61046927SAndroid Build Coastguard Worker       break;
866*61046927SAndroid Build Coastguard Worker 
867*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_HEVC:
868*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncPictureParameterBufferTypeHEVC(drv, context, buf);
869*61046927SAndroid Build Coastguard Worker       break;
870*61046927SAndroid Build Coastguard Worker 
871*61046927SAndroid Build Coastguard Worker #if VA_CHECK_VERSION(1, 16, 0)
872*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_AV1:
873*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncPictureParameterBufferTypeAV1(drv, context, buf);
874*61046927SAndroid Build Coastguard Worker       break;
875*61046927SAndroid Build Coastguard Worker #endif
876*61046927SAndroid Build Coastguard Worker 
877*61046927SAndroid Build Coastguard Worker    default:
878*61046927SAndroid Build Coastguard Worker       break;
879*61046927SAndroid Build Coastguard Worker    }
880*61046927SAndroid Build Coastguard Worker 
881*61046927SAndroid Build Coastguard Worker    return status;
882*61046927SAndroid Build Coastguard Worker }
883*61046927SAndroid Build Coastguard Worker 
884*61046927SAndroid Build Coastguard Worker static VAStatus
handleVAEncSliceParameterBufferType(vlVaDriver * drv,vlVaContext * context,vlVaBuffer * buf)885*61046927SAndroid Build Coastguard Worker handleVAEncSliceParameterBufferType(vlVaDriver *drv, vlVaContext *context, vlVaBuffer *buf)
886*61046927SAndroid Build Coastguard Worker {
887*61046927SAndroid Build Coastguard Worker    VAStatus status = VA_STATUS_SUCCESS;
888*61046927SAndroid Build Coastguard Worker 
889*61046927SAndroid Build Coastguard Worker    switch (u_reduce_video_profile(context->templat.profile)) {
890*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_MPEG4_AVC:
891*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncSliceParameterBufferTypeH264(drv, context, buf);
892*61046927SAndroid Build Coastguard Worker       break;
893*61046927SAndroid Build Coastguard Worker 
894*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_HEVC:
895*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncSliceParameterBufferTypeHEVC(drv, context, buf);
896*61046927SAndroid Build Coastguard Worker       break;
897*61046927SAndroid Build Coastguard Worker 
898*61046927SAndroid Build Coastguard Worker #if VA_CHECK_VERSION(1, 16, 0)
899*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_AV1:
900*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncSliceParameterBufferTypeAV1(drv, context, buf);
901*61046927SAndroid Build Coastguard Worker       break;
902*61046927SAndroid Build Coastguard Worker #endif
903*61046927SAndroid Build Coastguard Worker 
904*61046927SAndroid Build Coastguard Worker    default:
905*61046927SAndroid Build Coastguard Worker       break;
906*61046927SAndroid Build Coastguard Worker    }
907*61046927SAndroid Build Coastguard Worker 
908*61046927SAndroid Build Coastguard Worker    return status;
909*61046927SAndroid Build Coastguard Worker }
910*61046927SAndroid Build Coastguard Worker 
911*61046927SAndroid Build Coastguard Worker static VAStatus
handleVAEncPackedHeaderParameterBufferType(vlVaContext * context,vlVaBuffer * buf)912*61046927SAndroid Build Coastguard Worker handleVAEncPackedHeaderParameterBufferType(vlVaContext *context, vlVaBuffer *buf)
913*61046927SAndroid Build Coastguard Worker {
914*61046927SAndroid Build Coastguard Worker    VAEncPackedHeaderParameterBuffer *param = buf->data;
915*61046927SAndroid Build Coastguard Worker 
916*61046927SAndroid Build Coastguard Worker    context->packed_header_emulation_bytes = param->has_emulation_bytes;
917*61046927SAndroid Build Coastguard Worker    context->packed_header_type = param->type;
918*61046927SAndroid Build Coastguard Worker 
919*61046927SAndroid Build Coastguard Worker    return VA_STATUS_SUCCESS;
920*61046927SAndroid Build Coastguard Worker }
921*61046927SAndroid Build Coastguard Worker 
922*61046927SAndroid Build Coastguard Worker static VAStatus
handleVAEncPackedHeaderDataBufferType(vlVaContext * context,vlVaBuffer * buf)923*61046927SAndroid Build Coastguard Worker handleVAEncPackedHeaderDataBufferType(vlVaContext *context, vlVaBuffer *buf)
924*61046927SAndroid Build Coastguard Worker {
925*61046927SAndroid Build Coastguard Worker    VAStatus status = VA_STATUS_SUCCESS;
926*61046927SAndroid Build Coastguard Worker 
927*61046927SAndroid Build Coastguard Worker    switch (u_reduce_video_profile(context->templat.profile)) {
928*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_MPEG4_AVC:
929*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncPackedHeaderDataBufferTypeH264(context, buf);
930*61046927SAndroid Build Coastguard Worker       break;
931*61046927SAndroid Build Coastguard Worker 
932*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_HEVC:
933*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncPackedHeaderDataBufferTypeHEVC(context, buf);
934*61046927SAndroid Build Coastguard Worker       break;
935*61046927SAndroid Build Coastguard Worker 
936*61046927SAndroid Build Coastguard Worker #if VA_CHECK_VERSION(1, 16, 0)
937*61046927SAndroid Build Coastguard Worker    case PIPE_VIDEO_FORMAT_AV1:
938*61046927SAndroid Build Coastguard Worker       status = vlVaHandleVAEncPackedHeaderDataBufferTypeAV1(context, buf);
939*61046927SAndroid Build Coastguard Worker       break;
940*61046927SAndroid Build Coastguard Worker #endif
941*61046927SAndroid Build Coastguard Worker 
942*61046927SAndroid Build Coastguard Worker    default:
943*61046927SAndroid Build Coastguard Worker       break;
944*61046927SAndroid Build Coastguard Worker    }
945*61046927SAndroid Build Coastguard Worker 
946*61046927SAndroid Build Coastguard Worker    return status;
947*61046927SAndroid Build Coastguard Worker }
948*61046927SAndroid Build Coastguard Worker 
949*61046927SAndroid Build Coastguard Worker static VAStatus
handleVAStatsStatisticsBufferType(VADriverContextP ctx,vlVaContext * context,vlVaBuffer * buf)950*61046927SAndroid Build Coastguard Worker handleVAStatsStatisticsBufferType(VADriverContextP ctx, vlVaContext *context, vlVaBuffer *buf)
951*61046927SAndroid Build Coastguard Worker {
952*61046927SAndroid Build Coastguard Worker    if (context->decoder->entrypoint != PIPE_VIDEO_ENTRYPOINT_ENCODE)
953*61046927SAndroid Build Coastguard Worker       return VA_STATUS_ERROR_UNIMPLEMENTED;
954*61046927SAndroid Build Coastguard Worker 
955*61046927SAndroid Build Coastguard Worker    vlVaDriver *drv;
956*61046927SAndroid Build Coastguard Worker    drv = VL_VA_DRIVER(ctx);
957*61046927SAndroid Build Coastguard Worker 
958*61046927SAndroid Build Coastguard Worker    if (!drv)
959*61046927SAndroid Build Coastguard Worker       return VA_STATUS_ERROR_INVALID_CONTEXT;
960*61046927SAndroid Build Coastguard Worker 
961*61046927SAndroid Build Coastguard Worker    if (!buf->derived_surface.resource)
962*61046927SAndroid Build Coastguard Worker       buf->derived_surface.resource = pipe_buffer_create(drv->pipe->screen, PIPE_BIND_VERTEX_BUFFER,
963*61046927SAndroid Build Coastguard Worker                                             PIPE_USAGE_STREAM, buf->size);
964*61046927SAndroid Build Coastguard Worker 
965*61046927SAndroid Build Coastguard Worker    context->target->statistics_data = buf->derived_surface.resource;
966*61046927SAndroid Build Coastguard Worker 
967*61046927SAndroid Build Coastguard Worker    return VA_STATUS_SUCCESS;
968*61046927SAndroid Build Coastguard Worker }
969*61046927SAndroid Build Coastguard Worker 
970*61046927SAndroid Build Coastguard Worker VAStatus
vlVaRenderPicture(VADriverContextP ctx,VAContextID context_id,VABufferID * buffers,int num_buffers)971*61046927SAndroid Build Coastguard Worker vlVaRenderPicture(VADriverContextP ctx, VAContextID context_id, VABufferID *buffers, int num_buffers)
972*61046927SAndroid Build Coastguard Worker {
973*61046927SAndroid Build Coastguard Worker    vlVaDriver *drv;
974*61046927SAndroid Build Coastguard Worker    vlVaContext *context;
975*61046927SAndroid Build Coastguard Worker    VAStatus vaStatus = VA_STATUS_SUCCESS;
976*61046927SAndroid Build Coastguard Worker 
977*61046927SAndroid Build Coastguard Worker    unsigned i;
978*61046927SAndroid Build Coastguard Worker 
979*61046927SAndroid Build Coastguard Worker    if (!ctx)
980*61046927SAndroid Build Coastguard Worker       return VA_STATUS_ERROR_INVALID_CONTEXT;
981*61046927SAndroid Build Coastguard Worker 
982*61046927SAndroid Build Coastguard Worker    drv = VL_VA_DRIVER(ctx);
983*61046927SAndroid Build Coastguard Worker    if (!drv)
984*61046927SAndroid Build Coastguard Worker       return VA_STATUS_ERROR_INVALID_CONTEXT;
985*61046927SAndroid Build Coastguard Worker 
986*61046927SAndroid Build Coastguard Worker    mtx_lock(&drv->mutex);
987*61046927SAndroid Build Coastguard Worker    context = handle_table_get(drv->htab, context_id);
988*61046927SAndroid Build Coastguard Worker    if (!context) {
989*61046927SAndroid Build Coastguard Worker       mtx_unlock(&drv->mutex);
990*61046927SAndroid Build Coastguard Worker       return VA_STATUS_ERROR_INVALID_CONTEXT;
991*61046927SAndroid Build Coastguard Worker    }
992*61046927SAndroid Build Coastguard Worker 
993*61046927SAndroid Build Coastguard Worker    /* Always process VAProtectedSliceDataBufferType first because it changes the state */
994*61046927SAndroid Build Coastguard Worker    for (i = 0; i < num_buffers; ++i) {
995*61046927SAndroid Build Coastguard Worker       vlVaBuffer *buf = handle_table_get(drv->htab, buffers[i]);
996*61046927SAndroid Build Coastguard Worker       if (!buf) {
997*61046927SAndroid Build Coastguard Worker          mtx_unlock(&drv->mutex);
998*61046927SAndroid Build Coastguard Worker          return VA_STATUS_ERROR_INVALID_BUFFER;
999*61046927SAndroid Build Coastguard Worker       }
1000*61046927SAndroid Build Coastguard Worker 
1001*61046927SAndroid Build Coastguard Worker       if (buf->type == VAProtectedSliceDataBufferType)
1002*61046927SAndroid Build Coastguard Worker          handleVAProtectedSliceDataBufferType(context, buf);
1003*61046927SAndroid Build Coastguard Worker    }
1004*61046927SAndroid Build Coastguard Worker 
1005*61046927SAndroid Build Coastguard Worker    for (i = 0; i < num_buffers && vaStatus == VA_STATUS_SUCCESS; ++i) {
1006*61046927SAndroid Build Coastguard Worker       vlVaBuffer *buf = handle_table_get(drv->htab, buffers[i]);
1007*61046927SAndroid Build Coastguard Worker 
1008*61046927SAndroid Build Coastguard Worker       switch (buf->type) {
1009*61046927SAndroid Build Coastguard Worker       case VAPictureParameterBufferType:
1010*61046927SAndroid Build Coastguard Worker          vaStatus = handlePictureParameterBuffer(drv, context, buf);
1011*61046927SAndroid Build Coastguard Worker          break;
1012*61046927SAndroid Build Coastguard Worker 
1013*61046927SAndroid Build Coastguard Worker       case VAIQMatrixBufferType:
1014*61046927SAndroid Build Coastguard Worker          handleIQMatrixBuffer(context, buf);
1015*61046927SAndroid Build Coastguard Worker          break;
1016*61046927SAndroid Build Coastguard Worker 
1017*61046927SAndroid Build Coastguard Worker       case VASliceParameterBufferType:
1018*61046927SAndroid Build Coastguard Worker          handleSliceParameterBuffer(context, buf);
1019*61046927SAndroid Build Coastguard Worker          context->have_slice_params = true;
1020*61046927SAndroid Build Coastguard Worker          break;
1021*61046927SAndroid Build Coastguard Worker 
1022*61046927SAndroid Build Coastguard Worker       case VASliceDataBufferType:
1023*61046927SAndroid Build Coastguard Worker          vaStatus = handleVASliceDataBufferType(context, buf);
1024*61046927SAndroid Build Coastguard Worker          /* Workaround for apps sending single slice data buffer followed
1025*61046927SAndroid Build Coastguard Worker           * by multiple slice parameter buffers. */
1026*61046927SAndroid Build Coastguard Worker          if (context->have_slice_params)
1027*61046927SAndroid Build Coastguard Worker             context->slice_data_offset += buf->size;
1028*61046927SAndroid Build Coastguard Worker          break;
1029*61046927SAndroid Build Coastguard Worker 
1030*61046927SAndroid Build Coastguard Worker       case VAProcPipelineParameterBufferType:
1031*61046927SAndroid Build Coastguard Worker          vaStatus = vlVaHandleVAProcPipelineParameterBufferType(drv, context, buf);
1032*61046927SAndroid Build Coastguard Worker          break;
1033*61046927SAndroid Build Coastguard Worker 
1034*61046927SAndroid Build Coastguard Worker       case VAEncSequenceParameterBufferType:
1035*61046927SAndroid Build Coastguard Worker          vaStatus = handleVAEncSequenceParameterBufferType(drv, context, buf);
1036*61046927SAndroid Build Coastguard Worker          break;
1037*61046927SAndroid Build Coastguard Worker 
1038*61046927SAndroid Build Coastguard Worker       case VAEncMiscParameterBufferType:
1039*61046927SAndroid Build Coastguard Worker          vaStatus = handleVAEncMiscParameterBufferType(context, buf);
1040*61046927SAndroid Build Coastguard Worker          break;
1041*61046927SAndroid Build Coastguard Worker 
1042*61046927SAndroid Build Coastguard Worker       case VAEncPictureParameterBufferType:
1043*61046927SAndroid Build Coastguard Worker          vaStatus = handleVAEncPictureParameterBufferType(drv, context, buf);
1044*61046927SAndroid Build Coastguard Worker          break;
1045*61046927SAndroid Build Coastguard Worker 
1046*61046927SAndroid Build Coastguard Worker       case VAEncSliceParameterBufferType:
1047*61046927SAndroid Build Coastguard Worker          vaStatus = handleVAEncSliceParameterBufferType(drv, context, buf);
1048*61046927SAndroid Build Coastguard Worker          break;
1049*61046927SAndroid Build Coastguard Worker 
1050*61046927SAndroid Build Coastguard Worker       case VAHuffmanTableBufferType:
1051*61046927SAndroid Build Coastguard Worker          vlVaHandleHuffmanTableBufferType(context, buf);
1052*61046927SAndroid Build Coastguard Worker          break;
1053*61046927SAndroid Build Coastguard Worker 
1054*61046927SAndroid Build Coastguard Worker       case VAEncPackedHeaderParameterBufferType:
1055*61046927SAndroid Build Coastguard Worker          handleVAEncPackedHeaderParameterBufferType(context, buf);
1056*61046927SAndroid Build Coastguard Worker          break;
1057*61046927SAndroid Build Coastguard Worker       case VAEncPackedHeaderDataBufferType:
1058*61046927SAndroid Build Coastguard Worker          handleVAEncPackedHeaderDataBufferType(context, buf);
1059*61046927SAndroid Build Coastguard Worker          break;
1060*61046927SAndroid Build Coastguard Worker 
1061*61046927SAndroid Build Coastguard Worker       case VAStatsStatisticsBufferType:
1062*61046927SAndroid Build Coastguard Worker          handleVAStatsStatisticsBufferType(ctx, context, buf);
1063*61046927SAndroid Build Coastguard Worker          break;
1064*61046927SAndroid Build Coastguard Worker 
1065*61046927SAndroid Build Coastguard Worker       default:
1066*61046927SAndroid Build Coastguard Worker          break;
1067*61046927SAndroid Build Coastguard Worker       }
1068*61046927SAndroid Build Coastguard Worker    }
1069*61046927SAndroid Build Coastguard Worker 
1070*61046927SAndroid Build Coastguard Worker    if (context->decoder &&
1071*61046927SAndroid Build Coastguard Worker        context->decoder->entrypoint == PIPE_VIDEO_ENTRYPOINT_BITSTREAM &&
1072*61046927SAndroid Build Coastguard Worker        context->bs.num_buffers) {
1073*61046927SAndroid Build Coastguard Worker       context->decoder->decode_bitstream(context->decoder, context->target, &context->desc.base,
1074*61046927SAndroid Build Coastguard Worker          context->bs.num_buffers, (const void * const*)context->bs.buffers, context->bs.sizes);
1075*61046927SAndroid Build Coastguard Worker       context->bs.num_buffers = 0;
1076*61046927SAndroid Build Coastguard Worker    }
1077*61046927SAndroid Build Coastguard Worker 
1078*61046927SAndroid Build Coastguard Worker    mtx_unlock(&drv->mutex);
1079*61046927SAndroid Build Coastguard Worker 
1080*61046927SAndroid Build Coastguard Worker    return vaStatus;
1081*61046927SAndroid Build Coastguard Worker }
1082*61046927SAndroid Build Coastguard Worker 
vlVaQueryApplyFilmGrainAV1(vlVaContext * context,int * output_id,struct pipe_video_buffer *** out_target)1083*61046927SAndroid Build Coastguard Worker static bool vlVaQueryApplyFilmGrainAV1(vlVaContext *context,
1084*61046927SAndroid Build Coastguard Worker                                  int *output_id,
1085*61046927SAndroid Build Coastguard Worker                                  struct pipe_video_buffer ***out_target)
1086*61046927SAndroid Build Coastguard Worker {
1087*61046927SAndroid Build Coastguard Worker    struct pipe_av1_picture_desc *av1 = NULL;
1088*61046927SAndroid Build Coastguard Worker 
1089*61046927SAndroid Build Coastguard Worker    if (u_reduce_video_profile(context->templat.profile) != PIPE_VIDEO_FORMAT_AV1 ||
1090*61046927SAndroid Build Coastguard Worker        context->decoder->entrypoint != PIPE_VIDEO_ENTRYPOINT_BITSTREAM)
1091*61046927SAndroid Build Coastguard Worker       return false;
1092*61046927SAndroid Build Coastguard Worker 
1093*61046927SAndroid Build Coastguard Worker    av1 = &context->desc.av1;
1094*61046927SAndroid Build Coastguard Worker    if (!av1->picture_parameter.film_grain_info.film_grain_info_fields.apply_grain)
1095*61046927SAndroid Build Coastguard Worker       return false;
1096*61046927SAndroid Build Coastguard Worker 
1097*61046927SAndroid Build Coastguard Worker    *output_id = av1->picture_parameter.current_frame_id;
1098*61046927SAndroid Build Coastguard Worker    *out_target = &av1->film_grain_target;
1099*61046927SAndroid Build Coastguard Worker    return true;
1100*61046927SAndroid Build Coastguard Worker }
1101*61046927SAndroid Build Coastguard Worker 
vlVaClearRawHeaders(struct util_dynarray * headers)1102*61046927SAndroid Build Coastguard Worker static void vlVaClearRawHeaders(struct util_dynarray *headers)
1103*61046927SAndroid Build Coastguard Worker {
1104*61046927SAndroid Build Coastguard Worker    util_dynarray_foreach(headers, struct pipe_enc_raw_header, header)
1105*61046927SAndroid Build Coastguard Worker       FREE(header->buffer);
1106*61046927SAndroid Build Coastguard Worker    util_dynarray_clear(headers);
1107*61046927SAndroid Build Coastguard Worker }
1108*61046927SAndroid Build Coastguard Worker 
1109*61046927SAndroid Build Coastguard Worker VAStatus
vlVaEndPicture(VADriverContextP ctx,VAContextID context_id)1110*61046927SAndroid Build Coastguard Worker vlVaEndPicture(VADriverContextP ctx, VAContextID context_id)
1111*61046927SAndroid Build Coastguard Worker {
1112*61046927SAndroid Build Coastguard Worker    vlVaDriver *drv;
1113*61046927SAndroid Build Coastguard Worker    vlVaContext *context;
1114*61046927SAndroid Build Coastguard Worker    vlVaBuffer *coded_buf;
1115*61046927SAndroid Build Coastguard Worker    vlVaSurface *surf;
1116*61046927SAndroid Build Coastguard Worker    void *feedback = NULL;
1117*61046927SAndroid Build Coastguard Worker    struct pipe_screen *screen;
1118*61046927SAndroid Build Coastguard Worker    bool supported;
1119*61046927SAndroid Build Coastguard Worker    bool realloc = false;
1120*61046927SAndroid Build Coastguard Worker    bool apply_av1_fg = false;
1121*61046927SAndroid Build Coastguard Worker    enum pipe_format format;
1122*61046927SAndroid Build Coastguard Worker    struct pipe_video_buffer **out_target;
1123*61046927SAndroid Build Coastguard Worker    int output_id;
1124*61046927SAndroid Build Coastguard Worker 
1125*61046927SAndroid Build Coastguard Worker    if (!ctx)
1126*61046927SAndroid Build Coastguard Worker       return VA_STATUS_ERROR_INVALID_CONTEXT;
1127*61046927SAndroid Build Coastguard Worker 
1128*61046927SAndroid Build Coastguard Worker    drv = VL_VA_DRIVER(ctx);
1129*61046927SAndroid Build Coastguard Worker    if (!drv)
1130*61046927SAndroid Build Coastguard Worker       return VA_STATUS_ERROR_INVALID_CONTEXT;
1131*61046927SAndroid Build Coastguard Worker 
1132*61046927SAndroid Build Coastguard Worker    mtx_lock(&drv->mutex);
1133*61046927SAndroid Build Coastguard Worker    context = handle_table_get(drv->htab, context_id);
1134*61046927SAndroid Build Coastguard Worker    mtx_unlock(&drv->mutex);
1135*61046927SAndroid Build Coastguard Worker    if (!context)
1136*61046927SAndroid Build Coastguard Worker       return VA_STATUS_ERROR_INVALID_CONTEXT;
1137*61046927SAndroid Build Coastguard Worker 
1138*61046927SAndroid Build Coastguard Worker    if (!context->decoder) {
1139*61046927SAndroid Build Coastguard Worker       if (context->templat.profile != PIPE_VIDEO_PROFILE_UNKNOWN)
1140*61046927SAndroid Build Coastguard Worker          return VA_STATUS_ERROR_INVALID_CONTEXT;
1141*61046927SAndroid Build Coastguard Worker 
1142*61046927SAndroid Build Coastguard Worker       /* VPP */
1143*61046927SAndroid Build Coastguard Worker       return VA_STATUS_SUCCESS;
1144*61046927SAndroid Build Coastguard Worker    }
1145*61046927SAndroid Build Coastguard Worker 
1146*61046927SAndroid Build Coastguard Worker    output_id = context->target_id;
1147*61046927SAndroid Build Coastguard Worker    out_target = &context->target;
1148*61046927SAndroid Build Coastguard Worker    apply_av1_fg = vlVaQueryApplyFilmGrainAV1(context, &output_id, &out_target);
1149*61046927SAndroid Build Coastguard Worker 
1150*61046927SAndroid Build Coastguard Worker    mtx_lock(&drv->mutex);
1151*61046927SAndroid Build Coastguard Worker    surf = handle_table_get(drv->htab, output_id);
1152*61046927SAndroid Build Coastguard Worker    vlVaGetSurfaceBuffer(drv, surf);
1153*61046927SAndroid Build Coastguard Worker    if (!surf || !surf->buffer) {
1154*61046927SAndroid Build Coastguard Worker       mtx_unlock(&drv->mutex);
1155*61046927SAndroid Build Coastguard Worker       return VA_STATUS_ERROR_INVALID_SURFACE;
1156*61046927SAndroid Build Coastguard Worker    }
1157*61046927SAndroid Build Coastguard Worker 
1158*61046927SAndroid Build Coastguard Worker    if (apply_av1_fg) {
1159*61046927SAndroid Build Coastguard Worker       vlVaSetSurfaceContext(drv, surf, context);
1160*61046927SAndroid Build Coastguard Worker       *out_target = surf->buffer;
1161*61046927SAndroid Build Coastguard Worker    }
1162*61046927SAndroid Build Coastguard Worker 
1163*61046927SAndroid Build Coastguard Worker    context->mpeg4.frame_num++;
1164*61046927SAndroid Build Coastguard Worker 
1165*61046927SAndroid Build Coastguard Worker    screen = context->decoder->context->screen;
1166*61046927SAndroid Build Coastguard Worker    supported = screen->get_video_param(screen, context->decoder->profile,
1167*61046927SAndroid Build Coastguard Worker                                        context->decoder->entrypoint,
1168*61046927SAndroid Build Coastguard Worker                                        surf->buffer->interlaced ?
1169*61046927SAndroid Build Coastguard Worker                                        PIPE_VIDEO_CAP_SUPPORTS_INTERLACED :
1170*61046927SAndroid Build Coastguard Worker                                        PIPE_VIDEO_CAP_SUPPORTS_PROGRESSIVE);
1171*61046927SAndroid Build Coastguard Worker 
1172*61046927SAndroid Build Coastguard Worker    if (!supported) {
1173*61046927SAndroid Build Coastguard Worker       surf->templat.interlaced = screen->get_video_param(screen,
1174*61046927SAndroid Build Coastguard Worker                                        context->decoder->profile,
1175*61046927SAndroid Build Coastguard Worker                                        context->decoder->entrypoint,
1176*61046927SAndroid Build Coastguard Worker                                        PIPE_VIDEO_CAP_PREFERS_INTERLACED);
1177*61046927SAndroid Build Coastguard Worker       realloc = true;
1178*61046927SAndroid Build Coastguard Worker    }
1179*61046927SAndroid Build Coastguard Worker 
1180*61046927SAndroid Build Coastguard Worker    format = screen->get_video_param(screen, context->decoder->profile,
1181*61046927SAndroid Build Coastguard Worker                                     context->decoder->entrypoint,
1182*61046927SAndroid Build Coastguard Worker                                     PIPE_VIDEO_CAP_PREFERED_FORMAT);
1183*61046927SAndroid Build Coastguard Worker 
1184*61046927SAndroid Build Coastguard Worker    if (surf->buffer->buffer_format != format &&
1185*61046927SAndroid Build Coastguard Worker        surf->buffer->buffer_format == PIPE_FORMAT_NV12) {
1186*61046927SAndroid Build Coastguard Worker       /* check originally as NV12 only */
1187*61046927SAndroid Build Coastguard Worker       surf->templat.buffer_format = format;
1188*61046927SAndroid Build Coastguard Worker       realloc = true;
1189*61046927SAndroid Build Coastguard Worker    }
1190*61046927SAndroid Build Coastguard Worker 
1191*61046927SAndroid Build Coastguard Worker    if (u_reduce_video_profile(context->templat.profile) == PIPE_VIDEO_FORMAT_JPEG) {
1192*61046927SAndroid Build Coastguard Worker       if (surf->buffer->buffer_format == PIPE_FORMAT_NV12 &&
1193*61046927SAndroid Build Coastguard Worker           context->mjpeg.sampling_factor != MJPEG_SAMPLING_FACTOR_NV12) {
1194*61046927SAndroid Build Coastguard Worker          /* workaround to reallocate surface buffer with right format
1195*61046927SAndroid Build Coastguard Worker           * if it doesnt match with sampling_factor. ffmpeg doesnt
1196*61046927SAndroid Build Coastguard Worker           * use VASurfaceAttribPixelFormat and defaults to NV12.
1197*61046927SAndroid Build Coastguard Worker           */
1198*61046927SAndroid Build Coastguard Worker          switch (context->mjpeg.sampling_factor) {
1199*61046927SAndroid Build Coastguard Worker             case MJPEG_SAMPLING_FACTOR_YUV422:
1200*61046927SAndroid Build Coastguard Worker             case MJPEG_SAMPLING_FACTOR_YUY2:
1201*61046927SAndroid Build Coastguard Worker                surf->templat.buffer_format = PIPE_FORMAT_YUYV;
1202*61046927SAndroid Build Coastguard Worker                break;
1203*61046927SAndroid Build Coastguard Worker             case MJPEG_SAMPLING_FACTOR_YUV444:
1204*61046927SAndroid Build Coastguard Worker                surf->templat.buffer_format = PIPE_FORMAT_Y8_U8_V8_444_UNORM;
1205*61046927SAndroid Build Coastguard Worker                break;
1206*61046927SAndroid Build Coastguard Worker             case MJPEG_SAMPLING_FACTOR_YUV400:
1207*61046927SAndroid Build Coastguard Worker                surf->templat.buffer_format = PIPE_FORMAT_Y8_400_UNORM;
1208*61046927SAndroid Build Coastguard Worker                break;
1209*61046927SAndroid Build Coastguard Worker             default:
1210*61046927SAndroid Build Coastguard Worker                mtx_unlock(&drv->mutex);
1211*61046927SAndroid Build Coastguard Worker                return VA_STATUS_ERROR_INVALID_SURFACE;
1212*61046927SAndroid Build Coastguard Worker          }
1213*61046927SAndroid Build Coastguard Worker          realloc = true;
1214*61046927SAndroid Build Coastguard Worker       }
1215*61046927SAndroid Build Coastguard Worker       /* check if format is supported before proceeding with realloc,
1216*61046927SAndroid Build Coastguard Worker        * also avoid submission if hardware doesnt support the format and
1217*61046927SAndroid Build Coastguard Worker        * applcation failed to check the supported rt_formats.
1218*61046927SAndroid Build Coastguard Worker        */
1219*61046927SAndroid Build Coastguard Worker       if (!screen->is_video_format_supported(screen, surf->templat.buffer_format,
1220*61046927SAndroid Build Coastguard Worker           PIPE_VIDEO_PROFILE_JPEG_BASELINE, PIPE_VIDEO_ENTRYPOINT_BITSTREAM)) {
1221*61046927SAndroid Build Coastguard Worker          mtx_unlock(&drv->mutex);
1222*61046927SAndroid Build Coastguard Worker          return VA_STATUS_ERROR_INVALID_SURFACE;
1223*61046927SAndroid Build Coastguard Worker       }
1224*61046927SAndroid Build Coastguard Worker    }
1225*61046927SAndroid Build Coastguard Worker 
1226*61046927SAndroid Build Coastguard Worker    if ((bool)(surf->templat.bind & PIPE_BIND_PROTECTED) != context->desc.base.protected_playback) {
1227*61046927SAndroid Build Coastguard Worker       if (context->desc.base.protected_playback) {
1228*61046927SAndroid Build Coastguard Worker          surf->templat.bind |= PIPE_BIND_PROTECTED;
1229*61046927SAndroid Build Coastguard Worker       }
1230*61046927SAndroid Build Coastguard Worker       else
1231*61046927SAndroid Build Coastguard Worker          surf->templat.bind &= ~PIPE_BIND_PROTECTED;
1232*61046927SAndroid Build Coastguard Worker       realloc = true;
1233*61046927SAndroid Build Coastguard Worker    }
1234*61046927SAndroid Build Coastguard Worker 
1235*61046927SAndroid Build Coastguard Worker    if (u_reduce_video_profile(context->templat.profile) == PIPE_VIDEO_FORMAT_AV1 &&
1236*61046927SAndroid Build Coastguard Worker        surf->buffer->buffer_format == PIPE_FORMAT_NV12 &&
1237*61046927SAndroid Build Coastguard Worker        context->decoder->entrypoint == PIPE_VIDEO_ENTRYPOINT_BITSTREAM) {
1238*61046927SAndroid Build Coastguard Worker       if (context->desc.av1.picture_parameter.bit_depth_idx == 1) {
1239*61046927SAndroid Build Coastguard Worker          surf->templat.buffer_format = PIPE_FORMAT_P010;
1240*61046927SAndroid Build Coastguard Worker          realloc = true;
1241*61046927SAndroid Build Coastguard Worker       }
1242*61046927SAndroid Build Coastguard Worker    }
1243*61046927SAndroid Build Coastguard Worker 
1244*61046927SAndroid Build Coastguard Worker    if (realloc) {
1245*61046927SAndroid Build Coastguard Worker       struct pipe_video_buffer *old_buf = surf->buffer;
1246*61046927SAndroid Build Coastguard Worker 
1247*61046927SAndroid Build Coastguard Worker       if (vlVaHandleSurfaceAllocate(drv, surf, &surf->templat, NULL, 0) != VA_STATUS_SUCCESS) {
1248*61046927SAndroid Build Coastguard Worker          mtx_unlock(&drv->mutex);
1249*61046927SAndroid Build Coastguard Worker          return VA_STATUS_ERROR_ALLOCATION_FAILED;
1250*61046927SAndroid Build Coastguard Worker       }
1251*61046927SAndroid Build Coastguard Worker 
1252*61046927SAndroid Build Coastguard Worker       if (context->decoder->entrypoint == PIPE_VIDEO_ENTRYPOINT_ENCODE) {
1253*61046927SAndroid Build Coastguard Worker          if (old_buf->interlaced) {
1254*61046927SAndroid Build Coastguard Worker             struct u_rect src_rect, dst_rect;
1255*61046927SAndroid Build Coastguard Worker 
1256*61046927SAndroid Build Coastguard Worker             dst_rect.x0 = src_rect.x0 = 0;
1257*61046927SAndroid Build Coastguard Worker             dst_rect.y0 = src_rect.y0 = 0;
1258*61046927SAndroid Build Coastguard Worker             dst_rect.x1 = src_rect.x1 = surf->templat.width;
1259*61046927SAndroid Build Coastguard Worker             dst_rect.y1 = src_rect.y1 = surf->templat.height;
1260*61046927SAndroid Build Coastguard Worker             vl_compositor_yuv_deint_full(&drv->cstate, &drv->compositor,
1261*61046927SAndroid Build Coastguard Worker                                          old_buf, surf->buffer,
1262*61046927SAndroid Build Coastguard Worker                                          &src_rect, &dst_rect, VL_COMPOSITOR_WEAVE);
1263*61046927SAndroid Build Coastguard Worker          } else {
1264*61046927SAndroid Build Coastguard Worker             /* Can't convert from progressive to interlaced yet */
1265*61046927SAndroid Build Coastguard Worker             mtx_unlock(&drv->mutex);
1266*61046927SAndroid Build Coastguard Worker             return VA_STATUS_ERROR_INVALID_SURFACE;
1267*61046927SAndroid Build Coastguard Worker          }
1268*61046927SAndroid Build Coastguard Worker       }
1269*61046927SAndroid Build Coastguard Worker 
1270*61046927SAndroid Build Coastguard Worker       old_buf->destroy(old_buf);
1271*61046927SAndroid Build Coastguard Worker       *out_target = surf->buffer;
1272*61046927SAndroid Build Coastguard Worker    }
1273*61046927SAndroid Build Coastguard Worker 
1274*61046927SAndroid Build Coastguard Worker    if (context->decoder->entrypoint == PIPE_VIDEO_ENTRYPOINT_ENCODE) {
1275*61046927SAndroid Build Coastguard Worker       context->desc.base.fence = &surf->fence;
1276*61046927SAndroid Build Coastguard Worker       struct pipe_screen *screen = context->decoder->context->screen;
1277*61046927SAndroid Build Coastguard Worker       coded_buf = context->coded_buf;
1278*61046927SAndroid Build Coastguard Worker       if (u_reduce_video_profile(context->templat.profile) == PIPE_VIDEO_FORMAT_MPEG4_AVC)
1279*61046927SAndroid Build Coastguard Worker          context->desc.h264enc.frame_num_cnt++;
1280*61046927SAndroid Build Coastguard Worker 
1281*61046927SAndroid Build Coastguard Worker       if (surf->efc_surface) {
1282*61046927SAndroid Build Coastguard Worker          assert(surf == drv->last_efc_surface);
1283*61046927SAndroid Build Coastguard Worker          context->target = surf->efc_surface->buffer;
1284*61046927SAndroid Build Coastguard Worker          context->desc.base.input_format = surf->efc_surface->buffer->buffer_format;
1285*61046927SAndroid Build Coastguard Worker          context->desc.base.output_format = surf->buffer->buffer_format;
1286*61046927SAndroid Build Coastguard Worker          surf->efc_surface = NULL;
1287*61046927SAndroid Build Coastguard Worker          drv->last_efc_surface = NULL;
1288*61046927SAndroid Build Coastguard Worker       } else {
1289*61046927SAndroid Build Coastguard Worker          context->desc.base.input_format = surf->buffer->buffer_format;
1290*61046927SAndroid Build Coastguard Worker          context->desc.base.output_format = surf->buffer->buffer_format;
1291*61046927SAndroid Build Coastguard Worker       }
1292*61046927SAndroid Build Coastguard Worker       context->desc.base.input_full_range = surf->full_range;
1293*61046927SAndroid Build Coastguard Worker 
1294*61046927SAndroid Build Coastguard Worker       if (screen->is_video_target_buffer_supported &&
1295*61046927SAndroid Build Coastguard Worker           !screen->is_video_target_buffer_supported(screen,
1296*61046927SAndroid Build Coastguard Worker                                                     context->desc.base.output_format,
1297*61046927SAndroid Build Coastguard Worker                                                     context->target,
1298*61046927SAndroid Build Coastguard Worker                                                     context->decoder->profile,
1299*61046927SAndroid Build Coastguard Worker                                                     context->decoder->entrypoint)) {
1300*61046927SAndroid Build Coastguard Worker             mtx_unlock(&drv->mutex);
1301*61046927SAndroid Build Coastguard Worker             return VA_STATUS_ERROR_INVALID_SURFACE;
1302*61046927SAndroid Build Coastguard Worker       }
1303*61046927SAndroid Build Coastguard Worker 
1304*61046927SAndroid Build Coastguard Worker       int driver_metadata_support = drv->pipe->screen->get_video_param(drv->pipe->screen,
1305*61046927SAndroid Build Coastguard Worker                                                                        context->decoder->profile,
1306*61046927SAndroid Build Coastguard Worker                                                                        context->decoder->entrypoint,
1307*61046927SAndroid Build Coastguard Worker                                                                        PIPE_VIDEO_CAP_ENC_SUPPORTS_FEEDBACK_METADATA);
1308*61046927SAndroid Build Coastguard Worker       if (u_reduce_video_profile(context->templat.profile) == PIPE_VIDEO_FORMAT_MPEG4_AVC)
1309*61046927SAndroid Build Coastguard Worker          context->desc.h264enc.requested_metadata = driver_metadata_support;
1310*61046927SAndroid Build Coastguard Worker       else if (u_reduce_video_profile(context->templat.profile) == PIPE_VIDEO_FORMAT_HEVC)
1311*61046927SAndroid Build Coastguard Worker          context->desc.h265enc.requested_metadata = driver_metadata_support;
1312*61046927SAndroid Build Coastguard Worker       else if (u_reduce_video_profile(context->templat.profile) == PIPE_VIDEO_FORMAT_AV1)
1313*61046927SAndroid Build Coastguard Worker          context->desc.av1enc.requested_metadata = driver_metadata_support;
1314*61046927SAndroid Build Coastguard Worker 
1315*61046927SAndroid Build Coastguard Worker       context->decoder->begin_frame(context->decoder, context->target, &context->desc.base);
1316*61046927SAndroid Build Coastguard Worker       context->decoder->encode_bitstream(context->decoder, context->target,
1317*61046927SAndroid Build Coastguard Worker                                          coded_buf->derived_surface.resource, &feedback);
1318*61046927SAndroid Build Coastguard Worker       coded_buf->feedback = feedback;
1319*61046927SAndroid Build Coastguard Worker       coded_buf->ctx = context_id;
1320*61046927SAndroid Build Coastguard Worker       surf->feedback = feedback;
1321*61046927SAndroid Build Coastguard Worker       surf->coded_buf = coded_buf;
1322*61046927SAndroid Build Coastguard Worker       coded_buf->associated_encode_input_surf = context->target_id;
1323*61046927SAndroid Build Coastguard Worker    } else if (context->decoder->entrypoint == PIPE_VIDEO_ENTRYPOINT_BITSTREAM) {
1324*61046927SAndroid Build Coastguard Worker       context->desc.base.fence = &surf->fence;
1325*61046927SAndroid Build Coastguard Worker    } else if (context->decoder->entrypoint == PIPE_VIDEO_ENTRYPOINT_PROCESSING) {
1326*61046927SAndroid Build Coastguard Worker       context->desc.base.fence = &surf->fence;
1327*61046927SAndroid Build Coastguard Worker    }
1328*61046927SAndroid Build Coastguard Worker 
1329*61046927SAndroid Build Coastguard Worker    /* when there are external handles, we can't set PIPE_FLUSH_ASYNC */
1330*61046927SAndroid Build Coastguard Worker    if (context->desc.base.fence)
1331*61046927SAndroid Build Coastguard Worker       context->desc.base.flush_flags = drv->has_external_handles ? 0 : PIPE_FLUSH_ASYNC;
1332*61046927SAndroid Build Coastguard Worker 
1333*61046927SAndroid Build Coastguard Worker    if (context->decoder->end_frame(context->decoder, context->target, &context->desc.base) != 0) {
1334*61046927SAndroid Build Coastguard Worker       mtx_unlock(&drv->mutex);
1335*61046927SAndroid Build Coastguard Worker       return VA_STATUS_ERROR_OPERATION_FAILED;
1336*61046927SAndroid Build Coastguard Worker    }
1337*61046927SAndroid Build Coastguard Worker 
1338*61046927SAndroid Build Coastguard Worker    if (drv->pipe->screen->get_video_param(drv->pipe->screen,
1339*61046927SAndroid Build Coastguard Worker                            context->decoder->profile,
1340*61046927SAndroid Build Coastguard Worker                            context->decoder->entrypoint,
1341*61046927SAndroid Build Coastguard Worker                            PIPE_VIDEO_CAP_REQUIRES_FLUSH_ON_END_FRAME))
1342*61046927SAndroid Build Coastguard Worker       context->decoder->flush(context->decoder);
1343*61046927SAndroid Build Coastguard Worker    else {
1344*61046927SAndroid Build Coastguard Worker       if (context->decoder->entrypoint == PIPE_VIDEO_ENTRYPOINT_ENCODE &&
1345*61046927SAndroid Build Coastguard Worker          u_reduce_video_profile(context->templat.profile) == PIPE_VIDEO_FORMAT_MPEG4_AVC) {
1346*61046927SAndroid Build Coastguard Worker          int idr_period = context->desc.h264enc.gop_size / context->gop_coeff;
1347*61046927SAndroid Build Coastguard Worker          int p_remain_in_idr = idr_period - context->desc.h264enc.frame_num;
1348*61046927SAndroid Build Coastguard Worker          surf->frame_num_cnt = context->desc.h264enc.frame_num_cnt;
1349*61046927SAndroid Build Coastguard Worker          surf->force_flushed = false;
1350*61046927SAndroid Build Coastguard Worker          if (context->first_single_submitted) {
1351*61046927SAndroid Build Coastguard Worker             context->decoder->flush(context->decoder);
1352*61046927SAndroid Build Coastguard Worker             context->first_single_submitted = false;
1353*61046927SAndroid Build Coastguard Worker             surf->force_flushed = true;
1354*61046927SAndroid Build Coastguard Worker          }
1355*61046927SAndroid Build Coastguard Worker          if (p_remain_in_idr == 1) {
1356*61046927SAndroid Build Coastguard Worker             if ((context->desc.h264enc.frame_num_cnt % 2) != 0) {
1357*61046927SAndroid Build Coastguard Worker                context->decoder->flush(context->decoder);
1358*61046927SAndroid Build Coastguard Worker                context->first_single_submitted = true;
1359*61046927SAndroid Build Coastguard Worker             }
1360*61046927SAndroid Build Coastguard Worker             else
1361*61046927SAndroid Build Coastguard Worker                context->first_single_submitted = false;
1362*61046927SAndroid Build Coastguard Worker             surf->force_flushed = true;
1363*61046927SAndroid Build Coastguard Worker          }
1364*61046927SAndroid Build Coastguard Worker       }
1365*61046927SAndroid Build Coastguard Worker    }
1366*61046927SAndroid Build Coastguard Worker 
1367*61046927SAndroid Build Coastguard Worker    if (context->decoder->entrypoint == PIPE_VIDEO_ENTRYPOINT_ENCODE) {
1368*61046927SAndroid Build Coastguard Worker       switch (u_reduce_video_profile(context->templat.profile)) {
1369*61046927SAndroid Build Coastguard Worker       case PIPE_VIDEO_FORMAT_AV1:
1370*61046927SAndroid Build Coastguard Worker          context->desc.av1enc.frame_num++;
1371*61046927SAndroid Build Coastguard Worker          break;
1372*61046927SAndroid Build Coastguard Worker       case PIPE_VIDEO_FORMAT_HEVC:
1373*61046927SAndroid Build Coastguard Worker          context->desc.h265enc.frame_num++;
1374*61046927SAndroid Build Coastguard Worker          vlVaClearRawHeaders(&context->desc.h265enc.raw_headers);
1375*61046927SAndroid Build Coastguard Worker          break;
1376*61046927SAndroid Build Coastguard Worker       case PIPE_VIDEO_FORMAT_MPEG4_AVC:
1377*61046927SAndroid Build Coastguard Worker          if (!context->desc.h264enc.not_referenced)
1378*61046927SAndroid Build Coastguard Worker             context->desc.h264enc.frame_num++;
1379*61046927SAndroid Build Coastguard Worker          vlVaClearRawHeaders(&context->desc.h264enc.raw_headers);
1380*61046927SAndroid Build Coastguard Worker          break;
1381*61046927SAndroid Build Coastguard Worker       default:
1382*61046927SAndroid Build Coastguard Worker          break;
1383*61046927SAndroid Build Coastguard Worker       }
1384*61046927SAndroid Build Coastguard Worker    }
1385*61046927SAndroid Build Coastguard Worker 
1386*61046927SAndroid Build Coastguard Worker    mtx_unlock(&drv->mutex);
1387*61046927SAndroid Build Coastguard Worker    return VA_STATUS_SUCCESS;
1388*61046927SAndroid Build Coastguard Worker }
1389*61046927SAndroid Build Coastguard Worker 
1390*61046927SAndroid Build Coastguard Worker void
vlVaAddRawHeader(struct util_dynarray * headers,uint8_t type,uint32_t size,uint8_t * buf,bool is_slice,uint32_t emulation_bytes_start)1391*61046927SAndroid Build Coastguard Worker vlVaAddRawHeader(struct util_dynarray *headers, uint8_t type, uint32_t size,
1392*61046927SAndroid Build Coastguard Worker                  uint8_t *buf, bool is_slice, uint32_t emulation_bytes_start)
1393*61046927SAndroid Build Coastguard Worker {
1394*61046927SAndroid Build Coastguard Worker    struct pipe_enc_raw_header header = {
1395*61046927SAndroid Build Coastguard Worker       .type = type,
1396*61046927SAndroid Build Coastguard Worker       .is_slice = is_slice,
1397*61046927SAndroid Build Coastguard Worker    };
1398*61046927SAndroid Build Coastguard Worker    if (emulation_bytes_start) {
1399*61046927SAndroid Build Coastguard Worker       uint32_t pos = emulation_bytes_start, num_zeros = 0;
1400*61046927SAndroid Build Coastguard Worker       header.buffer = MALLOC(size * 3 / 2);
1401*61046927SAndroid Build Coastguard Worker       memcpy(header.buffer, buf, emulation_bytes_start);
1402*61046927SAndroid Build Coastguard Worker       for (uint32_t i = emulation_bytes_start; i < size; i++) {
1403*61046927SAndroid Build Coastguard Worker          uint8_t byte = buf[i];
1404*61046927SAndroid Build Coastguard Worker          if (num_zeros >= 2 && byte >= 0x00 && byte <= 0x03) {
1405*61046927SAndroid Build Coastguard Worker             header.buffer[pos++] = 0x03;
1406*61046927SAndroid Build Coastguard Worker             num_zeros = 0;
1407*61046927SAndroid Build Coastguard Worker          }
1408*61046927SAndroid Build Coastguard Worker          header.buffer[pos++] = byte;
1409*61046927SAndroid Build Coastguard Worker          num_zeros = byte == 0x00 ? num_zeros + 1 : 0;
1410*61046927SAndroid Build Coastguard Worker       }
1411*61046927SAndroid Build Coastguard Worker       header.size = pos;
1412*61046927SAndroid Build Coastguard Worker    } else {
1413*61046927SAndroid Build Coastguard Worker       header.size = size;
1414*61046927SAndroid Build Coastguard Worker       header.buffer = MALLOC(header.size);
1415*61046927SAndroid Build Coastguard Worker       memcpy(header.buffer, buf, size);
1416*61046927SAndroid Build Coastguard Worker    }
1417*61046927SAndroid Build Coastguard Worker    util_dynarray_append(headers, struct pipe_enc_raw_header, header);
1418*61046927SAndroid Build Coastguard Worker }
1419