1*fbb6c848SEzequiel Garcia // SPDX-License-Identifier: GPL-2.0
2*fbb6c848SEzequiel Garcia /*
3*fbb6c848SEzequiel Garcia  * Hantro VP8 codec driver
4*fbb6c848SEzequiel Garcia  *
5*fbb6c848SEzequiel Garcia  * Copyright (C) 2019 Rockchip Electronics Co., Ltd.
6*fbb6c848SEzequiel Garcia  *	ZhiChao Yu <zhichao.yu@rock-chips.com>
7*fbb6c848SEzequiel Garcia  *
8*fbb6c848SEzequiel Garcia  * Copyright (C) 2019 Google, Inc.
9*fbb6c848SEzequiel Garcia  *	Tomasz Figa <tfiga@chromium.org>
10*fbb6c848SEzequiel Garcia  */
11*fbb6c848SEzequiel Garcia 
12*fbb6c848SEzequiel Garcia #include <media/v4l2-mem2mem.h>
13*fbb6c848SEzequiel Garcia 
14*fbb6c848SEzequiel Garcia #include "hantro_hw.h"
15*fbb6c848SEzequiel Garcia #include "hantro.h"
16*fbb6c848SEzequiel Garcia #include "hantro_g1_regs.h"
17*fbb6c848SEzequiel Garcia 
18*fbb6c848SEzequiel Garcia /* DCT partition base address regs */
19*fbb6c848SEzequiel Garcia static const struct hantro_reg vp8_dec_dct_base[8] = {
20*fbb6c848SEzequiel Garcia 	{ G1_REG_ADDR_STR, 0, 0xffffffff },
21*fbb6c848SEzequiel Garcia 	{ G1_REG_ADDR_REF(8), 0, 0xffffffff },
22*fbb6c848SEzequiel Garcia 	{ G1_REG_ADDR_REF(9), 0, 0xffffffff },
23*fbb6c848SEzequiel Garcia 	{ G1_REG_ADDR_REF(10), 0, 0xffffffff },
24*fbb6c848SEzequiel Garcia 	{ G1_REG_ADDR_REF(11), 0, 0xffffffff },
25*fbb6c848SEzequiel Garcia 	{ G1_REG_ADDR_REF(12), 0, 0xffffffff },
26*fbb6c848SEzequiel Garcia 	{ G1_REG_ADDR_REF(14), 0, 0xffffffff },
27*fbb6c848SEzequiel Garcia 	{ G1_REG_ADDR_REF(15), 0, 0xffffffff },
28*fbb6c848SEzequiel Garcia };
29*fbb6c848SEzequiel Garcia 
30*fbb6c848SEzequiel Garcia /* Loop filter level regs */
31*fbb6c848SEzequiel Garcia static const struct hantro_reg vp8_dec_lf_level[4] = {
32*fbb6c848SEzequiel Garcia 	{ G1_REG_REF_PIC(2), 18, 0x3f },
33*fbb6c848SEzequiel Garcia 	{ G1_REG_REF_PIC(2), 12, 0x3f },
34*fbb6c848SEzequiel Garcia 	{ G1_REG_REF_PIC(2), 6, 0x3f },
35*fbb6c848SEzequiel Garcia 	{ G1_REG_REF_PIC(2), 0, 0x3f },
36*fbb6c848SEzequiel Garcia };
37*fbb6c848SEzequiel Garcia 
38*fbb6c848SEzequiel Garcia /* Macroblock loop filter level adjustment regs */
39*fbb6c848SEzequiel Garcia static const struct hantro_reg vp8_dec_mb_adj[4] = {
40*fbb6c848SEzequiel Garcia 	{ G1_REG_REF_PIC(0), 21, 0x7f },
41*fbb6c848SEzequiel Garcia 	{ G1_REG_REF_PIC(0), 14, 0x7f },
42*fbb6c848SEzequiel Garcia 	{ G1_REG_REF_PIC(0), 7, 0x7f },
43*fbb6c848SEzequiel Garcia 	{ G1_REG_REF_PIC(0), 0, 0x7f },
44*fbb6c848SEzequiel Garcia };
45*fbb6c848SEzequiel Garcia 
46*fbb6c848SEzequiel Garcia /* Reference frame adjustment regs */
47*fbb6c848SEzequiel Garcia static const struct hantro_reg vp8_dec_ref_adj[4] = {
48*fbb6c848SEzequiel Garcia 	{ G1_REG_REF_PIC(1), 21, 0x7f },
49*fbb6c848SEzequiel Garcia 	{ G1_REG_REF_PIC(1), 14, 0x7f },
50*fbb6c848SEzequiel Garcia 	{ G1_REG_REF_PIC(1), 7, 0x7f },
51*fbb6c848SEzequiel Garcia 	{ G1_REG_REF_PIC(1), 0, 0x7f },
52*fbb6c848SEzequiel Garcia };
53*fbb6c848SEzequiel Garcia 
54*fbb6c848SEzequiel Garcia /* Quantizer */
55*fbb6c848SEzequiel Garcia static const struct hantro_reg vp8_dec_quant[4] = {
56*fbb6c848SEzequiel Garcia 	{ G1_REG_REF_PIC(3), 11, 0x7ff },
57*fbb6c848SEzequiel Garcia 	{ G1_REG_REF_PIC(3), 0, 0x7ff },
58*fbb6c848SEzequiel Garcia 	{ G1_REG_BD_REF_PIC(4), 11, 0x7ff },
59*fbb6c848SEzequiel Garcia 	{ G1_REG_BD_REF_PIC(4), 0, 0x7ff },
60*fbb6c848SEzequiel Garcia };
61*fbb6c848SEzequiel Garcia 
62*fbb6c848SEzequiel Garcia /* Quantizer delta regs */
63*fbb6c848SEzequiel Garcia static const struct hantro_reg vp8_dec_quant_delta[5] = {
64*fbb6c848SEzequiel Garcia 	{ G1_REG_REF_PIC(3), 27, 0x1f },
65*fbb6c848SEzequiel Garcia 	{ G1_REG_REF_PIC(3), 22, 0x1f },
66*fbb6c848SEzequiel Garcia 	{ G1_REG_BD_REF_PIC(4), 27, 0x1f },
67*fbb6c848SEzequiel Garcia 	{ G1_REG_BD_REF_PIC(4), 22, 0x1f },
68*fbb6c848SEzequiel Garcia 	{ G1_REG_BD_P_REF_PIC, 27, 0x1f },
69*fbb6c848SEzequiel Garcia };
70*fbb6c848SEzequiel Garcia 
71*fbb6c848SEzequiel Garcia /* DCT partition start bits regs */
72*fbb6c848SEzequiel Garcia static const struct hantro_reg vp8_dec_dct_start_bits[8] = {
73*fbb6c848SEzequiel Garcia 	{ G1_REG_DEC_CTRL2, 26, 0x3f }, { G1_REG_DEC_CTRL4, 26, 0x3f },
74*fbb6c848SEzequiel Garcia 	{ G1_REG_DEC_CTRL4, 20, 0x3f }, { G1_REG_DEC_CTRL7, 24, 0x3f },
75*fbb6c848SEzequiel Garcia 	{ G1_REG_DEC_CTRL7, 18, 0x3f }, { G1_REG_DEC_CTRL7, 12, 0x3f },
76*fbb6c848SEzequiel Garcia 	{ G1_REG_DEC_CTRL7, 6, 0x3f },  { G1_REG_DEC_CTRL7, 0, 0x3f },
77*fbb6c848SEzequiel Garcia };
78*fbb6c848SEzequiel Garcia 
79*fbb6c848SEzequiel Garcia /* Precision filter tap regs */
80*fbb6c848SEzequiel Garcia static const struct hantro_reg vp8_dec_pred_bc_tap[8][4] = {
81*fbb6c848SEzequiel Garcia 	{
82*fbb6c848SEzequiel Garcia 		{ G1_REG_PRED_FLT, 22, 0x3ff },
83*fbb6c848SEzequiel Garcia 		{ G1_REG_PRED_FLT, 12, 0x3ff },
84*fbb6c848SEzequiel Garcia 		{ G1_REG_PRED_FLT, 2, 0x3ff },
85*fbb6c848SEzequiel Garcia 		{ G1_REG_REF_PIC(4), 22, 0x3ff },
86*fbb6c848SEzequiel Garcia 	},
87*fbb6c848SEzequiel Garcia 	{
88*fbb6c848SEzequiel Garcia 		{ G1_REG_REF_PIC(4), 12, 0x3ff },
89*fbb6c848SEzequiel Garcia 		{ G1_REG_REF_PIC(4), 2, 0x3ff },
90*fbb6c848SEzequiel Garcia 		{ G1_REG_REF_PIC(5), 22, 0x3ff },
91*fbb6c848SEzequiel Garcia 		{ G1_REG_REF_PIC(5), 12, 0x3ff },
92*fbb6c848SEzequiel Garcia 	},
93*fbb6c848SEzequiel Garcia 	{
94*fbb6c848SEzequiel Garcia 		{ G1_REG_REF_PIC(5), 2, 0x3ff },
95*fbb6c848SEzequiel Garcia 		{ G1_REG_REF_PIC(6), 22, 0x3ff },
96*fbb6c848SEzequiel Garcia 		{ G1_REG_REF_PIC(6), 12, 0x3ff },
97*fbb6c848SEzequiel Garcia 		{ G1_REG_REF_PIC(6), 2, 0x3ff },
98*fbb6c848SEzequiel Garcia 	},
99*fbb6c848SEzequiel Garcia 	{
100*fbb6c848SEzequiel Garcia 		{ G1_REG_REF_PIC(7), 22, 0x3ff },
101*fbb6c848SEzequiel Garcia 		{ G1_REG_REF_PIC(7), 12, 0x3ff },
102*fbb6c848SEzequiel Garcia 		{ G1_REG_REF_PIC(7), 2, 0x3ff },
103*fbb6c848SEzequiel Garcia 		{ G1_REG_LT_REF, 22, 0x3ff },
104*fbb6c848SEzequiel Garcia 	},
105*fbb6c848SEzequiel Garcia 	{
106*fbb6c848SEzequiel Garcia 		{ G1_REG_LT_REF, 12, 0x3ff },
107*fbb6c848SEzequiel Garcia 		{ G1_REG_LT_REF, 2, 0x3ff },
108*fbb6c848SEzequiel Garcia 		{ G1_REG_VALID_REF, 22, 0x3ff },
109*fbb6c848SEzequiel Garcia 		{ G1_REG_VALID_REF, 12, 0x3ff },
110*fbb6c848SEzequiel Garcia 	},
111*fbb6c848SEzequiel Garcia 	{
112*fbb6c848SEzequiel Garcia 		{ G1_REG_VALID_REF, 2, 0x3ff },
113*fbb6c848SEzequiel Garcia 		{ G1_REG_BD_REF_PIC(0), 22, 0x3ff },
114*fbb6c848SEzequiel Garcia 		{ G1_REG_BD_REF_PIC(0), 12, 0x3ff },
115*fbb6c848SEzequiel Garcia 		{ G1_REG_BD_REF_PIC(0), 2, 0x3ff },
116*fbb6c848SEzequiel Garcia 	},
117*fbb6c848SEzequiel Garcia 	{
118*fbb6c848SEzequiel Garcia 		{ G1_REG_BD_REF_PIC(1), 22, 0x3ff },
119*fbb6c848SEzequiel Garcia 		{ G1_REG_BD_REF_PIC(1), 12, 0x3ff },
120*fbb6c848SEzequiel Garcia 		{ G1_REG_BD_REF_PIC(1), 2, 0x3ff },
121*fbb6c848SEzequiel Garcia 		{ G1_REG_BD_REF_PIC(2), 22, 0x3ff },
122*fbb6c848SEzequiel Garcia 	},
123*fbb6c848SEzequiel Garcia 	{
124*fbb6c848SEzequiel Garcia 		{ G1_REG_BD_REF_PIC(2), 12, 0x3ff },
125*fbb6c848SEzequiel Garcia 		{ G1_REG_BD_REF_PIC(2), 2, 0x3ff },
126*fbb6c848SEzequiel Garcia 		{ G1_REG_BD_REF_PIC(3), 22, 0x3ff },
127*fbb6c848SEzequiel Garcia 		{ G1_REG_BD_REF_PIC(3), 12, 0x3ff },
128*fbb6c848SEzequiel Garcia 	},
129*fbb6c848SEzequiel Garcia };
130*fbb6c848SEzequiel Garcia 
131*fbb6c848SEzequiel Garcia /*
132*fbb6c848SEzequiel Garcia  * Set loop filters
133*fbb6c848SEzequiel Garcia  */
cfg_lf(struct hantro_ctx * ctx,const struct v4l2_ctrl_vp8_frame * hdr)134*fbb6c848SEzequiel Garcia static void cfg_lf(struct hantro_ctx *ctx,
135*fbb6c848SEzequiel Garcia 		   const struct v4l2_ctrl_vp8_frame *hdr)
136*fbb6c848SEzequiel Garcia {
137*fbb6c848SEzequiel Garcia 	const struct v4l2_vp8_segment *seg = &hdr->segment;
138*fbb6c848SEzequiel Garcia 	const struct v4l2_vp8_loop_filter *lf = &hdr->lf;
139*fbb6c848SEzequiel Garcia 	struct hantro_dev *vpu = ctx->dev;
140*fbb6c848SEzequiel Garcia 	unsigned int i;
141*fbb6c848SEzequiel Garcia 	u32 reg;
142*fbb6c848SEzequiel Garcia 
143*fbb6c848SEzequiel Garcia 	if (!(seg->flags & V4L2_VP8_SEGMENT_FLAG_ENABLED)) {
144*fbb6c848SEzequiel Garcia 		hantro_reg_write(vpu, &vp8_dec_lf_level[0], lf->level);
145*fbb6c848SEzequiel Garcia 	} else if (seg->flags & V4L2_VP8_SEGMENT_FLAG_DELTA_VALUE_MODE) {
146*fbb6c848SEzequiel Garcia 		for (i = 0; i < 4; i++) {
147*fbb6c848SEzequiel Garcia 			u32 lf_level = clamp(lf->level + seg->lf_update[i],
148*fbb6c848SEzequiel Garcia 					     0, 63);
149*fbb6c848SEzequiel Garcia 
150*fbb6c848SEzequiel Garcia 			hantro_reg_write(vpu, &vp8_dec_lf_level[i], lf_level);
151*fbb6c848SEzequiel Garcia 		}
152*fbb6c848SEzequiel Garcia 	} else {
153*fbb6c848SEzequiel Garcia 		for (i = 0; i < 4; i++)
154*fbb6c848SEzequiel Garcia 			hantro_reg_write(vpu, &vp8_dec_lf_level[i],
155*fbb6c848SEzequiel Garcia 					 seg->lf_update[i]);
156*fbb6c848SEzequiel Garcia 	}
157*fbb6c848SEzequiel Garcia 
158*fbb6c848SEzequiel Garcia 	reg = G1_REG_REF_PIC_FILT_SHARPNESS(lf->sharpness_level);
159*fbb6c848SEzequiel Garcia 	if (lf->flags & V4L2_VP8_LF_FILTER_TYPE_SIMPLE)
160*fbb6c848SEzequiel Garcia 		reg |= G1_REG_REF_PIC_FILT_TYPE_E;
161*fbb6c848SEzequiel Garcia 	vdpu_write_relaxed(vpu, reg, G1_REG_REF_PIC(0));
162*fbb6c848SEzequiel Garcia 
163*fbb6c848SEzequiel Garcia 	if (lf->flags & V4L2_VP8_LF_ADJ_ENABLE) {
164*fbb6c848SEzequiel Garcia 		for (i = 0; i < 4; i++) {
165*fbb6c848SEzequiel Garcia 			hantro_reg_write(vpu, &vp8_dec_mb_adj[i],
166*fbb6c848SEzequiel Garcia 					 lf->mb_mode_delta[i]);
167*fbb6c848SEzequiel Garcia 			hantro_reg_write(vpu, &vp8_dec_ref_adj[i],
168*fbb6c848SEzequiel Garcia 					 lf->ref_frm_delta[i]);
169*fbb6c848SEzequiel Garcia 		}
170*fbb6c848SEzequiel Garcia 	}
171*fbb6c848SEzequiel Garcia }
172*fbb6c848SEzequiel Garcia 
173*fbb6c848SEzequiel Garcia /*
174*fbb6c848SEzequiel Garcia  * Set quantization parameters
175*fbb6c848SEzequiel Garcia  */
cfg_qp(struct hantro_ctx * ctx,const struct v4l2_ctrl_vp8_frame * hdr)176*fbb6c848SEzequiel Garcia static void cfg_qp(struct hantro_ctx *ctx,
177*fbb6c848SEzequiel Garcia 		   const struct v4l2_ctrl_vp8_frame *hdr)
178*fbb6c848SEzequiel Garcia {
179*fbb6c848SEzequiel Garcia 	const struct v4l2_vp8_quantization *q = &hdr->quant;
180*fbb6c848SEzequiel Garcia 	const struct v4l2_vp8_segment *seg = &hdr->segment;
181*fbb6c848SEzequiel Garcia 	struct hantro_dev *vpu = ctx->dev;
182*fbb6c848SEzequiel Garcia 	unsigned int i;
183*fbb6c848SEzequiel Garcia 
184*fbb6c848SEzequiel Garcia 	if (!(seg->flags & V4L2_VP8_SEGMENT_FLAG_ENABLED)) {
185*fbb6c848SEzequiel Garcia 		hantro_reg_write(vpu, &vp8_dec_quant[0], q->y_ac_qi);
186*fbb6c848SEzequiel Garcia 	} else if (seg->flags & V4L2_VP8_SEGMENT_FLAG_DELTA_VALUE_MODE) {
187*fbb6c848SEzequiel Garcia 		for (i = 0; i < 4; i++) {
188*fbb6c848SEzequiel Garcia 			u32 quant = clamp(q->y_ac_qi + seg->quant_update[i],
189*fbb6c848SEzequiel Garcia 					  0, 127);
190*fbb6c848SEzequiel Garcia 
191*fbb6c848SEzequiel Garcia 			hantro_reg_write(vpu, &vp8_dec_quant[i], quant);
192*fbb6c848SEzequiel Garcia 		}
193*fbb6c848SEzequiel Garcia 	} else {
194*fbb6c848SEzequiel Garcia 		for (i = 0; i < 4; i++)
195*fbb6c848SEzequiel Garcia 			hantro_reg_write(vpu, &vp8_dec_quant[i],
196*fbb6c848SEzequiel Garcia 					 seg->quant_update[i]);
197*fbb6c848SEzequiel Garcia 	}
198*fbb6c848SEzequiel Garcia 
199*fbb6c848SEzequiel Garcia 	hantro_reg_write(vpu, &vp8_dec_quant_delta[0], q->y_dc_delta);
200*fbb6c848SEzequiel Garcia 	hantro_reg_write(vpu, &vp8_dec_quant_delta[1], q->y2_dc_delta);
201*fbb6c848SEzequiel Garcia 	hantro_reg_write(vpu, &vp8_dec_quant_delta[2], q->y2_ac_delta);
202*fbb6c848SEzequiel Garcia 	hantro_reg_write(vpu, &vp8_dec_quant_delta[3], q->uv_dc_delta);
203*fbb6c848SEzequiel Garcia 	hantro_reg_write(vpu, &vp8_dec_quant_delta[4], q->uv_ac_delta);
204*fbb6c848SEzequiel Garcia }
205*fbb6c848SEzequiel Garcia 
206*fbb6c848SEzequiel Garcia /*
207*fbb6c848SEzequiel Garcia  * set control partition and DCT partition regs
208*fbb6c848SEzequiel Garcia  *
209*fbb6c848SEzequiel Garcia  * VP8 frame stream data layout:
210*fbb6c848SEzequiel Garcia  *
211*fbb6c848SEzequiel Garcia  *	                     first_part_size          parttion_sizes[0]
212*fbb6c848SEzequiel Garcia  *                              ^                     ^
213*fbb6c848SEzequiel Garcia  * src_dma                      |                     |
214*fbb6c848SEzequiel Garcia  * ^                   +--------+------+        +-----+-----+
215*fbb6c848SEzequiel Garcia  * |                   | control part  |        |           |
216*fbb6c848SEzequiel Garcia  * +--------+----------------+------------------+-----------+-----+-----------+
217*fbb6c848SEzequiel Garcia  * | tag 3B | extra 7B | hdr | mb_data | DCT sz | DCT part0 | ... | DCT partn |
218*fbb6c848SEzequiel Garcia  * +--------+-----------------------------------+-----------+-----+-----------+
219*fbb6c848SEzequiel Garcia  *                           |         |        |                             |
220*fbb6c848SEzequiel Garcia  *                           v         +----+---+                             v
221*fbb6c848SEzequiel Garcia  *                           mb_start       |                       src_dma_end
222*fbb6c848SEzequiel Garcia  *                                          v
223*fbb6c848SEzequiel Garcia  *                                       DCT size part
224*fbb6c848SEzequiel Garcia  *                                      (num_dct-1)*3B
225*fbb6c848SEzequiel Garcia  * Note:
226*fbb6c848SEzequiel Garcia  *   1. only key-frames have extra 7-bytes
227*fbb6c848SEzequiel Garcia  *   2. all offsets are base on src_dma
228*fbb6c848SEzequiel Garcia  *   3. number of DCT parts is 1, 2, 4 or 8
229*fbb6c848SEzequiel Garcia  *   4. the addresses set to the VPU must be 64-bits aligned
230*fbb6c848SEzequiel Garcia  */
cfg_parts(struct hantro_ctx * ctx,const struct v4l2_ctrl_vp8_frame * hdr)231*fbb6c848SEzequiel Garcia static void cfg_parts(struct hantro_ctx *ctx,
232*fbb6c848SEzequiel Garcia 		      const struct v4l2_ctrl_vp8_frame *hdr)
233*fbb6c848SEzequiel Garcia {
234*fbb6c848SEzequiel Garcia 	struct hantro_dev *vpu = ctx->dev;
235*fbb6c848SEzequiel Garcia 	struct vb2_v4l2_buffer *vb2_src;
236*fbb6c848SEzequiel Garcia 	u32 first_part_offset = V4L2_VP8_FRAME_IS_KEY_FRAME(hdr) ? 10 : 3;
237*fbb6c848SEzequiel Garcia 	u32 mb_size, mb_offset_bytes, mb_offset_bits, mb_start_bits;
238*fbb6c848SEzequiel Garcia 	u32 dct_size_part_size, dct_part_offset;
239*fbb6c848SEzequiel Garcia 	struct hantro_reg reg;
240*fbb6c848SEzequiel Garcia 	dma_addr_t src_dma;
241*fbb6c848SEzequiel Garcia 	u32 dct_part_total_len = 0;
242*fbb6c848SEzequiel Garcia 	u32 count = 0;
243*fbb6c848SEzequiel Garcia 	unsigned int i;
244*fbb6c848SEzequiel Garcia 
245*fbb6c848SEzequiel Garcia 	vb2_src = hantro_get_src_buf(ctx);
246*fbb6c848SEzequiel Garcia 	src_dma = vb2_dma_contig_plane_dma_addr(&vb2_src->vb2_buf, 0);
247*fbb6c848SEzequiel Garcia 
248*fbb6c848SEzequiel Garcia 	/*
249*fbb6c848SEzequiel Garcia 	 * Calculate control partition mb data info
250*fbb6c848SEzequiel Garcia 	 * @first_part_header_bits:	bits offset of mb data from first
251*fbb6c848SEzequiel Garcia 	 *				part start pos
252*fbb6c848SEzequiel Garcia 	 * @mb_offset_bits:		bits offset of mb data from src_dma
253*fbb6c848SEzequiel Garcia 	 *				base addr
254*fbb6c848SEzequiel Garcia 	 * @mb_offset_byte:		bytes offset of mb data from src_dma
255*fbb6c848SEzequiel Garcia 	 *				base addr
256*fbb6c848SEzequiel Garcia 	 * @mb_start_bits:		bits offset of mb data from mb data
257*fbb6c848SEzequiel Garcia 	 *				64bits alignment addr
258*fbb6c848SEzequiel Garcia 	 */
259*fbb6c848SEzequiel Garcia 	mb_offset_bits = first_part_offset * 8 +
260*fbb6c848SEzequiel Garcia 			 hdr->first_part_header_bits + 8;
261*fbb6c848SEzequiel Garcia 	mb_offset_bytes = mb_offset_bits / 8;
262*fbb6c848SEzequiel Garcia 	mb_start_bits = mb_offset_bits -
263*fbb6c848SEzequiel Garcia 			(mb_offset_bytes & (~DEC_8190_ALIGN_MASK)) * 8;
264*fbb6c848SEzequiel Garcia 	mb_size = hdr->first_part_size -
265*fbb6c848SEzequiel Garcia 		  (mb_offset_bytes - first_part_offset) +
266*fbb6c848SEzequiel Garcia 		  (mb_offset_bytes & DEC_8190_ALIGN_MASK);
267*fbb6c848SEzequiel Garcia 
268*fbb6c848SEzequiel Garcia 	/* Macroblock data aligned base addr */
269*fbb6c848SEzequiel Garcia 	vdpu_write_relaxed(vpu, (mb_offset_bytes & (~DEC_8190_ALIGN_MASK))
270*fbb6c848SEzequiel Garcia 				+ src_dma, G1_REG_ADDR_REF(13));
271*fbb6c848SEzequiel Garcia 
272*fbb6c848SEzequiel Garcia 	/* Macroblock data start bits */
273*fbb6c848SEzequiel Garcia 	reg.base = G1_REG_DEC_CTRL2;
274*fbb6c848SEzequiel Garcia 	reg.mask = 0x3f;
275*fbb6c848SEzequiel Garcia 	reg.shift = 18;
276*fbb6c848SEzequiel Garcia 	hantro_reg_write(vpu, &reg, mb_start_bits);
277*fbb6c848SEzequiel Garcia 
278*fbb6c848SEzequiel Garcia 	/* Macroblock aligned data length */
279*fbb6c848SEzequiel Garcia 	reg.base = G1_REG_DEC_CTRL6;
280*fbb6c848SEzequiel Garcia 	reg.mask = 0x3fffff;
281*fbb6c848SEzequiel Garcia 	reg.shift = 0;
282*fbb6c848SEzequiel Garcia 	hantro_reg_write(vpu, &reg, mb_size + 1);
283*fbb6c848SEzequiel Garcia 
284*fbb6c848SEzequiel Garcia 	/*
285*fbb6c848SEzequiel Garcia 	 * Calculate DCT partition info
286*fbb6c848SEzequiel Garcia 	 * @dct_size_part_size: Containing sizes of DCT part, every DCT part
287*fbb6c848SEzequiel Garcia 	 *			has 3 bytes to store its size, except the last
288*fbb6c848SEzequiel Garcia 	 *			DCT part
289*fbb6c848SEzequiel Garcia 	 * @dct_part_offset:	bytes offset of DCT parts from src_dma base addr
290*fbb6c848SEzequiel Garcia 	 * @dct_part_total_len: total size of all DCT parts
291*fbb6c848SEzequiel Garcia 	 */
292*fbb6c848SEzequiel Garcia 	dct_size_part_size = (hdr->num_dct_parts - 1) * 3;
293*fbb6c848SEzequiel Garcia 	dct_part_offset = first_part_offset + hdr->first_part_size;
294*fbb6c848SEzequiel Garcia 	for (i = 0; i < hdr->num_dct_parts; i++)
295*fbb6c848SEzequiel Garcia 		dct_part_total_len += hdr->dct_part_sizes[i];
296*fbb6c848SEzequiel Garcia 	dct_part_total_len += dct_size_part_size;
297*fbb6c848SEzequiel Garcia 	dct_part_total_len += (dct_part_offset & DEC_8190_ALIGN_MASK);
298*fbb6c848SEzequiel Garcia 
299*fbb6c848SEzequiel Garcia 	/* Number of DCT partitions */
300*fbb6c848SEzequiel Garcia 	reg.base = G1_REG_DEC_CTRL6;
301*fbb6c848SEzequiel Garcia 	reg.mask = 0xf;
302*fbb6c848SEzequiel Garcia 	reg.shift = 24;
303*fbb6c848SEzequiel Garcia 	hantro_reg_write(vpu, &reg, hdr->num_dct_parts - 1);
304*fbb6c848SEzequiel Garcia 
305*fbb6c848SEzequiel Garcia 	/* DCT partition length */
306*fbb6c848SEzequiel Garcia 	vdpu_write_relaxed(vpu,
307*fbb6c848SEzequiel Garcia 			   G1_REG_DEC_CTRL3_STREAM_LEN(dct_part_total_len),
308*fbb6c848SEzequiel Garcia 			   G1_REG_DEC_CTRL3);
309*fbb6c848SEzequiel Garcia 
310*fbb6c848SEzequiel Garcia 	/* DCT partitions base address */
311*fbb6c848SEzequiel Garcia 	for (i = 0; i < hdr->num_dct_parts; i++) {
312*fbb6c848SEzequiel Garcia 		u32 byte_offset = dct_part_offset + dct_size_part_size + count;
313*fbb6c848SEzequiel Garcia 		u32 base_addr = byte_offset + src_dma;
314*fbb6c848SEzequiel Garcia 
315*fbb6c848SEzequiel Garcia 		hantro_reg_write(vpu, &vp8_dec_dct_base[i],
316*fbb6c848SEzequiel Garcia 				 base_addr & (~DEC_8190_ALIGN_MASK));
317*fbb6c848SEzequiel Garcia 
318*fbb6c848SEzequiel Garcia 		hantro_reg_write(vpu, &vp8_dec_dct_start_bits[i],
319*fbb6c848SEzequiel Garcia 				 (byte_offset & DEC_8190_ALIGN_MASK) * 8);
320*fbb6c848SEzequiel Garcia 
321*fbb6c848SEzequiel Garcia 		count += hdr->dct_part_sizes[i];
322*fbb6c848SEzequiel Garcia 	}
323*fbb6c848SEzequiel Garcia }
324*fbb6c848SEzequiel Garcia 
325*fbb6c848SEzequiel Garcia /*
326*fbb6c848SEzequiel Garcia  * prediction filter taps
327*fbb6c848SEzequiel Garcia  * normal 6-tap filters
328*fbb6c848SEzequiel Garcia  */
cfg_tap(struct hantro_ctx * ctx,const struct v4l2_ctrl_vp8_frame * hdr)329*fbb6c848SEzequiel Garcia static void cfg_tap(struct hantro_ctx *ctx,
330*fbb6c848SEzequiel Garcia 		    const struct v4l2_ctrl_vp8_frame *hdr)
331*fbb6c848SEzequiel Garcia {
332*fbb6c848SEzequiel Garcia 	struct hantro_dev *vpu = ctx->dev;
333*fbb6c848SEzequiel Garcia 	struct hantro_reg reg;
334*fbb6c848SEzequiel Garcia 	u32 val = 0;
335*fbb6c848SEzequiel Garcia 	int i, j;
336*fbb6c848SEzequiel Garcia 
337*fbb6c848SEzequiel Garcia 	reg.base = G1_REG_BD_REF_PIC(3);
338*fbb6c848SEzequiel Garcia 	reg.mask = 0xf;
339*fbb6c848SEzequiel Garcia 
340*fbb6c848SEzequiel Garcia 	if ((hdr->version & 0x03) != 0)
341*fbb6c848SEzequiel Garcia 		return; /* Tap filter not used. */
342*fbb6c848SEzequiel Garcia 
343*fbb6c848SEzequiel Garcia 	for (i = 0; i < 8; i++) {
344*fbb6c848SEzequiel Garcia 		val = (hantro_vp8_dec_mc_filter[i][0] << 2) |
345*fbb6c848SEzequiel Garcia 		       hantro_vp8_dec_mc_filter[i][5];
346*fbb6c848SEzequiel Garcia 
347*fbb6c848SEzequiel Garcia 		for (j = 0; j < 4; j++)
348*fbb6c848SEzequiel Garcia 			hantro_reg_write(vpu, &vp8_dec_pred_bc_tap[i][j],
349*fbb6c848SEzequiel Garcia 					 hantro_vp8_dec_mc_filter[i][j + 1]);
350*fbb6c848SEzequiel Garcia 
351*fbb6c848SEzequiel Garcia 		switch (i) {
352*fbb6c848SEzequiel Garcia 		case 2:
353*fbb6c848SEzequiel Garcia 			reg.shift = 8;
354*fbb6c848SEzequiel Garcia 			break;
355*fbb6c848SEzequiel Garcia 		case 4:
356*fbb6c848SEzequiel Garcia 			reg.shift = 4;
357*fbb6c848SEzequiel Garcia 			break;
358*fbb6c848SEzequiel Garcia 		case 6:
359*fbb6c848SEzequiel Garcia 			reg.shift = 0;
360*fbb6c848SEzequiel Garcia 			break;
361*fbb6c848SEzequiel Garcia 		default:
362*fbb6c848SEzequiel Garcia 			continue;
363*fbb6c848SEzequiel Garcia 		}
364*fbb6c848SEzequiel Garcia 
365*fbb6c848SEzequiel Garcia 		hantro_reg_write(vpu, &reg, val);
366*fbb6c848SEzequiel Garcia 	}
367*fbb6c848SEzequiel Garcia }
368*fbb6c848SEzequiel Garcia 
cfg_ref(struct hantro_ctx * ctx,const struct v4l2_ctrl_vp8_frame * hdr,struct vb2_v4l2_buffer * vb2_dst)369*fbb6c848SEzequiel Garcia static void cfg_ref(struct hantro_ctx *ctx,
370*fbb6c848SEzequiel Garcia 		    const struct v4l2_ctrl_vp8_frame *hdr,
371*fbb6c848SEzequiel Garcia 		    struct vb2_v4l2_buffer *vb2_dst)
372*fbb6c848SEzequiel Garcia {
373*fbb6c848SEzequiel Garcia 	struct hantro_dev *vpu = ctx->dev;
374*fbb6c848SEzequiel Garcia 	dma_addr_t ref;
375*fbb6c848SEzequiel Garcia 
376*fbb6c848SEzequiel Garcia 
377*fbb6c848SEzequiel Garcia 	ref = hantro_get_ref(ctx, hdr->last_frame_ts);
378*fbb6c848SEzequiel Garcia 	if (!ref) {
379*fbb6c848SEzequiel Garcia 		vpu_debug(0, "failed to find last frame ts=%llu\n",
380*fbb6c848SEzequiel Garcia 			  hdr->last_frame_ts);
381*fbb6c848SEzequiel Garcia 		ref = vb2_dma_contig_plane_dma_addr(&vb2_dst->vb2_buf, 0);
382*fbb6c848SEzequiel Garcia 	}
383*fbb6c848SEzequiel Garcia 	vdpu_write_relaxed(vpu, ref, G1_REG_ADDR_REF(0));
384*fbb6c848SEzequiel Garcia 
385*fbb6c848SEzequiel Garcia 	ref = hantro_get_ref(ctx, hdr->golden_frame_ts);
386*fbb6c848SEzequiel Garcia 	if (!ref && hdr->golden_frame_ts)
387*fbb6c848SEzequiel Garcia 		vpu_debug(0, "failed to find golden frame ts=%llu\n",
388*fbb6c848SEzequiel Garcia 			  hdr->golden_frame_ts);
389*fbb6c848SEzequiel Garcia 	if (!ref)
390*fbb6c848SEzequiel Garcia 		ref = vb2_dma_contig_plane_dma_addr(&vb2_dst->vb2_buf, 0);
391*fbb6c848SEzequiel Garcia 	if (hdr->flags & V4L2_VP8_FRAME_FLAG_SIGN_BIAS_GOLDEN)
392*fbb6c848SEzequiel Garcia 		ref |= G1_REG_ADDR_REF_TOPC_E;
393*fbb6c848SEzequiel Garcia 	vdpu_write_relaxed(vpu, ref, G1_REG_ADDR_REF(4));
394*fbb6c848SEzequiel Garcia 
395*fbb6c848SEzequiel Garcia 	ref = hantro_get_ref(ctx, hdr->alt_frame_ts);
396*fbb6c848SEzequiel Garcia 	if (!ref && hdr->alt_frame_ts)
397*fbb6c848SEzequiel Garcia 		vpu_debug(0, "failed to find alt frame ts=%llu\n",
398*fbb6c848SEzequiel Garcia 			  hdr->alt_frame_ts);
399*fbb6c848SEzequiel Garcia 	if (!ref)
400*fbb6c848SEzequiel Garcia 		ref = vb2_dma_contig_plane_dma_addr(&vb2_dst->vb2_buf, 0);
401*fbb6c848SEzequiel Garcia 	if (hdr->flags & V4L2_VP8_FRAME_FLAG_SIGN_BIAS_ALT)
402*fbb6c848SEzequiel Garcia 		ref |= G1_REG_ADDR_REF_TOPC_E;
403*fbb6c848SEzequiel Garcia 	vdpu_write_relaxed(vpu, ref, G1_REG_ADDR_REF(5));
404*fbb6c848SEzequiel Garcia }
405*fbb6c848SEzequiel Garcia 
cfg_buffers(struct hantro_ctx * ctx,const struct v4l2_ctrl_vp8_frame * hdr,struct vb2_v4l2_buffer * vb2_dst)406*fbb6c848SEzequiel Garcia static void cfg_buffers(struct hantro_ctx *ctx,
407*fbb6c848SEzequiel Garcia 			const struct v4l2_ctrl_vp8_frame *hdr,
408*fbb6c848SEzequiel Garcia 			struct vb2_v4l2_buffer *vb2_dst)
409*fbb6c848SEzequiel Garcia {
410*fbb6c848SEzequiel Garcia 	const struct v4l2_vp8_segment *seg = &hdr->segment;
411*fbb6c848SEzequiel Garcia 	struct hantro_dev *vpu = ctx->dev;
412*fbb6c848SEzequiel Garcia 	dma_addr_t dst_dma;
413*fbb6c848SEzequiel Garcia 	u32 reg;
414*fbb6c848SEzequiel Garcia 
415*fbb6c848SEzequiel Garcia 	/* Set probability table buffer address */
416*fbb6c848SEzequiel Garcia 	vdpu_write_relaxed(vpu, ctx->vp8_dec.prob_tbl.dma,
417*fbb6c848SEzequiel Garcia 			   G1_REG_ADDR_QTABLE);
418*fbb6c848SEzequiel Garcia 
419*fbb6c848SEzequiel Garcia 	/* Set segment map address */
420*fbb6c848SEzequiel Garcia 	reg = G1_REG_FWD_PIC1_SEGMENT_BASE(ctx->vp8_dec.segment_map.dma);
421*fbb6c848SEzequiel Garcia 	if (seg->flags & V4L2_VP8_SEGMENT_FLAG_ENABLED) {
422*fbb6c848SEzequiel Garcia 		reg |= G1_REG_FWD_PIC1_SEGMENT_E;
423*fbb6c848SEzequiel Garcia 		if (seg->flags & V4L2_VP8_SEGMENT_FLAG_UPDATE_MAP)
424*fbb6c848SEzequiel Garcia 			reg |= G1_REG_FWD_PIC1_SEGMENT_UPD_E;
425*fbb6c848SEzequiel Garcia 	}
426*fbb6c848SEzequiel Garcia 	vdpu_write_relaxed(vpu, reg, G1_REG_FWD_PIC(0));
427*fbb6c848SEzequiel Garcia 
428*fbb6c848SEzequiel Garcia 	dst_dma = hantro_get_dec_buf_addr(ctx, &vb2_dst->vb2_buf);
429*fbb6c848SEzequiel Garcia 	vdpu_write_relaxed(vpu, dst_dma, G1_REG_ADDR_DST);
430*fbb6c848SEzequiel Garcia }
431*fbb6c848SEzequiel Garcia 
hantro_g1_vp8_dec_run(struct hantro_ctx * ctx)432*fbb6c848SEzequiel Garcia int hantro_g1_vp8_dec_run(struct hantro_ctx *ctx)
433*fbb6c848SEzequiel Garcia {
434*fbb6c848SEzequiel Garcia 	const struct v4l2_ctrl_vp8_frame *hdr;
435*fbb6c848SEzequiel Garcia 	struct hantro_dev *vpu = ctx->dev;
436*fbb6c848SEzequiel Garcia 	struct vb2_v4l2_buffer *vb2_dst;
437*fbb6c848SEzequiel Garcia 	size_t height = ctx->dst_fmt.height;
438*fbb6c848SEzequiel Garcia 	size_t width = ctx->dst_fmt.width;
439*fbb6c848SEzequiel Garcia 	u32 mb_width, mb_height;
440*fbb6c848SEzequiel Garcia 	u32 reg;
441*fbb6c848SEzequiel Garcia 
442*fbb6c848SEzequiel Garcia 	hantro_start_prepare_run(ctx);
443*fbb6c848SEzequiel Garcia 
444*fbb6c848SEzequiel Garcia 	hdr = hantro_get_ctrl(ctx, V4L2_CID_STATELESS_VP8_FRAME);
445*fbb6c848SEzequiel Garcia 	if (WARN_ON(!hdr))
446*fbb6c848SEzequiel Garcia 		return -EINVAL;
447*fbb6c848SEzequiel Garcia 
448*fbb6c848SEzequiel Garcia 	/* Reset segment_map buffer in keyframe */
449*fbb6c848SEzequiel Garcia 	if (V4L2_VP8_FRAME_IS_KEY_FRAME(hdr) && ctx->vp8_dec.segment_map.cpu)
450*fbb6c848SEzequiel Garcia 		memset(ctx->vp8_dec.segment_map.cpu, 0,
451*fbb6c848SEzequiel Garcia 		       ctx->vp8_dec.segment_map.size);
452*fbb6c848SEzequiel Garcia 
453*fbb6c848SEzequiel Garcia 	hantro_vp8_prob_update(ctx, hdr);
454*fbb6c848SEzequiel Garcia 
455*fbb6c848SEzequiel Garcia 	reg = G1_REG_CONFIG_DEC_TIMEOUT_E |
456*fbb6c848SEzequiel Garcia 	      G1_REG_CONFIG_DEC_STRENDIAN_E |
457*fbb6c848SEzequiel Garcia 	      G1_REG_CONFIG_DEC_INSWAP32_E |
458*fbb6c848SEzequiel Garcia 	      G1_REG_CONFIG_DEC_STRSWAP32_E |
459*fbb6c848SEzequiel Garcia 	      G1_REG_CONFIG_DEC_OUTSWAP32_E |
460*fbb6c848SEzequiel Garcia 	      G1_REG_CONFIG_DEC_CLK_GATE_E |
461*fbb6c848SEzequiel Garcia 	      G1_REG_CONFIG_DEC_IN_ENDIAN |
462*fbb6c848SEzequiel Garcia 	      G1_REG_CONFIG_DEC_OUT_ENDIAN |
463*fbb6c848SEzequiel Garcia 	      G1_REG_CONFIG_DEC_MAX_BURST(16);
464*fbb6c848SEzequiel Garcia 	vdpu_write_relaxed(vpu, reg, G1_REG_CONFIG);
465*fbb6c848SEzequiel Garcia 
466*fbb6c848SEzequiel Garcia 	reg = G1_REG_DEC_CTRL0_DEC_MODE(10) |
467*fbb6c848SEzequiel Garcia 	      G1_REG_DEC_CTRL0_DEC_AXI_AUTO;
468*fbb6c848SEzequiel Garcia 	if (!V4L2_VP8_FRAME_IS_KEY_FRAME(hdr))
469*fbb6c848SEzequiel Garcia 		reg |= G1_REG_DEC_CTRL0_PIC_INTER_E;
470*fbb6c848SEzequiel Garcia 	if (!(hdr->flags & V4L2_VP8_FRAME_FLAG_MB_NO_SKIP_COEFF))
471*fbb6c848SEzequiel Garcia 		reg |= G1_REG_DEC_CTRL0_SKIP_MODE;
472*fbb6c848SEzequiel Garcia 	if (hdr->lf.level == 0)
473*fbb6c848SEzequiel Garcia 		reg |= G1_REG_DEC_CTRL0_FILTERING_DIS;
474*fbb6c848SEzequiel Garcia 	vdpu_write_relaxed(vpu, reg, G1_REG_DEC_CTRL0);
475*fbb6c848SEzequiel Garcia 
476*fbb6c848SEzequiel Garcia 	/* Frame dimensions */
477*fbb6c848SEzequiel Garcia 	mb_width = MB_WIDTH(width);
478*fbb6c848SEzequiel Garcia 	mb_height = MB_HEIGHT(height);
479*fbb6c848SEzequiel Garcia 	reg = G1_REG_DEC_CTRL1_PIC_MB_WIDTH(mb_width) |
480*fbb6c848SEzequiel Garcia 	      G1_REG_DEC_CTRL1_PIC_MB_HEIGHT_P(mb_height) |
481*fbb6c848SEzequiel Garcia 	      G1_REG_DEC_CTRL1_PIC_MB_W_EXT(mb_width >> 9) |
482*fbb6c848SEzequiel Garcia 	      G1_REG_DEC_CTRL1_PIC_MB_H_EXT(mb_height >> 8);
483*fbb6c848SEzequiel Garcia 	vdpu_write_relaxed(vpu, reg, G1_REG_DEC_CTRL1);
484*fbb6c848SEzequiel Garcia 
485*fbb6c848SEzequiel Garcia 	/* Boolean decoder */
486*fbb6c848SEzequiel Garcia 	reg = G1_REG_DEC_CTRL2_BOOLEAN_RANGE(hdr->coder_state.range)
487*fbb6c848SEzequiel Garcia 		| G1_REG_DEC_CTRL2_BOOLEAN_VALUE(hdr->coder_state.value);
488*fbb6c848SEzequiel Garcia 	vdpu_write_relaxed(vpu, reg, G1_REG_DEC_CTRL2);
489*fbb6c848SEzequiel Garcia 
490*fbb6c848SEzequiel Garcia 	reg = 0;
491*fbb6c848SEzequiel Garcia 	if (hdr->version != 3)
492*fbb6c848SEzequiel Garcia 		reg |= G1_REG_DEC_CTRL4_VC1_HEIGHT_EXT;
493*fbb6c848SEzequiel Garcia 	if (hdr->version & 0x3)
494*fbb6c848SEzequiel Garcia 		reg |= G1_REG_DEC_CTRL4_BILIN_MC_E;
495*fbb6c848SEzequiel Garcia 	vdpu_write_relaxed(vpu, reg, G1_REG_DEC_CTRL4);
496*fbb6c848SEzequiel Garcia 
497*fbb6c848SEzequiel Garcia 	cfg_lf(ctx, hdr);
498*fbb6c848SEzequiel Garcia 	cfg_qp(ctx, hdr);
499*fbb6c848SEzequiel Garcia 	cfg_parts(ctx, hdr);
500*fbb6c848SEzequiel Garcia 	cfg_tap(ctx, hdr);
501*fbb6c848SEzequiel Garcia 
502*fbb6c848SEzequiel Garcia 	vb2_dst = hantro_get_dst_buf(ctx);
503*fbb6c848SEzequiel Garcia 	cfg_ref(ctx, hdr, vb2_dst);
504*fbb6c848SEzequiel Garcia 	cfg_buffers(ctx, hdr, vb2_dst);
505*fbb6c848SEzequiel Garcia 
506*fbb6c848SEzequiel Garcia 	hantro_end_prepare_run(ctx);
507*fbb6c848SEzequiel Garcia 
508*fbb6c848SEzequiel Garcia 	vdpu_write(vpu, G1_REG_INTERRUPT_DEC_E, G1_REG_INTERRUPT);
509*fbb6c848SEzequiel Garcia 
510*fbb6c848SEzequiel Garcia 	return 0;
511*fbb6c848SEzequiel Garcia }
512