1*0934d375SYunfei Dong // SPDX-License-Identifier: GPL-2.0
2*0934d375SYunfei Dong /*
3*0934d375SYunfei Dong * Copyright (c) 2021 MediaTek Inc.
4*0934d375SYunfei Dong * Author: Yunfei Dong <[email protected]>
5*0934d375SYunfei Dong */
6*0934d375SYunfei Dong
7*0934d375SYunfei Dong #include <linux/freezer.h>
8*0934d375SYunfei Dong #include <linux/interrupt.h>
9*0934d375SYunfei Dong #include <linux/kthread.h>
10*0934d375SYunfei Dong
11*0934d375SYunfei Dong #include "mtk_vcodec_dec_drv.h"
12*0934d375SYunfei Dong #include "mtk_vcodec_dec_pm.h"
13*0934d375SYunfei Dong #include "vdec_msg_queue.h"
14*0934d375SYunfei Dong
15*0934d375SYunfei Dong #define VDEC_MSG_QUEUE_TIMEOUT_MS 1500
16*0934d375SYunfei Dong
17*0934d375SYunfei Dong /* the size used to store lat slice header information */
18*0934d375SYunfei Dong #define VDEC_LAT_SLICE_HEADER_SZ (640 * SZ_1K)
19*0934d375SYunfei Dong
20*0934d375SYunfei Dong /* the size used to store avc error information */
21*0934d375SYunfei Dong #define VDEC_ERR_MAP_SZ_AVC (17 * SZ_1K)
22*0934d375SYunfei Dong
23*0934d375SYunfei Dong #define VDEC_RD_MV_BUFFER_SZ (((SZ_4K * 2304 >> 4) + SZ_1K) << 1)
24*0934d375SYunfei Dong #define VDEC_LAT_TILE_SZ (64 * V4L2_AV1_MAX_TILE_COUNT)
25*0934d375SYunfei Dong
26*0934d375SYunfei Dong /* core will read the trans buffer which decoded by lat to decode again.
27*0934d375SYunfei Dong * The trans buffer size of FHD and 4K bitstreams are different.
28*0934d375SYunfei Dong */
vde_msg_queue_get_trans_size(int width,int height)29*0934d375SYunfei Dong static int vde_msg_queue_get_trans_size(int width, int height)
30*0934d375SYunfei Dong {
31*0934d375SYunfei Dong if (width > 1920 || height > 1088)
32*0934d375SYunfei Dong return 30 * SZ_1M;
33*0934d375SYunfei Dong else
34*0934d375SYunfei Dong return 6 * SZ_1M;
35*0934d375SYunfei Dong }
36*0934d375SYunfei Dong
vdec_msg_queue_init_ctx(struct vdec_msg_queue_ctx * ctx,int hardware_index)37*0934d375SYunfei Dong void vdec_msg_queue_init_ctx(struct vdec_msg_queue_ctx *ctx, int hardware_index)
38*0934d375SYunfei Dong {
39*0934d375SYunfei Dong init_waitqueue_head(&ctx->ready_to_use);
40*0934d375SYunfei Dong INIT_LIST_HEAD(&ctx->ready_queue);
41*0934d375SYunfei Dong spin_lock_init(&ctx->ready_lock);
42*0934d375SYunfei Dong ctx->ready_num = 0;
43*0934d375SYunfei Dong ctx->hardware_index = hardware_index;
44*0934d375SYunfei Dong }
45*0934d375SYunfei Dong
vdec_get_buf_list(int hardware_index,struct vdec_lat_buf * buf)46*0934d375SYunfei Dong static struct list_head *vdec_get_buf_list(int hardware_index, struct vdec_lat_buf *buf)
47*0934d375SYunfei Dong {
48*0934d375SYunfei Dong switch (hardware_index) {
49*0934d375SYunfei Dong case MTK_VDEC_CORE:
50*0934d375SYunfei Dong return &buf->core_list;
51*0934d375SYunfei Dong case MTK_VDEC_LAT0:
52*0934d375SYunfei Dong return &buf->lat_list;
53*0934d375SYunfei Dong default:
54*0934d375SYunfei Dong return NULL;
55*0934d375SYunfei Dong }
56*0934d375SYunfei Dong }
57*0934d375SYunfei Dong
vdec_msg_queue_inc(struct vdec_msg_queue * msg_queue,int hardware_index)58*0934d375SYunfei Dong static void vdec_msg_queue_inc(struct vdec_msg_queue *msg_queue, int hardware_index)
59*0934d375SYunfei Dong {
60*0934d375SYunfei Dong if (hardware_index == MTK_VDEC_CORE)
61*0934d375SYunfei Dong atomic_inc(&msg_queue->core_list_cnt);
62*0934d375SYunfei Dong else
63*0934d375SYunfei Dong atomic_inc(&msg_queue->lat_list_cnt);
64*0934d375SYunfei Dong }
65*0934d375SYunfei Dong
vdec_msg_queue_dec(struct vdec_msg_queue * msg_queue,int hardware_index)66*0934d375SYunfei Dong static void vdec_msg_queue_dec(struct vdec_msg_queue *msg_queue, int hardware_index)
67*0934d375SYunfei Dong {
68*0934d375SYunfei Dong if (hardware_index == MTK_VDEC_CORE)
69*0934d375SYunfei Dong atomic_dec(&msg_queue->core_list_cnt);
70*0934d375SYunfei Dong else
71*0934d375SYunfei Dong atomic_dec(&msg_queue->lat_list_cnt);
72*0934d375SYunfei Dong }
73*0934d375SYunfei Dong
vdec_msg_queue_qbuf(struct vdec_msg_queue_ctx * msg_ctx,struct vdec_lat_buf * buf)74*0934d375SYunfei Dong int vdec_msg_queue_qbuf(struct vdec_msg_queue_ctx *msg_ctx, struct vdec_lat_buf *buf)
75*0934d375SYunfei Dong {
76*0934d375SYunfei Dong struct list_head *head;
77*0934d375SYunfei Dong
78*0934d375SYunfei Dong head = vdec_get_buf_list(msg_ctx->hardware_index, buf);
79*0934d375SYunfei Dong if (!head) {
80*0934d375SYunfei Dong mtk_v4l2_vdec_err(buf->ctx, "fail to qbuf: %d", msg_ctx->hardware_index);
81*0934d375SYunfei Dong return -EINVAL;
82*0934d375SYunfei Dong }
83*0934d375SYunfei Dong
84*0934d375SYunfei Dong spin_lock(&msg_ctx->ready_lock);
85*0934d375SYunfei Dong list_add_tail(head, &msg_ctx->ready_queue);
86*0934d375SYunfei Dong msg_ctx->ready_num++;
87*0934d375SYunfei Dong
88*0934d375SYunfei Dong vdec_msg_queue_inc(&buf->ctx->msg_queue, msg_ctx->hardware_index);
89*0934d375SYunfei Dong if (msg_ctx->hardware_index != MTK_VDEC_CORE) {
90*0934d375SYunfei Dong wake_up_all(&msg_ctx->ready_to_use);
91*0934d375SYunfei Dong } else {
92*0934d375SYunfei Dong if (!(buf->ctx->msg_queue.status & CONTEXT_LIST_QUEUED)) {
93*0934d375SYunfei Dong queue_work(buf->ctx->dev->core_workqueue, &buf->ctx->msg_queue.core_work);
94*0934d375SYunfei Dong buf->ctx->msg_queue.status |= CONTEXT_LIST_QUEUED;
95*0934d375SYunfei Dong }
96*0934d375SYunfei Dong }
97*0934d375SYunfei Dong
98*0934d375SYunfei Dong mtk_v4l2_vdec_dbg(3, buf->ctx, "enqueue buf type: %d addr: 0x%p num: %d",
99*0934d375SYunfei Dong msg_ctx->hardware_index, buf, msg_ctx->ready_num);
100*0934d375SYunfei Dong spin_unlock(&msg_ctx->ready_lock);
101*0934d375SYunfei Dong
102*0934d375SYunfei Dong return 0;
103*0934d375SYunfei Dong }
104*0934d375SYunfei Dong
vdec_msg_queue_wait_event(struct vdec_msg_queue_ctx * msg_ctx)105*0934d375SYunfei Dong static bool vdec_msg_queue_wait_event(struct vdec_msg_queue_ctx *msg_ctx)
106*0934d375SYunfei Dong {
107*0934d375SYunfei Dong int ret;
108*0934d375SYunfei Dong
109*0934d375SYunfei Dong ret = wait_event_timeout(msg_ctx->ready_to_use,
110*0934d375SYunfei Dong !list_empty(&msg_ctx->ready_queue),
111*0934d375SYunfei Dong msecs_to_jiffies(VDEC_MSG_QUEUE_TIMEOUT_MS));
112*0934d375SYunfei Dong if (!ret)
113*0934d375SYunfei Dong return false;
114*0934d375SYunfei Dong
115*0934d375SYunfei Dong return true;
116*0934d375SYunfei Dong }
117*0934d375SYunfei Dong
vdec_msg_queue_dqbuf(struct vdec_msg_queue_ctx * msg_ctx)118*0934d375SYunfei Dong struct vdec_lat_buf *vdec_msg_queue_dqbuf(struct vdec_msg_queue_ctx *msg_ctx)
119*0934d375SYunfei Dong {
120*0934d375SYunfei Dong struct vdec_lat_buf *buf;
121*0934d375SYunfei Dong struct list_head *head;
122*0934d375SYunfei Dong int ret;
123*0934d375SYunfei Dong
124*0934d375SYunfei Dong spin_lock(&msg_ctx->ready_lock);
125*0934d375SYunfei Dong if (list_empty(&msg_ctx->ready_queue)) {
126*0934d375SYunfei Dong spin_unlock(&msg_ctx->ready_lock);
127*0934d375SYunfei Dong
128*0934d375SYunfei Dong if (msg_ctx->hardware_index == MTK_VDEC_CORE)
129*0934d375SYunfei Dong return NULL;
130*0934d375SYunfei Dong
131*0934d375SYunfei Dong ret = vdec_msg_queue_wait_event(msg_ctx);
132*0934d375SYunfei Dong if (!ret)
133*0934d375SYunfei Dong return NULL;
134*0934d375SYunfei Dong spin_lock(&msg_ctx->ready_lock);
135*0934d375SYunfei Dong }
136*0934d375SYunfei Dong
137*0934d375SYunfei Dong if (msg_ctx->hardware_index == MTK_VDEC_CORE)
138*0934d375SYunfei Dong buf = list_first_entry(&msg_ctx->ready_queue,
139*0934d375SYunfei Dong struct vdec_lat_buf, core_list);
140*0934d375SYunfei Dong else
141*0934d375SYunfei Dong buf = list_first_entry(&msg_ctx->ready_queue,
142*0934d375SYunfei Dong struct vdec_lat_buf, lat_list);
143*0934d375SYunfei Dong
144*0934d375SYunfei Dong head = vdec_get_buf_list(msg_ctx->hardware_index, buf);
145*0934d375SYunfei Dong if (!head) {
146*0934d375SYunfei Dong spin_unlock(&msg_ctx->ready_lock);
147*0934d375SYunfei Dong mtk_v4l2_vdec_err(buf->ctx, "fail to dqbuf: %d", msg_ctx->hardware_index);
148*0934d375SYunfei Dong return NULL;
149*0934d375SYunfei Dong }
150*0934d375SYunfei Dong list_del(head);
151*0934d375SYunfei Dong vdec_msg_queue_dec(&buf->ctx->msg_queue, msg_ctx->hardware_index);
152*0934d375SYunfei Dong
153*0934d375SYunfei Dong msg_ctx->ready_num--;
154*0934d375SYunfei Dong mtk_v4l2_vdec_dbg(3, buf->ctx, "dqueue buf type:%d addr: 0x%p num: %d",
155*0934d375SYunfei Dong msg_ctx->hardware_index, buf, msg_ctx->ready_num);
156*0934d375SYunfei Dong spin_unlock(&msg_ctx->ready_lock);
157*0934d375SYunfei Dong
158*0934d375SYunfei Dong return buf;
159*0934d375SYunfei Dong }
160*0934d375SYunfei Dong
vdec_msg_queue_update_ube_rptr(struct vdec_msg_queue * msg_queue,uint64_t ube_rptr)161*0934d375SYunfei Dong void vdec_msg_queue_update_ube_rptr(struct vdec_msg_queue *msg_queue, uint64_t ube_rptr)
162*0934d375SYunfei Dong {
163*0934d375SYunfei Dong spin_lock(&msg_queue->lat_ctx.ready_lock);
164*0934d375SYunfei Dong msg_queue->wdma_rptr_addr = ube_rptr;
165*0934d375SYunfei Dong mtk_v4l2_vdec_dbg(3, msg_queue->ctx, "update ube rprt (0x%llx)", ube_rptr);
166*0934d375SYunfei Dong spin_unlock(&msg_queue->lat_ctx.ready_lock);
167*0934d375SYunfei Dong }
168*0934d375SYunfei Dong
vdec_msg_queue_update_ube_wptr(struct vdec_msg_queue * msg_queue,uint64_t ube_wptr)169*0934d375SYunfei Dong void vdec_msg_queue_update_ube_wptr(struct vdec_msg_queue *msg_queue, uint64_t ube_wptr)
170*0934d375SYunfei Dong {
171*0934d375SYunfei Dong spin_lock(&msg_queue->lat_ctx.ready_lock);
172*0934d375SYunfei Dong msg_queue->wdma_wptr_addr = ube_wptr;
173*0934d375SYunfei Dong mtk_v4l2_vdec_dbg(3, msg_queue->ctx, "update ube wprt: (0x%llx 0x%llx) offset: 0x%llx",
174*0934d375SYunfei Dong msg_queue->wdma_rptr_addr, msg_queue->wdma_wptr_addr,
175*0934d375SYunfei Dong ube_wptr);
176*0934d375SYunfei Dong spin_unlock(&msg_queue->lat_ctx.ready_lock);
177*0934d375SYunfei Dong }
178*0934d375SYunfei Dong
vdec_msg_queue_wait_lat_buf_full(struct vdec_msg_queue * msg_queue)179*0934d375SYunfei Dong bool vdec_msg_queue_wait_lat_buf_full(struct vdec_msg_queue *msg_queue)
180*0934d375SYunfei Dong {
181*0934d375SYunfei Dong if (atomic_read(&msg_queue->lat_list_cnt) == NUM_BUFFER_COUNT) {
182*0934d375SYunfei Dong mtk_v4l2_vdec_dbg(3, msg_queue->ctx, "wait buf full: (%d %d) ready:%d status:%d",
183*0934d375SYunfei Dong atomic_read(&msg_queue->lat_list_cnt),
184*0934d375SYunfei Dong atomic_read(&msg_queue->core_list_cnt),
185*0934d375SYunfei Dong msg_queue->lat_ctx.ready_num, msg_queue->status);
186*0934d375SYunfei Dong return true;
187*0934d375SYunfei Dong }
188*0934d375SYunfei Dong
189*0934d375SYunfei Dong msg_queue->flush_done = false;
190*0934d375SYunfei Dong vdec_msg_queue_qbuf(&msg_queue->core_ctx, &msg_queue->empty_lat_buf);
191*0934d375SYunfei Dong wait_event(msg_queue->core_dec_done, msg_queue->flush_done);
192*0934d375SYunfei Dong
193*0934d375SYunfei Dong mtk_v4l2_vdec_dbg(3, msg_queue->ctx, "flush done => ready_num:%d status:%d list(%d %d)",
194*0934d375SYunfei Dong msg_queue->lat_ctx.ready_num, msg_queue->status,
195*0934d375SYunfei Dong atomic_read(&msg_queue->lat_list_cnt),
196*0934d375SYunfei Dong atomic_read(&msg_queue->core_list_cnt));
197*0934d375SYunfei Dong
198*0934d375SYunfei Dong return false;
199*0934d375SYunfei Dong }
200*0934d375SYunfei Dong
vdec_msg_queue_deinit(struct vdec_msg_queue * msg_queue,struct mtk_vcodec_dec_ctx * ctx)201*0934d375SYunfei Dong void vdec_msg_queue_deinit(struct vdec_msg_queue *msg_queue,
202*0934d375SYunfei Dong struct mtk_vcodec_dec_ctx *ctx)
203*0934d375SYunfei Dong {
204*0934d375SYunfei Dong struct vdec_lat_buf *lat_buf;
205*0934d375SYunfei Dong struct mtk_vcodec_mem *mem;
206*0934d375SYunfei Dong int i;
207*0934d375SYunfei Dong
208*0934d375SYunfei Dong mem = &msg_queue->wdma_addr;
209*0934d375SYunfei Dong if (mem->va)
210*0934d375SYunfei Dong mtk_vcodec_mem_free(ctx, mem);
211*0934d375SYunfei Dong for (i = 0; i < NUM_BUFFER_COUNT; i++) {
212*0934d375SYunfei Dong lat_buf = &msg_queue->lat_buf[i];
213*0934d375SYunfei Dong
214*0934d375SYunfei Dong mem = &lat_buf->wdma_err_addr;
215*0934d375SYunfei Dong if (mem->va)
216*0934d375SYunfei Dong mtk_vcodec_mem_free(ctx, mem);
217*0934d375SYunfei Dong
218*0934d375SYunfei Dong mem = &lat_buf->slice_bc_addr;
219*0934d375SYunfei Dong if (mem->va)
220*0934d375SYunfei Dong mtk_vcodec_mem_free(ctx, mem);
221*0934d375SYunfei Dong
222*0934d375SYunfei Dong mem = &lat_buf->rd_mv_addr;
223*0934d375SYunfei Dong if (mem->va)
224*0934d375SYunfei Dong mtk_vcodec_mem_free(ctx, mem);
225*0934d375SYunfei Dong
226*0934d375SYunfei Dong mem = &lat_buf->tile_addr;
227*0934d375SYunfei Dong if (mem->va)
228*0934d375SYunfei Dong mtk_vcodec_mem_free(ctx, mem);
229*0934d375SYunfei Dong
230*0934d375SYunfei Dong kfree(lat_buf->private_data);
231*0934d375SYunfei Dong lat_buf->private_data = NULL;
232*0934d375SYunfei Dong }
233*0934d375SYunfei Dong
234*0934d375SYunfei Dong if (msg_queue->wdma_addr.size)
235*0934d375SYunfei Dong cancel_work_sync(&msg_queue->core_work);
236*0934d375SYunfei Dong }
237*0934d375SYunfei Dong
vdec_msg_queue_core_work(struct work_struct * work)238*0934d375SYunfei Dong static void vdec_msg_queue_core_work(struct work_struct *work)
239*0934d375SYunfei Dong {
240*0934d375SYunfei Dong struct vdec_msg_queue *msg_queue =
241*0934d375SYunfei Dong container_of(work, struct vdec_msg_queue, core_work);
242*0934d375SYunfei Dong struct mtk_vcodec_dec_ctx *ctx =
243*0934d375SYunfei Dong container_of(msg_queue, struct mtk_vcodec_dec_ctx, msg_queue);
244*0934d375SYunfei Dong struct mtk_vcodec_dec_dev *dev = ctx->dev;
245*0934d375SYunfei Dong struct vdec_lat_buf *lat_buf;
246*0934d375SYunfei Dong
247*0934d375SYunfei Dong spin_lock(&msg_queue->core_ctx.ready_lock);
248*0934d375SYunfei Dong ctx->msg_queue.status &= ~CONTEXT_LIST_QUEUED;
249*0934d375SYunfei Dong spin_unlock(&msg_queue->core_ctx.ready_lock);
250*0934d375SYunfei Dong
251*0934d375SYunfei Dong lat_buf = vdec_msg_queue_dqbuf(&msg_queue->core_ctx);
252*0934d375SYunfei Dong if (!lat_buf)
253*0934d375SYunfei Dong return;
254*0934d375SYunfei Dong
255*0934d375SYunfei Dong if (lat_buf->is_last_frame) {
256*0934d375SYunfei Dong ctx->msg_queue.status = CONTEXT_LIST_DEC_DONE;
257*0934d375SYunfei Dong msg_queue->flush_done = true;
258*0934d375SYunfei Dong wake_up(&ctx->msg_queue.core_dec_done);
259*0934d375SYunfei Dong
260*0934d375SYunfei Dong return;
261*0934d375SYunfei Dong }
262*0934d375SYunfei Dong
263*0934d375SYunfei Dong ctx = lat_buf->ctx;
264*0934d375SYunfei Dong mtk_vcodec_dec_enable_hardware(ctx, MTK_VDEC_CORE);
265*0934d375SYunfei Dong mtk_vcodec_set_curr_ctx(dev, ctx, MTK_VDEC_CORE);
266*0934d375SYunfei Dong
267*0934d375SYunfei Dong lat_buf->core_decode(lat_buf);
268*0934d375SYunfei Dong
269*0934d375SYunfei Dong mtk_vcodec_set_curr_ctx(dev, NULL, MTK_VDEC_CORE);
270*0934d375SYunfei Dong mtk_vcodec_dec_disable_hardware(ctx, MTK_VDEC_CORE);
271*0934d375SYunfei Dong vdec_msg_queue_qbuf(&ctx->msg_queue.lat_ctx, lat_buf);
272*0934d375SYunfei Dong
273*0934d375SYunfei Dong if (!(ctx->msg_queue.status & CONTEXT_LIST_QUEUED) &&
274*0934d375SYunfei Dong atomic_read(&msg_queue->core_list_cnt)) {
275*0934d375SYunfei Dong spin_lock(&msg_queue->core_ctx.ready_lock);
276*0934d375SYunfei Dong ctx->msg_queue.status |= CONTEXT_LIST_QUEUED;
277*0934d375SYunfei Dong spin_unlock(&msg_queue->core_ctx.ready_lock);
278*0934d375SYunfei Dong queue_work(ctx->dev->core_workqueue, &msg_queue->core_work);
279*0934d375SYunfei Dong }
280*0934d375SYunfei Dong }
281*0934d375SYunfei Dong
vdec_msg_queue_init(struct vdec_msg_queue * msg_queue,struct mtk_vcodec_dec_ctx * ctx,core_decode_cb_t core_decode,int private_size)282*0934d375SYunfei Dong int vdec_msg_queue_init(struct vdec_msg_queue *msg_queue,
283*0934d375SYunfei Dong struct mtk_vcodec_dec_ctx *ctx, core_decode_cb_t core_decode,
284*0934d375SYunfei Dong int private_size)
285*0934d375SYunfei Dong {
286*0934d375SYunfei Dong struct vdec_lat_buf *lat_buf;
287*0934d375SYunfei Dong int i, err;
288*0934d375SYunfei Dong
289*0934d375SYunfei Dong /* already init msg queue */
290*0934d375SYunfei Dong if (msg_queue->wdma_addr.size)
291*0934d375SYunfei Dong return 0;
292*0934d375SYunfei Dong
293*0934d375SYunfei Dong vdec_msg_queue_init_ctx(&msg_queue->lat_ctx, MTK_VDEC_LAT0);
294*0934d375SYunfei Dong vdec_msg_queue_init_ctx(&msg_queue->core_ctx, MTK_VDEC_CORE);
295*0934d375SYunfei Dong INIT_WORK(&msg_queue->core_work, vdec_msg_queue_core_work);
296*0934d375SYunfei Dong
297*0934d375SYunfei Dong atomic_set(&msg_queue->lat_list_cnt, 0);
298*0934d375SYunfei Dong atomic_set(&msg_queue->core_list_cnt, 0);
299*0934d375SYunfei Dong init_waitqueue_head(&msg_queue->core_dec_done);
300*0934d375SYunfei Dong msg_queue->status = CONTEXT_LIST_EMPTY;
301*0934d375SYunfei Dong
302*0934d375SYunfei Dong msg_queue->wdma_addr.size =
303*0934d375SYunfei Dong vde_msg_queue_get_trans_size(ctx->picinfo.buf_w,
304*0934d375SYunfei Dong ctx->picinfo.buf_h);
305*0934d375SYunfei Dong err = mtk_vcodec_mem_alloc(ctx, &msg_queue->wdma_addr);
306*0934d375SYunfei Dong if (err) {
307*0934d375SYunfei Dong mtk_v4l2_vdec_err(ctx, "failed to allocate wdma_addr buf");
308*0934d375SYunfei Dong msg_queue->wdma_addr.size = 0;
309*0934d375SYunfei Dong return -ENOMEM;
310*0934d375SYunfei Dong }
311*0934d375SYunfei Dong msg_queue->wdma_rptr_addr = msg_queue->wdma_addr.dma_addr;
312*0934d375SYunfei Dong msg_queue->wdma_wptr_addr = msg_queue->wdma_addr.dma_addr;
313*0934d375SYunfei Dong
314*0934d375SYunfei Dong msg_queue->empty_lat_buf.ctx = ctx;
315*0934d375SYunfei Dong msg_queue->empty_lat_buf.core_decode = NULL;
316*0934d375SYunfei Dong msg_queue->empty_lat_buf.is_last_frame = true;
317*0934d375SYunfei Dong
318*0934d375SYunfei Dong msg_queue->ctx = ctx;
319*0934d375SYunfei Dong for (i = 0; i < NUM_BUFFER_COUNT; i++) {
320*0934d375SYunfei Dong lat_buf = &msg_queue->lat_buf[i];
321*0934d375SYunfei Dong
322*0934d375SYunfei Dong lat_buf->wdma_err_addr.size = VDEC_ERR_MAP_SZ_AVC;
323*0934d375SYunfei Dong err = mtk_vcodec_mem_alloc(ctx, &lat_buf->wdma_err_addr);
324*0934d375SYunfei Dong if (err) {
325*0934d375SYunfei Dong mtk_v4l2_vdec_err(ctx, "failed to allocate wdma_err_addr buf[%d]", i);
326*0934d375SYunfei Dong goto mem_alloc_err;
327*0934d375SYunfei Dong }
328*0934d375SYunfei Dong
329*0934d375SYunfei Dong lat_buf->slice_bc_addr.size = VDEC_LAT_SLICE_HEADER_SZ;
330*0934d375SYunfei Dong err = mtk_vcodec_mem_alloc(ctx, &lat_buf->slice_bc_addr);
331*0934d375SYunfei Dong if (err) {
332*0934d375SYunfei Dong mtk_v4l2_vdec_err(ctx, "failed to allocate wdma_addr buf[%d]", i);
333*0934d375SYunfei Dong goto mem_alloc_err;
334*0934d375SYunfei Dong }
335*0934d375SYunfei Dong
336*0934d375SYunfei Dong if (ctx->current_codec == V4L2_PIX_FMT_AV1_FRAME) {
337*0934d375SYunfei Dong lat_buf->rd_mv_addr.size = VDEC_RD_MV_BUFFER_SZ;
338*0934d375SYunfei Dong err = mtk_vcodec_mem_alloc(ctx, &lat_buf->rd_mv_addr);
339*0934d375SYunfei Dong if (err) {
340*0934d375SYunfei Dong mtk_v4l2_vdec_err(ctx, "failed to allocate rd_mv_addr buf[%d]", i);
341*0934d375SYunfei Dong goto mem_alloc_err;
342*0934d375SYunfei Dong }
343*0934d375SYunfei Dong
344*0934d375SYunfei Dong lat_buf->tile_addr.size = VDEC_LAT_TILE_SZ;
345*0934d375SYunfei Dong err = mtk_vcodec_mem_alloc(ctx, &lat_buf->tile_addr);
346*0934d375SYunfei Dong if (err) {
347*0934d375SYunfei Dong mtk_v4l2_vdec_err(ctx, "failed to allocate tile_addr buf[%d]", i);
348*0934d375SYunfei Dong goto mem_alloc_err;
349*0934d375SYunfei Dong }
350*0934d375SYunfei Dong }
351*0934d375SYunfei Dong
352*0934d375SYunfei Dong lat_buf->private_data = kzalloc(private_size, GFP_KERNEL);
353*0934d375SYunfei Dong if (!lat_buf->private_data) {
354*0934d375SYunfei Dong err = -ENOMEM;
355*0934d375SYunfei Dong goto mem_alloc_err;
356*0934d375SYunfei Dong }
357*0934d375SYunfei Dong
358*0934d375SYunfei Dong lat_buf->ctx = ctx;
359*0934d375SYunfei Dong lat_buf->core_decode = core_decode;
360*0934d375SYunfei Dong lat_buf->is_last_frame = false;
361*0934d375SYunfei Dong err = vdec_msg_queue_qbuf(&msg_queue->lat_ctx, lat_buf);
362*0934d375SYunfei Dong if (err) {
363*0934d375SYunfei Dong mtk_v4l2_vdec_err(ctx, "failed to qbuf buf[%d]", i);
364*0934d375SYunfei Dong goto mem_alloc_err;
365*0934d375SYunfei Dong }
366*0934d375SYunfei Dong }
367*0934d375SYunfei Dong return 0;
368*0934d375SYunfei Dong
369*0934d375SYunfei Dong mem_alloc_err:
370*0934d375SYunfei Dong vdec_msg_queue_deinit(msg_queue, ctx);
371*0934d375SYunfei Dong return err;
372*0934d375SYunfei Dong }
373