123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566 |
- /*
- * Copyright (C) STMicroelectronics SA 2015
- * Author: Hugues Fruchet <hugues.fruchet@st.com> for STMicroelectronics.
- * License terms: GNU General Public License (GPL), version 2
- */
- #ifndef DELTA_H
- #define DELTA_H
- #include <linux/rpmsg.h>
- #include <media/v4l2-device.h>
- #include <media/v4l2-mem2mem.h>
- #include "delta-cfg.h"
- /*
- * enum delta_state - state of decoding instance
- *
- *@DELTA_STATE_WF_FORMAT:
- * Wait for compressed format to be set by V4L2 client in order
- * to know what is the relevant decoder to open.
- *
- *@DELTA_STATE_WF_STREAMINFO:
- * Wait for stream information to be available (bitstream
- * header parsing is done).
- *
- *@DELTA_STATE_READY:
- * Decoding instance is ready to decode compressed access unit.
- *
- *@DELTA_STATE_WF_EOS:
- * Decoding instance is waiting for EOS (End Of Stream) completion.
- *
- *@DELTA_STATE_EOS:
- * EOS (End Of Stream) is completed (signaled to user). Decoding instance
- * should then be closed.
- */
- enum delta_state {
- DELTA_STATE_WF_FORMAT,
- DELTA_STATE_WF_STREAMINFO,
- DELTA_STATE_READY,
- DELTA_STATE_WF_EOS,
- DELTA_STATE_EOS
- };
- /*
- * struct delta_streaminfo - information about stream to decode
- *
- * @flags: validity of fields (crop, pixelaspect, other)
- * @width: width of video stream
- * @height: height ""
- * @streamformat: fourcc compressed format of video (MJPEG, MPEG2, ...)
- * @dpb: number of frames needed to decode a single frame
- * (h264 dpb, up to 16)
- * @crop: cropping window inside decoded frame (1920x1080@0,0
- * inside 1920x1088 frame for ex.)
- * @pixelaspect: pixel aspect ratio of video (4/3, 5/4)
- * @field: interlaced or not
- * @profile: profile string
- * @level: level string
- * @other: other string information from codec
- * @colorspace: colorspace identifier
- * @xfer_func: transfer function identifier
- * @ycbcr_enc: Y'CbCr encoding identifier
- * @quantization: quantization identifier
- */
- struct delta_streaminfo {
- u32 flags;
- u32 streamformat;
- u32 width;
- u32 height;
- u32 dpb;
- struct v4l2_rect crop;
- struct v4l2_fract pixelaspect;
- enum v4l2_field field;
- u8 profile[32];
- u8 level[32];
- u8 other[32];
- enum v4l2_colorspace colorspace;
- enum v4l2_xfer_func xfer_func;
- enum v4l2_ycbcr_encoding ycbcr_enc;
- enum v4l2_quantization quantization;
- };
- #define DELTA_STREAMINFO_FLAG_CROP 0x0001
- #define DELTA_STREAMINFO_FLAG_PIXELASPECT 0x0002
- #define DELTA_STREAMINFO_FLAG_OTHER 0x0004
- /*
- * struct delta_au - access unit structure.
- *
- * @vbuf: video buffer information for V4L2
- * @list: V4L2 m2m list that the frame belongs to
- * @prepared: if set vaddr/paddr are resolved
- * @vaddr: virtual address (kernel can read/write)
- * @paddr: physical address (for hardware)
- * @flags: access unit type (V4L2_BUF_FLAG_KEYFRAME/PFRAME/BFRAME)
- * @dts: decoding timestamp of this access unit
- */
- struct delta_au {
- struct vb2_v4l2_buffer vbuf; /* keep first */
- struct list_head list; /* keep second */
- bool prepared;
- u32 size;
- void *vaddr;
- dma_addr_t paddr;
- u32 flags;
- u64 dts;
- };
- /*
- * struct delta_frameinfo - information about decoded frame
- *
- * @flags: validity of fields (crop, pixelaspect)
- * @pixelformat: fourcc code for uncompressed video format
- * @width: width of frame
- * @height: height of frame
- * @aligned_width: width of frame (with encoder or decoder alignment
- * constraint)
- * @aligned_height: height of frame (with encoder or decoder alignment
- * constraint)
- * @size: maximum size in bytes required for data
- * @crop: cropping window inside frame (1920x1080@0,0
- * inside 1920x1088 frame for ex.)
- * @pixelaspect: pixel aspect ratio of video (4/3, 5/4)
- * @field: interlaced mode
- * @colorspace: colorspace identifier
- * @xfer_func: transfer function identifier
- * @ycbcr_enc: Y'CbCr encoding identifier
- * @quantization: quantization identifier
- */
- struct delta_frameinfo {
- u32 flags;
- u32 pixelformat;
- u32 width;
- u32 height;
- u32 aligned_width;
- u32 aligned_height;
- u32 size;
- struct v4l2_rect crop;
- struct v4l2_fract pixelaspect;
- enum v4l2_field field;
- enum v4l2_colorspace colorspace;
- enum v4l2_xfer_func xfer_func;
- enum v4l2_ycbcr_encoding ycbcr_enc;
- enum v4l2_quantization quantization;
- };
- #define DELTA_FRAMEINFO_FLAG_CROP 0x0001
- #define DELTA_FRAMEINFO_FLAG_PIXELASPECT 0x0002
- /*
- * struct delta_frame - frame structure.
- *
- * @vbuf: video buffer information for V4L2
- * @list: V4L2 m2m list that the frame belongs to
- * @info: frame information (width, height, format, alignment...)
- * @prepared: if set pix/vaddr/paddr are resolved
- * @index: frame index, aligned on V4L2 wow
- * @vaddr: virtual address (kernel can read/write)
- * @paddr: physical address (for hardware)
- * @state: frame state for frame lifecycle tracking
- * (DELTA_FRAME_FREE/DEC/OUT/REC/...)
- * @flags: frame type (V4L2_BUF_FLAG_KEYFRAME/PFRAME/BFRAME)
- * @dts: decoding timestamp of this frame
- * @field: field order for interlaced frame
- */
- struct delta_frame {
- struct vb2_v4l2_buffer vbuf; /* keep first */
- struct list_head list; /* keep second */
- struct delta_frameinfo info;
- bool prepared;
- u32 index;
- void *vaddr;
- dma_addr_t paddr;
- u32 state;
- u32 flags;
- u64 dts;
- enum v4l2_field field;
- };
- /* frame state for frame lifecycle tracking */
- #define DELTA_FRAME_FREE 0x00 /* is free and can be used for decoding */
- #define DELTA_FRAME_REF 0x01 /* is a reference frame */
- #define DELTA_FRAME_BSY 0x02 /* is owned by decoder and busy */
- #define DELTA_FRAME_DEC 0x04 /* contains decoded content */
- #define DELTA_FRAME_OUT 0x08 /* has been given to user */
- #define DELTA_FRAME_RDY 0x10 /* is ready but still held by decoder */
- #define DELTA_FRAME_M2M 0x20 /* is owned by mem2mem framework */
- /*
- * struct delta_dts - decoding timestamp.
- *
- * @list: list to chain timestamps
- * @val: timestamp in microseconds
- */
- struct delta_dts {
- struct list_head list;
- u64 val;
- };
- struct delta_buf {
- u32 size;
- void *vaddr;
- dma_addr_t paddr;
- const char *name;
- unsigned long attrs;
- };
- struct delta_ipc_ctx {
- int cb_err;
- u32 copro_hdl;
- struct completion done;
- struct delta_buf ipc_buf_struct;
- struct delta_buf *ipc_buf;
- };
- struct delta_ipc_param {
- u32 size;
- void *data;
- };
- struct delta_ctx;
- /*
- * struct delta_dec - decoder structure.
- *
- * @name: name of this decoder
- * @streamformat: input stream format that this decoder support
- * @pixelformat: pixel format of decoded frame that this decoder support
- * @max_width: (optional) maximum width that can decode this decoder
- * if not set, maximum width is DELTA_MAX_WIDTH
- * @max_height: (optional) maximum height that can decode this decoder
- * if not set, maximum height is DELTA_MAX_HEIGHT
- * @pm: (optional) if set, decoder will manage power on its own
- * @open: open this decoder
- * @close: close this decoder
- * @setup_frame: setup frame to be used by decoder, see below
- * @get_streaminfo: get stream related infos, see below
- * @get_frameinfo: get decoded frame related infos, see below
- * @set_frameinfo: (optional) set decoded frame related infos, see below
- * @setup_frame: setup frame to be used by decoder, see below
- * @decode: decode a single access unit, see below
- * @get_frame: get the next decoded frame available, see below
- * @recycle: recycle the given frame, see below
- * @flush: (optional) flush decoder, see below
- * @drain: (optional) drain decoder, see below
- */
- struct delta_dec {
- const char *name;
- u32 streamformat;
- u32 pixelformat;
- u32 max_width;
- u32 max_height;
- bool pm;
- /*
- * decoder ops
- */
- int (*open)(struct delta_ctx *ctx);
- int (*close)(struct delta_ctx *ctx);
- /*
- * setup_frame() - setup frame to be used by decoder
- * @ctx: (in) instance
- * @frame: (in) frame to use
- * @frame.index (in) identifier of frame
- * @frame.vaddr (in) virtual address (kernel can read/write)
- * @frame.paddr (in) physical address (for hardware)
- *
- * Frame is to be allocated by caller, then given
- * to decoder through this call.
- * Several frames must be given to decoder (dpb),
- * each frame is identified using its index.
- */
- int (*setup_frame)(struct delta_ctx *ctx, struct delta_frame *frame);
- /*
- * get_streaminfo() - get stream related infos
- * @ctx: (in) instance
- * @streaminfo: (out) width, height, dpb,...
- *
- * Precondition: stream header must have been successfully
- * parsed to have this call successful & @streaminfo valid.
- * Header parsing must be done using decode(), giving
- * explicitly header access unit or first access unit of bitstream.
- * If no valid header is found, get_streaminfo will return -ENODATA,
- * in this case the next bistream access unit must be decoded till
- * get_streaminfo becomes successful.
- */
- int (*get_streaminfo)(struct delta_ctx *ctx,
- struct delta_streaminfo *streaminfo);
- /*
- * get_frameinfo() - get decoded frame related infos
- * @ctx: (in) instance
- * @frameinfo: (out) width, height, alignment, crop, ...
- *
- * Precondition: get_streaminfo() must be successful
- */
- int (*get_frameinfo)(struct delta_ctx *ctx,
- struct delta_frameinfo *frameinfo);
- /*
- * set_frameinfo() - set decoded frame related infos
- * @ctx: (in) instance
- * @frameinfo: (out) width, height, alignment, crop, ...
- *
- * Optional.
- * Typically used to negotiate with decoder the output
- * frame if decoder can do post-processing.
- */
- int (*set_frameinfo)(struct delta_ctx *ctx,
- struct delta_frameinfo *frameinfo);
- /*
- * decode() - decode a single access unit
- * @ctx: (in) instance
- * @au: (in/out) access unit
- * @au.size (in) size of au to decode
- * @au.vaddr (in) virtual address (kernel can read/write)
- * @au.paddr (in) physical address (for hardware)
- * @au.flags (out) au type (V4L2_BUF_FLAG_KEYFRAME/
- * PFRAME/BFRAME)
- *
- * Decode the access unit given. Decode is synchronous;
- * access unit memory is no more needed after this call.
- * After this call, none, one or several frames could
- * have been decoded, which can be retrieved using
- * get_frame().
- */
- int (*decode)(struct delta_ctx *ctx, struct delta_au *au);
- /*
- * get_frame() - get the next decoded frame available
- * @ctx: (in) instance
- * @frame: (out) frame with decoded data:
- * @frame.index (out) identifier of frame
- * @frame.field (out) field order for interlaced frame
- * @frame.state (out) frame state for frame lifecycle tracking
- * @frame.flags (out) frame type (V4L2_BUF_FLAG_KEYFRAME/
- * PFRAME/BFRAME)
- *
- * Get the next available decoded frame.
- * If no frame is available, -ENODATA is returned.
- * If a frame is available, frame structure is filled with
- * relevant data, frame.index identifying this exact frame.
- * When this frame is no more needed by upper layers,
- * recycle() must be called giving this frame identifier.
- */
- int (*get_frame)(struct delta_ctx *ctx, struct delta_frame **frame);
- /*
- * recycle() - recycle the given frame
- * @ctx: (in) instance
- * @frame: (in) frame to recycle:
- * @frame.index (in) identifier of frame
- *
- * recycle() is to be called by user when the decoded frame
- * is no more needed (composition/display done).
- * This frame will then be reused by decoder to proceed
- * with next frame decoding.
- * If not enough frames have been provided through setup_frame(),
- * or recycle() is not called fast enough, the decoder can run out
- * of available frames to proceed with decoding (starvation).
- * This case is guarded by wq_recycle wait queue which ensures that
- * decoder is called only if at least one frame is available.
- */
- int (*recycle)(struct delta_ctx *ctx, struct delta_frame *frame);
- /*
- * flush() - flush decoder
- * @ctx: (in) instance
- *
- * Optional.
- * Reset decoder context and discard all internal buffers.
- * This allows implementation of seek, which leads to discontinuity
- * of input bitstream that decoder must know to restart its internal
- * decoding logic.
- */
- int (*flush)(struct delta_ctx *ctx);
- /*
- * drain() - drain decoder
- * @ctx: (in) instance
- *
- * Optional.
- * Mark decoder pending frames (decoded but not yet output) as ready
- * so that they can be output to client at EOS (End Of Stream).
- * get_frame() is to be called in a loop right after drain() to
- * get all those pending frames.
- */
- int (*drain)(struct delta_ctx *ctx);
- };
- struct delta_dev;
- /*
- * struct delta_ctx - instance structure.
- *
- * @flags: validity of fields (streaminfo)
- * @fh: V4L2 file handle
- * @dev: device context
- * @dec: selected decoder context for this instance
- * @ipc_ctx: context of IPC communication with firmware
- * @state: instance state
- * @frame_num: frame number
- * @au_num: access unit number
- * @max_au_size: max size of an access unit
- * @streaminfo: stream information (width, height, dpb, interlacing...)
- * @frameinfo: frame information (width, height, format, alignment...)
- * @nb_of_frames: number of frames available for decoding
- * @frames: array of decoding frames to keep track of frame
- * state and manage frame recycling
- * @decoded_frames: nb of decoded frames from opening
- * @output_frames: nb of output frames from opening
- * @dropped_frames: nb of frames dropped (ie access unit not parsed
- * or frame decoded but not output)
- * @stream_errors: nb of stream errors (corrupted, not supported, ...)
- * @decode_errors: nb of decode errors (firmware error)
- * @sys_errors: nb of system errors (memory, ipc, ...)
- * @dts: FIFO of decoding timestamp.
- * output frames are timestamped with incoming access
- * unit timestamps using this fifo.
- * @name: string naming this instance (debug purpose)
- * @run_work: decoding work
- * @lock: lock for decoding work serialization
- * @aborting: true if current job aborted
- * @priv: private decoder context for this instance, allocated
- * by decoder @open time.
- */
- struct delta_ctx {
- u32 flags;
- struct v4l2_fh fh;
- struct delta_dev *dev;
- const struct delta_dec *dec;
- struct delta_ipc_ctx ipc_ctx;
- enum delta_state state;
- u32 frame_num;
- u32 au_num;
- size_t max_au_size;
- struct delta_streaminfo streaminfo;
- struct delta_frameinfo frameinfo;
- u32 nb_of_frames;
- struct delta_frame *frames[DELTA_MAX_FRAMES];
- u32 decoded_frames;
- u32 output_frames;
- u32 dropped_frames;
- u32 stream_errors;
- u32 decode_errors;
- u32 sys_errors;
- struct list_head dts;
- char name[100];
- struct work_struct run_work;
- struct mutex lock;
- bool aborting;
- void *priv;
- };
- #define DELTA_FLAG_STREAMINFO 0x0001
- #define DELTA_FLAG_FRAMEINFO 0x0002
- #define DELTA_MAX_FORMATS DELTA_MAX_DECODERS
- /*
- * struct delta_dev - device struct, 1 per probe (so single one for
- * all platform life)
- *
- * @v4l2_dev: v4l2 device
- * @vdev: v4l2 video device
- * @pdev: platform device
- * @dev: device
- * @m2m_dev: memory-to-memory V4L2 device
- * @lock: device lock, for crit section & V4L2 ops serialization.
- * @clk_delta: delta main clock
- * @clk_st231: st231 coprocessor main clock
- * @clk_flash_promip: flash promip clock
- * @decoders: list of registered decoders
- * @nb_of_decoders: nb of registered decoders
- * @pixelformats: supported uncompressed video formats
- * @nb_of_pixelformats: number of supported umcompressed video formats
- * @streamformats: supported compressed video formats
- * @nb_of_streamformats:number of supported compressed video formats
- * @instance_id: rolling counter identifying an instance (debug purpose)
- * @work_queue: decoding job work queue
- * @rpmsg_driver: rpmsg IPC driver
- * @rpmsg_device: rpmsg IPC device
- */
- struct delta_dev {
- struct v4l2_device v4l2_dev;
- struct video_device *vdev;
- struct platform_device *pdev;
- struct device *dev;
- struct v4l2_m2m_dev *m2m_dev;
- struct mutex lock;
- struct clk *clk_delta;
- struct clk *clk_st231;
- struct clk *clk_flash_promip;
- const struct delta_dec *decoders[DELTA_MAX_DECODERS];
- u32 nb_of_decoders;
- u32 pixelformats[DELTA_MAX_FORMATS];
- u32 nb_of_pixelformats;
- u32 streamformats[DELTA_MAX_FORMATS];
- u32 nb_of_streamformats;
- u8 instance_id;
- struct workqueue_struct *work_queue;
- struct rpmsg_driver rpmsg_driver;
- struct rpmsg_device *rpmsg_device;
- };
- static inline char *frame_type_str(u32 flags)
- {
- if (flags & V4L2_BUF_FLAG_KEYFRAME)
- return "I";
- if (flags & V4L2_BUF_FLAG_PFRAME)
- return "P";
- if (flags & V4L2_BUF_FLAG_BFRAME)
- return "B";
- if (flags & V4L2_BUF_FLAG_LAST)
- return "EOS";
- return "?";
- }
- static inline char *frame_field_str(enum v4l2_field field)
- {
- if (field == V4L2_FIELD_NONE)
- return "-";
- if (field == V4L2_FIELD_TOP)
- return "T";
- if (field == V4L2_FIELD_BOTTOM)
- return "B";
- if (field == V4L2_FIELD_INTERLACED)
- return "I";
- if (field == V4L2_FIELD_INTERLACED_TB)
- return "TB";
- if (field == V4L2_FIELD_INTERLACED_BT)
- return "BT";
- return "?";
- }
- static inline char *frame_state_str(u32 state, char *str, unsigned int len)
- {
- snprintf(str, len, "%s %s %s %s %s %s",
- (state & DELTA_FRAME_REF) ? "ref" : " ",
- (state & DELTA_FRAME_BSY) ? "bsy" : " ",
- (state & DELTA_FRAME_DEC) ? "dec" : " ",
- (state & DELTA_FRAME_OUT) ? "out" : " ",
- (state & DELTA_FRAME_M2M) ? "m2m" : " ",
- (state & DELTA_FRAME_RDY) ? "rdy" : " ");
- return str;
- }
- int delta_get_frameinfo_default(struct delta_ctx *ctx,
- struct delta_frameinfo *frameinfo);
- int delta_recycle_default(struct delta_ctx *pctx,
- struct delta_frame *frame);
- int delta_get_free_frame(struct delta_ctx *ctx,
- struct delta_frame **pframe);
- int delta_get_sync(struct delta_ctx *ctx);
- void delta_put_autosuspend(struct delta_ctx *ctx);
- #endif /* DELTA_H */
|