• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /**************************************************************************
2  *
3  * Copyright 2013 Advanced Micro Devices, Inc.
4  * All Rights Reserved.
5  *
6  * Permission is hereby granted, free of charge, to any person obtaining a
7  * copy of this software and associated documentation files (the
8  * "Software"), to deal in the Software without restriction, including
9  * without limitation the rights to use, copy, modify, merge, publish,
10  * distribute, sub license, and/or sell copies of the Software, and to
11  * permit persons to whom the Software is furnished to do so, subject to
12  * the following conditions:
13  *
14  * The above copyright notice and this permission notice (including the
15  * next paragraph) shall be included in all copies or substantial portions
16  * of the Software.
17  *
18  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
19  * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
20  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT.
21  * IN NO EVENT SHALL THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR
22  * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
23  * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
24  * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
25  *
26  **************************************************************************/
27 
28 #include "vid_enc_common.h"
29 
30 #include "vl/vl_video_buffer.h"
31 #include "tgsi/tgsi_text.h"
32 
enc_ReleaseTasks(struct list_head * head)33 void enc_ReleaseTasks(struct list_head *head)
34 {
35    struct encode_task *i, *next;
36 
37    if (!head || !list_is_linked(head))
38       return;
39 
40    LIST_FOR_EACH_ENTRY_SAFE(i, next, head, list) {
41       pipe_resource_reference(&i->bitstream, NULL);
42       i->buf->destroy(i->buf);
43       FREE(i);
44    }
45 }
46 
enc_MoveTasks(struct list_head * from,struct list_head * to)47 void enc_MoveTasks(struct list_head *from, struct list_head *to)
48 {
49    to->prev->next = from->next;
50    from->next->prev = to->prev;
51    from->prev->next = to;
52    to->prev = from->prev;
53    list_inithead(from);
54 }
55 
enc_GetPictureParamPreset(struct pipe_h264_enc_picture_desc * picture)56 static void enc_GetPictureParamPreset(struct pipe_h264_enc_picture_desc *picture)
57 {
58    picture->motion_est.enc_disable_sub_mode = 0x000000fe;
59    picture->motion_est.enc_ime2_search_range_x = 0x00000001;
60    picture->motion_est.enc_ime2_search_range_y = 0x00000001;
61    picture->pic_ctrl.enc_constraint_set_flags = 0x00000040;
62 }
63 
enc_TranslateOMXProfileToPipe(unsigned omx_profile)64 enum pipe_video_profile enc_TranslateOMXProfileToPipe(unsigned omx_profile)
65 {
66    switch (omx_profile) {
67    case OMX_VIDEO_AVCProfileBaseline:
68       return PIPE_VIDEO_PROFILE_MPEG4_AVC_BASELINE;
69    case OMX_VIDEO_AVCProfileMain:
70       return PIPE_VIDEO_PROFILE_MPEG4_AVC_MAIN;
71    case OMX_VIDEO_AVCProfileExtended:
72       return PIPE_VIDEO_PROFILE_MPEG4_AVC_EXTENDED;
73    case OMX_VIDEO_AVCProfileHigh:
74       return PIPE_VIDEO_PROFILE_MPEG4_AVC_HIGH;
75    case OMX_VIDEO_AVCProfileHigh10:
76       return PIPE_VIDEO_PROFILE_MPEG4_AVC_HIGH10;
77    case OMX_VIDEO_AVCProfileHigh422:
78       return PIPE_VIDEO_PROFILE_MPEG4_AVC_HIGH422;
79    case OMX_VIDEO_AVCProfileHigh444:
80       return PIPE_VIDEO_PROFILE_MPEG4_AVC_HIGH444;
81    default:
82       return PIPE_VIDEO_PROFILE_UNKNOWN;
83    }
84 }
85 
enc_TranslateOMXLevelToPipe(unsigned omx_level)86 unsigned enc_TranslateOMXLevelToPipe(unsigned omx_level)
87 {
88    switch (omx_level) {
89    case OMX_VIDEO_AVCLevel1:
90    case OMX_VIDEO_AVCLevel1b:
91       return 10;
92    case OMX_VIDEO_AVCLevel11:
93       return 11;
94    case OMX_VIDEO_AVCLevel12:
95       return 12;
96    case OMX_VIDEO_AVCLevel13:
97       return 13;
98    case OMX_VIDEO_AVCLevel2:
99       return 20;
100    case OMX_VIDEO_AVCLevel21:
101       return 21;
102    case OMX_VIDEO_AVCLevel22:
103       return 22;
104    case OMX_VIDEO_AVCLevel3:
105       return 30;
106    case OMX_VIDEO_AVCLevel31:
107       return 31;
108    case OMX_VIDEO_AVCLevel32:
109       return 32;
110    case OMX_VIDEO_AVCLevel4:
111       return 40;
112    case OMX_VIDEO_AVCLevel41:
113       return 41;
114    default:
115    case OMX_VIDEO_AVCLevel42:
116       return 42;
117    case OMX_VIDEO_AVCLevel5:
118       return 50;
119    case OMX_VIDEO_AVCLevel51:
120       return 51;
121    }
122 }
123 
vid_enc_BufferEncoded_common(vid_enc_PrivateType * priv,OMX_BUFFERHEADERTYPE * input,OMX_BUFFERHEADERTYPE * output)124 void vid_enc_BufferEncoded_common(vid_enc_PrivateType * priv, OMX_BUFFERHEADERTYPE* input, OMX_BUFFERHEADERTYPE* output)
125 {
126    struct output_buf_private *outp = output->pOutputPortPrivate;
127    struct input_buf_private *inp = input->pInputPortPrivate;
128    struct encode_task *task;
129    struct pipe_box box = {};
130    unsigned size;
131 
132 #if ENABLE_ST_OMX_BELLAGIO
133    if (!inp || list_is_empty(&inp->tasks)) {
134       input->nFilledLen = 0; /* mark buffer as empty */
135       enc_MoveTasks(&priv->used_tasks, &inp->tasks);
136       return;
137    }
138 #endif
139 
140    task = list_entry(inp->tasks.next, struct encode_task, list);
141    list_del(&task->list);
142    list_addtail(&task->list, &priv->used_tasks);
143 
144    if (!task->bitstream)
145       return;
146 
147    /* ------------- map result buffer ----------------- */
148 
149    if (outp->transfer)
150       pipe_buffer_unmap(priv->t_pipe, outp->transfer);
151 
152    pipe_resource_reference(&outp->bitstream, task->bitstream);
153    pipe_resource_reference(&task->bitstream, NULL);
154 
155    box.width = outp->bitstream->width0;
156    box.height = outp->bitstream->height0;
157    box.depth = outp->bitstream->depth0;
158 
159    output->pBuffer = priv->t_pipe->buffer_map(priv->t_pipe, outp->bitstream, 0,
160                                                 PIPE_MAP_READ_WRITE,
161                                                 &box, &outp->transfer);
162 
163    /* ------------- get size of result ----------------- */
164 
165    priv->codec->get_feedback(priv->codec, task->feedback, &size);
166 
167    output->nOffset = 0;
168    output->nFilledLen = size; /* mark buffer as full */
169 
170    /* all output buffers contain exactly one frame */
171    output->nFlags = OMX_BUFFERFLAG_ENDOFFRAME;
172 
173 #if ENABLE_ST_OMX_TIZONIA
174    input->nFilledLen = 0; /* mark buffer as empty */
175    enc_MoveTasks(&priv->used_tasks, &inp->tasks);
176 #endif
177 }
178 
179 
enc_NeedTask_common(vid_enc_PrivateType * priv,OMX_VIDEO_PORTDEFINITIONTYPE * def)180 struct encode_task *enc_NeedTask_common(vid_enc_PrivateType * priv, OMX_VIDEO_PORTDEFINITIONTYPE *def)
181 {
182    struct pipe_video_buffer templat = {};
183    struct encode_task *task;
184 
185    if (!list_is_empty(&priv->free_tasks)) {
186       task = list_entry(priv->free_tasks.next, struct encode_task, list);
187       list_del(&task->list);
188       return task;
189    }
190 
191    /* allocate a new one */
192    task = CALLOC_STRUCT(encode_task);
193    if (!task)
194       return NULL;
195 
196    templat.buffer_format = PIPE_FORMAT_NV12;
197    templat.width = def->nFrameWidth;
198    templat.height = def->nFrameHeight;
199    templat.interlaced = false;
200 
201    task->buf = priv->s_pipe->create_video_buffer(priv->s_pipe, &templat);
202    if (!task->buf) {
203       FREE(task);
204       return NULL;
205    }
206 
207    return task;
208 }
209 
enc_ScaleInput_common(vid_enc_PrivateType * priv,OMX_VIDEO_PORTDEFINITIONTYPE * def,struct pipe_video_buffer ** vbuf,unsigned * size)210 void enc_ScaleInput_common(vid_enc_PrivateType * priv, OMX_VIDEO_PORTDEFINITIONTYPE *def,
211                                   struct pipe_video_buffer **vbuf, unsigned *size)
212 {
213    struct pipe_video_buffer *src_buf = *vbuf;
214    struct vl_compositor *compositor = &priv->compositor;
215    struct vl_compositor_state *s = &priv->cstate;
216    struct pipe_sampler_view **views;
217    struct pipe_surface **dst_surface;
218    unsigned i;
219 
220    if (!priv->scale_buffer[priv->current_scale_buffer])
221       return;
222 
223    views = src_buf->get_sampler_view_planes(src_buf);
224    dst_surface = priv->scale_buffer[priv->current_scale_buffer]->get_surfaces
225                  (priv->scale_buffer[priv->current_scale_buffer]);
226    vl_compositor_clear_layers(s);
227 
228    for (i = 0; i < VL_MAX_SURFACES; ++i) {
229       struct u_rect src_rect;
230       if (!views[i] || !dst_surface[i])
231          continue;
232       src_rect.x0 = 0;
233       src_rect.y0 = 0;
234       src_rect.x1 = def->nFrameWidth;
235       src_rect.y1 = def->nFrameHeight;
236       if (i > 0) {
237          src_rect.x1 /= 2;
238          src_rect.y1 /= 2;
239       }
240       vl_compositor_set_rgba_layer(s, compositor, 0, views[i], &src_rect, NULL, NULL);
241       vl_compositor_render(s, compositor, dst_surface[i], NULL, false);
242    }
243    *size  = priv->scale.xWidth * priv->scale.xHeight * 2;
244    *vbuf = priv->scale_buffer[priv->current_scale_buffer++];
245    priv->current_scale_buffer %= OMX_VID_ENC_NUM_SCALING_BUFFERS;
246 }
247 
enc_ControlPicture_common(vid_enc_PrivateType * priv,struct pipe_h264_enc_picture_desc * picture)248 void enc_ControlPicture_common(vid_enc_PrivateType * priv, struct pipe_h264_enc_picture_desc *picture)
249 {
250    struct pipe_h264_enc_rate_control *rate_ctrl = &picture->rate_ctrl[0];
251 
252    /* Get bitrate from port */
253    switch (priv->bitrate.eControlRate) {
254    case OMX_Video_ControlRateVariable:
255       rate_ctrl->rate_ctrl_method = PIPE_H2645_ENC_RATE_CONTROL_METHOD_VARIABLE;
256       break;
257    case OMX_Video_ControlRateConstant:
258       rate_ctrl->rate_ctrl_method = PIPE_H2645_ENC_RATE_CONTROL_METHOD_CONSTANT;
259       break;
260    case OMX_Video_ControlRateVariableSkipFrames:
261       rate_ctrl->rate_ctrl_method = PIPE_H2645_ENC_RATE_CONTROL_METHOD_VARIABLE_SKIP;
262       break;
263    case OMX_Video_ControlRateConstantSkipFrames:
264       rate_ctrl->rate_ctrl_method = PIPE_H2645_ENC_RATE_CONTROL_METHOD_CONSTANT_SKIP;
265       break;
266    default:
267       rate_ctrl->rate_ctrl_method = PIPE_H2645_ENC_RATE_CONTROL_METHOD_DISABLE;
268       break;
269    }
270 
271    rate_ctrl->frame_rate_den = OMX_VID_ENC_CONTROL_FRAME_RATE_DEN_DEFAULT;
272    rate_ctrl->frame_rate_num = ((priv->frame_rate) >> 16) * rate_ctrl->frame_rate_den;
273 
274    if (rate_ctrl->rate_ctrl_method != PIPE_H2645_ENC_RATE_CONTROL_METHOD_DISABLE) {
275       if (priv->bitrate.nTargetBitrate < OMX_VID_ENC_BITRATE_MIN)
276          rate_ctrl->target_bitrate = OMX_VID_ENC_BITRATE_MIN;
277       else if (priv->bitrate.nTargetBitrate < OMX_VID_ENC_BITRATE_MAX)
278          rate_ctrl->target_bitrate = priv->bitrate.nTargetBitrate;
279       else
280          rate_ctrl->target_bitrate = OMX_VID_ENC_BITRATE_MAX;
281       rate_ctrl->peak_bitrate = rate_ctrl->target_bitrate;
282       if (rate_ctrl->target_bitrate < OMX_VID_ENC_BITRATE_MEDIAN)
283          rate_ctrl->vbv_buffer_size = MIN2((rate_ctrl->target_bitrate * 2.75), OMX_VID_ENC_BITRATE_MEDIAN);
284       else
285          rate_ctrl->vbv_buffer_size = rate_ctrl->target_bitrate;
286 
287       if (rate_ctrl->frame_rate_num) {
288          unsigned long long t = rate_ctrl->target_bitrate;
289          t *= rate_ctrl->frame_rate_den;
290          rate_ctrl->target_bits_picture = t / rate_ctrl->frame_rate_num;
291       } else {
292          rate_ctrl->target_bits_picture = rate_ctrl->target_bitrate;
293       }
294       rate_ctrl->peak_bits_picture_integer = rate_ctrl->target_bits_picture;
295       rate_ctrl->peak_bits_picture_fraction = 0;
296    }
297 
298    picture->quant_i_frames = priv->quant.nQpI;
299    picture->quant_p_frames = priv->quant.nQpP;
300    picture->quant_b_frames = priv->quant.nQpB;
301 
302    picture->frame_num = priv->frame_num;
303    picture->num_ref_idx_l0_active_minus1 = 0;
304    picture->ref_idx_l0_list[0] = priv->ref_idx_l0;
305    picture->num_ref_idx_l1_active_minus1 = 0;
306    picture->ref_idx_l1_list[0] = priv->ref_idx_l1;
307    picture->enable_vui = (picture->rate_ctrl[0].frame_rate_num != 0);
308    enc_GetPictureParamPreset(picture);
309 }
310 
create_compute_state(struct pipe_context * pipe,const char * source)311 static void *create_compute_state(struct pipe_context *pipe,
312                                   const char *source)
313 {
314    struct tgsi_token tokens[1024];
315    struct pipe_compute_state state = {0};
316 
317    if (!tgsi_text_translate(source, tokens, ARRAY_SIZE(tokens))) {
318            assert(false);
319            return NULL;
320    }
321 
322    state.ir_type = PIPE_SHADER_IR_TGSI;
323    state.prog = tokens;
324 
325    return pipe->create_compute_state(pipe, &state);
326 }
327 
enc_InitCompute_common(vid_enc_PrivateType * priv)328 void enc_InitCompute_common(vid_enc_PrivateType *priv)
329 {
330    struct pipe_context *pipe = priv->s_pipe;
331    struct pipe_screen *screen = pipe->screen;
332 
333    /* We need the partial last block support. */
334    if (!screen->get_param(screen, PIPE_CAP_COMPUTE_GRID_INFO_LAST_BLOCK))
335       return;
336 
337    static const char *copy_y =
338          "COMP\n"
339          "PROPERTY CS_FIXED_BLOCK_WIDTH 64\n"
340          "PROPERTY CS_FIXED_BLOCK_HEIGHT 1\n"
341          "PROPERTY CS_FIXED_BLOCK_DEPTH 1\n"
342          "DCL SV[0], THREAD_ID\n"
343          "DCL SV[1], BLOCK_ID\n"
344          "DCL IMAGE[0], 2D, PIPE_FORMAT_R8_UINT\n"
345          "DCL IMAGE[1], 2D, PIPE_FORMAT_R8_UINT, WR\n"
346          "DCL TEMP[0..1]\n"
347          "IMM[0] UINT32 {64, 0, 0, 0}\n"
348 
349          "UMAD TEMP[0].x, SV[1], IMM[0], SV[0]\n"
350          "MOV TEMP[0].y, SV[1]\n"
351          "LOAD TEMP[1].x, IMAGE[0], TEMP[0], 2D, PIPE_FORMAT_R8_UINT\n"
352          "STORE IMAGE[1].x, TEMP[0], TEMP[1], 2D, PIPE_FORMAT_R8_UINT\n"
353          "END\n";
354 
355    static const char *copy_uv =
356          "COMP\n"
357          "PROPERTY CS_FIXED_BLOCK_WIDTH 64\n"
358          "PROPERTY CS_FIXED_BLOCK_HEIGHT 1\n"
359          "PROPERTY CS_FIXED_BLOCK_DEPTH 1\n"
360          "DCL SV[0], THREAD_ID\n"
361          "DCL SV[1], BLOCK_ID\n"
362          "DCL IMAGE[0], 2D, PIPE_FORMAT_R8_UINT\n"
363          "DCL IMAGE[2], 2D, PIPE_FORMAT_R8G8_UINT, WR\n"
364          "DCL CONST[0][0]\n" /* .x = offset of the UV portion in the y direction */
365          "DCL TEMP[0..4]\n"
366          "IMM[0] UINT32 {64, 0, 2, 1}\n"
367          /* Destination R8G8 coordinates */
368          "UMAD TEMP[0].x, SV[1], IMM[0], SV[0]\n"
369          "MOV TEMP[0].y, SV[1]\n"
370          /* Source R8 coordinates of U */
371          "UMUL TEMP[1].x, TEMP[0], IMM[0].zzzz\n"
372          "UADD TEMP[1].y, TEMP[0], CONST[0].xxxx\n"
373          /* Source R8 coordinates of V */
374          "UADD TEMP[2].x, TEMP[1], IMM[0].wwww\n"
375          "MOV TEMP[2].y, TEMP[1]\n"
376 
377          "LOAD TEMP[3].x, IMAGE[0], TEMP[1], 2D, PIPE_FORMAT_R8_UINT\n"
378          "LOAD TEMP[4].x, IMAGE[0], TEMP[2], 2D, PIPE_FORMAT_R8_UINT\n"
379          "MOV TEMP[3].y, TEMP[4].xxxx\n"
380          "STORE IMAGE[2], TEMP[0], TEMP[3], 2D, PIPE_FORMAT_R8G8_UINT\n"
381          "END\n";
382 
383    priv->copy_y_shader = create_compute_state(pipe, copy_y);
384    priv->copy_uv_shader = create_compute_state(pipe, copy_uv);
385 }
386 
enc_ReleaseCompute_common(vid_enc_PrivateType * priv)387 void enc_ReleaseCompute_common(vid_enc_PrivateType *priv)
388 {
389    struct pipe_context *pipe = priv->s_pipe;
390 
391    if (priv->copy_y_shader)
392       pipe->delete_compute_state(pipe, priv->copy_y_shader);
393    if (priv->copy_uv_shader)
394       pipe->delete_compute_state(pipe, priv->copy_uv_shader);
395 }
396 
enc_LoadImage_common(vid_enc_PrivateType * priv,OMX_VIDEO_PORTDEFINITIONTYPE * def,OMX_BUFFERHEADERTYPE * buf,struct pipe_video_buffer * vbuf)397 OMX_ERRORTYPE enc_LoadImage_common(vid_enc_PrivateType * priv, OMX_VIDEO_PORTDEFINITIONTYPE *def,
398                                    OMX_BUFFERHEADERTYPE *buf,
399                                    struct pipe_video_buffer *vbuf)
400 {
401    struct pipe_context *pipe = priv->s_pipe;
402    struct pipe_box box = {};
403    struct input_buf_private *inp = buf->pInputPortPrivate;
404 
405    if (!inp->resource) {
406       struct pipe_sampler_view **views;
407       void *ptr;
408 
409       views = vbuf->get_sampler_view_planes(vbuf);
410       if (!views)
411          return OMX_ErrorInsufficientResources;
412 
413       ptr = buf->pBuffer;
414       box.width = def->nFrameWidth;
415       box.height = def->nFrameHeight;
416       box.depth = 1;
417       pipe->texture_subdata(pipe, views[0]->texture, 0,
418                             PIPE_MAP_WRITE, &box,
419                             ptr, def->nStride, 0);
420       ptr = ((uint8_t*)buf->pBuffer) + (def->nStride * box.height);
421       box.width = def->nFrameWidth / 2;
422       box.height = def->nFrameHeight / 2;
423       box.depth = 1;
424       pipe->texture_subdata(pipe, views[1]->texture, 0,
425                             PIPE_MAP_WRITE, &box,
426                             ptr, def->nStride, 0);
427    } else {
428       struct vl_video_buffer *dst_buf = (struct vl_video_buffer *)vbuf;
429 
430       pipe_texture_unmap(pipe, inp->transfer);
431 
432       /* inp->resource uses PIPE_FORMAT_I8 and the layout looks like this:
433        *
434        * def->nFrameWidth = 4, def->nFrameHeight = 4:
435        * |----|
436        * |YYYY|
437        * |YYYY|
438        * |YYYY|
439        * |YYYY|
440        * |UVUV|
441        * |UVUV|
442        * |----|
443        *
444        * The copy has 2 steps:
445        * - Copy Y to dst_buf->resources[0] as R8.
446        * - Copy UV to dst_buf->resources[1] as R8G8.
447        */
448       if (priv->copy_y_shader && priv->copy_uv_shader) {
449          /* Compute path */
450          /* Set shader images for both copies. */
451          struct pipe_image_view image[3] = {0};
452          image[0].resource = inp->resource;
453          image[0].shader_access = image[0].access = PIPE_IMAGE_ACCESS_READ;
454          image[0].format = PIPE_FORMAT_R8_UINT;
455 
456          image[1].resource = dst_buf->resources[0];
457          image[1].shader_access = image[1].access = PIPE_IMAGE_ACCESS_WRITE;
458          image[1].format = PIPE_FORMAT_R8_UINT;
459 
460          image[2].resource = dst_buf->resources[1];
461          image[2].shader_access = image[1].access = PIPE_IMAGE_ACCESS_WRITE;
462          image[2].format = PIPE_FORMAT_R8G8_UINT;
463 
464          pipe->set_shader_images(pipe, PIPE_SHADER_COMPUTE, 0, 3, 0, image);
465 
466          /* Set the constant buffer. */
467          uint32_t constants[4] = {def->nFrameHeight};
468          struct pipe_constant_buffer cb = {};
469 
470          cb.buffer_size = sizeof(constants);
471          cb.user_buffer = constants;
472          pipe->set_constant_buffer(pipe, PIPE_SHADER_COMPUTE, 0, false, &cb);
473 
474          /* Use the optimal block size for the linear image layout. */
475          struct pipe_grid_info info = {};
476          info.block[0] = 64;
477          info.block[1] = 1;
478          info.block[2] = 1;
479          info.grid[2] = 1;
480 
481          /* Copy Y */
482          pipe->bind_compute_state(pipe, priv->copy_y_shader);
483 
484          info.grid[0] = DIV_ROUND_UP(def->nFrameWidth, 64);
485          info.grid[1] = def->nFrameHeight;
486          info.last_block[0] = def->nFrameWidth % 64;
487          pipe->launch_grid(pipe, &info);
488 
489          /* Copy UV */
490          pipe->bind_compute_state(pipe, priv->copy_uv_shader);
491 
492          info.grid[0] = DIV_ROUND_UP(def->nFrameWidth / 2, 64);
493          info.grid[1] = def->nFrameHeight / 2;
494          info.last_block[0] = (def->nFrameWidth / 2) % 64;
495          pipe->launch_grid(pipe, &info);
496 
497          /* Make the result visible to all clients. */
498          pipe->memory_barrier(pipe, PIPE_BARRIER_ALL);
499 
500          /* Unbind. */
501          pipe->set_shader_images(pipe, PIPE_SHADER_COMPUTE, 0, 0, 3, NULL);
502          pipe->set_constant_buffer(pipe, PIPE_SHADER_COMPUTE, 0, false, NULL);
503          pipe->bind_compute_state(pipe, NULL);
504       } else {
505          /* Graphics path */
506          struct pipe_blit_info blit;
507 
508          box.width = def->nFrameWidth;
509          box.height = def->nFrameHeight;
510          box.depth = 1;
511 
512          /* Copy Y */
513          pipe->resource_copy_region(pipe,
514                                     dst_buf->resources[0],
515                                     0, 0, 0, 0, inp->resource, 0, &box);
516 
517          /* Copy U */
518          memset(&blit, 0, sizeof(blit));
519          blit.src.resource = inp->resource;
520          blit.src.format = inp->resource->format;
521 
522          blit.src.box.x = -1;
523          blit.src.box.y = def->nFrameHeight;
524          blit.src.box.width = def->nFrameWidth;
525          blit.src.box.height = def->nFrameHeight / 2 ;
526          blit.src.box.depth = 1;
527 
528          blit.dst.resource = dst_buf->resources[1];
529          blit.dst.format = blit.dst.resource->format;
530 
531          blit.dst.box.width = def->nFrameWidth / 2;
532          blit.dst.box.height = def->nFrameHeight / 2;
533          blit.dst.box.depth = 1;
534          blit.filter = PIPE_TEX_FILTER_NEAREST;
535 
536          blit.mask = PIPE_MASK_R;
537          pipe->blit(pipe, &blit);
538 
539          /* Copy V */
540          blit.src.box.x = 0;
541          blit.mask = PIPE_MASK_G;
542          pipe->blit(pipe, &blit);
543       }
544 
545       pipe->flush(pipe, NULL, 0);
546 
547       box.width = inp->resource->width0;
548       box.height = inp->resource->height0;
549       box.depth = inp->resource->depth0;
550       buf->pBuffer = pipe->texture_map(pipe, inp->resource, 0,
551                                         PIPE_MAP_WRITE, &box,
552                                         &inp->transfer);
553    }
554 
555    return OMX_ErrorNone;
556 }
557