On Thu, 2018-11-22 at 19:20 +0900, Tomasz Figa wrote: > Hi Ezequiel, > > On Thu, Nov 22, 2018 at 4:59 AM Ezequiel Garcia <ezequiel@xxxxxxxxxxxxx> wrote: > > Add a mem2mem driver for the VPU available on Rockchip SoCs. > > Currently only JPEG encoding is supported, for RK3399 and RK3288 > > platforms. > > > > Signed-off-by: Ezequiel Garcia <ezequiel@xxxxxxxxxxxxx> > > Sorry for being late to the party. Please see my comments inline. > No problem! Thanks for the detailed review. > [snip] > > diff --git a/drivers/staging/media/rockchip/vpu/rk3288_vpu_hw.c b/drivers/staging/media/rockchip/vpu/rk3288_vpu_hw.c > > new file mode 100644 > > index 000000000000..75b7abbd3aca > > --- /dev/null > > +++ b/drivers/staging/media/rockchip/vpu/rk3288_vpu_hw.c > > @@ -0,0 +1,118 @@ > > +// SPDX-License-Identifier: GPL-2.0 > > +/* > > + * Rockchip VPU codec driver > > + * > > + * Copyright (C) 2018 Rockchip Electronics Co., Ltd. > > + * Jeffy Chen <jeffy.chen@xxxxxxxxxxxxxx> > > + */ > > + > > +#include <linux/clk.h> > > + > > +#include "rockchip_vpu.h" > > +#include "rockchip_vpu_jpeg.h" > > +#include "rk3288_vpu_regs.h" > > + > > +#define RK3288_ACLK_MAX_FREQ (400 * 1000 * 1000) > > + > > +/* > > + * Supported formats. > > + */ > > + > > +static const struct rockchip_vpu_fmt rk3288_vpu_enc_fmts[] = { > > + { > > + .fourcc = V4L2_PIX_FMT_YUV420M, > > + .codec_mode = RK_VPU_MODE_NONE, > > + .enc_fmt = RK3288_VPU_ENC_FMT_YUV420P, > > + }, > > + { > > + .fourcc = V4L2_PIX_FMT_NV12M, > > + .codec_mode = RK_VPU_MODE_NONE, > > + .enc_fmt = RK3288_VPU_ENC_FMT_YUV420SP, > > + }, > > + { > > + .fourcc = V4L2_PIX_FMT_YUYV, > > + .codec_mode = RK_VPU_MODE_NONE, > > + .enc_fmt = RK3288_VPU_ENC_FMT_YUYV422, > > + }, > > + { > > + .fourcc = V4L2_PIX_FMT_UYVY, > > + .codec_mode = RK_VPU_MODE_NONE, > > + .enc_fmt = RK3288_VPU_ENC_FMT_UYVY422, > > + }, > > + { > > + .fourcc = V4L2_PIX_FMT_JPEG, > > + .codec_mode = RK_VPU_MODE_JPEG_ENC, > > + .max_depth = 2, > > + .header_size = JPEG_HEADER_SIZE, > > + .frmsize = { > > + .min_width = 96, > > + .max_width = 8192, > > + .step_width = MB_DIM, > > + .min_height = 32, > > + .max_height = 8192, > > + .step_height = MB_DIM, > > + }, > > + }, > > +}; > > + > > +static irqreturn_t rk3288_vepu_irq(int irq, void *dev_id) > > +{ > > + struct rockchip_vpu_dev *vpu = dev_id; > > + enum vb2_buffer_state state; > > + u32 status, bytesused; > > + > > + status = vepu_read(vpu, VEPU_REG_INTERRUPT); > > + bytesused = vepu_read(vpu, VEPU_REG_STR_BUF_LIMIT) / 8; > > + state = (status & VEPU_REG_INTERRUPT_FRAME_RDY) ? > > + VB2_BUF_STATE_DONE : VB2_BUF_STATE_ERROR; > > + > > + vepu_write(vpu, 0, VEPU_REG_INTERRUPT); > > + vepu_write(vpu, 0, VEPU_REG_AXI_CTRL); > > + > > + rockchip_vpu_irq_done(vpu, bytesused, state); > > + > > + return IRQ_HANDLED; > > +} > > + > > +static int rk3288_vpu_hw_init(struct rockchip_vpu_dev *vpu) > > +{ > > + /* Bump ACLK to max. possible freq. to improve performance. */ > > + clk_set_rate(vpu->clocks[0].clk, RK3288_ACLK_MAX_FREQ); > > + return 0; > > +} > > + > > +static void rk3288_vpu_enc_reset(struct rockchip_vpu_ctx *ctx) > > +{ > > + struct rockchip_vpu_dev *vpu = ctx->dev; > > + > > + vepu_write(vpu, VEPU_REG_INTERRUPT_DIS_BIT, VEPU_REG_INTERRUPT); > > + vepu_write(vpu, 0, VEPU_REG_ENC_CTRL); > > + vepu_write(vpu, 0, VEPU_REG_AXI_CTRL); > > +} > > + > > +/* > > + * Supported codec ops. > > + */ > > + > > +static const struct rockchip_vpu_codec_ops rk3288_vpu_codec_ops[] = { > > + [RK_VPU_MODE_JPEG_ENC] = { > > + .run = rk3288_vpu_jpeg_enc_run, > > + .reset = rk3288_vpu_enc_reset, > > + }, > > +}; > > + > > +/* > > + * VPU variant. > > + */ > > + > > +const struct rockchip_vpu_variant rk3288_vpu_variant = { > > + .enc_offset = 0x0, > > + .enc_fmts = rk3288_vpu_enc_fmts, > > + .num_enc_fmts = ARRAY_SIZE(rk3288_vpu_enc_fmts), > > + .codec_ops = rk3288_vpu_codec_ops, > > + .codec = RK_VPU_CODEC_JPEG, > > + .vepu_irq = rk3288_vepu_irq, > > + .init = rk3288_vpu_hw_init, > > + .clk_names = {"aclk", "hclk"}, > > nit: Spaces inside the brackets. > You mean you this style is prefered? .clk_names = { "aclk", "hclk" }, Grepping thru sources, it seems there is no convention on this, so it's your call. + .num_clocks = 2 > > +}; > > diff --git a/drivers/staging/media/rockchip/vpu/rk3288_vpu_hw_jpeg_enc.c b/drivers/staging/media/rockchip/vpu/rk3288_vpu_hw_jpeg_enc.c > > new file mode 100644 > > index 000000000000..1ea60bd5e1e6 > > --- /dev/null > > +++ b/drivers/staging/media/rockchip/vpu/rk3288_vpu_hw_jpeg_enc.c > > @@ -0,0 +1,133 @@ > > +// SPDX-License-Identifier: GPL-2.0 > > +/* > > + * Rockchip VPU codec driver > > + * > > + * Copyright (C) 2018 Rockchip Electronics Co., Ltd. > > + */ > > + > > +#include <asm/unaligned.h> > > +#include <media/v4l2-mem2mem.h> > > +#include "rockchip_vpu_jpeg.h" > > +#include "rockchip_vpu.h" > > +#include "rockchip_vpu_common.h" > > +#include "rockchip_vpu_hw.h" > > +#include "rk3288_vpu_regs.h" > > + > > +#define VEPU_JPEG_QUANT_TABLE_COUNT 16 > > + > > +static void rk3288_vpu_set_src_img_ctrl(struct rockchip_vpu_dev *vpu, > > + struct rockchip_vpu_ctx *ctx) > > +{ > > + struct v4l2_pix_format_mplane *pix_fmt = &ctx->src_fmt; > > + u32 reg; > > + > > + reg = VEPU_REG_IN_IMG_CTRL_ROW_LEN(pix_fmt->width) > > + | VEPU_REG_IN_IMG_CTRL_OVRFLR_D4(0) > > + | VEPU_REG_IN_IMG_CTRL_OVRFLB_D4(0) > > + | VEPU_REG_IN_IMG_CTRL_FMT(ctx->vpu_src_fmt->enc_fmt); > > + vepu_write_relaxed(vpu, reg, VEPU_REG_IN_IMG_CTRL); > > +} > > + > > +static void rk3288_vpu_jpeg_enc_set_buffers(struct rockchip_vpu_dev *vpu, > > + struct rockchip_vpu_ctx *ctx, > > + struct vb2_buffer *src_buf) > > +{ > > + struct v4l2_pix_format_mplane *pix_fmt = &ctx->src_fmt; > > + dma_addr_t src[3]; > > + > > + WARN_ON(pix_fmt->num_planes > 3); > > + > > + vepu_write_relaxed(vpu, ctx->bounce_dma_addr, > > + VEPU_REG_ADDR_OUTPUT_STREAM); > > + vepu_write_relaxed(vpu, ctx->bounce_size, > > + VEPU_REG_STR_BUF_LIMIT); > > + > > + if (pix_fmt->num_planes == 1) { > > + src[0] = vb2_dma_contig_plane_dma_addr(src_buf, 0); > > + /* single plane formats we supported are all interlaced */ > > + src[1] = src[0]; > > + src[2] = src[0]; > > + } else if (pix_fmt->num_planes == 2) { > > + src[0] = vb2_dma_contig_plane_dma_addr(src_buf, 0); > > + src[1] = vb2_dma_contig_plane_dma_addr(src_buf, 1); > > + src[2] = src[1]; > > In my testing, the value for VEPU_REG_ADDR_IN_CR seemed to be ignored > for NV12, so possibly the registers are just misnamed and should be > called VEPU_REG_ADDR_IN_PLANE0, 1, 2? > Right, and only write those that are needed. > > + } else { > > + src[0] = vb2_dma_contig_plane_dma_addr(src_buf, 0); > > + src[1] = vb2_dma_contig_plane_dma_addr(src_buf, 1); > > + src[2] = vb2_dma_contig_plane_dma_addr(src_buf, 2); > > + } > > + > > + vepu_write_relaxed(vpu, src[0], VEPU_REG_ADDR_IN_LUMA); > > + vepu_write_relaxed(vpu, src[2], VEPU_REG_ADDR_IN_CR); > > + vepu_write_relaxed(vpu, src[1], VEPU_REG_ADDR_IN_CB); > > nit: Any reason not to swap the last 2 lines? > I think it's just a leftover that felt thru the cracks. > > +} > > + > > +static void > > +rk3288_vpu_jpeg_enc_set_qtable(struct rockchip_vpu_dev *vpu, > > + unsigned char *luma_qtable, > > + unsigned char *chroma_qtable) > > +{ > > + __be32 *luma_qtable_p; > > + __be32 *chroma_qtable_p; > > + u32 reg, i; > > + > > + luma_qtable_p = (__be32 *)luma_qtable; > > + chroma_qtable_p = (__be32 *)chroma_qtable; > > + > > + for (i = 0; i < VEPU_JPEG_QUANT_TABLE_COUNT; i++) { > > + reg = get_unaligned_be32(&luma_qtable[i]); > > + vepu_write_relaxed(vpu, reg, VEPU_REG_JPEG_LUMA_QUAT(i)); > > + > > + reg = get_unaligned_be32(&chroma_qtable[i]); > > + vepu_write_relaxed(vpu, reg, VEPU_REG_JPEG_CHROMA_QUAT(i)); > > + } > > +} > > + > > +void rk3288_vpu_jpeg_enc_run(struct rockchip_vpu_ctx *ctx) > > +{ > > + struct rockchip_vpu_dev *vpu = ctx->dev; > > + struct vb2_buffer *src_buf, *dst_buf; > > + struct rockchip_vpu_jpeg_ctx jpeg_ctx; > > + u32 reg; > > + > > + src_buf = v4l2_m2m_next_src_buf(ctx->fh.m2m_ctx); > > + dst_buf = v4l2_m2m_next_dst_buf(ctx->fh.m2m_ctx); > > + > > + memset(&jpeg_ctx, 0, sizeof(jpeg_ctx)); > > + jpeg_ctx.buffer = vb2_plane_vaddr(dst_buf, 0); > > + jpeg_ctx.width = ctx->dst_fmt.width; > > + jpeg_ctx.height = ctx->dst_fmt.height; > > + jpeg_ctx.quality = ctx->jpeg_quality; > > + rockchip_vpu_jpeg_render(&jpeg_ctx); > > + > > + /* Switch to JPEG encoder mode before writing registers */ > > + vepu_write_relaxed(vpu, VEPU_REG_ENC_CTRL_ENC_MODE_JPEG, > > + VEPU_REG_ENC_CTRL); > > + > > + rk3288_vpu_set_src_img_ctrl(vpu, ctx); > > + rk3288_vpu_jpeg_enc_set_buffers(vpu, ctx, src_buf); > > + rk3288_vpu_jpeg_enc_set_qtable(vpu, > > + rockchip_vpu_jpeg_get_qtable(&jpeg_ctx, 0), > > + rockchip_vpu_jpeg_get_qtable(&jpeg_ctx, 1)); > > + > > + /* Make sure that all registers are written at this point. */ > > + wmb(); > > + > > Perhaps the next vepu_write_relaxed() should be turned into > vepu_write() instead? writel() basically starts with a wmb() on > arm/arm64. > Ack. > > + reg = VEPU_REG_AXI_CTRL_OUTPUT_SWAP16 > > + | VEPU_REG_AXI_CTRL_INPUT_SWAP16 > > + | VEPU_REG_AXI_CTRL_BURST_LEN(16) > > + | VEPU_REG_AXI_CTRL_OUTPUT_SWAP32 > > + | VEPU_REG_AXI_CTRL_INPUT_SWAP32 > > + | VEPU_REG_AXI_CTRL_OUTPUT_SWAP8 > > + | VEPU_REG_AXI_CTRL_INPUT_SWAP8; > > + vepu_write_relaxed(vpu, reg, VEPU_REG_AXI_CTRL); > > + > > + reg = VEPU_REG_ENC_CTRL_WIDTH(MB_WIDTH(ctx->src_fmt.width)) > > + | VEPU_REG_ENC_CTRL_HEIGHT(MB_HEIGHT(ctx->src_fmt.height)) > > + | VEPU_REG_ENC_CTRL_ENC_MODE_JPEG > > + | VEPU_REG_ENC_PIC_INTRA > > + | VEPU_REG_ENC_CTRL_EN_BIT; > > + /* Kick the watchdog and start encoding */ > > + schedule_delayed_work(&vpu->watchdog_work, msecs_to_jiffies(2000)); > > + vepu_write(vpu, reg, VEPU_REG_ENC_CTRL); > > +} > > diff --git a/drivers/staging/media/rockchip/vpu/rk3288_vpu_regs.h b/drivers/staging/media/rockchip/vpu/rk3288_vpu_regs.h > > new file mode 100644 > > index 000000000000..b5a464844dce > > --- /dev/null > > +++ b/drivers/staging/media/rockchip/vpu/rk3288_vpu_regs.h > > @@ -0,0 +1,442 @@ > > +/* SPDX-License-Identifier: GPL-2.0 */ > > +/* > > + * Rockchip VPU codec driver > > + * > > + * Copyright (C) 2018 Google, Inc. > > Should be: > > Copyright 2018 Google LLC. > Ack. > > + * Tomasz Figa <tfiga@xxxxxxxxxxxx> > > + */ > > + > [snip] > > diff --git a/drivers/staging/media/rockchip/vpu/rk3399_vpu_hw.c b/drivers/staging/media/rockchip/vpu/rk3399_vpu_hw.c > > new file mode 100644 > > index 000000000000..f9338745afe9 > > --- /dev/null > > +++ b/drivers/staging/media/rockchip/vpu/rk3399_vpu_hw.c > > @@ -0,0 +1,118 @@ > > +// SPDX-License-Identifier: GPL-2.0 > > +/* > > + * Rockchip VPU codec driver > > + * > > + * Copyright (C) 2018 Rockchip Electronics Co., Ltd. > > + * Jeffy Chen <jeffy.chen@xxxxxxxxxxxxxx> > > + */ > > + > > +#include <linux/clk.h> > > + > > +#include "rockchip_vpu.h" > > +#include "rockchip_vpu_jpeg.h" > > +#include "rk3399_vpu_regs.h" > > + > > +#define RK3399_ACLK_MAX_FREQ (400 * 1000 * 1000) > > + > > +/* > > + * Supported formats. > > + */ > > + > > +static const struct rockchip_vpu_fmt rk3399_vpu_enc_fmts[] = { > > + { > > + .fourcc = V4L2_PIX_FMT_YUV420M, > > + .codec_mode = RK_VPU_MODE_NONE, > > + .enc_fmt = RK3288_VPU_ENC_FMT_YUV420P, > > + }, > > + { > > + .fourcc = V4L2_PIX_FMT_NV12M, > > + .codec_mode = RK_VPU_MODE_NONE, > > + .enc_fmt = RK3288_VPU_ENC_FMT_YUV420SP, > > + }, > > + { > > + .fourcc = V4L2_PIX_FMT_YUYV, > > + .codec_mode = RK_VPU_MODE_NONE, > > + .enc_fmt = RK3288_VPU_ENC_FMT_YUYV422, > > + }, > > + { > > + .fourcc = V4L2_PIX_FMT_UYVY, > > + .codec_mode = RK_VPU_MODE_NONE, > > + .enc_fmt = RK3288_VPU_ENC_FMT_UYVY422, > > + }, > > + { > > + .fourcc = V4L2_PIX_FMT_JPEG, > > + .codec_mode = RK_VPU_MODE_JPEG_ENC, > > + .max_depth = 2, > > + .header_size = JPEG_HEADER_SIZE, > > + .frmsize = { > > + .min_width = 96, > > + .max_width = 8192, > > + .step_width = MB_DIM, > > + .min_height = 32, > > + .max_height = 8192, > > + .step_height = MB_DIM, > > + }, > > + }, > > +}; > > + > > +static irqreturn_t rk3399_vepu_irq(int irq, void *dev_id) > > +{ > > + struct rockchip_vpu_dev *vpu = dev_id; > > + enum vb2_buffer_state state; > > + u32 status, bytesused; > > + > > + status = vepu_read(vpu, VEPU_REG_INTERRUPT); > > + bytesused = vepu_read(vpu, VEPU_REG_STR_BUF_LIMIT) / 8; > > + state = (status & VEPU_REG_INTERRUPT_FRAME_READY) ? > > + VB2_BUF_STATE_DONE : VB2_BUF_STATE_ERROR; > > + > > + vepu_write(vpu, 0, VEPU_REG_INTERRUPT); > > + vepu_write(vpu, 0, VEPU_REG_AXI_CTRL); > > + > > + rockchip_vpu_irq_done(vpu, bytesused, state); > > + > > + return IRQ_HANDLED; > > +} > > + > > +static int rk3399_vpu_hw_init(struct rockchip_vpu_dev *vpu) > > +{ > > + /* Bump ACLK to max. possible freq. to improve performance. */ > > + clk_set_rate(vpu->clocks[0].clk, RK3399_ACLK_MAX_FREQ); > > + return 0; > > +} > > + > > +static void rk3399_vpu_enc_reset(struct rockchip_vpu_ctx *ctx) > > +{ > > + struct rockchip_vpu_dev *vpu = ctx->dev; > > + > > + vepu_write(vpu, VEPU_REG_INTERRUPT_DIS_BIT, VEPU_REG_INTERRUPT); > > + vepu_write(vpu, 0, VEPU_REG_ENCODE_START); > > + vepu_write(vpu, 0, VEPU_REG_AXI_CTRL); > > +} > > + > > +/* > > + * Supported codec ops. > > + */ > > + > > +static const struct rockchip_vpu_codec_ops rk3399_vpu_codec_ops[] = { > > + [RK_VPU_MODE_JPEG_ENC] = { > > + .run = rk3399_vpu_jpeg_enc_run, > > + .reset = rk3399_vpu_enc_reset, > > + }, > > +}; > > + > > +/* > > + * VPU variant. > > + */ > > + > > +const struct rockchip_vpu_variant rk3399_vpu_variant = { > > + .enc_offset = 0x0, > > + .enc_fmts = rk3399_vpu_enc_fmts, > > + .num_enc_fmts = ARRAY_SIZE(rk3399_vpu_enc_fmts), > > + .codec = RK_VPU_CODEC_JPEG, > > + .codec_ops = rk3399_vpu_codec_ops, > > + .vepu_irq = rk3399_vepu_irq, > > + .init = rk3399_vpu_hw_init, > > + .clk_names = {"aclk", "hclk"}, > > nit: Spaces inside the brackets. > > > + .num_clocks = 2 > > +}; > > diff --git a/drivers/staging/media/rockchip/vpu/rk3399_vpu_hw_jpeg_enc.c b/drivers/staging/media/rockchip/vpu/rk3399_vpu_hw_jpeg_enc.c > > new file mode 100644 > > index 000000000000..56d2da314c0e > > --- /dev/null > > +++ b/drivers/staging/media/rockchip/vpu/rk3399_vpu_hw_jpeg_enc.c > > @@ -0,0 +1,160 @@ > > +// SPDX-License-Identifier: GPL-2.0 > > +/* > > + * Rockchip VPU codec driver > > + * > > + * Copyright (C) 2018 Rockchip Electronics Co., Ltd. > > + * > > + * JPEG encoder > > + * ------------ > > + * The VPU JPEG encoder produces JPEG baseline sequential format. > > + * The quantization coefficients are 8-bit values, complying with > > + * the baseline specification. Therefore, it requires > > + * luma and chroma quantization tables. The hardware does entrophy > > entropy > Ack. > > + * encoding using internal Huffman tables, as specified in the JPEG > > + * specification. > > + * > > + * In other words, only the luma and chroma quantization tables are > > + * required for the encoding operation. > > + * > > + * Quantization luma table values are written to registers > > + * VEPU_swreg_0-VEPU_swreg_15, and chroma table values to > > + * VEPU_swreg_16-VEPU_swreg_31. > > + * > > + * JPEG zigzag order is expected on the quantization tables. > > + */ > > + > > +#include <asm/unaligned.h> > > +#include <media/v4l2-mem2mem.h> > > +#include "rockchip_vpu_jpeg.h" > > +#include "rockchip_vpu.h" > > +#include "rockchip_vpu_common.h" > > +#include "rockchip_vpu_hw.h" > > +#include "rk3399_vpu_regs.h" > > + > > +#define VEPU_JPEG_QUANT_TABLE_COUNT 16 > > + > > +static void rk3399_vpu_set_src_img_ctrl(struct rockchip_vpu_dev *vpu, > > + struct rockchip_vpu_ctx *ctx) > > +{ > > + struct v4l2_pix_format_mplane *pix_fmt = &ctx->src_fmt; > > + u32 reg; > > + > > + /* The pix fmt width/height are already MiB aligned > > nit: /* > * The pix fmt width/height are already MiB aligned > * ... > */ > > Also, is "MiB" right here? I think it was about macroblocks? > Ah, good catch. > > + * by .vidioc_s_fmt_vid_cap_mplane() callback > > + */ > > + reg = VEPU_REG_IN_IMG_CTRL_ROW_LEN(pix_fmt->width); > > + vepu_write_relaxed(vpu, reg, VEPU_REG_INPUT_LUMA_INFO); > > + > > + reg = VEPU_REG_IN_IMG_CTRL_OVRFLR_D4(0) | > > + VEPU_REG_IN_IMG_CTRL_OVRFLB(0); > > For reference, this register controls the input crop, as the offset > from the right/bottom within the last macroblock. The offset from the > right must be divided by 4 and so the crop must be aligned to 4 pixels > horizontally. > OK, I'll add a comment. > > + vepu_write_relaxed(vpu, reg, VEPU_REG_ENC_OVER_FILL_STRM_OFFSET); > > + > > + reg = VEPU_REG_IN_IMG_CTRL_FMT(ctx->vpu_src_fmt->enc_fmt); > > + vepu_write_relaxed(vpu, reg, VEPU_REG_ENC_CTRL1); > > +} > > + > > +static void rk3399_vpu_jpeg_enc_set_buffers(struct rockchip_vpu_dev *vpu, > > + struct rockchip_vpu_ctx *ctx, > > + struct vb2_buffer *src_buf) > > +{ > > + struct v4l2_pix_format_mplane *pix_fmt = &ctx->src_fmt; > > + dma_addr_t src[3]; > > + > > + WARN_ON(pix_fmt->num_planes > 3); > > + > > + vepu_write_relaxed(vpu, ctx->bounce_dma_addr, > > + VEPU_REG_ADDR_OUTPUT_STREAM); > > + vepu_write_relaxed(vpu, ctx->bounce_size, > > + VEPU_REG_STR_BUF_LIMIT); > > + > > + if (pix_fmt->num_planes == 1) { > > + src[0] = vb2_dma_contig_plane_dma_addr(src_buf, 0); > > + src[1] = src[0]; > > + src[2] = src[0]; > > + } else if (pix_fmt->num_planes == 2) { > > + src[0] = vb2_dma_contig_plane_dma_addr(src_buf, 0); > > + src[1] = vb2_dma_contig_plane_dma_addr(src_buf, 1); > > + src[2] = src[1]; > > + } else { > > + src[0] = vb2_dma_contig_plane_dma_addr(src_buf, 0); > > + src[1] = vb2_dma_contig_plane_dma_addr(src_buf, 1); > > + src[2] = vb2_dma_contig_plane_dma_addr(src_buf, 2); > > + } > > + > > + vepu_write_relaxed(vpu, src[0], VEPU_REG_ADDR_IN_LUMA); > > + vepu_write_relaxed(vpu, src[2], VEPU_REG_ADDR_IN_CR); > > + vepu_write_relaxed(vpu, src[1], VEPU_REG_ADDR_IN_CB); > > Same comments as for 3288 in this function. > > > +} > > + > > +static void > > +rk3399_vpu_jpeg_enc_set_qtable(struct rockchip_vpu_dev *vpu, > > + unsigned char *luma_qtable, > > + unsigned char *chroma_qtable) > > +{ > > + __be32 *luma_qtable_p; > > + __be32 *chroma_qtable_p; > > + u32 reg, i; > > + > > + luma_qtable_p = (__be32 *)luma_qtable; > > + chroma_qtable_p = (__be32 *)chroma_qtable; > > + > > + for (i = 0; i < VEPU_JPEG_QUANT_TABLE_COUNT; i++) { > > + reg = get_unaligned_be32(&luma_qtable[i]); > > + vepu_write_relaxed(vpu, reg, VEPU_REG_JPEG_LUMA_QUAT(i)); > > + > > + reg = get_unaligned_be32(&chroma_qtable[i]); > > + vepu_write_relaxed(vpu, reg, VEPU_REG_JPEG_CHROMA_QUAT(i)); > > + } > > +} > > + > > +void rk3399_vpu_jpeg_enc_run(struct rockchip_vpu_ctx *ctx) > > +{ > > + struct rockchip_vpu_dev *vpu = ctx->dev; > > + struct vb2_buffer *src_buf, *dst_buf; > > + struct rockchip_vpu_jpeg_ctx jpeg_ctx; > > + u32 reg; > > + > > + src_buf = v4l2_m2m_next_src_buf(ctx->fh.m2m_ctx); > > + dst_buf = v4l2_m2m_next_dst_buf(ctx->fh.m2m_ctx); > > + > > + memset(&jpeg_ctx, 0, sizeof(jpeg_ctx)); > > + jpeg_ctx.buffer = vb2_plane_vaddr(dst_buf, 0); > > + jpeg_ctx.width = ctx->dst_fmt.width; > > + jpeg_ctx.height = ctx->dst_fmt.height; > > + jpeg_ctx.quality = ctx->jpeg_quality; > > + rockchip_vpu_jpeg_render(&jpeg_ctx); > > + > > + /* Switch to JPEG encoder mode before writing registers */ > > + vepu_write_relaxed(vpu, VEPU_REG_ENCODE_FORMAT_JPEG, > > + VEPU_REG_ENCODE_START); > > + > > + rk3399_vpu_set_src_img_ctrl(vpu, ctx); > > + rk3399_vpu_jpeg_enc_set_buffers(vpu, ctx, src_buf); > > + rk3399_vpu_jpeg_enc_set_qtable(vpu, > > + rockchip_vpu_jpeg_get_qtable(&jpeg_ctx, 0), > > + rockchip_vpu_jpeg_get_qtable(&jpeg_ctx, 1)); > > + > > + /* Make sure that all registers are written at this point. */ > > + wmb(); > > Similar comment to 3288 here too. > > > + > > + reg = VEPU_REG_OUTPUT_SWAP32 > > + | VEPU_REG_OUTPUT_SWAP16 > > + | VEPU_REG_OUTPUT_SWAP8 > > + | VEPU_REG_INPUT_SWAP8 > > + | VEPU_REG_INPUT_SWAP16 > > + | VEPU_REG_INPUT_SWAP32; > > + vepu_write_relaxed(vpu, reg, VEPU_REG_DATA_ENDIAN); > > + > > + reg = VEPU_REG_AXI_CTRL_BURST_LEN(16); > > + vepu_write_relaxed(vpu, reg, VEPU_REG_AXI_CTRL); > > + > > + reg = VEPU_REG_MB_WIDTH(MB_WIDTH(ctx->src_fmt.width)) > > + | VEPU_REG_MB_HEIGHT(MB_HEIGHT(ctx->src_fmt.height)) > > + | VEPU_REG_FRAME_TYPE_INTRA > > + | VEPU_REG_ENCODE_FORMAT_JPEG > > + | VEPU_REG_ENCODE_ENABLE; > > + > > + /* Kick the watchdog and start encoding */ > > + schedule_delayed_work(&vpu->watchdog_work, msecs_to_jiffies(2000)); > > + vepu_write(vpu, reg, VEPU_REG_ENCODE_START); > > +} > [snip] > > diff --git a/drivers/staging/media/rockchip/vpu/rockchip_vpu.h b/drivers/staging/media/rockchip/vpu/rockchip_vpu.h > > new file mode 100644 > > index 000000000000..acc90cfe3102 > > --- /dev/null > > +++ b/drivers/staging/media/rockchip/vpu/rockchip_vpu.h > > @@ -0,0 +1,237 @@ > > +/* SPDX-License-Identifier: GPL-2.0 */ > > +/* > > + * Rockchip VPU codec driver > > + * > > + * Copyright (C) 2018 Google, Inc. > > + * Tomasz Figa <tfiga@xxxxxxxxxxxx> > > + * > > + * Based on s5p-mfc driver by Samsung Electronics Co., Ltd. > > + * Copyright (C) 2011 Samsung Electronics Co., Ltd. > > + */ > > + > > +#ifndef ROCKCHIP_VPU_H_ > > +#define ROCKCHIP_VPU_H_ > > + > > +#include <linux/platform_device.h> > > +#include <linux/videodev2.h> > > +#include <linux/wait.h> > > +#include <linux/clk.h> > > + > > +#include <media/v4l2-ctrls.h> > > +#include <media/v4l2-device.h> > > +#include <media/v4l2-ioctl.h> > > +#include <media/videobuf2-core.h> > > +#include <media/videobuf2-dma-contig.h> > > + > > +#include "rockchip_vpu_hw.h" > > + > > +#define ROCKCHIP_VPU_MAX_CLOCKS 4 > > + > > +#define MB_DIM 16 > > +#define MB_WIDTH(x_size) DIV_ROUND_UP(x_size, MB_DIM) > > +#define MB_HEIGHT(y_size) DIV_ROUND_UP(y_size, MB_DIM) > > +#define SB_DIM 64 > > +#define SB_WIDTH(x_size) DIV_ROUND_UP(x_size, SB_DIM) > > +#define SB_HEIGHT(y_size) DIV_ROUND_UP(y_size, SB_DIM) > > These are specific to video compression formats. H.264/VP8/VP9 share > the macroblock size of 16, while the superblock size of 64 is a VP9 > thing. for JPEG, it might be 8x8 (4:4:4), 16x8 (4:2:2) or 16x16 > (4:2:0) [1]. Perhaps this should be defined on a per-codec basis? > > [1] https://en.wikipedia.org/wiki/JPEG#Block_splitting > Good catch. > > + > > +struct rockchip_vpu_ctx; > > +struct rockchip_vpu_codec_ops; > > + > > +#define RK_VPU_CODEC_JPEG BIT(0) > > nit: Space instead of tab after "#define" > Ack. > > + > > +/** > > + * struct rockchip_vpu_variant - information about VPU hardware variant > > + * > > + * @enc_offset: Offset from VPU base to encoder registers. > > + * @enc_fmts: Encoder formats. > > + * @num_enc_fmts: Number of encoder formats. > > + * @codec: Supported codecs > > + * @codec_ops: Codec ops. > > + * @init: Initialize hardware. > > + * @vepu_irq: encoder interrupt handler > > + * @clocks: array of clock names > > + * @num_clocks: number of clocks in the array > > + */ > > +struct rockchip_vpu_variant { > > + unsigned int enc_offset; > > + const struct rockchip_vpu_fmt *enc_fmts; > > + unsigned int num_enc_fmts; > > + unsigned int codec; > > + const struct rockchip_vpu_codec_ops *codec_ops; > > + int (*init)(struct rockchip_vpu_dev *vpu); > > + irqreturn_t (*vepu_irq)(int irq, void *priv); > > + const char *clk_names[ROCKCHIP_VPU_MAX_CLOCKS]; > > + int num_clocks; > > +}; > > + > > +/** > > + * enum rockchip_vpu_codec_mode - codec operating mode. > > + * @RK_VPU_MODE_NONE: No operating mode. Used for RAW video formats. > > + * @RK_VPU_MODE_JPEG_ENC: JPEG encoder. > > + */ > > +enum rockchip_vpu_codec_mode { > > + RK_VPU_MODE_NONE = -1, > > + RK_VPU_MODE_JPEG_ENC, > > +}; > > + > > +/** > > + * struct rockchip_vpu_dev - driver data > > + * @v4l2_dev: V4L2 device to register video devices for. > > + * @vfd_enc: Video device for encoder. > > + * @pdev: Pointer to VPU platform device. > > + * @dev: Pointer to device for convenient logging using > > + * dev_ macros. > > + * @clocks: Array of clock handles. > > + * @base: Mapped address of VPU registers. > > + * @enc_base: Mapped address of VPU encoder register for convenience. > > + * @vpu_mutex: Mutex to synchronize V4L2 calls. > > + * @irqlock: Spinlock to synchronize access to data structures > > + * shared with interrupt handlers. > > + * @variant: Hardware variant-specific parameters. > > + * @watchdog_work: Delayed work for hardware timeout handling. > > + */ > > +struct rockchip_vpu_dev { > > + struct v4l2_device v4l2_dev; > > + struct v4l2_m2m_dev *m2m_dev; > > + struct media_device mdev; > > These two are not documented in the comment above. > Will fix all these. > > + struct video_device *vfd_enc; > > + struct platform_device *pdev; > > + struct device *dev; > > + struct clk_bulk_data clocks[ROCKCHIP_VPU_MAX_CLOCKS]; > > + void __iomem *base; > > + void __iomem *enc_base; > > + void __iomem *dec_base; > > This one is not either. > > > + > > + struct mutex vpu_mutex; /* video_device lock */ > > + spinlock_t irqlock; > > + const struct rockchip_vpu_variant *variant; > > + struct delayed_work watchdog_work; > > +}; > > + > > +/** > > + * struct rockchip_vpu_ctx - Context (instance) private data. > > + * > > + * @dev: VPU driver data to which the context belongs. > > + * @fh: V4L2 file handler. > > + * > > + * @sequence_cap: Sequence counter for capture queue > > + * @sequence_out: Sequence counter for output queue > > + * @codec_mode: Active codec mode > > There is no such field in the struct. > > > + * > > + * @vpu_src_fmt: Descriptor of active source format. > > + * @src_fmt: V4L2 pixel format of active source format. > > + * @vpu_dst_fmt: Descriptor of active destination format. > > + * @dst_fmt: V4L2 pixel format of active destination format. > > + * > > + * @ctrls: Array containing pointer to registered controls. > > No such field in the struct. > > > + * @ctrl_handler: Control handler used to register controls. > > + * @num_ctrls: Number of registered controls. > > No such field in the struct. > > > + * > > + * @codec_ops: Set of operations related to codec mode. > > + */ > > +struct rockchip_vpu_ctx { > > + struct rockchip_vpu_dev *dev; > > + struct v4l2_fh fh; > > + > > + u32 sequence_cap; > > + u32 sequence_out; > > + > > + const struct rockchip_vpu_fmt *vpu_src_fmt; > > + struct v4l2_pix_format_mplane src_fmt; > > + const struct rockchip_vpu_fmt *vpu_dst_fmt; > > + struct v4l2_pix_format_mplane dst_fmt; > > + > > + enum v4l2_colorspace colorspace; > > + enum v4l2_ycbcr_encoding ycbcr_enc; > > + enum v4l2_quantization quantization; > > + enum v4l2_xfer_func xfer_func; > > These 4 are not documented. > > > + > > + struct v4l2_ctrl_handler ctrl_handler; > > + int jpeg_quality; > > This one is not documented. > > > + > > + const struct rockchip_vpu_codec_ops *codec_ops; > > + > > + dma_addr_t bounce_dma_addr; > > + void *bounce_buf; > > + size_t bounce_size; > > These 3 are not documented. > > > +}; > > + > > +/** > > + * struct rockchip_vpu_fmt - information about supported video formats. > > + * @name: Human readable name of the format. > > + * @fourcc: FourCC code of the format. See V4L2_PIX_FMT_*. > > + * @codec_mode: Codec mode related to this format. See > > + * enum rockchip_vpu_codec_mode. > > + * @header_size: Optional header size. Currently used by JPEG encoder. > > + * @max_depth: Maximum depth, for bitstream formats > > + * @enc_fmt: Format identifier for encoder registers. > > + * @frmsize: Supported range of frame sizes (only for bitstream formats). > > + */ > > +struct rockchip_vpu_fmt { > > + char *name; > > + u32 fourcc; > > + enum rockchip_vpu_codec_mode codec_mode; > > + int header_size; > > + int max_depth; > > + enum rockchip_vpu_enc_fmt enc_fmt; > > + struct v4l2_frmsize_stepwise frmsize; > > +}; > > + > > +/* Logging helpers */ > > + > > +/** > > + * debug - Module parameter to control level of debugging messages. > > + * > > + * Level of debugging messages can be controlled by bits of > > + * module parameter called "debug". Meaning of particular > > + * bits is as follows: > > + * > > + * bit 0 - global information: mode, size, init, release > > + * bit 1 - each run start/result information > > + * bit 2 - contents of small controls from userspace > > + * bit 3 - contents of big controls from userspace > > + * bit 4 - detail fmt, ctrl, buffer q/dq information > > + * bit 5 - detail function enter/leave trace information > > + * bit 6 - register write/read information > > + */ > > +extern int rockchip_vpu_debug; > > + > > +#define vpu_debug(level, fmt, args...) \ > > + do { \ > > + if (rockchip_vpu_debug & BIT(level)) \ > > + pr_info("%s:%d: " fmt, \ > > + __func__, __LINE__, ##args); \ > > + } while (0) > > + > > +#define vpu_err(fmt, args...) \ > > + pr_err("%s:%d: " fmt, __func__, __LINE__, ##args) > > + > > +/* Structure access helpers. */ > > +static inline struct rockchip_vpu_ctx *fh_to_ctx(struct v4l2_fh *fh) > > +{ > > + return container_of(fh, struct rockchip_vpu_ctx, fh); > > +} > > + > > +/* Register accessors. */ > > +static inline void vepu_write_relaxed(struct rockchip_vpu_dev *vpu, > > + u32 val, u32 reg) > > +{ > > + vpu_debug(6, "MARK: set reg[%03d]: %08x\n", reg / 4, val); > > + writel_relaxed(val, vpu->enc_base + reg); > > +} > > + > > +static inline void vepu_write(struct rockchip_vpu_dev *vpu, u32 val, u32 reg) > > +{ > > + vpu_debug(6, "MARK: set reg[%03d]: %08x\n", reg / 4, val); > > + writel(val, vpu->enc_base + reg); > > +} > > + > > +static inline u32 vepu_read(struct rockchip_vpu_dev *vpu, u32 reg) > > +{ > > + u32 val = readl(vpu->enc_base + reg); > > + > > + vpu_debug(6, "MARK: get reg[%03d]: %08x\n", reg / 4, val); > > I remember seeing this "MARK" in the logs when debugging. I don't > think it's desired here. > > How about printing "%s(%03d) = %08x\n" for reads and "%s(%08x, > %03d)\n" for writes? > Makes sense, but why a %s string format? > > + return val; > > +} > > + > > +#endif /* ROCKCHIP_VPU_H_ */ > > diff --git a/drivers/staging/media/rockchip/vpu/rockchip_vpu_common.h b/drivers/staging/media/rockchip/vpu/rockchip_vpu_common.h > > new file mode 100644 > > index 000000000000..dc59e0796f5a > > --- /dev/null > > +++ b/drivers/staging/media/rockchip/vpu/rockchip_vpu_common.h > > @@ -0,0 +1,29 @@ > > +/* SPDX-License-Identifier: GPL-2.0 */ > > +/* > > + * Rockchip VPU codec driver > > + */ > > + * Copyright (C) 2018 Rockchip Electronics Co., Ltd. > > + * Alpha Lin <Alpha.Lin@xxxxxxxxxxxxxx> > > + * Jeffy Chen <jeffy.chen@xxxxxxxxxxxxxx> > > + * > > + * Copyright (C) 2018 Google, Inc. > > Copyright 2018 Google LLC > > > + * Tomasz Figa <tfiga@xxxxxxxxxxxx> > > + * > > + * Based on s5p-mfc driver by Samsung Electronics Co., Ltd. > > + * Copyright (C) 2011 Samsung Electronics Co., Ltd. > > + */ > > + > > +#ifndef ROCKCHIP_VPU_COMMON_H_ > > +#define ROCKCHIP_VPU_COMMON_H_ > > + > > +#include "rockchip_vpu.h" > > + > > +extern const struct v4l2_ioctl_ops rockchip_vpu_enc_ioctl_ops; > > +extern const struct vb2_ops rockchip_vpu_enc_queue_ops; > > + > > +void rockchip_vpu_enc_reset_src_fmt(struct rockchip_vpu_dev *vpu, > > + struct rockchip_vpu_ctx *ctx); > > +void rockchip_vpu_enc_reset_dst_fmt(struct rockchip_vpu_dev *vpu, > > + struct rockchip_vpu_ctx *ctx); > > + > > +#endif /* ROCKCHIP_VPU_COMMON_H_ */ > > diff --git a/drivers/staging/media/rockchip/vpu/rockchip_vpu_drv.c b/drivers/staging/media/rockchip/vpu/rockchip_vpu_drv.c > > new file mode 100644 > > index 000000000000..a355ccb678e8 > > --- /dev/null > > +++ b/drivers/staging/media/rockchip/vpu/rockchip_vpu_drv.c > > @@ -0,0 +1,535 @@ > > +// SPDX-License-Identifier: GPL-2.0 > > +/* > > + * Rockchip VPU codec driver > > + * > > + * Copyright (C) 2018 Collabora, Ltd. > > + * Copyright (C) 2014 Google, Inc. > > Ditto. > > > + * Tomasz Figa <tfiga@xxxxxxxxxxxx> > > + * > > + * Based on s5p-mfc driver by Samsung Electronics Co., Ltd. > > + * Copyright (C) 2011 Samsung Electronics Co., Ltd. > > + */ > > + > > +#include <linux/clk.h> > > +#include <linux/module.h> > > +#include <linux/of.h> > > +#include <linux/platform_device.h> > > +#include <linux/pm.h> > > +#include <linux/pm_runtime.h> > > +#include <linux/slab.h> > > +#include <linux/videodev2.h> > > +#include <linux/workqueue.h> > > +#include <media/v4l2-event.h> > > +#include <media/v4l2-mem2mem.h> > > +#include <media/videobuf2-core.h> > > +#include <media/videobuf2-core.h> > > +#include <media/videobuf2-vmalloc.h> > > + > > +#include "rockchip_vpu_common.h" > > +#include "rockchip_vpu.h" > > +#include "rockchip_vpu_hw.h" > > + > > +#define DRIVER_NAME "rockchip-vpu" > > + > > +int rockchip_vpu_debug; > > +module_param_named(debug, rockchip_vpu_debug, int, 0644); > > +MODULE_PARM_DESC(debug, > > + "Debug level - higher value produces more verbose messages"); > > + > > +static void rockchip_vpu_job_finish(struct rockchip_vpu_dev *vpu, > > + struct rockchip_vpu_ctx *ctx, > > + unsigned int bytesused, > > + enum vb2_buffer_state result) > > +{ > > + struct vb2_v4l2_buffer *src, *dst; > > + > > + pm_runtime_mark_last_busy(vpu->dev); > > + pm_runtime_put_autosuspend(vpu->dev); > > + > > + src = v4l2_m2m_src_buf_remove(ctx->fh.m2m_ctx); > > + dst = v4l2_m2m_dst_buf_remove(ctx->fh.m2m_ctx); > > + > > + if (WARN_ON(!src)) > > + return; > > + if (WARN_ON(!dst)) > > + return; > > + > > + src->sequence = ctx->sequence_out++; > > + dst->sequence = ctx->sequence_cap++; > > + > > + dst->field = src->field; > > + dst->timecode = src->timecode; > > Time code is only valid if the buffer has V4L2_BUF_FLAG_TIMECODE set. > I don't think there is any use case for mem2mem devices for it. > Right. Other mem2mem drivers seem to pass thru the timecode like this: if (in_vb->flags & V4L2_BUF_FLAG_TIMECODE) out_vb->timecode = in_vb->timecode; It fails a v4l2-compliance test without it. > > + dst->vb2_buf.timestamp = src->vb2_buf.timestamp; > > + dst->flags &= ~V4L2_BUF_FLAG_TSTAMP_SRC_MASK; > > + dst->flags |= src->flags & V4L2_BUF_FLAG_TSTAMP_SRC_MASK; > > Not V4L2_BUF_FLAG_TIMESTAMP_COPY? > I believe v4l core should take care of it in __fill_v4l2_buffer, as timestamp_flags is set when the vb2_queue structs are init'ed. > > + > > + if (bytesused) { > > Should we check whether bytesused (read from hardware) is not bigger > than size of the buffer? > Good catch, makes sense. OTOH, if bytesused is bigger than the dst buffer, it is also bigger than the bounce buffer. I guess the IOMMU helps prevents nasty issues? > > + if (ctx->bounce_buf) { > > + memcpy(vb2_plane_vaddr(&dst->vb2_buf, 0) + > > + ctx->vpu_dst_fmt->header_size, > > + ctx->bounce_buf, bytesused); > > + } > > + dst->vb2_buf.planes[0].bytesused = > > + ctx->vpu_dst_fmt->header_size + bytesused; > > + } > > + > > + v4l2_m2m_buf_done(src, result); > > + v4l2_m2m_buf_done(dst, result); > > + > > + v4l2_m2m_job_finish(vpu->m2m_dev, ctx->fh.m2m_ctx); > > +} > > + > > +void rockchip_vpu_irq_done(struct rockchip_vpu_dev *vpu, > > + unsigned int bytesused, > > + enum vb2_buffer_state result) > > +{ > > + struct rockchip_vpu_ctx *ctx = > > + (struct rockchip_vpu_ctx *)v4l2_m2m_get_curr_priv(vpu->m2m_dev); > > I don't think we need to cast from void *? > Right. > > + > > + /* Atomic watchdog cancel. The worker may still be > > + * running after calling this. > > + */ > > Wrong multi-line comment style. > Right. > > + cancel_delayed_work(&vpu->watchdog_work); > > + if (ctx) > > + rockchip_vpu_job_finish(vpu, ctx, bytesused, result); > > +} > > + > > +void rockchip_vpu_watchdog(struct work_struct *work) > > +{ > > + struct rockchip_vpu_dev *vpu; > > + struct rockchip_vpu_ctx *ctx; > > + > > + vpu = container_of(to_delayed_work(work), > > + struct rockchip_vpu_dev, watchdog_work); > > + ctx = (struct rockchip_vpu_ctx *)v4l2_m2m_get_curr_priv(vpu->m2m_dev); > > Ditto. > > > + if (ctx) { > > Is !ctx possible here? > Yes, it's possible because cancel_delayed_work doesn't flush the worker, so the top-half competes with the watchdog delayed worker. > > + vpu_err("frame processing timed out!\n"); > > + ctx->codec_ops->reset(ctx); > > + rockchip_vpu_job_finish(vpu, ctx, 0, VB2_BUF_STATE_ERROR); > > + } > > +} > > + > > +static void device_run(void *priv) > > +{ > > + struct rockchip_vpu_ctx *ctx = priv; > > + > > + pm_runtime_get_sync(ctx->dev->dev); > > Shouldn't we handle errors here? > Yes, definitely. > > + > > + ctx->codec_ops->run(ctx); > > +} > > + > > +static struct v4l2_m2m_ops vpu_m2m_ops = { > > + .device_run = device_run, > > +}; > > + > > +static int > > +enc_queue_init(void *priv, struct vb2_queue *src_vq, struct vb2_queue *dst_vq) > > +{ > > + struct rockchip_vpu_ctx *ctx = priv; > > + int ret; > > + > > + src_vq->type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE; > > + src_vq->io_modes = VB2_MMAP | VB2_DMABUF; > > + src_vq->drv_priv = ctx; > > + src_vq->ops = &rockchip_vpu_enc_queue_ops; > > + src_vq->mem_ops = &vb2_dma_contig_memops; > > + src_vq->dma_attrs = DMA_ATTR_ALLOC_SINGLE_PAGES | > > + DMA_ATTR_NO_KERNEL_MAPPING; > > + src_vq->buf_struct_size = sizeof(struct v4l2_m2m_buffer); > > + src_vq->timestamp_flags = V4L2_BUF_FLAG_TIMESTAMP_COPY; > > + src_vq->lock = &ctx->dev->vpu_mutex; > > + src_vq->dev = ctx->dev->v4l2_dev.dev; > > + > > + ret = vb2_queue_init(src_vq); > > + if (ret) > > + return ret; > > + > > + /* The CAPTURE queue doesn't need dma memory, > > + * as the CPU needs to create the JPEG frames, > > + * from the hardware-produced JPEG payload. > > + * > > + * For the DMA destination buffer, we use > > + * a bounce buffer. > > Alternatively we could use a normal buffer and memmove() the payload > to make space for the headers, as we used to do in the VP8 encoder on > rk3288. Either is fine and perhaps we could even do away without that > with some smart trick. Something for the TODO list I guess. > Perhaps we can re-discuss this in IRC? Anyway, it's a nice optimization to keep in mind. > > + */ > > + dst_vq->type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE; > > + dst_vq->io_modes = VB2_MMAP | VB2_DMABUF; > > + dst_vq->drv_priv = ctx; > > + dst_vq->ops = &rockchip_vpu_enc_queue_ops; > > + dst_vq->mem_ops = &vb2_vmalloc_memops; > > + dst_vq->buf_struct_size = sizeof(struct v4l2_m2m_buffer); > > + dst_vq->timestamp_flags = V4L2_BUF_FLAG_TIMESTAMP_COPY; > > + dst_vq->lock = &ctx->dev->vpu_mutex; > > + dst_vq->dev = ctx->dev->v4l2_dev.dev; > > + > > + return vb2_queue_init(dst_vq); > > +} > > + > > +static int rockchip_vpu_s_ctrl(struct v4l2_ctrl *ctrl) > > +{ > > + struct rockchip_vpu_ctx *ctx; > > + > > + ctx = container_of(ctrl->handler, > > + struct rockchip_vpu_ctx, ctrl_handler); > > + > > + vpu_debug(1, "s_ctrl: id = %d, val = %d\n", ctrl->id, ctrl->val); > > + > > + switch (ctrl->id) { > > + case V4L2_CID_JPEG_COMPRESSION_QUALITY: > > + ctx->jpeg_quality = ctrl->val; > > + break; > > + default: > > + vpu_err("Invalid control id = %d, val = %d\n", > > + ctrl->id, ctrl->val); > > + return -EINVAL; > > + } > > + > > + return 0; > > +} > > + > > +static const struct v4l2_ctrl_ops rockchip_vpu_ctrl_ops = { > > + .s_ctrl = rockchip_vpu_s_ctrl, > > +}; > > + > > +static int rockchip_vpu_ctrls_setup(struct rockchip_vpu_dev *vpu, > > + struct rockchip_vpu_ctx *ctx) > > +{ > > + v4l2_ctrl_handler_init(&ctx->ctrl_handler, 1); > > + if (ctx->ctrl_handler.error) { > > + vpu_err("v4l2_ctrl_handler_init failed (%d)\n", > > + ctx->ctrl_handler.error); > > + return ctx->ctrl_handler.error; > > + } > > No need to check for the error every single operation. The > v4l2_ctrl_new_std() call will bail out if the handler is in an error > condition. > OK. > > + > > + if (vpu->variant->codec & RK_VPU_CODEC_JPEG) { > > + v4l2_ctrl_new_std(&ctx->ctrl_handler, &rockchip_vpu_ctrl_ops, > > + V4L2_CID_JPEG_COMPRESSION_QUALITY, > > + 5, 100, 1, 50); > > + if (ctx->ctrl_handler.error) { > > + vpu_err("Adding JPEG control failed %d\n", > > + ctx->ctrl_handler.error); > > + v4l2_ctrl_handler_free(&ctx->ctrl_handler); > > + return ctx->ctrl_handler.error; > > + } > > + } > > + > > + return v4l2_ctrl_handler_setup(&ctx->ctrl_handler); > > +} > > + > > +/* > > + * V4L2 file operations. > > + */ > > + > > +static int rockchip_vpu_open(struct file *filp) > > +{ > > + struct rockchip_vpu_dev *vpu = video_drvdata(filp); > > + struct video_device *vdev = video_devdata(filp); > > + struct rockchip_vpu_ctx *ctx; > > + int ret; > > + > > + /* > > + * We do not need any extra locking here, because we operate only > > + * on local data here, except reading few fields from dev, which > > + * do not change through device's lifetime (which is guaranteed by > > + * reference on module from open()) and V4L2 internal objects (such > > + * as vdev and ctx->fh), which have proper locking done in respective > > + * helper functions used here. > > + */ > > + > > + ctx = kzalloc(sizeof(*ctx), GFP_KERNEL); > > + if (!ctx) > > + return -ENOMEM; > > + > > + ctx->dev = vpu; > > + if (vdev == vpu->vfd_enc) > > + ctx->fh.m2m_ctx = v4l2_m2m_ctx_init(vpu->m2m_dev, ctx, > > + &enc_queue_init); > > + else > > + ctx->fh.m2m_ctx = ERR_PTR(-ENODEV); > > + if (IS_ERR(ctx->fh.m2m_ctx)) { > > + ret = PTR_ERR(ctx->fh.m2m_ctx); > > + kfree(ctx); > > + return ret; > > + } > > + > > + v4l2_fh_init(&ctx->fh, vdev); > > + filp->private_data = &ctx->fh; > > + v4l2_fh_add(&ctx->fh); > > + > > + if (vdev == vpu->vfd_enc) { > > + rockchip_vpu_enc_reset_dst_fmt(vpu, ctx); > > + rockchip_vpu_enc_reset_src_fmt(vpu, ctx); > > + } > > + > > + ret = rockchip_vpu_ctrls_setup(vpu, ctx); > > + if (ret) { > > + vpu_err("Failed to set up controls\n"); > > + goto err_fh_free; > > + } > > + ctx->fh.ctrl_handler = &ctx->ctrl_handler; > > + > > + return 0; > > + > > +err_fh_free: > > + v4l2_fh_del(&ctx->fh); > > + v4l2_fh_exit(&ctx->fh); > > + kfree(ctx); > > + return ret; > > +} > > + > > +static int rockchip_vpu_release(struct file *filp) > > +{ > > + struct rockchip_vpu_ctx *ctx = > > + container_of(filp->private_data, struct rockchip_vpu_ctx, fh); > > + > > + /* > > + * No need for extra locking because this was the last reference > > + * to this file. > > + */ > > + v4l2_m2m_ctx_release(ctx->fh.m2m_ctx); > > + v4l2_fh_del(&ctx->fh); > > + v4l2_fh_exit(&ctx->fh); > > + v4l2_ctrl_handler_free(&ctx->ctrl_handler); > > + kfree(ctx); > > + > > + return 0; > > +} > > + > > +static const struct v4l2_file_operations rockchip_vpu_fops = { > > + .owner = THIS_MODULE, > > + .open = rockchip_vpu_open, > > + .release = rockchip_vpu_release, > > + .poll = v4l2_m2m_fop_poll, > > + .unlocked_ioctl = video_ioctl2, > > + .mmap = v4l2_m2m_fop_mmap, > > +}; > > + > > +static const struct of_device_id of_rockchip_vpu_match[] = { > > + { .compatible = "rockchip,rk3399-vpu", .data = &rk3399_vpu_variant, }, > > + { .compatible = "rockchip,rk3288-vpu", .data = &rk3288_vpu_variant, }, > > + { /* sentinel */ } > > +}; > > +MODULE_DEVICE_TABLE(of, of_rockchip_vpu_match); > > + > > +static int rockchip_vpu_video_device_register(struct rockchip_vpu_dev *vpu) > > +{ > > + const struct of_device_id *match; > > + struct video_device *vfd; > > + int function, ret; > > + > > + match = of_match_node(of_rockchip_vpu_match, vpu->dev->of_node); > > + vfd = video_device_alloc(); > > + if (!vfd) { > > + v4l2_err(&vpu->v4l2_dev, "Failed to allocate video device\n"); > > + return -ENOMEM; > > + } > > + > > + vfd->fops = &rockchip_vpu_fops; > > + vfd->release = video_device_release; > > + vfd->lock = &vpu->vpu_mutex; > > + vfd->v4l2_dev = &vpu->v4l2_dev; > > + vfd->vfl_dir = VFL_DIR_M2M; > > + vfd->device_caps = V4L2_CAP_STREAMING | V4L2_CAP_VIDEO_M2M_MPLANE; > > + vfd->ioctl_ops = &rockchip_vpu_enc_ioctl_ops; > > + snprintf(vfd->name, sizeof(vfd->name), "%s-enc", match->compatible); > > + vpu->vfd_enc = vfd; > > + video_set_drvdata(vfd, vpu); > > + > > + ret = video_register_device(vfd, VFL_TYPE_GRABBER, 0); > > + if (ret) { > > + v4l2_err(&vpu->v4l2_dev, "Failed to register video device\n"); > > + goto err_free_dev; > > + } > > + v4l2_info(&vpu->v4l2_dev, "registered as /dev/video%d\n", vfd->num); > > + > > + function = MEDIA_ENT_F_PROC_VIDEO_ENCODER; > > + ret = v4l2_m2m_register_media_controller(vpu->m2m_dev, vfd, function); > > + if (ret) { > > + v4l2_err(&vpu->v4l2_dev, "Failed to init mem2mem media controller\n"); > > + goto err_unreg_video; > > + } > > + return 0; > > + > > +err_unreg_video: > > + video_unregister_device(vfd); > > +err_free_dev: > > + video_device_release(vfd); > > + return ret; > > +} > > + > > +static int rockchip_vpu_probe(struct platform_device *pdev) > > +{ > > + const struct of_device_id *match; > > + struct rockchip_vpu_dev *vpu; > > + struct resource *res; > > + int i, ret; > > + > > + vpu = devm_kzalloc(&pdev->dev, sizeof(*vpu), GFP_KERNEL); > > + if (!vpu) > > + return -ENOMEM; > > + > > + vpu->dev = &pdev->dev; > > + vpu->pdev = pdev; > > + mutex_init(&vpu->vpu_mutex); > > + spin_lock_init(&vpu->irqlock); > > + > > + match = of_match_node(of_rockchip_vpu_match, pdev->dev.of_node); > > + vpu->variant = match->data; > > + > > + INIT_DELAYED_WORK(&vpu->watchdog_work, rockchip_vpu_watchdog); > > + > > + for (i = 0; i < vpu->variant->num_clocks; i++) > > + vpu->clocks[i].id = vpu->variant->clk_names[i]; > > + ret = devm_clk_bulk_get(&pdev->dev, vpu->variant->num_clocks, > > + vpu->clocks); > > + if (ret) > > + return ret; > > + > > + res = platform_get_resource(vpu->pdev, IORESOURCE_MEM, 0); > > + vpu->base = devm_ioremap_resource(vpu->dev, res); > > + if (IS_ERR(vpu->base)) > > + return PTR_ERR(vpu->base); > > + vpu->enc_base = vpu->base + vpu->variant->enc_offset; > > + > > + ret = dma_set_coherent_mask(vpu->dev, DMA_BIT_MASK(32)); > > + if (ret) { > > + dev_err(vpu->dev, "Could not set DMA coherent mask.\n"); > > + return ret; > > + } > > + > > + if (vpu->variant->vepu_irq) { > > + int irq; > > + > > + irq = platform_get_irq_byname(vpu->pdev, "vepu"); > > + if (irq <= 0) { > > + dev_err(vpu->dev, "Could not get vepu IRQ.\n"); > > + return -ENXIO; > > + } > > + > > + ret = devm_request_irq(vpu->dev, irq, vpu->variant->vepu_irq, > > + 0, dev_name(vpu->dev), vpu); > > + if (ret) { > > + dev_err(vpu->dev, "Could not request vepu IRQ.\n"); > > + return ret; > > + } > > + } > > + > > + ret = vpu->variant->init(vpu); > > + if (ret) { > > + dev_err(&pdev->dev, "Failed to init VPU hardware\n"); > > + return ret; > > + } > > + > > + pm_runtime_set_autosuspend_delay(vpu->dev, 100); > > + pm_runtime_use_autosuspend(vpu->dev); > > + pm_runtime_enable(vpu->dev); > > + > > + ret = clk_bulk_prepare(vpu->variant->num_clocks, vpu->clocks); > > + if (ret) { > > + dev_err(&pdev->dev, "Failed to prepare clocks\n"); > > + return ret; > > + } > > + > > + ret = v4l2_device_register(&pdev->dev, &vpu->v4l2_dev); > > + if (ret) { > > + dev_err(&pdev->dev, "Failed to register v4l2 device\n"); > > + goto err_clk_unprepare; > > + } > > + platform_set_drvdata(pdev, vpu); > > + > > + vpu->m2m_dev = v4l2_m2m_init(&vpu_m2m_ops); > > + if (IS_ERR(vpu->m2m_dev)) { > > + v4l2_err(&vpu->v4l2_dev, "Failed to init mem2mem device\n"); > > + ret = PTR_ERR(vpu->m2m_dev); > > + goto err_v4l2_unreg; > > + } > > + > > + vpu->mdev.dev = vpu->dev; > > + strlcpy(vpu->mdev.model, DRIVER_NAME, sizeof(vpu->mdev.model)); > > + media_device_init(&vpu->mdev); > > + vpu->v4l2_dev.mdev = &vpu->mdev; > > + > > + ret = rockchip_vpu_video_device_register(vpu); > > + if (ret) { > > + dev_err(&pdev->dev, "Failed to register encoder\n"); > > + goto err_m2m_rel; > > + } > > + > > + ret = media_device_register(&vpu->mdev); > > + if (ret) { > > + v4l2_err(&vpu->v4l2_dev, "Failed to register mem2mem media device\n"); > > + goto err_video_dev_unreg; > > + } > > + return 0; > > +err_video_dev_unreg: > > + if (vpu->vfd_enc) { > > + video_unregister_device(vpu->vfd_enc); > > + video_device_release(vpu->vfd_enc); > > + } > > +err_m2m_rel: > > + v4l2_m2m_release(vpu->m2m_dev); > > +err_v4l2_unreg: > > + v4l2_device_unregister(&vpu->v4l2_dev); > > +err_clk_unprepare: > > + clk_bulk_unprepare(vpu->variant->num_clocks, vpu->clocks); > > + pm_runtime_disable(vpu->dev); > > + return ret; > > +} > > + > > +static int rockchip_vpu_remove(struct platform_device *pdev) > > +{ > > + struct rockchip_vpu_dev *vpu = platform_get_drvdata(pdev); > > + > > + v4l2_info(&vpu->v4l2_dev, "Removing %s\n", pdev->name); > > + > > + media_device_unregister(&vpu->mdev); > > + v4l2_m2m_unregister_media_controller(vpu->m2m_dev); > > + v4l2_m2m_release(vpu->m2m_dev); > > + media_device_cleanup(&vpu->mdev); > > + if (vpu->vfd_enc) { > > + video_unregister_device(vpu->vfd_enc); > > + video_device_release(vpu->vfd_enc); > > + } > > + v4l2_device_unregister(&vpu->v4l2_dev); > > + clk_bulk_unprepare(vpu->variant->num_clocks, vpu->clocks); > > + pm_runtime_disable(vpu->dev); > > + return 0; > > +} > > + > > +static int __maybe_unused rockchip_vpu_runtime_suspend(struct device *dev) > > +{ > > + struct rockchip_vpu_dev *vpu = dev_get_drvdata(dev); > > + > > + clk_bulk_disable(vpu->variant->num_clocks, vpu->clocks); > > + return 0; > > +} > > + > > +static int __maybe_unused rockchip_vpu_runtime_resume(struct device *dev) > > +{ > > + struct rockchip_vpu_dev *vpu = dev_get_drvdata(dev); > > + > > + return clk_bulk_enable(vpu->variant->num_clocks, vpu->clocks); > > Something for the TODO list: We should disable the clocks as soon as > the hardware becomes idle, because it's super cheap and the delay > between the idle and autosuspend is quite significant. > You mean getting rid of autosuspend, right? > > +} > > + > > +static const struct dev_pm_ops rockchip_vpu_pm_ops = { > > + SET_SYSTEM_SLEEP_PM_OPS(pm_runtime_force_suspend, > > + pm_runtime_force_resume) > > + SET_RUNTIME_PM_OPS(rockchip_vpu_runtime_suspend, > > + rockchip_vpu_runtime_resume, NULL) > > +}; > > + > > +static struct platform_driver rockchip_vpu_driver = { > > + .probe = rockchip_vpu_probe, > > + .remove = rockchip_vpu_remove, > > + .driver = { > > + .name = DRIVER_NAME, > > + .of_match_table = of_match_ptr(of_rockchip_vpu_match), > > + .pm = &rockchip_vpu_pm_ops, > > + }, > > +}; > > +module_platform_driver(rockchip_vpu_driver); > > + > > +MODULE_LICENSE("GPL v2"); > > +MODULE_AUTHOR("Alpha Lin <Alpha.Lin@xxxxxxxxxxxxxx>"); > > +MODULE_AUTHOR("Tomasz Figa <tfiga@xxxxxxxxxxxx>"); > > +MODULE_AUTHOR("Ezequiel Garcia <ezequiel@xxxxxxxxxxxxx>"); > > +MODULE_DESCRIPTION("Rockchip VPU codec driver"); > > diff --git a/drivers/staging/media/rockchip/vpu/rockchip_vpu_enc.c b/drivers/staging/media/rockchip/vpu/rockchip_vpu_enc.c > > new file mode 100644 > > index 000000000000..374fea20a71d > > --- /dev/null > > +++ b/drivers/staging/media/rockchip/vpu/rockchip_vpu_enc.c > > @@ -0,0 +1,702 @@ > > +// SPDX-License-Identifier: GPL-2.0 > > +/* > > + * Rockchip VPU codec driver > > + * > > + * Copyright (C) 2018 Collabora, Ltd. > > + * Copyright (C) 2018 Rockchip Electronics Co., Ltd. > > + * Alpha Lin <Alpha.Lin@xxxxxxxxxxxxxx> > > + * Jeffy Chen <jeffy.chen@xxxxxxxxxxxxxx> > > + * > > + * Copyright (C) 2018 Google, Inc. > > Ditto. > > > + * Tomasz Figa <tfiga@xxxxxxxxxxxx> > > + * > > + * Based on s5p-mfc driver by Samsung Electronics Co., Ltd. > > + * Copyright (C) 2010-2011 Samsung Electronics Co., Ltd. > > + */ > > + > > +#include <linux/interrupt.h> > > +#include <linux/io.h> > > +#include <linux/module.h> > > +#include <linux/pm_runtime.h> > > +#include <linux/videodev2.h> > > +#include <linux/workqueue.h> > > +#include <media/v4l2-ctrls.h> > > +#include <media/v4l2-event.h> > > +#include <media/v4l2-mem2mem.h> > > +#include <media/videobuf2-core.h> > > +#include <media/videobuf2-dma-sg.h> > > + > > +#include "rockchip_vpu.h" > > +#include "rockchip_vpu_hw.h" > > +#include "rockchip_vpu_common.h" > > + > > +/** > > + * struct v4l2_format_info - information about a V4L2 format > > + * @format: 4CC format identifier (V4L2_PIX_FMT_*) > > + * @header_size: Size of header, optional and used by compressed formats > > + * @num_planes: Number of planes (1 to 3) > > + * @cpp: Number of bytes per pixel (per plane) > > + * @hsub: Horizontal chroma subsampling factor > > + * @vsub: Vertical chroma subsampling factor > > + * @is_compressed: Is it a compressed format? > > + * @multiplanar: Is it a multiplanar variant format? (e.g. NV12M) > > + */ > > +struct v4l2_format_info { > > + u32 format; > > + u32 header_size; > > + u8 num_planes; > > + u8 cpp[3]; > > + u8 hsub; > > + u8 vsub; > > + u8 is_compressed; > > + u8 multiplanar; > > +}; > > + > > +static const struct v4l2_format_info * > > +v4l2_format_info(u32 format) > > +{ > > + static const struct v4l2_format_info formats[] = { > > + { .format = V4L2_PIX_FMT_YUV420M, .num_planes = 3, .cpp = { 1, 1, 1 }, .hsub = 2, .vsub = 2, .multiplanar = 1 }, > > + { .format = V4L2_PIX_FMT_NV12M, .num_planes = 2, .cpp = { 1, 2, 0 }, .hsub = 2, .vsub = 2, .multiplanar = 1 }, > > + { .format = V4L2_PIX_FMT_YUYV, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 2, .vsub = 1 }, > > + { .format = V4L2_PIX_FMT_UYVY, .num_planes = 1, .cpp = { 2, 0, 0 }, .hsub = 2, .vsub = 1 }, > > + }; > > + unsigned int i; > > + > > + for (i = 0; i < ARRAY_SIZE(formats); ++i) { > > + if (formats[i].format == format) > > + return &formats[i]; > > + } > > + > > + vpu_err("Unsupported V4L 4CC format (%08x)\n", format); > > + return NULL; > > +} > > + > > +static void > > +fill_pixfmt_mp(struct v4l2_pix_format_mplane *pixfmt, > > + int pixelformat, int width, int height) > > +{ > > + const struct v4l2_format_info *info; > > + struct v4l2_plane_pix_format *plane; > > + int i; > > + > > + info = v4l2_format_info(pixelformat); > > + if (!info) > > + return; > > + > > + pixfmt->width = width; > > + pixfmt->height = height; > > + pixfmt->pixelformat = pixelformat; > > + > > + if (!info->multiplanar) { > > + pixfmt->num_planes = 1; > > + plane = &pixfmt->plane_fmt[0]; > > + plane->bytesperline = info->is_compressed ? > > + 0 : width * info->cpp[0]; > > + plane->sizeimage = info->header_size; > > + for (i = 0; i < info->num_planes; i++) { > > + unsigned int hsub = (i == 0) ? 1 : info->hsub; > > + unsigned int vsub = (i == 0) ? 1 : info->vsub; > > + > > + plane->sizeimage += > > + width * height * info->cpp[i] / (hsub * vsub); > > In general, I'd say it should be more like > > DIV_ROUND_UP(width, hsub) * DIV_ROUND_UP(height, vsub) * info->cpp[i] > > to avoid rounding problems. Although for this driver, there would be > no difference, because macroblock alignment is applied. > Ack. > > + } > > + } else { > > + pixfmt->num_planes = info->num_planes; > > + for (i = 0; i < info->num_planes; i++) { > > + unsigned int hsub = (i == 0) ? 1 : info->hsub; > > + unsigned int vsub = (i == 0) ? 1 : info->vsub; > > + > > + plane = &pixfmt->plane_fmt[i]; > > + plane->bytesperline = width * info->cpp[i] / hsub; > > + plane->sizeimage = > > + width * height * info->cpp[i] / (hsub * vsub); > > Perhaps > > plane->bytesperline * DIV_ROUND_UP(height, vsub) > Ack. > ? > > > + } > > + } > > +} > > + > > +static const struct rockchip_vpu_fmt * > > +rockchip_vpu_find_format(struct rockchip_vpu_ctx *ctx, u32 fourcc) > > +{ > > + struct rockchip_vpu_dev *dev = ctx->dev; > > + const struct rockchip_vpu_fmt *formats; > > + unsigned int num_fmts, i; > > + > > + formats = dev->variant->enc_fmts; > > + num_fmts = dev->variant->num_enc_fmts; > > + for (i = 0; i < num_fmts; i++) > > + if (formats[i].fourcc == fourcc) > > + return &formats[i]; > > + return NULL; > > +} > > + > > +static const struct rockchip_vpu_fmt * > > +rockchip_vpu_get_default_fmt(struct rockchip_vpu_ctx *ctx, bool bitstream) > > +{ > > + struct rockchip_vpu_dev *dev = ctx->dev; > > + const struct rockchip_vpu_fmt *formats; > > + unsigned int num_fmts, i; > > + > > + formats = dev->variant->enc_fmts; > > + num_fmts = dev->variant->num_enc_fmts; > > + for (i = 0; i < num_fmts; i++) { > > + if (bitstream == (formats[i].codec_mode != RK_VPU_MODE_NONE)) > > + return &formats[i]; > > + } > > + return NULL; > > +} > > + > > +static int vidioc_querycap(struct file *file, void *priv, > > + struct v4l2_capability *cap) > > +{ > > + struct rockchip_vpu_dev *vpu = video_drvdata(file); > > + > > + strscpy(cap->driver, vpu->dev->driver->name, sizeof(cap->driver)); > > + strscpy(cap->card, vpu->vfd_enc->name, sizeof(cap->card)); > > + snprintf(cap->bus_info, sizeof(cap->bus_info), "platform: %s", > > + vpu->dev->driver->name); > > + return 0; > > +} > > + > > +static int vidioc_enum_framesizes(struct file *file, void *priv, > > + struct v4l2_frmsizeenum *fsize) > > +{ > > + struct rockchip_vpu_ctx *ctx = fh_to_ctx(priv); > > + const struct rockchip_vpu_fmt *fmt; > > + > > + if (fsize->index != 0) { > > + vpu_debug(0, "invalid frame size index (expected 0, got %d)\n", > > + fsize->index); > > + return -EINVAL; > > + } > > + > > + fmt = rockchip_vpu_find_format(ctx, fsize->pixel_format); > > + if (!fmt) { > > + vpu_debug(0, "unsupported bitstream format (%08x)\n", > > + fsize->pixel_format); > > + return -EINVAL; > > + } > > + > > + /* This only makes sense for codec formats */ > > typo: coded > Ack. > > + if (fmt->codec_mode == RK_VPU_MODE_NONE) > > + return -EINVAL; > > + > > + fsize->type = V4L2_FRMSIZE_TYPE_STEPWISE; > > + fsize->stepwise = fmt->frmsize; > > + > > + return 0; > > +} > > + > > +static int vidioc_enum_fmt_vid_cap_mplane(struct file *file, void *priv, > > + struct v4l2_fmtdesc *f) > > +{ > > + struct rockchip_vpu_dev *dev = video_drvdata(file); > > + const struct rockchip_vpu_fmt *fmt; > > + const struct rockchip_vpu_fmt *formats; > > + int num_fmts, i, j = 0; > > + > > + formats = dev->variant->enc_fmts; > > + num_fmts = dev->variant->num_enc_fmts; > > + for (i = 0; i < num_fmts; i++) { > > + /* Skip uncompressed formats */ > > + if (formats[i].codec_mode == RK_VPU_MODE_NONE) > > + continue; > > + if (j == f->index) { > > + fmt = &formats[i]; > > + f->pixelformat = fmt->fourcc; > > + return 0; > > + } > > + ++j; > > + } > > + return -EINVAL; > > +} > > + > > +static int vidioc_enum_fmt_vid_out_mplane(struct file *file, void *priv, > > + struct v4l2_fmtdesc *f) > > +{ > > + struct rockchip_vpu_dev *dev = video_drvdata(file); > > + const struct rockchip_vpu_fmt *formats; > > + const struct rockchip_vpu_fmt *fmt; > > + int num_fmts, i, j = 0; > > + > > + formats = dev->variant->enc_fmts; > > + num_fmts = dev->variant->num_enc_fmts; > > + for (i = 0; i < num_fmts; i++) { > > + if (formats[i].codec_mode != RK_VPU_MODE_NONE) > > + continue; > > + if (j == f->index) { > > + fmt = &formats[i]; > > + f->pixelformat = fmt->fourcc; > > + return 0; > > + } > > + ++j; > > + } > > + return -EINVAL; > > +} > > The two functions above are almost the same, with the exception of the > condition being negated. Could be abstracted into a common function. > I'll give it a shot, and see how it looks. > > + > > +static int vidioc_g_fmt_out_mplane(struct file *file, void *priv, > > + struct v4l2_format *f) > > +{ > > + struct v4l2_pix_format_mplane *pix_mp = &f->fmt.pix_mp; > > + struct rockchip_vpu_ctx *ctx = fh_to_ctx(priv); > > + > > + vpu_debug(4, "f->type = %d\n", f->type); > > + > > + *pix_mp = ctx->src_fmt; > > + pix_mp->colorspace = ctx->colorspace; > > + pix_mp->ycbcr_enc = ctx->ycbcr_enc; > > + pix_mp->xfer_func = ctx->xfer_func; > > + pix_mp->quantization = ctx->quantization; > > Why do we need to set these 4 manually rather than just using whatever > is in ctx->src_fmt? > This should be cleaned up. > > + > > + return 0; > > +} > > + > > +static int vidioc_g_fmt_cap_mplane(struct file *file, void *priv, > > + struct v4l2_format *f) > > +{ > > + struct v4l2_pix_format_mplane *pix_mp = &f->fmt.pix_mp; > > + struct rockchip_vpu_ctx *ctx = fh_to_ctx(priv); > > + > > + vpu_debug(4, "f->type = %d\n", f->type); > > + > > + *pix_mp = ctx->dst_fmt; > > + pix_mp->colorspace = ctx->colorspace; > > + pix_mp->ycbcr_enc = ctx->ycbcr_enc; > > + pix_mp->xfer_func = ctx->xfer_func; > > + pix_mp->quantization = ctx->quantization; > > + > > + return 0; > > +} > > + > > +static int > > +vidioc_try_fmt_cap_mplane(struct file *file, void *priv, struct v4l2_format *f) > > +{ > > + struct rockchip_vpu_ctx *ctx = fh_to_ctx(priv); > > + struct v4l2_pix_format_mplane *pix_mp = &f->fmt.pix_mp; > > + const struct rockchip_vpu_fmt *fmt; > > + > > + vpu_debug(4, "%c%c%c%c\n", > > + (pix_mp->pixelformat & 0x7f), > > + (pix_mp->pixelformat >> 8) & 0x7f, > > + (pix_mp->pixelformat >> 16) & 0x7f, > > + (pix_mp->pixelformat >> 24) & 0x7f); > > + > > + fmt = rockchip_vpu_find_format(ctx, pix_mp->pixelformat); > > + if (!fmt) { > > + fmt = rockchip_vpu_get_default_fmt(ctx, true); > > + f->fmt.pix.pixelformat = fmt->fourcc; > > + } > > + > > + pix_mp->num_planes = 1; > > + pix_mp->field = V4L2_FIELD_NONE; > > + pix_mp->width = clamp(pix_mp->width, > > + fmt->frmsize.min_width, > > + fmt->frmsize.max_width); > > + pix_mp->height = clamp(pix_mp->height, > > + fmt->frmsize.min_height, > > + fmt->frmsize.max_height); > > Don't we also need to align to macroblocks? > > > + pix_mp->plane_fmt[0].sizeimage = fmt->header_size + > > + pix_mp->width * pix_mp->height * fmt->max_depth; > > I suppose this is a hint for the potential maximum compressed size? > Indeed. > I don't like the idea of enforcing one particular size on the user > space. Or even a minimum size. For example, the user may know that the > image is well-compressible and want to use smaller buffers. Or may > want to try with a smaller buffer first and reallocate to a bigger one > if it turns out to be too small. > > I'd just leave this kind of logic to the user space. > Right. However, it seems to me we still need to set a value, and return it to userspace when userspace doesn't provide any. > > + memset(pix_mp->plane_fmt[0].reserved, 0, > > + sizeof(pix_mp->plane_fmt[0].reserved)); > > Does every driver really need to memset() this to 0 on its own? Sounds crazy. > Indeed. It's something to consider fixing in the core. > > + return 0; > > +} > > + > > +static int > > +vidioc_try_fmt_out_mplane(struct file *file, void *priv, struct v4l2_format *f) > > +{ > > + struct rockchip_vpu_ctx *ctx = fh_to_ctx(priv); > > + struct v4l2_pix_format_mplane *pix_mp = &f->fmt.pix_mp; > > + const struct rockchip_vpu_fmt *fmt; > > + unsigned int width, height; > > + unsigned long dma_align; > > + bool need_alignment; > > + int i; > > + > > + vpu_debug(4, "%c%c%c%c\n", > > + (pix_mp->pixelformat & 0x7f), > > + (pix_mp->pixelformat >> 8) & 0x7f, > > + (pix_mp->pixelformat >> 16) & 0x7f, > > + (pix_mp->pixelformat >> 24) & 0x7f); > > + > > + fmt = rockchip_vpu_find_format(ctx, pix_mp->pixelformat); > > + if (!fmt) { > > + fmt = rockchip_vpu_get_default_fmt(ctx, false); > > + f->fmt.pix.pixelformat = fmt->fourcc; > > + } > > + > > + pix_mp->field = V4L2_FIELD_NONE; > > + width = clamp(pix_mp->width, > > + ctx->vpu_dst_fmt->frmsize.min_width, > > + ctx->vpu_dst_fmt->frmsize.max_width); > > + height = clamp(pix_mp->height, > > + ctx->vpu_dst_fmt->frmsize.min_height, > > + ctx->vpu_dst_fmt->frmsize.max_height); > > + /* Round up to macroblocks. */ > > + width = round_up(width, MB_DIM); > > + height = round_up(height, MB_DIM); > > + > > + /* Fill remaining fields */ > > + fill_pixfmt_mp(pix_mp, fmt->fourcc, width, height); > > + > > + for (i = 0; i < pix_mp->num_planes; i++) { > > + memset(pix_mp->plane_fmt[i].reserved, 0, > > + sizeof(pix_mp->plane_fmt[i].reserved)); > > + } > > + > > + dma_align = dma_get_cache_alignment(); > > + need_alignment = false; > > + for (i = 0; i < pix_mp->num_planes; i++) { > > + if (!IS_ALIGNED(pix_mp->plane_fmt[i].sizeimage, > > + dma_align)) { > > + need_alignment = true; > > + break; > > + } > > + } > > + if (!need_alignment) > > + return 0; > > This alignment thing was here only for USERPTR. Since we currently > don't support USERPTR, we don't really care. > OK. > > + > > + pix_mp->height = round_up(pix_mp->height, dma_align * 4 / MB_DIM); > > + if (pix_mp->height > ctx->vpu_dst_fmt->frmsize.max_height) { > > + vpu_err("Aligned height higher than maximum.\n"); > > + return -EINVAL; > > Just FYI since we're going to remove this, we can't fail in this > function. Instead we need to align to the closest supported format. > Ah, good catch. > > + } > > + /* Fill in remaining fields, again */ > > + fill_pixfmt_mp(pix_mp, fmt->fourcc, pix_mp->width, pix_mp->height); > > + return 0; > > +} > > + > > +void rockchip_vpu_enc_reset_dst_fmt(struct rockchip_vpu_dev *vpu, > > + struct rockchip_vpu_ctx *ctx) > > +{ > > + struct v4l2_pix_format_mplane *fmt = &ctx->dst_fmt; > > + > > + ctx->vpu_dst_fmt = rockchip_vpu_get_default_fmt(ctx, true); > > + > > + memset(fmt, 0, sizeof(*fmt)); > > + > > + fmt->num_planes = 1; > > + fmt->width = clamp(fmt->width, ctx->vpu_dst_fmt->frmsize.min_width, > > + ctx->vpu_dst_fmt->frmsize.max_width); > > + fmt->height = clamp(fmt->height, ctx->vpu_dst_fmt->frmsize.min_height, > > + ctx->vpu_dst_fmt->frmsize.max_height); > > + fmt->pixelformat = ctx->vpu_dst_fmt->fourcc; > > + fmt->field = V4L2_FIELD_NONE; > > + fmt->colorspace = ctx->colorspace; > > + fmt->ycbcr_enc = ctx->ycbcr_enc; > > + fmt->xfer_func = ctx->xfer_func; > > + fmt->quantization = ctx->quantization; > > If this is a reset, shouldn't these 4 be reset to default values too? > Right. > > + > > + fmt->plane_fmt[0].sizeimage = ctx->vpu_dst_fmt->header_size + > > + fmt->width * fmt->height * ctx->vpu_dst_fmt->max_depth; > > +} > > + > > +void rockchip_vpu_enc_reset_src_fmt(struct rockchip_vpu_dev *vpu, > > + struct rockchip_vpu_ctx *ctx) > > +{ > > + struct v4l2_pix_format_mplane *fmt = &ctx->src_fmt; > > + unsigned int width, height; > > + > > + ctx->vpu_src_fmt = rockchip_vpu_get_default_fmt(ctx, false); > > + > > + memset(fmt, 0, sizeof(*fmt)); > > + > > + width = clamp(fmt->width, ctx->vpu_dst_fmt->frmsize.min_width, > > + ctx->vpu_dst_fmt->frmsize.max_width); > > + height = clamp(fmt->height, ctx->vpu_dst_fmt->frmsize.min_height, > > + ctx->vpu_dst_fmt->frmsize.max_height); > > + fmt->field = V4L2_FIELD_NONE; > > + fmt->colorspace = ctx->colorspace; > > + fmt->ycbcr_enc = ctx->ycbcr_enc; > > + fmt->xfer_func = ctx->xfer_func; > > + fmt->quantization = ctx->quantization; > > Ditto. > > > + > > + fill_pixfmt_mp(fmt, ctx->vpu_src_fmt->fourcc, width, height); > > +} > > These two don't seem to be very encoder-specific. In particular, > rockchip_vpu_enc_reset_dst_fmt() is not even used by the encoder after > the context is initialized, but it's going to be used by the decoder > to reset the raw format when the coded format changes (similarly to > rockchip_vpu_enc_reset_src_fmt() for encoder). > Yeah, I'm aware there might be some commong logic here, but since the driver is an encoder only for now, I decided to let it be. However, I've made sure the driver is ready to support decoding and other variants, with fairly simple changes. > > + > > +static int > > +vidioc_s_fmt_out_mplane(struct file *file, void *priv, struct v4l2_format *f) > > +{ > > + struct v4l2_pix_format_mplane *pix_mp = &f->fmt.pix_mp; > > + struct rockchip_vpu_ctx *ctx = fh_to_ctx(priv); > > + struct vb2_queue *vq, *peer_vq; > > + int ret; > > + > > + /* Change not allowed if queue is streaming. */ > > + vq = v4l2_m2m_get_vq(ctx->fh.m2m_ctx, f->type); > > + if (vb2_is_streaming(vq)) > > + return -EBUSY; > > + > > + ctx->colorspace = pix_mp->colorspace; > > + ctx->ycbcr_enc = pix_mp->ycbcr_enc; > > + ctx->xfer_func = pix_mp->xfer_func; > > + ctx->quantization = pix_mp->quantization; > > Why do we need to store these 4 in separate fields, rather than just > inside ctx->src_fmt? > I don't think we need to. Let me try to get rid of them. > > + > > + /* > > + * Pixel format change is not allowed when the other queue has > > + * buffers allocated. > > + */ > > + peer_vq = v4l2_m2m_get_vq(ctx->fh.m2m_ctx, > > + V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE); > > + if (vb2_is_busy(peer_vq) && > > + pix_mp->pixelformat != ctx->src_fmt.pixelformat) > > + return -EBUSY; > > This is not true for the OUTPUT queue. It's just not possible to > change the coded format, if raw queue is already busy. > Got it. > > + > > + ret = vidioc_try_fmt_out_mplane(file, priv, f); > > + if (ret) > > + return ret; > > + > > + ctx->vpu_src_fmt = rockchip_vpu_find_format(ctx, pix_mp->pixelformat); > > + ctx->src_fmt = *pix_mp; > > + > > + vpu_debug(0, "OUTPUT codec mode: %d\n", ctx->vpu_src_fmt->codec_mode); > > + vpu_debug(0, "fmt - w: %d, h: %d, mb - w: %d, h: %d\n", > > + pix_mp->width, pix_mp->height, > > + MB_WIDTH(pix_mp->width), > > + MB_HEIGHT(pix_mp->height)); > > + return 0; > > +} > > + > > +static int > > +vidioc_s_fmt_cap_mplane(struct file *file, void *priv, struct v4l2_format *f) > > +{ > > + struct v4l2_pix_format_mplane *pix_mp = &f->fmt.pix_mp; > > + struct rockchip_vpu_ctx *ctx = fh_to_ctx(priv); > > + struct rockchip_vpu_dev *vpu = ctx->dev; > > + struct vb2_queue *vq, *peer_vq; > > + int ret; > > + > > + /* Change not allowed if queue is streaming. */ > > + vq = v4l2_m2m_get_vq(ctx->fh.m2m_ctx, f->type); > > + if (vb2_is_streaming(vq)) > > + return -EBUSY; > > + > > + ctx->colorspace = pix_mp->colorspace; > > + ctx->ycbcr_enc = pix_mp->ycbcr_enc; > > + ctx->xfer_func = pix_mp->xfer_func; > > + ctx->quantization = pix_mp->quantization; > > Ditto. > > > + > > + /* > > + * Pixel format change is not allowed when the other queue has > > + * buffers allocated. > > + */ > > + peer_vq = v4l2_m2m_get_vq(ctx->fh.m2m_ctx, > > + V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE); > > + if (vb2_is_busy(peer_vq) && > > + pix_mp->pixelformat != ctx->dst_fmt.pixelformat) > > + return -EBUSY; > > I don't think this is only about pixel format. For an encoder, the > CAPTURE queue commits the state, which may mean resetting the format > on the OUTPUT queue, but we can't do that if it's busy. > Right. > > + > > + ret = vidioc_try_fmt_cap_mplane(file, priv, f); > > + if (ret) > > + return ret; > > + > > + ctx->vpu_dst_fmt = rockchip_vpu_find_format(ctx, pix_mp->pixelformat); > > + ctx->dst_fmt = *pix_mp; > > + > > + vpu_debug(0, "CAPTURE codec mode: %d\n", ctx->vpu_dst_fmt->codec_mode); > > + vpu_debug(0, "fmt - w: %d, h: %d, mb - w: %d, h: %d\n", > > + pix_mp->width, pix_mp->height, > > + MB_WIDTH(pix_mp->width), > > + MB_HEIGHT(pix_mp->height)); > > + > > + /* > > + * Current raw format might have become invalid with newly > > + * selected codec, so reset it to default just to be safe and > > + * keep internal driver state sane. User is mandated to set > > + * the raw format again after we return, so we don't need > > + * anything smarter. > > + */ > > + rockchip_vpu_enc_reset_src_fmt(vpu, ctx); > > + return 0; > > +} > > + > > +const struct v4l2_ioctl_ops rockchip_vpu_enc_ioctl_ops = { > > + .vidioc_querycap = vidioc_querycap, > > + .vidioc_enum_framesizes = vidioc_enum_framesizes, > > + > > + .vidioc_try_fmt_vid_cap_mplane = vidioc_try_fmt_cap_mplane, > > + .vidioc_try_fmt_vid_out_mplane = vidioc_try_fmt_out_mplane, > > + .vidioc_s_fmt_vid_out_mplane = vidioc_s_fmt_out_mplane, > > + .vidioc_s_fmt_vid_cap_mplane = vidioc_s_fmt_cap_mplane, > > + .vidioc_g_fmt_vid_out_mplane = vidioc_g_fmt_out_mplane, > > + .vidioc_g_fmt_vid_cap_mplane = vidioc_g_fmt_cap_mplane, > > + .vidioc_enum_fmt_vid_out_mplane = vidioc_enum_fmt_vid_out_mplane, > > + .vidioc_enum_fmt_vid_cap_mplane = vidioc_enum_fmt_vid_cap_mplane, > > + > > + .vidioc_reqbufs = v4l2_m2m_ioctl_reqbufs, > > + .vidioc_querybuf = v4l2_m2m_ioctl_querybuf, > > + .vidioc_qbuf = v4l2_m2m_ioctl_qbuf, > > + .vidioc_dqbuf = v4l2_m2m_ioctl_dqbuf, > > + .vidioc_prepare_buf = v4l2_m2m_ioctl_prepare_buf, > > + .vidioc_create_bufs = v4l2_m2m_ioctl_create_bufs, > > + .vidioc_expbuf = v4l2_m2m_ioctl_expbuf, > > + > > + .vidioc_subscribe_event = v4l2_ctrl_subscribe_event, > > + .vidioc_unsubscribe_event = v4l2_event_unsubscribe, > > + > > + .vidioc_streamon = v4l2_m2m_ioctl_streamon, > > + .vidioc_streamoff = v4l2_m2m_ioctl_streamoff, > > +}; > > + > > +static int > > +rockchip_vpu_queue_setup(struct vb2_queue *vq, > > + unsigned int *num_buffers, > > + unsigned int *num_planes, > > + unsigned int sizes[], > > + struct device *alloc_devs[]) > > +{ > > + struct rockchip_vpu_ctx *ctx = vb2_get_drv_priv(vq); > > + const struct rockchip_vpu_fmt *vpu_fmt; > > + struct v4l2_pix_format_mplane *pixfmt; > > + int i; > > + > > + switch (vq->type) { > > + case V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE: > > + vpu_fmt = ctx->vpu_dst_fmt; > > + pixfmt = &ctx->dst_fmt; > > + break; > > + case V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE: > > + vpu_fmt = ctx->vpu_src_fmt; > > + pixfmt = &ctx->src_fmt; > > + break; > > + default: > > + vpu_err("invalid queue type: %d\n", vq->type); > > + return -EINVAL; > > + } > > + > > + if (*num_planes) { > > + if (*num_planes != pixfmt->num_planes) > > nit: Double space. > Ack. > > + return -EINVAL; > > + for (i = 0; i < pixfmt->num_planes; ++i) > > + if (sizes[i] < pixfmt->plane_fmt[i].sizeimage) > > + return -EINVAL; > > + return 0; > > + } > > + > > + *num_planes = pixfmt->num_planes; > > + for (i = 0; i < pixfmt->num_planes; ++i) > > + sizes[i] = pixfmt->plane_fmt[i].sizeimage; > > + return 0; > > +} > > + > > +static int rockchip_vpu_buf_prepare(struct vb2_buffer *vb) > > +{ > > + struct vb2_v4l2_buffer *vbuf = to_vb2_v4l2_buffer(vb); > > + struct vb2_queue *vq = vb->vb2_queue; > > + struct rockchip_vpu_ctx *ctx = vb2_get_drv_priv(vq); > > + const struct rockchip_vpu_fmt *vpu_fmt; > > + struct v4l2_pix_format_mplane *pixfmt; > > + unsigned int sz; > > + int ret = 0; > > + int i; > > + > > + switch (vq->type) { > > + case V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE: > > + vpu_fmt = ctx->vpu_dst_fmt; > > + pixfmt = &ctx->dst_fmt; > > + break; > > + case V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE: > > + vpu_fmt = ctx->vpu_src_fmt; > > + pixfmt = &ctx->src_fmt; > > + > > + if (vbuf->field == V4L2_FIELD_ANY) > > + vbuf->field = V4L2_FIELD_NONE; > > + if (vbuf->field != V4L2_FIELD_NONE) { > > + vpu_debug(4, "field %d not supported\n", > > + vbuf->field); > > + return -EINVAL; > > + } > > + break; > > + default: > > + vpu_err("invalid queue type: %d\n", vq->type); > > + return -EINVAL; > > + } > > + > > + for (i = 0; i < pixfmt->num_planes; ++i) { > > + sz = pixfmt->plane_fmt[i].sizeimage; > > + vpu_debug(4, "plane %d size: %ld, sizeimage: %u\n", > > + i, vb2_plane_size(vb, i), sz); > > + if (vb2_plane_size(vb, i) < sz) { > > + vpu_err("plane %d is too small\n", i); > > + ret = -EINVAL; > > + break; > > + } > > + } > > + > > + return ret; > > +} > > + > > +static void rockchip_vpu_buf_queue(struct vb2_buffer *vb) > > +{ > > + struct rockchip_vpu_ctx *ctx = vb2_get_drv_priv(vb->vb2_queue); > > + struct vb2_v4l2_buffer *vbuf = to_vb2_v4l2_buffer(vb); > > + > > + v4l2_m2m_buf_queue(ctx->fh.m2m_ctx, vbuf); > > +} > > + > > +static int rockchip_vpu_start_streaming(struct vb2_queue *q, unsigned int count) > > +{ > > + struct rockchip_vpu_ctx *ctx = vb2_get_drv_priv(q); > > + enum rockchip_vpu_codec_mode codec_mode; > > + > > + if (V4L2_TYPE_IS_OUTPUT(q->type)) > > + ctx->sequence_out = 0; > > + else > > + ctx->sequence_cap = 0; > > + > > + /* Set codec_ops for the chosen destination format */ > > + codec_mode = ctx->vpu_dst_fmt->codec_mode; > > + > > + vpu_debug(4, "Codec mode = %d\n", codec_mode); > > + ctx->codec_ops = &ctx->dev->variant->codec_ops[codec_mode]; > > + > > + /* A bounce buffer is needed for the JPEG payload */ > > + if (!V4L2_TYPE_IS_OUTPUT(q->type)) { > > + ctx->bounce_size = ctx->dst_fmt.plane_fmt[0].sizeimage - > > + ctx->vpu_dst_fmt->header_size; > > + ctx->bounce_buf = dma_alloc_attrs(ctx->dev->dev, > > + ctx->bounce_size, > > + &ctx->bounce_dma_addr, > > + GFP_KERNEL, > > + DMA_ATTR_ALLOC_SINGLE_PAGES); > > + } > > + return 0; > > +} > > + > > +static void rockchip_vpu_stop_streaming(struct vb2_queue *q) > > +{ > > + struct rockchip_vpu_ctx *ctx = vb2_get_drv_priv(q); > > + > > + if (!V4L2_TYPE_IS_OUTPUT(q->type)) > > + dma_free_attrs(ctx->dev->dev, > > + ctx->bounce_size, > > + ctx->bounce_buf, > > + ctx->bounce_dma_addr, > > + DMA_ATTR_ALLOC_SINGLE_PAGES); > > + > > + /* The mem2mem framework calls v4l2_m2m_cancel_job before > > + * .stop_streaming, so there isn't any job running and > > + * it is safe to return all the buffers. > > + */ > > + for (;;) { > > + struct vb2_v4l2_buffer *vbuf; > > + > > + if (V4L2_TYPE_IS_OUTPUT(q->type)) > > + vbuf = v4l2_m2m_src_buf_remove(ctx->fh.m2m_ctx); > > + else > > + vbuf = v4l2_m2m_dst_buf_remove(ctx->fh.m2m_ctx); > > + if (!vbuf) > > + break; > > + v4l2_m2m_buf_done(vbuf, VB2_BUF_STATE_ERROR); > > + } > > +} > > + > > +const struct vb2_ops rockchip_vpu_enc_queue_ops = { > > + .queue_setup = rockchip_vpu_queue_setup, > > + .buf_prepare = rockchip_vpu_buf_prepare, > > + .buf_queue = rockchip_vpu_buf_queue, > > + .start_streaming = rockchip_vpu_start_streaming, > > + .stop_streaming = rockchip_vpu_stop_streaming, > > + .wait_prepare = vb2_ops_wait_prepare, > > + .wait_finish = vb2_ops_wait_finish, > > +}; > > diff --git a/drivers/staging/media/rockchip/vpu/rockchip_vpu_hw.h b/drivers/staging/media/rockchip/vpu/rockchip_vpu_hw.h > > new file mode 100644 > > index 000000000000..77c5a974c2d9 > > --- /dev/null > > +++ b/drivers/staging/media/rockchip/vpu/rockchip_vpu_hw.h > > @@ -0,0 +1,58 @@ > > +/* SPDX-License-Identifier: GPL-2.0 */ > > +/* > > + * Rockchip VPU codec driver > > + * > > + * Copyright (C) 2018 Google, Inc. > > + * Tomasz Figa <tfiga@xxxxxxxxxxxx> > > + */ > > + > > +#ifndef ROCKCHIP_VPU_HW_H_ > > +#define ROCKCHIP_VPU_HW_H_ > > + > > +#include <linux/interrupt.h> > > +#include <linux/v4l2-controls.h> > > +#include <media/videobuf2-core.h> > > + > > +struct rockchip_vpu_dev; > > +struct rockchip_vpu_ctx; > > +struct rockchip_vpu_buf; > > +struct rockchip_vpu_variant; > > + > > +/** > > + * struct rockchip_vpu_codec_ops - codec mode specific operations > > + * > > + * @run: Start single {en,de)coding job. Called from atomic context > > + * to indicate that a pair of buffers is ready and the hardware > > + * should be programmed and started. > > + * @done: Read back processing results and additional data from hardware. > > + * @reset: Reset the hardware in case of a timeout. > > + */ > > +struct rockchip_vpu_codec_ops { > > + void (*run)(struct rockchip_vpu_ctx *ctx); > > + void (*done)(struct rockchip_vpu_ctx *ctx, enum vb2_buffer_state); > > + void (*reset)(struct rockchip_vpu_ctx *ctx); > > +}; > > + > > +/** > > + * enum rockchip_vpu_enc_fmt - source format ID for hardware registers. > > + */ > > +enum rockchip_vpu_enc_fmt { > > + RK3288_VPU_ENC_FMT_YUV420P = 0, > > + RK3288_VPU_ENC_FMT_YUV420SP = 1, > > + RK3288_VPU_ENC_FMT_YUYV422 = 2, > > + RK3288_VPU_ENC_FMT_UYVY422 = 3, > > +}; > > + > > +extern const struct rockchip_vpu_variant rk3399_vpu_variant; > > +extern const struct rockchip_vpu_variant rk3288_vpu_variant; > > + > > +void rockchip_vpu_watchdog(struct work_struct *work); > > +void rockchip_vpu_run(struct rockchip_vpu_ctx *ctx); > > +void rockchip_vpu_irq_done(struct rockchip_vpu_dev *vpu, > > + unsigned int bytesused, > > + enum vb2_buffer_state result); > > + > > +void rk3288_vpu_jpeg_enc_run(struct rockchip_vpu_ctx *ctx); > > +void rk3399_vpu_jpeg_enc_run(struct rockchip_vpu_ctx *ctx); > > + > > +#endif /* ROCKCHIP_VPU_HW_H_ */ > > diff --git a/drivers/staging/media/rockchip/vpu/rockchip_vpu_jpeg.c b/drivers/staging/media/rockchip/vpu/rockchip_vpu_jpeg.c > > new file mode 100644 > > index 000000000000..da6a5cd5f4b1 > > --- /dev/null > > +++ b/drivers/staging/media/rockchip/vpu/rockchip_vpu_jpeg.c > > @@ -0,0 +1,290 @@ > > +// SPDX-License-Identifier: GPL-2.0+ > > +/* > > + * Copyright (C) Collabora, Ltd. > > + * > > + * Based on GSPCA and CODA drivers: > > + * Copyright (C) Jean-Francois Moine (http://moinejf.free.fr) > > + * Copyright (C) 2014 Philipp Zabel, Pengutronix > > + */ > > +#include <linux/kernel.h> > > +#include <linux/string.h> > > +#include "rockchip_vpu_jpeg.h" > > + > > +#define LUMA_QUANT_OFF 7 > > +#define CHROMA_QUANT_OFF 72 > > +#define HEIGHT_OFF 141 > > +#define WIDTH_OFF 143 > > + > > +#define HUFF_LUMA_DC_OFF 160 > > +#define HUFF_LUMA_AC_OFF 193 > > +#define HUFF_CHROMA_DC_OFF 376 > > +#define HUFF_CHROMA_AC_OFF 409 > > + > > +/* Default tables from JPEG ITU-T.81 > > + * (ISO/IEC 10918-1) Annex K.3, I > > + */ > > +static const unsigned char luma_q_table[] = { > > + 0x10, 0x0b, 0x0a, 0x10, 0x7c, 0x8c, 0x97, 0xa1, > > + 0x0c, 0x0c, 0x0e, 0x13, 0x7e, 0x9e, 0xa0, 0x9b, > > + 0x0e, 0x0d, 0x10, 0x18, 0x8c, 0x9d, 0xa9, 0x9c, > > + 0x0e, 0x11, 0x16, 0x1d, 0x97, 0xbb, 0xb4, 0xa2, > > + 0x12, 0x16, 0x25, 0x38, 0xa8, 0x6d, 0x67, 0xb1, > > + 0x18, 0x23, 0x37, 0x40, 0xb5, 0x68, 0x71, 0xc0, > > + 0x31, 0x40, 0x4e, 0x57, 0x67, 0x79, 0x78, 0x65, > > + 0x48, 0x5c, 0x5f, 0x62, 0x70, 0x64, 0x67, 0xc7, > > +}; > > + > > +static const unsigned char chroma_q_table[] = { > > + 0x11, 0x12, 0x18, 0x2f, 0x63, 0x63, 0x63, 0x63, > > + 0x12, 0x15, 0x1a, 0x42, 0x63, 0x63, 0x63, 0x63, > > + 0x18, 0x1a, 0x38, 0x63, 0x63, 0x63, 0x63, 0x63, > > + 0x2f, 0x42, 0x63, 0x63, 0x63, 0x63, 0x63, 0x63, > > + 0x63, 0x63, 0x63, 0x63, 0x63, 0x63, 0x63, 0x63, > > + 0x63, 0x63, 0x63, 0x63, 0x63, 0x63, 0x63, 0x63, > > + 0x63, 0x63, 0x63, 0x63, 0x63, 0x63, 0x63, 0x63, > > + 0x63, 0x63, 0x63, 0x63, 0x63, 0x63, 0x63, 0x63 > > +}; > > + > > +/* Huffman tables are shared with CODA */ > > +static const unsigned char luma_dc_table[] = { > > + 0x00, 0x01, 0x05, 0x01, 0x01, 0x01, 0x01, 0x01, > > + 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, > > + 0x08, 0x09, 0x0a, 0x0b, > > +}; > > + > > +static const unsigned char chroma_dc_table[] = { > > + 0x00, 0x03, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, > > + 0x01, 0x01, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, > > + 0x08, 0x09, 0x0a, 0x0b, > > +}; > > + > > +static const unsigned char luma_ac_table[] = { > > + 0x00, 0x02, 0x01, 0x03, 0x03, 0x02, 0x04, 0x03, > > + 0x05, 0x05, 0x04, 0x04, 0x00, 0x00, 0x01, 0x7d, > > + 0x01, 0x02, 0x03, 0x00, 0x04, 0x11, 0x05, 0x12, > > + 0x21, 0x31, 0x41, 0x06, 0x13, 0x51, 0x61, 0x07, > > + 0x22, 0x71, 0x14, 0x32, 0x81, 0x91, 0xa1, 0x08, > > + 0x23, 0x42, 0xb1, 0xc1, 0x15, 0x52, 0xd1, 0xf0, > > + 0x24, 0x33, 0x62, 0x72, 0x82, 0x09, 0x0a, 0x16, > > + 0x17, 0x18, 0x19, 0x1a, 0x25, 0x26, 0x27, 0x28, > > + 0x29, 0x2a, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, > > + 0x3a, 0x43, 0x44, 0x45, 0x46, 0x47, 0x48, 0x49, > > + 0x4a, 0x53, 0x54, 0x55, 0x56, 0x57, 0x58, 0x59, > > + 0x5a, 0x63, 0x64, 0x65, 0x66, 0x67, 0x68, 0x69, > > + 0x6a, 0x73, 0x74, 0x75, 0x76, 0x77, 0x78, 0x79, > > + 0x7a, 0x83, 0x84, 0x85, 0x86, 0x87, 0x88, 0x89, > > + 0x8a, 0x92, 0x93, 0x94, 0x95, 0x96, 0x97, 0x98, > > + 0x99, 0x9a, 0xa2, 0xa3, 0xa4, 0xa5, 0xa6, 0xa7, > > + 0xa8, 0xa9, 0xaa, 0xb2, 0xb3, 0xb4, 0xb5, 0xb6, > > + 0xb7, 0xb8, 0xb9, 0xba, 0xc2, 0xc3, 0xc4, 0xc5, > > + 0xc6, 0xc7, 0xc8, 0xc9, 0xca, 0xd2, 0xd3, 0xd4, > > + 0xd5, 0xd6, 0xd7, 0xd8, 0xd9, 0xda, 0xe1, 0xe2, > > + 0xe3, 0xe4, 0xe5, 0xe6, 0xe7, 0xe8, 0xe9, 0xea, > > + 0xf1, 0xf2, 0xf3, 0xf4, 0xf5, 0xf6, 0xf7, 0xf8, > > + 0xf9, 0xfa, > > +}; > > + > > +static const unsigned char chroma_ac_table[] = { > > + 0x00, 0x02, 0x01, 0x02, 0x04, 0x04, 0x03, 0x04, > > + 0x07, 0x05, 0x04, 0x04, 0x00, 0x01, 0x02, 0x77, > > + 0x00, 0x01, 0x02, 0x03, 0x11, 0x04, 0x05, 0x21, > > + 0x31, 0x06, 0x12, 0x41, 0x51, 0x07, 0x61, 0x71, > > + 0x13, 0x22, 0x32, 0x81, 0x08, 0x14, 0x42, 0x91, > > + 0xa1, 0xb1, 0xc1, 0x09, 0x23, 0x33, 0x52, 0xf0, > > + 0x15, 0x62, 0x72, 0xd1, 0x0a, 0x16, 0x24, 0x34, > > + 0xe1, 0x25, 0xf1, 0x17, 0x18, 0x19, 0x1a, 0x26, > > + 0x27, 0x28, 0x29, 0x2a, 0x35, 0x36, 0x37, 0x38, > > + 0x39, 0x3a, 0x43, 0x44, 0x45, 0x46, 0x47, 0x48, > > + 0x49, 0x4a, 0x53, 0x54, 0x55, 0x56, 0x57, 0x58, > > + 0x59, 0x5a, 0x63, 0x64, 0x65, 0x66, 0x67, 0x68, > > + 0x69, 0x6a, 0x73, 0x74, 0x75, 0x76, 0x77, 0x78, > > + 0x79, 0x7a, 0x82, 0x83, 0x84, 0x85, 0x86, 0x87, > > + 0x88, 0x89, 0x8a, 0x92, 0x93, 0x94, 0x95, 0x96, > > + 0x97, 0x98, 0x99, 0x9a, 0xa2, 0xa3, 0xa4, 0xa5, > > + 0xa6, 0xa7, 0xa8, 0xa9, 0xaa, 0xb2, 0xb3, 0xb4, > > + 0xb5, 0xb6, 0xb7, 0xb8, 0xb9, 0xba, 0xc2, 0xc3, > > + 0xc4, 0xc5, 0xc6, 0xc7, 0xc8, 0xc9, 0xca, 0xd2, > > + 0xd3, 0xd4, 0xd5, 0xd6, 0xd7, 0xd8, 0xd9, 0xda, > > + 0xe2, 0xe3, 0xe4, 0xe5, 0xe6, 0xe7, 0xe8, 0xe9, > > + 0xea, 0xf2, 0xf3, 0xf4, 0xf5, 0xf6, 0xf7, 0xf8, > > + 0xf9, 0xfa, > > +}; > > + > > +/* For simplicity, we keep a pre-formatted JPEG header, > > + * and we'll use fixed offsets to change the width, height > > + * quantization tables, etc. > > + */ > > +static const unsigned char rockchip_vpu_jpeg_header[JPEG_HEADER_SIZE] = { > > + /* SOI */ > > + 0xff, 0xd8, > > + > > + /* DQT */ > > + 0xff, 0xdb, 0x00, 0x84, > > + > > + 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + > > + 0x01, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + > > + /* SOF */ > > + 0xff, 0xc0, 0x00, 0x11, 0x08, 0x00, 0xf0, 0x01, > > + 0x40, 0x03, 0x01, 0x22, 0x00, 0x02, 0x11, 0x01, > > + 0x03, 0x11, 0x01, > > + > > + /* DHT */ > > + 0xff, 0xc4, 0x00, 0x1f, 0x00, > > + > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, > > + > > + /* DHT */ > > + 0xff, 0xc4, 0x00, 0xb5, 0x10, > > + > > + 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + > > + /* DHT */ > > + 0xff, 0xc4, 0x00, 0x1f, 0x01, > > + > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, > > + > > + /* DHT */ > > + 0xff, 0xc4, 0x00, 0xb5, 0x11, > > + > > + 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, > > + > > + /* SOS */ > > + 0xff, 0xda, 0x00, 0x0c, 0x03, 0x01, 0x00, 0x02, > > + 0x11, 0x03, 0x11, 0x00, 0x3f, 0x00, > > +}; > > + > > +static void > > +jpeg_scale_quant_table(unsigned char *q_tab, > > + const unsigned char *tab, int scale) > > +{ > > + unsigned int temp; > > + int i; > > + > > + for (i = 0; i < 64; i++) { > > + temp = DIV_ROUND_CLOSEST((unsigned int)tab[i] * scale, 100); > > + if (temp <= 0) > > + temp = 1; > > + if (temp > 255) > > + temp = 255; > > + q_tab[i] = (unsigned char)temp; > > + } > > +} > > + > > +static void jpeg_set_quality(unsigned char *buffer, int quality) > > +{ > > + int scale; > > + > > + /* > > + * Non-linear scaling factor: > > + * [5,50] -> [1000..100], [51,100] -> [98..0] > > + */ > > + if (quality < 50) > > + scale = 5000 / quality; > > + else > > + scale = 200 - 2 * quality; > > + > > + jpeg_scale_quant_table(buffer + LUMA_QUANT_OFF, > > + luma_q_table, scale); > > + jpeg_scale_quant_table(buffer + CHROMA_QUANT_OFF, > > + chroma_q_table, scale); > > +} > > + > > +unsigned char * > > +rockchip_vpu_jpeg_get_qtable(struct rockchip_vpu_jpeg_ctx *ctx, int index) > > +{ > > + if (index == 0) > > + return ctx->buffer + LUMA_QUANT_OFF; > > + return ctx->buffer + CHROMA_QUANT_OFF; > > +} > > + > > +void rockchip_vpu_jpeg_render(struct rockchip_vpu_jpeg_ctx *ctx) > > nit: I'm not sure "render" is the right word here. Maybe it's just me, > but I associate it with rendering of visible graphics. Perhaps > "assemble" would be better? > I don't think render is gfx only, but.. it might be confusing, so how about "rockchip_vpu_jpeg_header_prepare" or "rockchip_vpu_jpeg_header_set" ? This function might be moved to lib/jpeg.c, so getting the name right is not that much of a nitpick. > > +{ > > + char *buf = ctx->buffer; > > + > > + memcpy(buf, rockchip_vpu_jpeg_header, > > + sizeof(rockchip_vpu_jpeg_header)); > > + > > + buf[HEIGHT_OFF + 0] = ctx->height >> 8; > > + buf[HEIGHT_OFF + 1] = ctx->height; > > + buf[WIDTH_OFF + 0] = ctx->width >> 8; > > + buf[WIDTH_OFF + 1] = ctx->width; > > + > > + memcpy(buf + HUFF_LUMA_DC_OFF, luma_dc_table, sizeof(luma_dc_table)); > > + memcpy(buf + HUFF_LUMA_AC_OFF, luma_ac_table, sizeof(luma_ac_table)); > > + memcpy(buf + HUFF_CHROMA_DC_OFF, chroma_dc_table, > > + sizeof(chroma_dc_table)); > > + memcpy(buf + HUFF_CHROMA_AC_OFF, chroma_ac_table, > > + sizeof(chroma_ac_table)); > > + > > + jpeg_set_quality(buf, ctx->quality); > > +} > > diff --git a/drivers/staging/media/rockchip/vpu/rockchip_vpu_jpeg.h b/drivers/staging/media/rockchip/vpu/rockchip_vpu_jpeg.h > > new file mode 100644 > > index 000000000000..ebe34071851e > > --- /dev/null > > +++ b/drivers/staging/media/rockchip/vpu/rockchip_vpu_jpeg.h > > @@ -0,0 +1,14 @@ > > +/* SPDX-License-Identifier: GPL-2.0+ */ > > + > > +#define JPEG_HEADER_SIZE 601 > > Could we just use ARRAY_SIZE() instead? > > We can, just not sure how easily. Thanks a lot for the review! Eze