1*e7b8153eSMauro Carvalho Chehab /* SPDX-License-Identifier: GPL-2.0 */
2*e7b8153eSMauro Carvalho Chehab /*
3*e7b8153eSMauro Carvalho Chehab * Copyright (C) STMicroelectronics SA 2015
4*e7b8153eSMauro Carvalho Chehab * Author: Hugues Fruchet <hugues.fruchet@st.com> for STMicroelectronics.
5*e7b8153eSMauro Carvalho Chehab */
6*e7b8153eSMauro Carvalho Chehab
7*e7b8153eSMauro Carvalho Chehab #ifndef DELTA_H
8*e7b8153eSMauro Carvalho Chehab #define DELTA_H
9*e7b8153eSMauro Carvalho Chehab
10*e7b8153eSMauro Carvalho Chehab #include <linux/rpmsg.h>
11*e7b8153eSMauro Carvalho Chehab #include <media/v4l2-device.h>
12*e7b8153eSMauro Carvalho Chehab #include <media/v4l2-mem2mem.h>
13*e7b8153eSMauro Carvalho Chehab
14*e7b8153eSMauro Carvalho Chehab #include "delta-cfg.h"
15*e7b8153eSMauro Carvalho Chehab
16*e7b8153eSMauro Carvalho Chehab /*
17*e7b8153eSMauro Carvalho Chehab * enum delta_state - state of decoding instance
18*e7b8153eSMauro Carvalho Chehab *
19*e7b8153eSMauro Carvalho Chehab *@DELTA_STATE_WF_FORMAT:
20*e7b8153eSMauro Carvalho Chehab * Wait for compressed format to be set by V4L2 client in order
21*e7b8153eSMauro Carvalho Chehab * to know what is the relevant decoder to open.
22*e7b8153eSMauro Carvalho Chehab *
23*e7b8153eSMauro Carvalho Chehab *@DELTA_STATE_WF_STREAMINFO:
24*e7b8153eSMauro Carvalho Chehab * Wait for stream information to be available (bitstream
25*e7b8153eSMauro Carvalho Chehab * header parsing is done).
26*e7b8153eSMauro Carvalho Chehab *
27*e7b8153eSMauro Carvalho Chehab *@DELTA_STATE_READY:
28*e7b8153eSMauro Carvalho Chehab * Decoding instance is ready to decode compressed access unit.
29*e7b8153eSMauro Carvalho Chehab *
30*e7b8153eSMauro Carvalho Chehab *@DELTA_STATE_WF_EOS:
31*e7b8153eSMauro Carvalho Chehab * Decoding instance is waiting for EOS (End Of Stream) completion.
32*e7b8153eSMauro Carvalho Chehab *
33*e7b8153eSMauro Carvalho Chehab *@DELTA_STATE_EOS:
34*e7b8153eSMauro Carvalho Chehab * EOS (End Of Stream) is completed (signaled to user). Decoding instance
35*e7b8153eSMauro Carvalho Chehab * should then be closed.
36*e7b8153eSMauro Carvalho Chehab */
37*e7b8153eSMauro Carvalho Chehab enum delta_state {
38*e7b8153eSMauro Carvalho Chehab DELTA_STATE_WF_FORMAT,
39*e7b8153eSMauro Carvalho Chehab DELTA_STATE_WF_STREAMINFO,
40*e7b8153eSMauro Carvalho Chehab DELTA_STATE_READY,
41*e7b8153eSMauro Carvalho Chehab DELTA_STATE_WF_EOS,
42*e7b8153eSMauro Carvalho Chehab DELTA_STATE_EOS
43*e7b8153eSMauro Carvalho Chehab };
44*e7b8153eSMauro Carvalho Chehab
45*e7b8153eSMauro Carvalho Chehab /*
46*e7b8153eSMauro Carvalho Chehab * struct delta_streaminfo - information about stream to decode
47*e7b8153eSMauro Carvalho Chehab *
48*e7b8153eSMauro Carvalho Chehab * @flags: validity of fields (crop, pixelaspect, other)
49*e7b8153eSMauro Carvalho Chehab * @width: width of video stream
50*e7b8153eSMauro Carvalho Chehab * @height: height ""
51*e7b8153eSMauro Carvalho Chehab * @streamformat: fourcc compressed format of video (MJPEG, MPEG2, ...)
52*e7b8153eSMauro Carvalho Chehab * @dpb: number of frames needed to decode a single frame
53*e7b8153eSMauro Carvalho Chehab * (h264 dpb, up to 16)
54*e7b8153eSMauro Carvalho Chehab * @crop: cropping window inside decoded frame (1920x1080@0,0
55*e7b8153eSMauro Carvalho Chehab * inside 1920x1088 frame for ex.)
56*e7b8153eSMauro Carvalho Chehab * @pixelaspect: pixel aspect ratio of video (4/3, 5/4)
57*e7b8153eSMauro Carvalho Chehab * @field: interlaced or not
58*e7b8153eSMauro Carvalho Chehab * @profile: profile string
59*e7b8153eSMauro Carvalho Chehab * @level: level string
60*e7b8153eSMauro Carvalho Chehab * @other: other string information from codec
61*e7b8153eSMauro Carvalho Chehab * @colorspace: colorspace identifier
62*e7b8153eSMauro Carvalho Chehab * @xfer_func: transfer function identifier
63*e7b8153eSMauro Carvalho Chehab * @ycbcr_enc: Y'CbCr encoding identifier
64*e7b8153eSMauro Carvalho Chehab * @quantization: quantization identifier
65*e7b8153eSMauro Carvalho Chehab */
66*e7b8153eSMauro Carvalho Chehab struct delta_streaminfo {
67*e7b8153eSMauro Carvalho Chehab u32 flags;
68*e7b8153eSMauro Carvalho Chehab u32 streamformat;
69*e7b8153eSMauro Carvalho Chehab u32 width;
70*e7b8153eSMauro Carvalho Chehab u32 height;
71*e7b8153eSMauro Carvalho Chehab u32 dpb;
72*e7b8153eSMauro Carvalho Chehab struct v4l2_rect crop;
73*e7b8153eSMauro Carvalho Chehab struct v4l2_fract pixelaspect;
74*e7b8153eSMauro Carvalho Chehab enum v4l2_field field;
75*e7b8153eSMauro Carvalho Chehab u8 profile[32];
76*e7b8153eSMauro Carvalho Chehab u8 level[32];
77*e7b8153eSMauro Carvalho Chehab u8 other[32];
78*e7b8153eSMauro Carvalho Chehab enum v4l2_colorspace colorspace;
79*e7b8153eSMauro Carvalho Chehab enum v4l2_xfer_func xfer_func;
80*e7b8153eSMauro Carvalho Chehab enum v4l2_ycbcr_encoding ycbcr_enc;
81*e7b8153eSMauro Carvalho Chehab enum v4l2_quantization quantization;
82*e7b8153eSMauro Carvalho Chehab };
83*e7b8153eSMauro Carvalho Chehab
84*e7b8153eSMauro Carvalho Chehab #define DELTA_STREAMINFO_FLAG_CROP 0x0001
85*e7b8153eSMauro Carvalho Chehab #define DELTA_STREAMINFO_FLAG_PIXELASPECT 0x0002
86*e7b8153eSMauro Carvalho Chehab #define DELTA_STREAMINFO_FLAG_OTHER 0x0004
87*e7b8153eSMauro Carvalho Chehab
88*e7b8153eSMauro Carvalho Chehab /*
89*e7b8153eSMauro Carvalho Chehab * struct delta_au - access unit structure.
90*e7b8153eSMauro Carvalho Chehab *
91*e7b8153eSMauro Carvalho Chehab * @vbuf: video buffer information for V4L2
92*e7b8153eSMauro Carvalho Chehab * @list: V4L2 m2m list that the frame belongs to
93*e7b8153eSMauro Carvalho Chehab * @prepared: if set vaddr/paddr are resolved
94*e7b8153eSMauro Carvalho Chehab * @vaddr: virtual address (kernel can read/write)
95*e7b8153eSMauro Carvalho Chehab * @paddr: physical address (for hardware)
96*e7b8153eSMauro Carvalho Chehab * @flags: access unit type (V4L2_BUF_FLAG_KEYFRAME/PFRAME/BFRAME)
97*e7b8153eSMauro Carvalho Chehab * @dts: decoding timestamp of this access unit
98*e7b8153eSMauro Carvalho Chehab */
99*e7b8153eSMauro Carvalho Chehab struct delta_au {
100*e7b8153eSMauro Carvalho Chehab struct vb2_v4l2_buffer vbuf; /* keep first */
101*e7b8153eSMauro Carvalho Chehab struct list_head list; /* keep second */
102*e7b8153eSMauro Carvalho Chehab
103*e7b8153eSMauro Carvalho Chehab bool prepared;
104*e7b8153eSMauro Carvalho Chehab u32 size;
105*e7b8153eSMauro Carvalho Chehab void *vaddr;
106*e7b8153eSMauro Carvalho Chehab dma_addr_t paddr;
107*e7b8153eSMauro Carvalho Chehab u32 flags;
108*e7b8153eSMauro Carvalho Chehab u64 dts;
109*e7b8153eSMauro Carvalho Chehab };
110*e7b8153eSMauro Carvalho Chehab
111*e7b8153eSMauro Carvalho Chehab /*
112*e7b8153eSMauro Carvalho Chehab * struct delta_frameinfo - information about decoded frame
113*e7b8153eSMauro Carvalho Chehab *
114*e7b8153eSMauro Carvalho Chehab * @flags: validity of fields (crop, pixelaspect)
115*e7b8153eSMauro Carvalho Chehab * @pixelformat: fourcc code for uncompressed video format
116*e7b8153eSMauro Carvalho Chehab * @width: width of frame
117*e7b8153eSMauro Carvalho Chehab * @height: height of frame
118*e7b8153eSMauro Carvalho Chehab * @aligned_width: width of frame (with encoder or decoder alignment
119*e7b8153eSMauro Carvalho Chehab * constraint)
120*e7b8153eSMauro Carvalho Chehab * @aligned_height: height of frame (with encoder or decoder alignment
121*e7b8153eSMauro Carvalho Chehab * constraint)
122*e7b8153eSMauro Carvalho Chehab * @size: maximum size in bytes required for data
123*e7b8153eSMauro Carvalho Chehab * @crop: cropping window inside frame (1920x1080@0,0
124*e7b8153eSMauro Carvalho Chehab * inside 1920x1088 frame for ex.)
125*e7b8153eSMauro Carvalho Chehab * @pixelaspect: pixel aspect ratio of video (4/3, 5/4)
126*e7b8153eSMauro Carvalho Chehab * @field: interlaced mode
127*e7b8153eSMauro Carvalho Chehab * @colorspace: colorspace identifier
128*e7b8153eSMauro Carvalho Chehab * @xfer_func: transfer function identifier
129*e7b8153eSMauro Carvalho Chehab * @ycbcr_enc: Y'CbCr encoding identifier
130*e7b8153eSMauro Carvalho Chehab * @quantization: quantization identifier
131*e7b8153eSMauro Carvalho Chehab */
132*e7b8153eSMauro Carvalho Chehab struct delta_frameinfo {
133*e7b8153eSMauro Carvalho Chehab u32 flags;
134*e7b8153eSMauro Carvalho Chehab u32 pixelformat;
135*e7b8153eSMauro Carvalho Chehab u32 width;
136*e7b8153eSMauro Carvalho Chehab u32 height;
137*e7b8153eSMauro Carvalho Chehab u32 aligned_width;
138*e7b8153eSMauro Carvalho Chehab u32 aligned_height;
139*e7b8153eSMauro Carvalho Chehab u32 size;
140*e7b8153eSMauro Carvalho Chehab struct v4l2_rect crop;
141*e7b8153eSMauro Carvalho Chehab struct v4l2_fract pixelaspect;
142*e7b8153eSMauro Carvalho Chehab enum v4l2_field field;
143*e7b8153eSMauro Carvalho Chehab enum v4l2_colorspace colorspace;
144*e7b8153eSMauro Carvalho Chehab enum v4l2_xfer_func xfer_func;
145*e7b8153eSMauro Carvalho Chehab enum v4l2_ycbcr_encoding ycbcr_enc;
146*e7b8153eSMauro Carvalho Chehab enum v4l2_quantization quantization;
147*e7b8153eSMauro Carvalho Chehab };
148*e7b8153eSMauro Carvalho Chehab
149*e7b8153eSMauro Carvalho Chehab #define DELTA_FRAMEINFO_FLAG_CROP 0x0001
150*e7b8153eSMauro Carvalho Chehab #define DELTA_FRAMEINFO_FLAG_PIXELASPECT 0x0002
151*e7b8153eSMauro Carvalho Chehab
152*e7b8153eSMauro Carvalho Chehab /*
153*e7b8153eSMauro Carvalho Chehab * struct delta_frame - frame structure.
154*e7b8153eSMauro Carvalho Chehab *
155*e7b8153eSMauro Carvalho Chehab * @vbuf: video buffer information for V4L2
156*e7b8153eSMauro Carvalho Chehab * @list: V4L2 m2m list that the frame belongs to
157*e7b8153eSMauro Carvalho Chehab * @info: frame information (width, height, format, alignment...)
158*e7b8153eSMauro Carvalho Chehab * @prepared: if set pix/vaddr/paddr are resolved
159*e7b8153eSMauro Carvalho Chehab * @index: frame index, aligned on V4L2 wow
160*e7b8153eSMauro Carvalho Chehab * @vaddr: virtual address (kernel can read/write)
161*e7b8153eSMauro Carvalho Chehab * @paddr: physical address (for hardware)
162*e7b8153eSMauro Carvalho Chehab * @state: frame state for frame lifecycle tracking
163*e7b8153eSMauro Carvalho Chehab * (DELTA_FRAME_FREE/DEC/OUT/REC/...)
164*e7b8153eSMauro Carvalho Chehab * @flags: frame type (V4L2_BUF_FLAG_KEYFRAME/PFRAME/BFRAME)
165*e7b8153eSMauro Carvalho Chehab * @dts: decoding timestamp of this frame
166*e7b8153eSMauro Carvalho Chehab * @field: field order for interlaced frame
167*e7b8153eSMauro Carvalho Chehab */
168*e7b8153eSMauro Carvalho Chehab struct delta_frame {
169*e7b8153eSMauro Carvalho Chehab struct vb2_v4l2_buffer vbuf; /* keep first */
170*e7b8153eSMauro Carvalho Chehab struct list_head list; /* keep second */
171*e7b8153eSMauro Carvalho Chehab
172*e7b8153eSMauro Carvalho Chehab struct delta_frameinfo info;
173*e7b8153eSMauro Carvalho Chehab bool prepared;
174*e7b8153eSMauro Carvalho Chehab u32 index;
175*e7b8153eSMauro Carvalho Chehab void *vaddr;
176*e7b8153eSMauro Carvalho Chehab dma_addr_t paddr;
177*e7b8153eSMauro Carvalho Chehab u32 state;
178*e7b8153eSMauro Carvalho Chehab u32 flags;
179*e7b8153eSMauro Carvalho Chehab u64 dts;
180*e7b8153eSMauro Carvalho Chehab enum v4l2_field field;
181*e7b8153eSMauro Carvalho Chehab };
182*e7b8153eSMauro Carvalho Chehab
183*e7b8153eSMauro Carvalho Chehab /* frame state for frame lifecycle tracking */
184*e7b8153eSMauro Carvalho Chehab #define DELTA_FRAME_FREE 0x00 /* is free and can be used for decoding */
185*e7b8153eSMauro Carvalho Chehab #define DELTA_FRAME_REF 0x01 /* is a reference frame */
186*e7b8153eSMauro Carvalho Chehab #define DELTA_FRAME_BSY 0x02 /* is owned by decoder and busy */
187*e7b8153eSMauro Carvalho Chehab #define DELTA_FRAME_DEC 0x04 /* contains decoded content */
188*e7b8153eSMauro Carvalho Chehab #define DELTA_FRAME_OUT 0x08 /* has been given to user */
189*e7b8153eSMauro Carvalho Chehab #define DELTA_FRAME_RDY 0x10 /* is ready but still held by decoder */
190*e7b8153eSMauro Carvalho Chehab #define DELTA_FRAME_M2M 0x20 /* is owned by mem2mem framework */
191*e7b8153eSMauro Carvalho Chehab
192*e7b8153eSMauro Carvalho Chehab /*
193*e7b8153eSMauro Carvalho Chehab * struct delta_dts - decoding timestamp.
194*e7b8153eSMauro Carvalho Chehab *
195*e7b8153eSMauro Carvalho Chehab * @list: list to chain timestamps
196*e7b8153eSMauro Carvalho Chehab * @val: timestamp in microseconds
197*e7b8153eSMauro Carvalho Chehab */
198*e7b8153eSMauro Carvalho Chehab struct delta_dts {
199*e7b8153eSMauro Carvalho Chehab struct list_head list;
200*e7b8153eSMauro Carvalho Chehab u64 val;
201*e7b8153eSMauro Carvalho Chehab };
202*e7b8153eSMauro Carvalho Chehab
203*e7b8153eSMauro Carvalho Chehab struct delta_buf {
204*e7b8153eSMauro Carvalho Chehab u32 size;
205*e7b8153eSMauro Carvalho Chehab void *vaddr;
206*e7b8153eSMauro Carvalho Chehab dma_addr_t paddr;
207*e7b8153eSMauro Carvalho Chehab const char *name;
208*e7b8153eSMauro Carvalho Chehab unsigned long attrs;
209*e7b8153eSMauro Carvalho Chehab };
210*e7b8153eSMauro Carvalho Chehab
211*e7b8153eSMauro Carvalho Chehab struct delta_ipc_ctx {
212*e7b8153eSMauro Carvalho Chehab int cb_err;
213*e7b8153eSMauro Carvalho Chehab u32 copro_hdl;
214*e7b8153eSMauro Carvalho Chehab struct completion done;
215*e7b8153eSMauro Carvalho Chehab struct delta_buf ipc_buf_struct;
216*e7b8153eSMauro Carvalho Chehab struct delta_buf *ipc_buf;
217*e7b8153eSMauro Carvalho Chehab };
218*e7b8153eSMauro Carvalho Chehab
219*e7b8153eSMauro Carvalho Chehab struct delta_ipc_param {
220*e7b8153eSMauro Carvalho Chehab u32 size;
221*e7b8153eSMauro Carvalho Chehab void *data;
222*e7b8153eSMauro Carvalho Chehab };
223*e7b8153eSMauro Carvalho Chehab
224*e7b8153eSMauro Carvalho Chehab struct delta_ctx;
225*e7b8153eSMauro Carvalho Chehab
226*e7b8153eSMauro Carvalho Chehab /*
227*e7b8153eSMauro Carvalho Chehab * struct delta_dec - decoder structure.
228*e7b8153eSMauro Carvalho Chehab *
229*e7b8153eSMauro Carvalho Chehab * @name: name of this decoder
230*e7b8153eSMauro Carvalho Chehab * @streamformat: input stream format that this decoder support
231*e7b8153eSMauro Carvalho Chehab * @pixelformat: pixel format of decoded frame that this decoder support
232*e7b8153eSMauro Carvalho Chehab * @max_width: (optional) maximum width that can decode this decoder
233*e7b8153eSMauro Carvalho Chehab * if not set, maximum width is DELTA_MAX_WIDTH
234*e7b8153eSMauro Carvalho Chehab * @max_height: (optional) maximum height that can decode this decoder
235*e7b8153eSMauro Carvalho Chehab * if not set, maximum height is DELTA_MAX_HEIGHT
236*e7b8153eSMauro Carvalho Chehab * @pm: (optional) if set, decoder will manage power on its own
237*e7b8153eSMauro Carvalho Chehab * @open: open this decoder
238*e7b8153eSMauro Carvalho Chehab * @close: close this decoder
239*e7b8153eSMauro Carvalho Chehab * @setup_frame: setup frame to be used by decoder, see below
240*e7b8153eSMauro Carvalho Chehab * @get_streaminfo: get stream related infos, see below
241*e7b8153eSMauro Carvalho Chehab * @get_frameinfo: get decoded frame related infos, see below
242*e7b8153eSMauro Carvalho Chehab * @set_frameinfo: (optional) set decoded frame related infos, see below
243*e7b8153eSMauro Carvalho Chehab * @setup_frame: setup frame to be used by decoder, see below
244*e7b8153eSMauro Carvalho Chehab * @decode: decode a single access unit, see below
245*e7b8153eSMauro Carvalho Chehab * @get_frame: get the next decoded frame available, see below
246*e7b8153eSMauro Carvalho Chehab * @recycle: recycle the given frame, see below
247*e7b8153eSMauro Carvalho Chehab * @flush: (optional) flush decoder, see below
248*e7b8153eSMauro Carvalho Chehab * @drain: (optional) drain decoder, see below
249*e7b8153eSMauro Carvalho Chehab */
250*e7b8153eSMauro Carvalho Chehab struct delta_dec {
251*e7b8153eSMauro Carvalho Chehab const char *name;
252*e7b8153eSMauro Carvalho Chehab u32 streamformat;
253*e7b8153eSMauro Carvalho Chehab u32 pixelformat;
254*e7b8153eSMauro Carvalho Chehab u32 max_width;
255*e7b8153eSMauro Carvalho Chehab u32 max_height;
256*e7b8153eSMauro Carvalho Chehab bool pm;
257*e7b8153eSMauro Carvalho Chehab
258*e7b8153eSMauro Carvalho Chehab /*
259*e7b8153eSMauro Carvalho Chehab * decoder ops
260*e7b8153eSMauro Carvalho Chehab */
261*e7b8153eSMauro Carvalho Chehab int (*open)(struct delta_ctx *ctx);
262*e7b8153eSMauro Carvalho Chehab int (*close)(struct delta_ctx *ctx);
263*e7b8153eSMauro Carvalho Chehab
264*e7b8153eSMauro Carvalho Chehab /*
265*e7b8153eSMauro Carvalho Chehab * setup_frame() - setup frame to be used by decoder
266*e7b8153eSMauro Carvalho Chehab * @ctx: (in) instance
267*e7b8153eSMauro Carvalho Chehab * @frame: (in) frame to use
268*e7b8153eSMauro Carvalho Chehab * @frame.index (in) identifier of frame
269*e7b8153eSMauro Carvalho Chehab * @frame.vaddr (in) virtual address (kernel can read/write)
270*e7b8153eSMauro Carvalho Chehab * @frame.paddr (in) physical address (for hardware)
271*e7b8153eSMauro Carvalho Chehab *
272*e7b8153eSMauro Carvalho Chehab * Frame is to be allocated by caller, then given
273*e7b8153eSMauro Carvalho Chehab * to decoder through this call.
274*e7b8153eSMauro Carvalho Chehab * Several frames must be given to decoder (dpb),
275*e7b8153eSMauro Carvalho Chehab * each frame is identified using its index.
276*e7b8153eSMauro Carvalho Chehab */
277*e7b8153eSMauro Carvalho Chehab int (*setup_frame)(struct delta_ctx *ctx, struct delta_frame *frame);
278*e7b8153eSMauro Carvalho Chehab
279*e7b8153eSMauro Carvalho Chehab /*
280*e7b8153eSMauro Carvalho Chehab * get_streaminfo() - get stream related infos
281*e7b8153eSMauro Carvalho Chehab * @ctx: (in) instance
282*e7b8153eSMauro Carvalho Chehab * @streaminfo: (out) width, height, dpb,...
283*e7b8153eSMauro Carvalho Chehab *
284*e7b8153eSMauro Carvalho Chehab * Precondition: stream header must have been successfully
285*e7b8153eSMauro Carvalho Chehab * parsed to have this call successful & @streaminfo valid.
286*e7b8153eSMauro Carvalho Chehab * Header parsing must be done using decode(), giving
287*e7b8153eSMauro Carvalho Chehab * explicitly header access unit or first access unit of bitstream.
288*e7b8153eSMauro Carvalho Chehab * If no valid header is found, get_streaminfo will return -ENODATA,
289*e7b8153eSMauro Carvalho Chehab * in this case the next bitstream access unit must be decoded till
290*e7b8153eSMauro Carvalho Chehab * get_streaminfo becomes successful.
291*e7b8153eSMauro Carvalho Chehab */
292*e7b8153eSMauro Carvalho Chehab int (*get_streaminfo)(struct delta_ctx *ctx,
293*e7b8153eSMauro Carvalho Chehab struct delta_streaminfo *streaminfo);
294*e7b8153eSMauro Carvalho Chehab
295*e7b8153eSMauro Carvalho Chehab /*
296*e7b8153eSMauro Carvalho Chehab * get_frameinfo() - get decoded frame related infos
297*e7b8153eSMauro Carvalho Chehab * @ctx: (in) instance
298*e7b8153eSMauro Carvalho Chehab * @frameinfo: (out) width, height, alignment, crop, ...
299*e7b8153eSMauro Carvalho Chehab *
300*e7b8153eSMauro Carvalho Chehab * Precondition: get_streaminfo() must be successful
301*e7b8153eSMauro Carvalho Chehab */
302*e7b8153eSMauro Carvalho Chehab int (*get_frameinfo)(struct delta_ctx *ctx,
303*e7b8153eSMauro Carvalho Chehab struct delta_frameinfo *frameinfo);
304*e7b8153eSMauro Carvalho Chehab
305*e7b8153eSMauro Carvalho Chehab /*
306*e7b8153eSMauro Carvalho Chehab * set_frameinfo() - set decoded frame related infos
307*e7b8153eSMauro Carvalho Chehab * @ctx: (in) instance
308*e7b8153eSMauro Carvalho Chehab * @frameinfo: (out) width, height, alignment, crop, ...
309*e7b8153eSMauro Carvalho Chehab *
310*e7b8153eSMauro Carvalho Chehab * Optional.
311*e7b8153eSMauro Carvalho Chehab * Typically used to negotiate with decoder the output
312*e7b8153eSMauro Carvalho Chehab * frame if decoder can do post-processing.
313*e7b8153eSMauro Carvalho Chehab */
314*e7b8153eSMauro Carvalho Chehab int (*set_frameinfo)(struct delta_ctx *ctx,
315*e7b8153eSMauro Carvalho Chehab struct delta_frameinfo *frameinfo);
316*e7b8153eSMauro Carvalho Chehab
317*e7b8153eSMauro Carvalho Chehab /*
318*e7b8153eSMauro Carvalho Chehab * decode() - decode a single access unit
319*e7b8153eSMauro Carvalho Chehab * @ctx: (in) instance
320*e7b8153eSMauro Carvalho Chehab * @au: (in/out) access unit
321*e7b8153eSMauro Carvalho Chehab * @au.size (in) size of au to decode
322*e7b8153eSMauro Carvalho Chehab * @au.vaddr (in) virtual address (kernel can read/write)
323*e7b8153eSMauro Carvalho Chehab * @au.paddr (in) physical address (for hardware)
324*e7b8153eSMauro Carvalho Chehab * @au.flags (out) au type (V4L2_BUF_FLAG_KEYFRAME/
325*e7b8153eSMauro Carvalho Chehab * PFRAME/BFRAME)
326*e7b8153eSMauro Carvalho Chehab *
327*e7b8153eSMauro Carvalho Chehab * Decode the access unit given. Decode is synchronous;
328*e7b8153eSMauro Carvalho Chehab * access unit memory is no more needed after this call.
329*e7b8153eSMauro Carvalho Chehab * After this call, none, one or several frames could
330*e7b8153eSMauro Carvalho Chehab * have been decoded, which can be retrieved using
331*e7b8153eSMauro Carvalho Chehab * get_frame().
332*e7b8153eSMauro Carvalho Chehab */
333*e7b8153eSMauro Carvalho Chehab int (*decode)(struct delta_ctx *ctx, struct delta_au *au);
334*e7b8153eSMauro Carvalho Chehab
335*e7b8153eSMauro Carvalho Chehab /*
336*e7b8153eSMauro Carvalho Chehab * get_frame() - get the next decoded frame available
337*e7b8153eSMauro Carvalho Chehab * @ctx: (in) instance
338*e7b8153eSMauro Carvalho Chehab * @frame: (out) frame with decoded data:
339*e7b8153eSMauro Carvalho Chehab * @frame.index (out) identifier of frame
340*e7b8153eSMauro Carvalho Chehab * @frame.field (out) field order for interlaced frame
341*e7b8153eSMauro Carvalho Chehab * @frame.state (out) frame state for frame lifecycle tracking
342*e7b8153eSMauro Carvalho Chehab * @frame.flags (out) frame type (V4L2_BUF_FLAG_KEYFRAME/
343*e7b8153eSMauro Carvalho Chehab * PFRAME/BFRAME)
344*e7b8153eSMauro Carvalho Chehab *
345*e7b8153eSMauro Carvalho Chehab * Get the next available decoded frame.
346*e7b8153eSMauro Carvalho Chehab * If no frame is available, -ENODATA is returned.
347*e7b8153eSMauro Carvalho Chehab * If a frame is available, frame structure is filled with
348*e7b8153eSMauro Carvalho Chehab * relevant data, frame.index identifying this exact frame.
349*e7b8153eSMauro Carvalho Chehab * When this frame is no more needed by upper layers,
350*e7b8153eSMauro Carvalho Chehab * recycle() must be called giving this frame identifier.
351*e7b8153eSMauro Carvalho Chehab */
352*e7b8153eSMauro Carvalho Chehab int (*get_frame)(struct delta_ctx *ctx, struct delta_frame **frame);
353*e7b8153eSMauro Carvalho Chehab
354*e7b8153eSMauro Carvalho Chehab /*
355*e7b8153eSMauro Carvalho Chehab * recycle() - recycle the given frame
356*e7b8153eSMauro Carvalho Chehab * @ctx: (in) instance
357*e7b8153eSMauro Carvalho Chehab * @frame: (in) frame to recycle:
358*e7b8153eSMauro Carvalho Chehab * @frame.index (in) identifier of frame
359*e7b8153eSMauro Carvalho Chehab *
360*e7b8153eSMauro Carvalho Chehab * recycle() is to be called by user when the decoded frame
361*e7b8153eSMauro Carvalho Chehab * is no more needed (composition/display done).
362*e7b8153eSMauro Carvalho Chehab * This frame will then be reused by decoder to proceed
363*e7b8153eSMauro Carvalho Chehab * with next frame decoding.
364*e7b8153eSMauro Carvalho Chehab * If not enough frames have been provided through setup_frame(),
365*e7b8153eSMauro Carvalho Chehab * or recycle() is not called fast enough, the decoder can run out
366*e7b8153eSMauro Carvalho Chehab * of available frames to proceed with decoding (starvation).
367*e7b8153eSMauro Carvalho Chehab * This case is guarded by wq_recycle wait queue which ensures that
368*e7b8153eSMauro Carvalho Chehab * decoder is called only if at least one frame is available.
369*e7b8153eSMauro Carvalho Chehab */
370*e7b8153eSMauro Carvalho Chehab int (*recycle)(struct delta_ctx *ctx, struct delta_frame *frame);
371*e7b8153eSMauro Carvalho Chehab
372*e7b8153eSMauro Carvalho Chehab /*
373*e7b8153eSMauro Carvalho Chehab * flush() - flush decoder
374*e7b8153eSMauro Carvalho Chehab * @ctx: (in) instance
375*e7b8153eSMauro Carvalho Chehab *
376*e7b8153eSMauro Carvalho Chehab * Optional.
377*e7b8153eSMauro Carvalho Chehab * Reset decoder context and discard all internal buffers.
378*e7b8153eSMauro Carvalho Chehab * This allows implementation of seek, which leads to discontinuity
379*e7b8153eSMauro Carvalho Chehab * of input bitstream that decoder must know to restart its internal
380*e7b8153eSMauro Carvalho Chehab * decoding logic.
381*e7b8153eSMauro Carvalho Chehab */
382*e7b8153eSMauro Carvalho Chehab int (*flush)(struct delta_ctx *ctx);
383*e7b8153eSMauro Carvalho Chehab
384*e7b8153eSMauro Carvalho Chehab /*
385*e7b8153eSMauro Carvalho Chehab * drain() - drain decoder
386*e7b8153eSMauro Carvalho Chehab * @ctx: (in) instance
387*e7b8153eSMauro Carvalho Chehab *
388*e7b8153eSMauro Carvalho Chehab * Optional.
389*e7b8153eSMauro Carvalho Chehab * Mark decoder pending frames (decoded but not yet output) as ready
390*e7b8153eSMauro Carvalho Chehab * so that they can be output to client at EOS (End Of Stream).
391*e7b8153eSMauro Carvalho Chehab * get_frame() is to be called in a loop right after drain() to
392*e7b8153eSMauro Carvalho Chehab * get all those pending frames.
393*e7b8153eSMauro Carvalho Chehab */
394*e7b8153eSMauro Carvalho Chehab int (*drain)(struct delta_ctx *ctx);
395*e7b8153eSMauro Carvalho Chehab };
396*e7b8153eSMauro Carvalho Chehab
397*e7b8153eSMauro Carvalho Chehab struct delta_dev;
398*e7b8153eSMauro Carvalho Chehab
399*e7b8153eSMauro Carvalho Chehab /*
400*e7b8153eSMauro Carvalho Chehab * struct delta_ctx - instance structure.
401*e7b8153eSMauro Carvalho Chehab *
402*e7b8153eSMauro Carvalho Chehab * @flags: validity of fields (streaminfo)
403*e7b8153eSMauro Carvalho Chehab * @fh: V4L2 file handle
404*e7b8153eSMauro Carvalho Chehab * @dev: device context
405*e7b8153eSMauro Carvalho Chehab * @dec: selected decoder context for this instance
406*e7b8153eSMauro Carvalho Chehab * @ipc_ctx: context of IPC communication with firmware
407*e7b8153eSMauro Carvalho Chehab * @state: instance state
408*e7b8153eSMauro Carvalho Chehab * @frame_num: frame number
409*e7b8153eSMauro Carvalho Chehab * @au_num: access unit number
410*e7b8153eSMauro Carvalho Chehab * @max_au_size: max size of an access unit
411*e7b8153eSMauro Carvalho Chehab * @streaminfo: stream information (width, height, dpb, interlacing...)
412*e7b8153eSMauro Carvalho Chehab * @frameinfo: frame information (width, height, format, alignment...)
413*e7b8153eSMauro Carvalho Chehab * @nb_of_frames: number of frames available for decoding
414*e7b8153eSMauro Carvalho Chehab * @frames: array of decoding frames to keep track of frame
415*e7b8153eSMauro Carvalho Chehab * state and manage frame recycling
416*e7b8153eSMauro Carvalho Chehab * @decoded_frames: nb of decoded frames from opening
417*e7b8153eSMauro Carvalho Chehab * @output_frames: nb of output frames from opening
418*e7b8153eSMauro Carvalho Chehab * @dropped_frames: nb of frames dropped (ie access unit not parsed
419*e7b8153eSMauro Carvalho Chehab * or frame decoded but not output)
420*e7b8153eSMauro Carvalho Chehab * @stream_errors: nb of stream errors (corrupted, not supported, ...)
421*e7b8153eSMauro Carvalho Chehab * @decode_errors: nb of decode errors (firmware error)
422*e7b8153eSMauro Carvalho Chehab * @sys_errors: nb of system errors (memory, ipc, ...)
423*e7b8153eSMauro Carvalho Chehab * @dts: FIFO of decoding timestamp.
424*e7b8153eSMauro Carvalho Chehab * output frames are timestamped with incoming access
425*e7b8153eSMauro Carvalho Chehab * unit timestamps using this fifo.
426*e7b8153eSMauro Carvalho Chehab * @name: string naming this instance (debug purpose)
427*e7b8153eSMauro Carvalho Chehab * @run_work: decoding work
428*e7b8153eSMauro Carvalho Chehab * @lock: lock for decoding work serialization
429*e7b8153eSMauro Carvalho Chehab * @aborting: true if current job aborted
430*e7b8153eSMauro Carvalho Chehab * @priv: private decoder context for this instance, allocated
431*e7b8153eSMauro Carvalho Chehab * by decoder @open time.
432*e7b8153eSMauro Carvalho Chehab */
433*e7b8153eSMauro Carvalho Chehab struct delta_ctx {
434*e7b8153eSMauro Carvalho Chehab u32 flags;
435*e7b8153eSMauro Carvalho Chehab struct v4l2_fh fh;
436*e7b8153eSMauro Carvalho Chehab struct delta_dev *dev;
437*e7b8153eSMauro Carvalho Chehab const struct delta_dec *dec;
438*e7b8153eSMauro Carvalho Chehab struct delta_ipc_ctx ipc_ctx;
439*e7b8153eSMauro Carvalho Chehab
440*e7b8153eSMauro Carvalho Chehab enum delta_state state;
441*e7b8153eSMauro Carvalho Chehab u32 frame_num;
442*e7b8153eSMauro Carvalho Chehab u32 au_num;
443*e7b8153eSMauro Carvalho Chehab size_t max_au_size;
444*e7b8153eSMauro Carvalho Chehab struct delta_streaminfo streaminfo;
445*e7b8153eSMauro Carvalho Chehab struct delta_frameinfo frameinfo;
446*e7b8153eSMauro Carvalho Chehab u32 nb_of_frames;
447*e7b8153eSMauro Carvalho Chehab struct delta_frame *frames[DELTA_MAX_FRAMES];
448*e7b8153eSMauro Carvalho Chehab u32 decoded_frames;
449*e7b8153eSMauro Carvalho Chehab u32 output_frames;
450*e7b8153eSMauro Carvalho Chehab u32 dropped_frames;
451*e7b8153eSMauro Carvalho Chehab u32 stream_errors;
452*e7b8153eSMauro Carvalho Chehab u32 decode_errors;
453*e7b8153eSMauro Carvalho Chehab u32 sys_errors;
454*e7b8153eSMauro Carvalho Chehab struct list_head dts;
455*e7b8153eSMauro Carvalho Chehab char name[100];
456*e7b8153eSMauro Carvalho Chehab struct work_struct run_work;
457*e7b8153eSMauro Carvalho Chehab struct mutex lock;
458*e7b8153eSMauro Carvalho Chehab bool aborting;
459*e7b8153eSMauro Carvalho Chehab void *priv;
460*e7b8153eSMauro Carvalho Chehab };
461*e7b8153eSMauro Carvalho Chehab
462*e7b8153eSMauro Carvalho Chehab #define DELTA_FLAG_STREAMINFO 0x0001
463*e7b8153eSMauro Carvalho Chehab #define DELTA_FLAG_FRAMEINFO 0x0002
464*e7b8153eSMauro Carvalho Chehab
465*e7b8153eSMauro Carvalho Chehab #define DELTA_MAX_FORMATS DELTA_MAX_DECODERS
466*e7b8153eSMauro Carvalho Chehab
467*e7b8153eSMauro Carvalho Chehab /*
468*e7b8153eSMauro Carvalho Chehab * struct delta_dev - device struct, 1 per probe (so single one for
469*e7b8153eSMauro Carvalho Chehab * all platform life)
470*e7b8153eSMauro Carvalho Chehab *
471*e7b8153eSMauro Carvalho Chehab * @v4l2_dev: v4l2 device
472*e7b8153eSMauro Carvalho Chehab * @vdev: v4l2 video device
473*e7b8153eSMauro Carvalho Chehab * @pdev: platform device
474*e7b8153eSMauro Carvalho Chehab * @dev: device
475*e7b8153eSMauro Carvalho Chehab * @m2m_dev: memory-to-memory V4L2 device
476*e7b8153eSMauro Carvalho Chehab * @lock: device lock, for crit section & V4L2 ops serialization.
477*e7b8153eSMauro Carvalho Chehab * @clk_delta: delta main clock
478*e7b8153eSMauro Carvalho Chehab * @clk_st231: st231 coprocessor main clock
479*e7b8153eSMauro Carvalho Chehab * @clk_flash_promip: flash promip clock
480*e7b8153eSMauro Carvalho Chehab * @decoders: list of registered decoders
481*e7b8153eSMauro Carvalho Chehab * @nb_of_decoders: nb of registered decoders
482*e7b8153eSMauro Carvalho Chehab * @pixelformats: supported uncompressed video formats
483*e7b8153eSMauro Carvalho Chehab * @nb_of_pixelformats: number of supported umcompressed video formats
484*e7b8153eSMauro Carvalho Chehab * @streamformats: supported compressed video formats
485*e7b8153eSMauro Carvalho Chehab * @nb_of_streamformats:number of supported compressed video formats
486*e7b8153eSMauro Carvalho Chehab * @instance_id: rolling counter identifying an instance (debug purpose)
487*e7b8153eSMauro Carvalho Chehab * @work_queue: decoding job work queue
488*e7b8153eSMauro Carvalho Chehab * @rpmsg_driver: rpmsg IPC driver
489*e7b8153eSMauro Carvalho Chehab * @rpmsg_device: rpmsg IPC device
490*e7b8153eSMauro Carvalho Chehab */
491*e7b8153eSMauro Carvalho Chehab struct delta_dev {
492*e7b8153eSMauro Carvalho Chehab struct v4l2_device v4l2_dev;
493*e7b8153eSMauro Carvalho Chehab struct video_device *vdev;
494*e7b8153eSMauro Carvalho Chehab struct platform_device *pdev;
495*e7b8153eSMauro Carvalho Chehab struct device *dev;
496*e7b8153eSMauro Carvalho Chehab struct v4l2_m2m_dev *m2m_dev;
497*e7b8153eSMauro Carvalho Chehab struct mutex lock;
498*e7b8153eSMauro Carvalho Chehab struct clk *clk_delta;
499*e7b8153eSMauro Carvalho Chehab struct clk *clk_st231;
500*e7b8153eSMauro Carvalho Chehab struct clk *clk_flash_promip;
501*e7b8153eSMauro Carvalho Chehab const struct delta_dec *decoders[DELTA_MAX_DECODERS];
502*e7b8153eSMauro Carvalho Chehab u32 nb_of_decoders;
503*e7b8153eSMauro Carvalho Chehab u32 pixelformats[DELTA_MAX_FORMATS];
504*e7b8153eSMauro Carvalho Chehab u32 nb_of_pixelformats;
505*e7b8153eSMauro Carvalho Chehab u32 streamformats[DELTA_MAX_FORMATS];
506*e7b8153eSMauro Carvalho Chehab u32 nb_of_streamformats;
507*e7b8153eSMauro Carvalho Chehab u8 instance_id;
508*e7b8153eSMauro Carvalho Chehab struct workqueue_struct *work_queue;
509*e7b8153eSMauro Carvalho Chehab struct rpmsg_driver rpmsg_driver;
510*e7b8153eSMauro Carvalho Chehab struct rpmsg_device *rpmsg_device;
511*e7b8153eSMauro Carvalho Chehab };
512*e7b8153eSMauro Carvalho Chehab
frame_type_str(u32 flags)513*e7b8153eSMauro Carvalho Chehab static inline char *frame_type_str(u32 flags)
514*e7b8153eSMauro Carvalho Chehab {
515*e7b8153eSMauro Carvalho Chehab if (flags & V4L2_BUF_FLAG_KEYFRAME)
516*e7b8153eSMauro Carvalho Chehab return "I";
517*e7b8153eSMauro Carvalho Chehab if (flags & V4L2_BUF_FLAG_PFRAME)
518*e7b8153eSMauro Carvalho Chehab return "P";
519*e7b8153eSMauro Carvalho Chehab if (flags & V4L2_BUF_FLAG_BFRAME)
520*e7b8153eSMauro Carvalho Chehab return "B";
521*e7b8153eSMauro Carvalho Chehab if (flags & V4L2_BUF_FLAG_LAST)
522*e7b8153eSMauro Carvalho Chehab return "EOS";
523*e7b8153eSMauro Carvalho Chehab return "?";
524*e7b8153eSMauro Carvalho Chehab }
525*e7b8153eSMauro Carvalho Chehab
frame_field_str(enum v4l2_field field)526*e7b8153eSMauro Carvalho Chehab static inline char *frame_field_str(enum v4l2_field field)
527*e7b8153eSMauro Carvalho Chehab {
528*e7b8153eSMauro Carvalho Chehab if (field == V4L2_FIELD_NONE)
529*e7b8153eSMauro Carvalho Chehab return "-";
530*e7b8153eSMauro Carvalho Chehab if (field == V4L2_FIELD_TOP)
531*e7b8153eSMauro Carvalho Chehab return "T";
532*e7b8153eSMauro Carvalho Chehab if (field == V4L2_FIELD_BOTTOM)
533*e7b8153eSMauro Carvalho Chehab return "B";
534*e7b8153eSMauro Carvalho Chehab if (field == V4L2_FIELD_INTERLACED)
535*e7b8153eSMauro Carvalho Chehab return "I";
536*e7b8153eSMauro Carvalho Chehab if (field == V4L2_FIELD_INTERLACED_TB)
537*e7b8153eSMauro Carvalho Chehab return "TB";
538*e7b8153eSMauro Carvalho Chehab if (field == V4L2_FIELD_INTERLACED_BT)
539*e7b8153eSMauro Carvalho Chehab return "BT";
540*e7b8153eSMauro Carvalho Chehab return "?";
541*e7b8153eSMauro Carvalho Chehab }
542*e7b8153eSMauro Carvalho Chehab
frame_state_str(u32 state,char * str,unsigned int len)543*e7b8153eSMauro Carvalho Chehab static inline char *frame_state_str(u32 state, char *str, unsigned int len)
544*e7b8153eSMauro Carvalho Chehab {
545*e7b8153eSMauro Carvalho Chehab snprintf(str, len, "%s %s %s %s %s %s",
546*e7b8153eSMauro Carvalho Chehab (state & DELTA_FRAME_REF) ? "ref" : " ",
547*e7b8153eSMauro Carvalho Chehab (state & DELTA_FRAME_BSY) ? "bsy" : " ",
548*e7b8153eSMauro Carvalho Chehab (state & DELTA_FRAME_DEC) ? "dec" : " ",
549*e7b8153eSMauro Carvalho Chehab (state & DELTA_FRAME_OUT) ? "out" : " ",
550*e7b8153eSMauro Carvalho Chehab (state & DELTA_FRAME_M2M) ? "m2m" : " ",
551*e7b8153eSMauro Carvalho Chehab (state & DELTA_FRAME_RDY) ? "rdy" : " ");
552*e7b8153eSMauro Carvalho Chehab return str;
553*e7b8153eSMauro Carvalho Chehab }
554*e7b8153eSMauro Carvalho Chehab
555*e7b8153eSMauro Carvalho Chehab int delta_get_frameinfo_default(struct delta_ctx *ctx,
556*e7b8153eSMauro Carvalho Chehab struct delta_frameinfo *frameinfo);
557*e7b8153eSMauro Carvalho Chehab int delta_recycle_default(struct delta_ctx *pctx,
558*e7b8153eSMauro Carvalho Chehab struct delta_frame *frame);
559*e7b8153eSMauro Carvalho Chehab
560*e7b8153eSMauro Carvalho Chehab int delta_get_free_frame(struct delta_ctx *ctx,
561*e7b8153eSMauro Carvalho Chehab struct delta_frame **pframe);
562*e7b8153eSMauro Carvalho Chehab
563*e7b8153eSMauro Carvalho Chehab int delta_get_sync(struct delta_ctx *ctx);
564*e7b8153eSMauro Carvalho Chehab void delta_put_autosuspend(struct delta_ctx *ctx);
565*e7b8153eSMauro Carvalho Chehab
566*e7b8153eSMauro Carvalho Chehab #endif /* DELTA_H */
567