| // SPDX-License-Identifier: GPL-2.0 |
| /* |
| * Imagination E5010 JPEG Encoder driver. |
| * |
| * TODO: Add MMU and memory tiling support |
| * |
| * Copyright (C) 2023 Texas Instruments Incorporated - https://www.ti.com/ |
| * |
| * Author: David Huang <d-huang@ti.com> |
| * Author: Devarsh Thakkar <devarsht@ti.com> |
| */ |
| |
| #include <linux/clk.h> |
| #include <linux/dma-mapping.h> |
| #include <linux/err.h> |
| #include <linux/interrupt.h> |
| #include <linux/ioctl.h> |
| #include <linux/module.h> |
| #include <linux/of_device.h> |
| #include <linux/pm_runtime.h> |
| #include <media/jpeg.h> |
| #include <media/v4l2-common.h> |
| #include <media/v4l2-ctrls.h> |
| #include <media/v4l2-device.h> |
| #include <media/v4l2-event.h> |
| #include <media/v4l2-ioctl.h> |
| #include <media/v4l2-jpeg.h> |
| #include <media/v4l2-rect.h> |
| #include <media/v4l2-mem2mem.h> |
| #include <media/videobuf2-dma-contig.h> |
| #include <media/videobuf2-v4l2.h> |
| #include "e5010-jpeg-enc.h" |
| #include "e5010-jpeg-enc-hw.h" |
| |
| /* forward declarations */ |
| static const struct of_device_id e5010_of_match[]; |
| |
| static const struct v4l2_file_operations e5010_fops; |
| |
| static const struct v4l2_ioctl_ops e5010_ioctl_ops; |
| |
| static const struct vb2_ops e5010_video_ops; |
| |
| static const struct v4l2_m2m_ops e5010_m2m_ops; |
| |
| static struct e5010_fmt e5010_formats[] = { |
| { |
| .fourcc = V4L2_PIX_FMT_NV12, |
| .num_planes = 1, |
| .type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE, |
| .subsampling = V4L2_JPEG_CHROMA_SUBSAMPLING_420, |
| .chroma_order = CHROMA_ORDER_CB_CR, |
| .frmsize = { MIN_DIMENSION, MAX_DIMENSION, 64, |
| MIN_DIMENSION, MAX_DIMENSION, 8 }, |
| }, |
| { |
| .fourcc = V4L2_PIX_FMT_NV12M, |
| .num_planes = 2, |
| .type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE, |
| .subsampling = V4L2_JPEG_CHROMA_SUBSAMPLING_420, |
| .chroma_order = CHROMA_ORDER_CB_CR, |
| .frmsize = { MIN_DIMENSION, MAX_DIMENSION, 64, |
| MIN_DIMENSION, MAX_DIMENSION, 8 }, |
| }, |
| { |
| .fourcc = V4L2_PIX_FMT_NV21, |
| .num_planes = 1, |
| .type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE, |
| .subsampling = V4L2_JPEG_CHROMA_SUBSAMPLING_420, |
| .chroma_order = CHROMA_ORDER_CR_CB, |
| .frmsize = { MIN_DIMENSION, MAX_DIMENSION, 64, |
| MIN_DIMENSION, MAX_DIMENSION, 8 }, |
| }, |
| { |
| .fourcc = V4L2_PIX_FMT_NV21M, |
| .num_planes = 2, |
| .type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE, |
| .subsampling = V4L2_JPEG_CHROMA_SUBSAMPLING_420, |
| .chroma_order = CHROMA_ORDER_CR_CB, |
| .frmsize = { MIN_DIMENSION, MAX_DIMENSION, 64, |
| MIN_DIMENSION, MAX_DIMENSION, 8 }, |
| }, |
| { |
| .fourcc = V4L2_PIX_FMT_NV16, |
| .num_planes = 1, |
| .type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE, |
| .subsampling = V4L2_JPEG_CHROMA_SUBSAMPLING_422, |
| .chroma_order = CHROMA_ORDER_CB_CR, |
| .frmsize = { MIN_DIMENSION, MAX_DIMENSION, 64, |
| MIN_DIMENSION, MAX_DIMENSION, 8 }, |
| }, |
| { |
| .fourcc = V4L2_PIX_FMT_NV16M, |
| .num_planes = 2, |
| .type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE, |
| .subsampling = V4L2_JPEG_CHROMA_SUBSAMPLING_422, |
| .chroma_order = CHROMA_ORDER_CB_CR, |
| .frmsize = { MIN_DIMENSION, MAX_DIMENSION, 64, |
| MIN_DIMENSION, MAX_DIMENSION, 8 }, |
| |
| }, |
| { |
| .fourcc = V4L2_PIX_FMT_NV61, |
| .num_planes = 1, |
| .type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE, |
| .subsampling = V4L2_JPEG_CHROMA_SUBSAMPLING_422, |
| .chroma_order = CHROMA_ORDER_CR_CB, |
| .frmsize = { MIN_DIMENSION, MAX_DIMENSION, 64, |
| MIN_DIMENSION, MAX_DIMENSION, 8 }, |
| }, |
| { |
| .fourcc = V4L2_PIX_FMT_NV61M, |
| .num_planes = 2, |
| .type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE, |
| .subsampling = V4L2_JPEG_CHROMA_SUBSAMPLING_422, |
| .chroma_order = CHROMA_ORDER_CR_CB, |
| .frmsize = { MIN_DIMENSION, MAX_DIMENSION, 64, |
| MIN_DIMENSION, MAX_DIMENSION, 8 }, |
| }, |
| { |
| .fourcc = V4L2_PIX_FMT_JPEG, |
| .num_planes = 1, |
| .type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE, |
| .subsampling = 0, |
| .chroma_order = 0, |
| .frmsize = { MIN_DIMENSION, MAX_DIMENSION, 16, |
| MIN_DIMENSION, MAX_DIMENSION, 8 }, |
| }, |
| }; |
| |
| static unsigned int debug; |
| module_param(debug, uint, 0644); |
| MODULE_PARM_DESC(debug, "debug level"); |
| |
| #define dprintk(dev, lvl, fmt, arg...) \ |
| v4l2_dbg(lvl, debug, &(dev)->v4l2_dev, "%s: " fmt, __func__, ## arg) |
| |
| static const struct v4l2_event e5010_eos_event = { |
| .type = V4L2_EVENT_EOS |
| }; |
| |
| static const char *type_name(enum v4l2_buf_type type) |
| { |
| switch (type) { |
| case V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE: |
| return "Output"; |
| case V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE: |
| return "Capture"; |
| default: |
| return "Invalid"; |
| } |
| } |
| |
| static struct e5010_q_data *get_queue(struct e5010_context *ctx, enum v4l2_buf_type type) |
| { |
| return (type == V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE) ? &ctx->out_queue : &ctx->cap_queue; |
| } |
| |
| static void calculate_qp_tables(struct e5010_context *ctx) |
| { |
| long long luminosity, contrast; |
| int quality, i; |
| |
| quality = 50 - ctx->quality; |
| |
| luminosity = LUMINOSITY * quality / 50; |
| contrast = CONTRAST * quality / 50; |
| |
| if (quality > 0) { |
| luminosity *= INCREASE; |
| contrast *= INCREASE; |
| } |
| |
| for (i = 0; i < V4L2_JPEG_PIXELS_IN_BLOCK; i++) { |
| long long delta = v4l2_jpeg_ref_table_chroma_qt[i] * contrast + luminosity; |
| int val = (int)(v4l2_jpeg_ref_table_chroma_qt[i] + delta); |
| |
| clamp(val, 1, 255); |
| ctx->chroma_qp[i] = quality == -50 ? 1 : val; |
| |
| delta = v4l2_jpeg_ref_table_luma_qt[i] * contrast + luminosity; |
| val = (int)(v4l2_jpeg_ref_table_luma_qt[i] + delta); |
| clamp(val, 1, 255); |
| ctx->luma_qp[i] = quality == -50 ? 1 : val; |
| } |
| |
| ctx->update_qp = true; |
| } |
| |
| static int update_qp_tables(struct e5010_context *ctx) |
| { |
| struct e5010_dev *e5010 = ctx->e5010; |
| int i, ret = 0; |
| u32 lvalue, cvalue; |
| |
| lvalue = 0; |
| cvalue = 0; |
| |
| for (i = 0; i < QP_TABLE_SIZE; i++) { |
| lvalue |= ctx->luma_qp[i] << (8 * (i % 4)); |
| cvalue |= ctx->chroma_qp[i] << (8 * (i % 4)); |
| if (i % 4 == 3) { |
| ret |= e5010_hw_set_qpvalue(e5010->core_base, |
| JASPER_LUMA_QUANTIZATION_TABLE0_OFFSET |
| + QP_TABLE_FIELD_OFFSET * ((i - 3) / 4), |
| lvalue); |
| ret |= e5010_hw_set_qpvalue(e5010->core_base, |
| JASPER_CHROMA_QUANTIZATION_TABLE0_OFFSET |
| + QP_TABLE_FIELD_OFFSET * ((i - 3) / 4), |
| cvalue); |
| lvalue = 0; |
| cvalue = 0; |
| } |
| } |
| |
| return ret; |
| } |
| |
| static int e5010_set_input_subsampling(void __iomem *core_base, int subsampling) |
| { |
| switch (subsampling) { |
| case V4L2_JPEG_CHROMA_SUBSAMPLING_420: |
| return e5010_hw_set_input_subsampling(core_base, SUBSAMPLING_420); |
| case V4L2_JPEG_CHROMA_SUBSAMPLING_422: |
| return e5010_hw_set_input_subsampling(core_base, SUBSAMPLING_422); |
| default: |
| return -EINVAL; |
| }; |
| } |
| |
| static int e5010_querycap(struct file *file, void *priv, struct v4l2_capability *cap) |
| { |
| strscpy(cap->driver, E5010_MODULE_NAME, sizeof(cap->driver)); |
| strscpy(cap->card, E5010_MODULE_NAME, sizeof(cap->card)); |
| |
| return 0; |
| } |
| |
| static struct e5010_fmt *find_format(struct v4l2_format *f) |
| { |
| int i; |
| |
| for (i = 0; i < ARRAY_SIZE(e5010_formats); ++i) { |
| if (e5010_formats[i].fourcc == f->fmt.pix_mp.pixelformat && |
| e5010_formats[i].type == f->type) |
| return &e5010_formats[i]; |
| } |
| |
| return NULL; |
| } |
| |
| static int e5010_enum_fmt(struct file *file, void *priv, struct v4l2_fmtdesc *f) |
| { |
| int i, index = 0; |
| struct e5010_fmt *fmt = NULL; |
| struct e5010_context *ctx = file->private_data; |
| |
| if (!V4L2_TYPE_IS_MULTIPLANAR(f->type)) { |
| v4l2_err(&ctx->e5010->v4l2_dev, "ENUMFMT with Invalid type: %d\n", f->type); |
| return -EINVAL; |
| } |
| |
| for (i = 0; i < ARRAY_SIZE(e5010_formats); ++i) { |
| if (e5010_formats[i].type == f->type) { |
| if (index == f->index) { |
| fmt = &e5010_formats[i]; |
| break; |
| } |
| index++; |
| } |
| } |
| |
| if (!fmt) |
| return -EINVAL; |
| |
| f->pixelformat = fmt->fourcc; |
| return 0; |
| } |
| |
| static int e5010_g_fmt(struct file *file, void *priv, struct v4l2_format *f) |
| { |
| struct e5010_context *ctx = file->private_data; |
| struct e5010_q_data *queue; |
| int i; |
| struct v4l2_pix_format_mplane *pix_mp = &f->fmt.pix_mp; |
| struct v4l2_plane_pix_format *plane_fmt = pix_mp->plane_fmt; |
| |
| queue = get_queue(ctx, f->type); |
| |
| pix_mp->flags = 0; |
| pix_mp->field = V4L2_FIELD_NONE; |
| pix_mp->pixelformat = queue->fmt->fourcc; |
| pix_mp->width = queue->width_adjusted; |
| pix_mp->height = queue->height_adjusted; |
| pix_mp->num_planes = queue->fmt->num_planes; |
| |
| if (V4L2_TYPE_IS_OUTPUT(f->type)) { |
| if (!pix_mp->colorspace) |
| pix_mp->colorspace = V4L2_COLORSPACE_SRGB; |
| |
| for (i = 0; i < queue->fmt->num_planes; i++) { |
| plane_fmt[i].sizeimage = queue->sizeimage[i]; |
| plane_fmt[i].bytesperline = queue->bytesperline[i]; |
| } |
| |
| } else { |
| pix_mp->colorspace = V4L2_COLORSPACE_JPEG; |
| plane_fmt[0].bytesperline = 0; |
| plane_fmt[0].sizeimage = queue->sizeimage[0]; |
| } |
| pix_mp->ycbcr_enc = V4L2_YCBCR_ENC_DEFAULT; |
| pix_mp->xfer_func = V4L2_XFER_FUNC_DEFAULT; |
| pix_mp->quantization = V4L2_QUANTIZATION_DEFAULT; |
| |
| return 0; |
| } |
| |
| static int e5010_jpeg_try_fmt(struct v4l2_format *f, struct e5010_context *ctx) |
| { |
| struct e5010_fmt *fmt; |
| struct v4l2_pix_format_mplane *pix_mp = &f->fmt.pix_mp; |
| struct v4l2_plane_pix_format *plane_fmt = pix_mp->plane_fmt; |
| |
| fmt = find_format(f); |
| if (!fmt) { |
| if (V4L2_TYPE_IS_OUTPUT(f->type)) |
| pix_mp->pixelformat = V4L2_PIX_FMT_NV12; |
| else |
| pix_mp->pixelformat = V4L2_PIX_FMT_JPEG; |
| fmt = find_format(f); |
| if (!fmt) |
| return -EINVAL; |
| } |
| |
| if (V4L2_TYPE_IS_OUTPUT(f->type)) { |
| if (!pix_mp->colorspace) |
| pix_mp->colorspace = V4L2_COLORSPACE_JPEG; |
| if (!pix_mp->ycbcr_enc) |
| pix_mp->ycbcr_enc = V4L2_YCBCR_ENC_DEFAULT; |
| if (!pix_mp->quantization) |
| pix_mp->quantization = V4L2_QUANTIZATION_DEFAULT; |
| if (!pix_mp->xfer_func) |
| pix_mp->xfer_func = V4L2_XFER_FUNC_DEFAULT; |
| |
| v4l2_apply_frmsize_constraints(&pix_mp->width, |
| &pix_mp->height, |
| &fmt->frmsize); |
| |
| v4l2_fill_pixfmt_mp(pix_mp, pix_mp->pixelformat, |
| pix_mp->width, pix_mp->height); |
| |
| } else { |
| pix_mp->colorspace = V4L2_COLORSPACE_JPEG; |
| pix_mp->ycbcr_enc = V4L2_YCBCR_ENC_DEFAULT; |
| pix_mp->quantization = V4L2_QUANTIZATION_DEFAULT; |
| pix_mp->xfer_func = V4L2_XFER_FUNC_DEFAULT; |
| v4l2_apply_frmsize_constraints(&pix_mp->width, |
| &pix_mp->height, |
| &fmt->frmsize); |
| plane_fmt[0].sizeimage = pix_mp->width * pix_mp->height * JPEG_MAX_BYTES_PER_PIXEL; |
| plane_fmt[0].sizeimage += HEADER_SIZE; |
| plane_fmt[0].bytesperline = 0; |
| pix_mp->pixelformat = fmt->fourcc; |
| pix_mp->num_planes = fmt->num_planes; |
| } |
| pix_mp->flags = 0; |
| pix_mp->field = V4L2_FIELD_NONE; |
| |
| dprintk(ctx->e5010, 2, |
| "ctx: 0x%p: format type %s:, wxh: %dx%d (plane0 : %d bytes, plane1 : %d bytes),fmt: %c%c%c%c\n", |
| ctx, type_name(f->type), pix_mp->width, pix_mp->height, |
| plane_fmt[0].sizeimage, plane_fmt[1].sizeimage, |
| (fmt->fourcc & 0xff), |
| (fmt->fourcc >> 8) & 0xff, |
| (fmt->fourcc >> 16) & 0xff, |
| (fmt->fourcc >> 24) & 0xff); |
| |
| return 0; |
| } |
| |
| static int e5010_try_fmt(struct file *file, void *priv, struct v4l2_format *f) |
| { |
| struct e5010_context *ctx = file->private_data; |
| |
| return e5010_jpeg_try_fmt(f, ctx); |
| } |
| |
| static int e5010_s_fmt(struct file *file, void *priv, struct v4l2_format *f) |
| { |
| struct e5010_context *ctx = file->private_data; |
| struct vb2_queue *vq; |
| int ret = 0, i = 0; |
| struct v4l2_pix_format_mplane *pix_mp = &f->fmt.pix_mp; |
| struct v4l2_plane_pix_format *plane_fmt = pix_mp->plane_fmt; |
| struct e5010_q_data *queue; |
| struct e5010_fmt *fmt; |
| |
| vq = v4l2_m2m_get_vq(ctx->fh.m2m_ctx, f->type); |
| if (!vq) |
| return -EINVAL; |
| |
| if (vb2_is_busy(vq)) { |
| v4l2_err(&ctx->e5010->v4l2_dev, "queue busy\n"); |
| return -EBUSY; |
| } |
| |
| ret = e5010_jpeg_try_fmt(f, ctx); |
| if (ret) |
| return ret; |
| |
| fmt = find_format(f); |
| queue = get_queue(ctx, f->type); |
| |
| queue->fmt = fmt; |
| queue->width = pix_mp->width; |
| queue->height = pix_mp->height; |
| |
| if (V4L2_TYPE_IS_OUTPUT(f->type)) { |
| for (i = 0; i < fmt->num_planes; i++) { |
| queue->bytesperline[i] = plane_fmt[i].bytesperline; |
| queue->sizeimage[i] = plane_fmt[i].sizeimage; |
| } |
| queue->crop.left = 0; |
| queue->crop.top = 0; |
| queue->crop.width = queue->width; |
| queue->crop.height = queue->height; |
| } else { |
| queue->sizeimage[0] = plane_fmt[0].sizeimage; |
| queue->sizeimage[1] = 0; |
| queue->bytesperline[0] = 0; |
| queue->bytesperline[1] = 0; |
| } |
| |
| return 0; |
| } |
| |
| static int e5010_enum_framesizes(struct file *file, void *priv, struct v4l2_frmsizeenum *fsize) |
| { |
| struct v4l2_format f; |
| struct e5010_fmt *fmt; |
| |
| if (fsize->index != 0) |
| return -EINVAL; |
| |
| f.fmt.pix_mp.pixelformat = fsize->pixel_format; |
| if (f.fmt.pix_mp.pixelformat == V4L2_PIX_FMT_JPEG) |
| f.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE; |
| else |
| f.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE; |
| |
| fmt = find_format(&f); |
| if (!fmt) |
| return -EINVAL; |
| |
| fsize->type = V4L2_FRMSIZE_TYPE_STEPWISE; |
| fsize->stepwise = fmt->frmsize; |
| fsize->reserved[0] = 0; |
| fsize->reserved[1] = 0; |
| |
| return 0; |
| } |
| |
| static int e5010_g_selection(struct file *file, void *fh, struct v4l2_selection *s) |
| { |
| struct e5010_context *ctx = file->private_data; |
| struct e5010_q_data *queue; |
| |
| if (s->type != V4L2_BUF_TYPE_VIDEO_OUTPUT) |
| return -EINVAL; |
| |
| queue = get_queue(ctx, V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE); |
| |
| switch (s->target) { |
| case V4L2_SEL_TGT_CROP_DEFAULT: |
| case V4L2_SEL_TGT_CROP_BOUNDS: |
| s->r.left = 0; |
| s->r.top = 0; |
| s->r.width = queue->width; |
| s->r.height = queue->height; |
| break; |
| case V4L2_SEL_TGT_CROP: |
| memcpy(&s->r, &queue->crop, sizeof(s->r)); |
| break; |
| default: |
| return -EINVAL; |
| } |
| |
| return 0; |
| } |
| |
| static int e5010_s_selection(struct file *file, void *fh, struct v4l2_selection *s) |
| { |
| struct e5010_context *ctx = file->private_data; |
| struct e5010_q_data *queue; |
| struct vb2_queue *vq; |
| struct v4l2_rect base_rect; |
| |
| vq = v4l2_m2m_get_vq(ctx->fh.m2m_ctx, s->type); |
| if (!vq) |
| return -EINVAL; |
| |
| if (vb2_is_streaming(vq)) |
| return -EBUSY; |
| |
| if (s->target != V4L2_SEL_TGT_CROP || |
| s->type != V4L2_BUF_TYPE_VIDEO_OUTPUT) |
| return -EINVAL; |
| |
| queue = get_queue(ctx, V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE); |
| base_rect.top = 0; |
| base_rect.left = 0; |
| base_rect.width = queue->width; |
| base_rect.height = queue->height; |
| |
| switch (s->flags) { |
| case 0: |
| s->r.width = round_down(s->r.width, queue->fmt->frmsize.step_width); |
| s->r.height = round_down(s->r.height, queue->fmt->frmsize.step_height); |
| s->r.left = round_down(s->r.left, queue->fmt->frmsize.step_width); |
| s->r.top = round_down(s->r.top, 2); |
| |
| if (s->r.left + s->r.width > queue->width) |
| s->r.width = round_down(s->r.width + s->r.left - queue->width, |
| queue->fmt->frmsize.step_width); |
| if (s->r.top + s->r.height > queue->height) |
| s->r.top = round_down(s->r.top + s->r.height - queue->height, 2); |
| break; |
| case V4L2_SEL_FLAG_GE: |
| s->r.width = round_up(s->r.width, queue->fmt->frmsize.step_width); |
| s->r.height = round_up(s->r.height, queue->fmt->frmsize.step_height); |
| s->r.left = round_up(s->r.left, queue->fmt->frmsize.step_width); |
| s->r.top = round_up(s->r.top, 2); |
| break; |
| case V4L2_SEL_FLAG_LE: |
| s->r.width = round_down(s->r.width, queue->fmt->frmsize.step_width); |
| s->r.height = round_down(s->r.height, queue->fmt->frmsize.step_height); |
| s->r.left = round_down(s->r.left, queue->fmt->frmsize.step_width); |
| s->r.top = round_down(s->r.top, 2); |
| break; |
| case V4L2_SEL_FLAG_LE | V4L2_SEL_FLAG_GE: |
| if (!IS_ALIGNED(s->r.width, queue->fmt->frmsize.step_width) || |
| !IS_ALIGNED(s->r.height, queue->fmt->frmsize.step_height) || |
| !IS_ALIGNED(s->r.left, queue->fmt->frmsize.step_width) || |
| !IS_ALIGNED(s->r.top, 2)) |
| return -ERANGE; |
| break; |
| default: |
| return -EINVAL; |
| } |
| |
| if (!v4l2_rect_enclosed(&s->r, &base_rect)) |
| return -ERANGE; |
| |
| memcpy(&queue->crop, &s->r, sizeof(s->r)); |
| |
| if (!v4l2_rect_equal(&s->r, &base_rect)) |
| queue->crop_set = true; |
| |
| dprintk(ctx->e5010, 2, "ctx: 0x%p: crop rectangle: w: %d, h : %d, l : %d, t : %d\n", |
| ctx, queue->crop.width, queue->crop.height, queue->crop.left, queue->crop.top); |
| |
| return 0; |
| } |
| |
| static int e5010_subscribe_event(struct v4l2_fh *fh, const struct v4l2_event_subscription *sub) |
| { |
| switch (sub->type) { |
| case V4L2_EVENT_EOS: |
| return v4l2_event_subscribe(fh, sub, 0, NULL); |
| case V4L2_EVENT_CTRL: |
| return v4l2_ctrl_subscribe_event(fh, sub); |
| default: |
| return -EINVAL; |
| } |
| |
| return 0; |
| } |
| |
| static int queue_init(void *priv, struct vb2_queue *src_vq, struct vb2_queue *dst_vq) |
| { |
| struct e5010_context *ctx = priv; |
| struct e5010_dev *e5010 = ctx->e5010; |
| int ret = 0; |
| |
| /* src_vq */ |
| memset(src_vq, 0, sizeof(*src_vq)); |
| src_vq->type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE; |
| src_vq->io_modes = VB2_MMAP | VB2_DMABUF; |
| src_vq->drv_priv = ctx; |
| src_vq->buf_struct_size = sizeof(struct e5010_buffer); |
| src_vq->ops = &e5010_video_ops; |
| src_vq->mem_ops = &vb2_dma_contig_memops; |
| src_vq->timestamp_flags = V4L2_BUF_FLAG_TIMESTAMP_COPY; |
| src_vq->lock = &e5010->mutex; |
| src_vq->dev = e5010->v4l2_dev.dev; |
| |
| ret = vb2_queue_init(src_vq); |
| if (ret) |
| return ret; |
| |
| /* dst_vq */ |
| memset(dst_vq, 0, sizeof(*dst_vq)); |
| dst_vq->type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE; |
| dst_vq->io_modes = VB2_MMAP | VB2_DMABUF; |
| dst_vq->drv_priv = ctx; |
| dst_vq->buf_struct_size = sizeof(struct e5010_buffer); |
| dst_vq->ops = &e5010_video_ops; |
| dst_vq->mem_ops = &vb2_dma_contig_memops; |
| dst_vq->timestamp_flags = V4L2_BUF_FLAG_TIMESTAMP_COPY; |
| dst_vq->lock = &e5010->mutex; |
| dst_vq->dev = e5010->v4l2_dev.dev; |
| |
| ret = vb2_queue_init(dst_vq); |
| if (ret) { |
| vb2_queue_release(src_vq); |
| return ret; |
| } |
| |
| return 0; |
| } |
| |
| static int e5010_s_ctrl(struct v4l2_ctrl *ctrl) |
| { |
| struct e5010_context *ctx = |
| container_of(ctrl->handler, struct e5010_context, ctrl_handler); |
| |
| switch (ctrl->id) { |
| case V4L2_CID_JPEG_COMPRESSION_QUALITY: |
| ctx->quality = ctrl->val; |
| calculate_qp_tables(ctx); |
| dprintk(ctx->e5010, 2, "ctx: 0x%p compression quality set to : %d\n", ctx, |
| ctx->quality); |
| break; |
| default: |
| return -EINVAL; |
| } |
| |
| return 0; |
| } |
| |
| static const struct v4l2_ctrl_ops e5010_ctrl_ops = { |
| .s_ctrl = e5010_s_ctrl, |
| }; |
| |
| static void e5010_encode_ctrls(struct e5010_context *ctx) |
| { |
| v4l2_ctrl_new_std(&ctx->ctrl_handler, &e5010_ctrl_ops, |
| V4L2_CID_JPEG_COMPRESSION_QUALITY, 1, 100, 1, 75); |
| } |
| |
| static int e5010_ctrls_setup(struct e5010_context *ctx) |
| { |
| int err; |
| |
| v4l2_ctrl_handler_init(&ctx->ctrl_handler, 1); |
| |
| e5010_encode_ctrls(ctx); |
| |
| if (ctx->ctrl_handler.error) { |
| err = ctx->ctrl_handler.error; |
| v4l2_ctrl_handler_free(&ctx->ctrl_handler); |
| |
| return err; |
| } |
| |
| err = v4l2_ctrl_handler_setup(&ctx->ctrl_handler); |
| if (err) |
| v4l2_ctrl_handler_free(&ctx->ctrl_handler); |
| |
| return err; |
| } |
| |
| static void e5010_jpeg_set_default_params(struct e5010_context *ctx) |
| { |
| struct e5010_q_data *queue; |
| struct v4l2_format f; |
| struct e5010_fmt *fmt; |
| struct v4l2_pix_format_mplane *pix_mp = &f.fmt.pix_mp; |
| struct v4l2_plane_pix_format *plane_fmt = pix_mp->plane_fmt; |
| int i = 0; |
| |
| f.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE; |
| f.fmt.pix_mp.pixelformat = V4L2_PIX_FMT_NV12; |
| fmt = find_format(&f); |
| queue = &ctx->out_queue; |
| queue->fmt = fmt; |
| queue->width = DEFAULT_WIDTH; |
| queue->height = DEFAULT_HEIGHT; |
| pix_mp->width = queue->width; |
| pix_mp->height = queue->height; |
| queue->crop.left = 0; |
| queue->crop.top = 0; |
| queue->crop.width = queue->width; |
| queue->crop.height = queue->height; |
| v4l2_apply_frmsize_constraints(&pix_mp->width, |
| &pix_mp->height, |
| &fmt->frmsize); |
| v4l2_fill_pixfmt_mp(pix_mp, pix_mp->pixelformat, |
| pix_mp->width, pix_mp->height); |
| for (i = 0; i < fmt->num_planes; i++) { |
| queue->bytesperline[i] = plane_fmt[i].bytesperline; |
| queue->sizeimage[i] = plane_fmt[i].sizeimage; |
| } |
| queue->width_adjusted = pix_mp->width; |
| queue->height_adjusted = pix_mp->height; |
| |
| f.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE; |
| f.fmt.pix_mp.pixelformat = V4L2_PIX_FMT_JPEG; |
| fmt = find_format(&f); |
| queue = &ctx->cap_queue; |
| queue->fmt = fmt; |
| queue->width = DEFAULT_WIDTH; |
| queue->height = DEFAULT_HEIGHT; |
| pix_mp->width = queue->width; |
| pix_mp->height = queue->height; |
| v4l2_apply_frmsize_constraints(&pix_mp->width, |
| &pix_mp->height, |
| &fmt->frmsize); |
| queue->sizeimage[0] = pix_mp->width * pix_mp->height * JPEG_MAX_BYTES_PER_PIXEL; |
| queue->sizeimage[0] += HEADER_SIZE; |
| queue->sizeimage[1] = 0; |
| queue->bytesperline[0] = 0; |
| queue->bytesperline[1] = 0; |
| queue->width_adjusted = pix_mp->width; |
| queue->height_adjusted = pix_mp->height; |
| } |
| |
| static int e5010_open(struct file *file) |
| { |
| struct e5010_dev *e5010 = video_drvdata(file); |
| struct video_device *vdev = video_devdata(file); |
| struct e5010_context *ctx; |
| int ret = 0; |
| |
| ctx = kzalloc(sizeof(*ctx), GFP_KERNEL); |
| if (!ctx) |
| return -ENOMEM; |
| |
| if (mutex_lock_interruptible(&e5010->mutex)) { |
| ret = -ERESTARTSYS; |
| goto free; |
| } |
| |
| v4l2_fh_init(&ctx->fh, vdev); |
| file->private_data = ctx; |
| v4l2_fh_add(&ctx->fh); |
| |
| ctx->e5010 = e5010; |
| ctx->fh.m2m_ctx = v4l2_m2m_ctx_init(e5010->m2m_dev, ctx, queue_init); |
| if (IS_ERR(ctx->fh.m2m_ctx)) { |
| v4l2_err(&e5010->v4l2_dev, "failed to init m2m ctx\n"); |
| ret = PTR_ERR(ctx->fh.m2m_ctx); |
| goto exit; |
| } |
| |
| ret = e5010_ctrls_setup(ctx); |
| if (ret) { |
| v4l2_err(&e5010->v4l2_dev, "failed to setup e5010 jpeg controls\n"); |
| goto err_ctrls_setup; |
| } |
| ctx->fh.ctrl_handler = &ctx->ctrl_handler; |
| |
| e5010_jpeg_set_default_params(ctx); |
| |
| dprintk(e5010, 1, "Created instance: 0x%p, m2m_ctx: 0x%p\n", ctx, ctx->fh.m2m_ctx); |
| |
| mutex_unlock(&e5010->mutex); |
| return 0; |
| |
| err_ctrls_setup: |
| v4l2_m2m_ctx_release(ctx->fh.m2m_ctx); |
| exit: |
| v4l2_fh_del(&ctx->fh); |
| v4l2_fh_exit(&ctx->fh); |
| mutex_unlock(&e5010->mutex); |
| free: |
| kfree(ctx); |
| return ret; |
| } |
| |
| static int e5010_release(struct file *file) |
| { |
| struct e5010_dev *e5010 = video_drvdata(file); |
| struct e5010_context *ctx = file->private_data; |
| |
| dprintk(e5010, 1, "Releasing instance: 0x%p, m2m_ctx: 0x%p\n", ctx, ctx->fh.m2m_ctx); |
| mutex_lock(&e5010->mutex); |
| v4l2_ctrl_handler_free(&ctx->ctrl_handler); |
| v4l2_m2m_ctx_release(ctx->fh.m2m_ctx); |
| v4l2_fh_del(&ctx->fh); |
| v4l2_fh_exit(&ctx->fh); |
| kfree(ctx); |
| mutex_unlock(&e5010->mutex); |
| |
| return 0; |
| } |
| |
| static void header_write(struct e5010_context *ctx, u8 *addr, unsigned int *offset, |
| unsigned int no_bytes, unsigned long bits) |
| { |
| u8 *w_addr = addr + *offset; |
| int i; |
| |
| if ((*offset + no_bytes) > HEADER_SIZE) { |
| v4l2_warn(&ctx->e5010->v4l2_dev, "%s: %s: %d: Problem writing header. %d > HEADER_SIZE %d\n", |
| __FILE__, __func__, __LINE__, *offset + no_bytes, HEADER_SIZE); |
| return; |
| } |
| |
| for (i = no_bytes - 1; i >= 0; i--) |
| *(w_addr++) = ((u8 *)&bits)[i]; |
| |
| *offset += no_bytes; |
| } |
| |
| static void encode_marker_segment(struct e5010_context *ctx, void *addr, unsigned int *offset) |
| { |
| u8 *buffer = (u8 *)addr; |
| int i; |
| |
| header_write(ctx, buffer, offset, 2, START_OF_IMAGE); |
| header_write(ctx, buffer, offset, 2, DQT_MARKER); |
| header_write(ctx, buffer, offset, 3, LQPQ << 4); |
| for (i = 0; i < V4L2_JPEG_PIXELS_IN_BLOCK; i++) |
| header_write(ctx, buffer, offset, 1, ctx->luma_qp[v4l2_jpeg_zigzag_scan_index[i]]); |
| |
| header_write(ctx, buffer, offset, 2, DQT_MARKER); |
| header_write(ctx, buffer, offset, 3, (LQPQ << 4) | 1); |
| for (i = 0; i < V4L2_JPEG_PIXELS_IN_BLOCK; i++) |
| header_write(ctx, buffer, offset, 1, |
| ctx->chroma_qp[v4l2_jpeg_zigzag_scan_index[i]]); |
| |
| /* Huffman tables */ |
| header_write(ctx, buffer, offset, 2, DHT_MARKER); |
| header_write(ctx, buffer, offset, 2, LH_DC); |
| header_write(ctx, buffer, offset, 1, V4L2_JPEG_LUM_HT | V4L2_JPEG_DC_HT); |
| for (i = 0 ; i < V4L2_JPEG_REF_HT_DC_LEN; i++) |
| header_write(ctx, buffer, offset, 1, v4l2_jpeg_ref_table_luma_dc_ht[i]); |
| |
| header_write(ctx, buffer, offset, 2, DHT_MARKER); |
| header_write(ctx, buffer, offset, 2, LH_AC); |
| header_write(ctx, buffer, offset, 1, V4L2_JPEG_LUM_HT | V4L2_JPEG_AC_HT); |
| for (i = 0 ; i < V4L2_JPEG_REF_HT_AC_LEN; i++) |
| header_write(ctx, buffer, offset, 1, v4l2_jpeg_ref_table_luma_ac_ht[i]); |
| |
| header_write(ctx, buffer, offset, 2, DHT_MARKER); |
| header_write(ctx, buffer, offset, 2, LH_DC); |
| header_write(ctx, buffer, offset, 1, V4L2_JPEG_CHR_HT | V4L2_JPEG_DC_HT); |
| for (i = 0 ; i < V4L2_JPEG_REF_HT_DC_LEN; i++) |
| header_write(ctx, buffer, offset, 1, v4l2_jpeg_ref_table_chroma_dc_ht[i]); |
| |
| header_write(ctx, buffer, offset, 2, DHT_MARKER); |
| header_write(ctx, buffer, offset, 2, LH_AC); |
| header_write(ctx, buffer, offset, 1, V4L2_JPEG_CHR_HT | V4L2_JPEG_AC_HT); |
| for (i = 0 ; i < V4L2_JPEG_REF_HT_AC_LEN; i++) |
| header_write(ctx, buffer, offset, 1, v4l2_jpeg_ref_table_chroma_ac_ht[i]); |
| } |
| |
| static void encode_frame_header(struct e5010_context *ctx, void *addr, unsigned int *offset) |
| { |
| u8 *buffer = (u8 *)addr; |
| |
| header_write(ctx, buffer, offset, 2, SOF_BASELINE_DCT); |
| header_write(ctx, buffer, offset, 2, 8 + (3 * UC_NUM_COMP)); |
| header_write(ctx, buffer, offset, 1, PRECISION); |
| header_write(ctx, buffer, offset, 2, ctx->out_queue.crop.height); |
| header_write(ctx, buffer, offset, 2, ctx->out_queue.crop.width); |
| header_write(ctx, buffer, offset, 1, UC_NUM_COMP); |
| |
| /* Luma details */ |
| header_write(ctx, buffer, offset, 1, 1); |
| if (ctx->out_queue.fmt->subsampling == V4L2_JPEG_CHROMA_SUBSAMPLING_422) |
| header_write(ctx, buffer, offset, 1, |
| HORZ_SAMPLING_FACTOR | (VERT_SAMPLING_FACTOR_422)); |
| else |
| header_write(ctx, buffer, offset, 1, |
| HORZ_SAMPLING_FACTOR | (VERT_SAMPLING_FACTOR_420)); |
| header_write(ctx, buffer, offset, 1, 0); |
| /* Chroma details */ |
| header_write(ctx, buffer, offset, 1, 2); |
| header_write(ctx, buffer, offset, 1, (HORZ_SAMPLING_FACTOR >> 1) | 1); |
| header_write(ctx, buffer, offset, 1, 1); |
| header_write(ctx, buffer, offset, 1, 3); |
| header_write(ctx, buffer, offset, 1, (HORZ_SAMPLING_FACTOR >> 1) | 1); |
| header_write(ctx, buffer, offset, 1, 1); |
| } |
| |
| static void jpg_encode_sos_header(struct e5010_context *ctx, void *addr, unsigned int *offset) |
| { |
| u8 *buffer = (u8 *)addr; |
| int i; |
| |
| header_write(ctx, buffer, offset, 2, START_OF_SCAN); |
| header_write(ctx, buffer, offset, 2, 6 + (COMPONENTS_IN_SCAN << 1)); |
| header_write(ctx, buffer, offset, 1, COMPONENTS_IN_SCAN); |
| |
| for (i = 0; i < COMPONENTS_IN_SCAN; i++) { |
| header_write(ctx, buffer, offset, 1, i + 1); |
| if (i == 0) |
| header_write(ctx, buffer, offset, 1, 0); |
| else |
| header_write(ctx, buffer, offset, 1, 17); |
| } |
| |
| header_write(ctx, buffer, offset, 1, 0); |
| header_write(ctx, buffer, offset, 1, 63); |
| header_write(ctx, buffer, offset, 1, 0); |
| } |
| |
| static void write_header(struct e5010_context *ctx, void *addr) |
| { |
| unsigned int offset = 0; |
| |
| encode_marker_segment(ctx, addr, &offset); |
| encode_frame_header(ctx, addr, &offset); |
| jpg_encode_sos_header(ctx, addr, &offset); |
| } |
| |
| static irqreturn_t e5010_irq(int irq, void *data) |
| { |
| struct e5010_dev *e5010 = data; |
| struct e5010_context *ctx; |
| int output_size; |
| struct vb2_v4l2_buffer *src_buf, *dst_buf; |
| bool pic_done, out_addr_err; |
| |
| spin_lock(&e5010->hw_lock); |
| pic_done = e5010_hw_pic_done_irq(e5010->core_base); |
| out_addr_err = e5010_hw_output_address_irq(e5010->core_base); |
| |
| if (!pic_done && !out_addr_err) { |
| spin_unlock(&e5010->hw_lock); |
| return IRQ_NONE; |
| } |
| |
| ctx = v4l2_m2m_get_curr_priv(e5010->m2m_dev); |
| if (WARN_ON(!ctx)) |
| goto job_unlock; |
| |
| dst_buf = v4l2_m2m_dst_buf_remove(ctx->fh.m2m_ctx); |
| src_buf = v4l2_m2m_src_buf_remove(ctx->fh.m2m_ctx); |
| if (!dst_buf || !src_buf) { |
| v4l2_err(&e5010->v4l2_dev, "ctx: 0x%p No source or destination buffer\n", ctx); |
| goto job_unlock; |
| } |
| |
| if (out_addr_err) { |
| e5010_hw_clear_output_error(e5010->core_base, 1); |
| v4l2_warn(&e5010->v4l2_dev, |
| "ctx: 0x%p Output bitstream size exceeded max size\n", ctx); |
| v4l2_m2m_buf_done(src_buf, VB2_BUF_STATE_ERROR); |
| vb2_set_plane_payload(&dst_buf->vb2_buf, 0, dst_buf->planes[0].length); |
| v4l2_m2m_buf_done(dst_buf, VB2_BUF_STATE_ERROR); |
| if (v4l2_m2m_is_last_draining_src_buf(ctx->fh.m2m_ctx, src_buf)) { |
| dst_buf->flags |= V4L2_BUF_FLAG_LAST; |
| v4l2_m2m_mark_stopped(ctx->fh.m2m_ctx); |
| v4l2_event_queue_fh(&ctx->fh, &e5010_eos_event); |
| dprintk(e5010, 2, "ctx: 0x%p Sending EOS\n", ctx); |
| } |
| } |
| |
| if (pic_done) { |
| e5010_hw_clear_picture_done(e5010->core_base, 1); |
| dprintk(e5010, 3, "ctx: 0x%p Got output bitstream of size %d bytes\n", |
| ctx, readl(e5010->core_base + JASPER_OUTPUT_SIZE_OFFSET)); |
| |
| if (v4l2_m2m_is_last_draining_src_buf(ctx->fh.m2m_ctx, src_buf)) { |
| dst_buf->flags |= V4L2_BUF_FLAG_LAST; |
| v4l2_m2m_mark_stopped(ctx->fh.m2m_ctx); |
| v4l2_event_queue_fh(&ctx->fh, &e5010_eos_event); |
| dprintk(e5010, 2, "ctx: 0x%p Sending EOS\n", ctx); |
| } |
| v4l2_m2m_buf_done(src_buf, VB2_BUF_STATE_DONE); |
| output_size = e5010_hw_get_output_size(e5010->core_base); |
| vb2_set_plane_payload(&dst_buf->vb2_buf, 0, output_size + HEADER_SIZE); |
| v4l2_m2m_buf_done(dst_buf, VB2_BUF_STATE_DONE); |
| dprintk(e5010, 3, |
| "ctx: 0x%p frame done for dst_buf->sequence: %d src_buf->sequence: %d\n", |
| ctx, dst_buf->sequence, src_buf->sequence); |
| } |
| |
| v4l2_m2m_job_finish(e5010->m2m_dev, ctx->fh.m2m_ctx); |
| dprintk(e5010, 3, "ctx: 0x%p Finish job\n", ctx); |
| |
| job_unlock: |
| spin_unlock(&e5010->hw_lock); |
| return IRQ_HANDLED; |
| } |
| |
| static int e5010_init_device(struct e5010_dev *e5010) |
| { |
| int ret = 0; |
| |
| /*TODO: Set MMU in bypass mode until support for the same is added in driver*/ |
| e5010_hw_bypass_mmu(e5010->mmu_base, 1); |
| |
| if (e5010_hw_enable_auto_clock_gating(e5010->core_base, 1)) |
| v4l2_warn(&e5010->v4l2_dev, "failed to enable auto clock gating\n"); |
| |
| if (e5010_hw_enable_manual_clock_gating(e5010->core_base, 0)) |
| v4l2_warn(&e5010->v4l2_dev, "failed to disable manual clock gating\n"); |
| |
| if (e5010_hw_enable_crc_check(e5010->core_base, 0)) |
| v4l2_warn(&e5010->v4l2_dev, "failed to disable CRC check\n"); |
| |
| if (e5010_hw_enable_output_address_error_irq(e5010->core_base, 1)) |
| v4l2_err(&e5010->v4l2_dev, "failed to enable Output Address Error interrupts\n"); |
| |
| ret = e5010_hw_set_input_source_to_memory(e5010->core_base, 1); |
| if (ret) { |
| v4l2_err(&e5010->v4l2_dev, "failed to set input source to memory\n"); |
| return ret; |
| } |
| |
| ret = e5010_hw_enable_picture_done_irq(e5010->core_base, 1); |
| if (ret) |
| v4l2_err(&e5010->v4l2_dev, "failed to enable Picture Done interrupts\n"); |
| |
| return ret; |
| } |
| |
| static int e5010_probe(struct platform_device *pdev) |
| { |
| struct e5010_dev *e5010; |
| int irq, ret = 0; |
| struct device *dev = &pdev->dev; |
| |
| ret = dma_set_mask(dev, DMA_BIT_MASK(32)); |
| if (ret) |
| return dev_err_probe(dev, ret, "32-bit consistent DMA enable failed\n"); |
| |
| e5010 = devm_kzalloc(dev, sizeof(*e5010), GFP_KERNEL); |
| if (!e5010) |
| return -ENOMEM; |
| |
| platform_set_drvdata(pdev, e5010); |
| |
| e5010->dev = dev; |
| |
| mutex_init(&e5010->mutex); |
| spin_lock_init(&e5010->hw_lock); |
| |
| e5010->vdev = video_device_alloc(); |
| if (!e5010->vdev) { |
| dev_err(dev, "failed to allocate video device\n"); |
| return -ENOMEM; |
| } |
| |
| snprintf(e5010->vdev->name, sizeof(e5010->vdev->name), "%s", E5010_MODULE_NAME); |
| e5010->vdev->fops = &e5010_fops; |
| e5010->vdev->ioctl_ops = &e5010_ioctl_ops; |
| e5010->vdev->minor = -1; |
| e5010->vdev->release = video_device_release; |
| e5010->vdev->vfl_dir = VFL_DIR_M2M; |
| e5010->vdev->device_caps = V4L2_CAP_VIDEO_M2M_MPLANE | V4L2_CAP_STREAMING; |
| e5010->vdev->v4l2_dev = &e5010->v4l2_dev; |
| e5010->vdev->lock = &e5010->mutex; |
| |
| ret = v4l2_device_register(dev, &e5010->v4l2_dev); |
| if (ret) |
| return dev_err_probe(dev, ret, "failed to register v4l2 device\n"); |
| |
| e5010->m2m_dev = v4l2_m2m_init(&e5010_m2m_ops); |
| if (IS_ERR(e5010->m2m_dev)) { |
| ret = PTR_ERR(e5010->m2m_dev); |
| e5010->m2m_dev = NULL; |
| dev_err_probe(dev, ret, "failed to init mem2mem device\n"); |
| goto fail_after_v4l2_register; |
| } |
| |
| video_set_drvdata(e5010->vdev, e5010); |
| |
| e5010->core_base = devm_platform_ioremap_resource_byname(pdev, "core"); |
| if (IS_ERR(e5010->core_base)) { |
| ret = PTR_ERR(e5010->core_base); |
| dev_err_probe(dev, ret, "Missing 'core' resources area\n"); |
| goto fail_after_v4l2_register; |
| } |
| |
| e5010->mmu_base = devm_platform_ioremap_resource_byname(pdev, "mmu"); |
| if (IS_ERR(e5010->mmu_base)) { |
| ret = PTR_ERR(e5010->mmu_base); |
| dev_err_probe(dev, ret, "Missing 'mmu' resources area\n"); |
| goto fail_after_v4l2_register; |
| } |
| |
| e5010->last_context_run = NULL; |
| |
| irq = platform_get_irq(pdev, 0); |
| ret = devm_request_irq(dev, irq, e5010_irq, 0, |
| E5010_MODULE_NAME, e5010); |
| if (ret) { |
| dev_err_probe(dev, ret, "failed to register IRQ %d\n", irq); |
| goto fail_after_v4l2_register; |
| } |
| |
| e5010->clk = devm_clk_get(dev, NULL); |
| if (IS_ERR(e5010->clk)) { |
| ret = PTR_ERR(e5010->clk); |
| dev_err_probe(dev, ret, "failed to get clock\n"); |
| goto fail_after_v4l2_register; |
| } |
| |
| pm_runtime_enable(dev); |
| |
| ret = video_register_device(e5010->vdev, VFL_TYPE_VIDEO, 0); |
| if (ret) { |
| dev_err_probe(dev, ret, "failed to register video device\n"); |
| goto fail_after_video_register_device; |
| } |
| |
| v4l2_info(&e5010->v4l2_dev, "Device registered as /dev/video%d\n", |
| e5010->vdev->num); |
| |
| return 0; |
| |
| fail_after_video_register_device: |
| v4l2_m2m_release(e5010->m2m_dev); |
| fail_after_v4l2_register: |
| v4l2_device_unregister(&e5010->v4l2_dev); |
| return ret; |
| } |
| |
| static void e5010_remove(struct platform_device *pdev) |
| { |
| struct e5010_dev *e5010 = platform_get_drvdata(pdev); |
| |
| pm_runtime_disable(e5010->dev); |
| video_unregister_device(e5010->vdev); |
| v4l2_m2m_release(e5010->m2m_dev); |
| v4l2_device_unregister(&e5010->v4l2_dev); |
| } |
| |
| static void e5010_vb2_buffers_return(struct vb2_queue *q, enum vb2_buffer_state state) |
| { |
| struct vb2_v4l2_buffer *vbuf; |
| struct e5010_context *ctx = vb2_get_drv_priv(q); |
| |
| if (V4L2_TYPE_IS_OUTPUT(q->type)) { |
| while ((vbuf = v4l2_m2m_src_buf_remove(ctx->fh.m2m_ctx))) { |
| dprintk(ctx->e5010, 2, "ctx: 0x%p, buf type %s | index %d\n", |
| ctx, type_name(vbuf->vb2_buf.type), vbuf->vb2_buf.index); |
| v4l2_m2m_buf_done(vbuf, state); |
| } |
| } else { |
| while ((vbuf = v4l2_m2m_dst_buf_remove(ctx->fh.m2m_ctx))) { |
| dprintk(ctx->e5010, 2, "ctx: 0x%p, buf type %s | index %d\n", |
| ctx, type_name(vbuf->vb2_buf.type), vbuf->vb2_buf.index); |
| vb2_set_plane_payload(&vbuf->vb2_buf, 0, 0); |
| v4l2_m2m_buf_done(vbuf, state); |
| } |
| } |
| } |
| |
| static int e5010_queue_setup(struct vb2_queue *vq, unsigned int *nbuffers, unsigned int *nplanes, |
| unsigned int sizes[], struct device *alloc_devs[]) |
| { |
| struct e5010_context *ctx = vb2_get_drv_priv(vq); |
| struct e5010_q_data *queue; |
| int i; |
| |
| queue = get_queue(ctx, vq->type); |
| |
| if (*nplanes) { |
| if (*nplanes != queue->fmt->num_planes) |
| return -EINVAL; |
| for (i = 0; i < *nplanes; i++) { |
| if (sizes[i] < queue->sizeimage[i]) |
| return -EINVAL; |
| } |
| return 0; |
| } |
| |
| *nplanes = queue->fmt->num_planes; |
| for (i = 0; i < *nplanes; i++) |
| sizes[i] = queue->sizeimage[i]; |
| |
| dprintk(ctx->e5010, 2, |
| "ctx: 0x%p, type %s, buffer(s): %d, planes %d, plane1: bytes %d plane2: %d bytes\n", |
| ctx, type_name(vq->type), *nbuffers, *nplanes, sizes[0], sizes[1]); |
| |
| return 0; |
| } |
| |
| static void e5010_buf_finish(struct vb2_buffer *vb) |
| { |
| struct e5010_context *ctx = vb2_get_drv_priv(vb->vb2_queue); |
| void *d_addr; |
| |
| if (vb->state != VB2_BUF_STATE_DONE || V4L2_TYPE_IS_OUTPUT(vb->vb2_queue->type)) |
| return; |
| |
| d_addr = vb2_plane_vaddr(vb, 0); |
| write_header(ctx, d_addr); |
| } |
| |
| static int e5010_buf_out_validate(struct vb2_buffer *vb) |
| { |
| struct vb2_v4l2_buffer *vbuf = to_vb2_v4l2_buffer(vb); |
| struct e5010_context *ctx = vb2_get_drv_priv(vb->vb2_queue); |
| |
| if (vbuf->field != V4L2_FIELD_NONE) |
| dprintk(ctx->e5010, 1, "ctx: 0x%p, field isn't supported\n", ctx); |
| |
| vbuf->field = V4L2_FIELD_NONE; |
| |
| return 0; |
| } |
| |
| static int e5010_buf_prepare(struct vb2_buffer *vb) |
| { |
| struct e5010_context *ctx = vb2_get_drv_priv(vb->vb2_queue); |
| struct vb2_v4l2_buffer *vbuf = to_vb2_v4l2_buffer(vb); |
| struct e5010_q_data *queue; |
| int i; |
| |
| vbuf->field = V4L2_FIELD_NONE; |
| |
| queue = get_queue(ctx, vb->vb2_queue->type); |
| |
| for (i = 0; i < queue->fmt->num_planes; i++) { |
| if (vb2_plane_size(vb, i) < (unsigned long)queue->sizeimage[i]) { |
| v4l2_err(&ctx->e5010->v4l2_dev, "plane %d too small (%lu < %lu)", i, |
| vb2_plane_size(vb, i), (unsigned long)queue->sizeimage[i]); |
| |
| return -EINVAL; |
| } |
| } |
| |
| if (V4L2_TYPE_IS_CAPTURE(vb->vb2_queue->type)) { |
| vb2_set_plane_payload(vb, 0, 0); |
| vb2_set_plane_payload(vb, 1, 0); |
| } |
| |
| return 0; |
| } |
| |
| static void e5010_buf_queue(struct vb2_buffer *vb) |
| { |
| struct e5010_context *ctx = vb2_get_drv_priv(vb->vb2_queue); |
| struct vb2_v4l2_buffer *vbuf = to_vb2_v4l2_buffer(vb); |
| |
| if (V4L2_TYPE_IS_CAPTURE(vb->vb2_queue->type) && |
| vb2_is_streaming(vb->vb2_queue) && |
| v4l2_m2m_dst_buf_is_last(ctx->fh.m2m_ctx)) { |
| struct e5010_q_data *queue = get_queue(ctx, V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE); |
| |
| vbuf->sequence = queue->sequence++; |
| v4l2_m2m_last_buffer_done(ctx->fh.m2m_ctx, vbuf); |
| v4l2_event_queue_fh(&ctx->fh, &e5010_eos_event); |
| return; |
| } |
| |
| v4l2_m2m_buf_queue(ctx->fh.m2m_ctx, vbuf); |
| } |
| |
| static int e5010_encoder_cmd(struct file *file, void *priv, |
| struct v4l2_encoder_cmd *cmd) |
| { |
| struct e5010_context *ctx = file->private_data; |
| int ret; |
| struct vb2_queue *cap_vq; |
| |
| cap_vq = v4l2_m2m_get_vq(ctx->fh.m2m_ctx, V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE); |
| |
| ret = v4l2_m2m_ioctl_try_encoder_cmd(file, &ctx->fh, cmd); |
| if (ret < 0) |
| return ret; |
| |
| if (!vb2_is_streaming(v4l2_m2m_get_src_vq(ctx->fh.m2m_ctx)) || |
| !vb2_is_streaming(v4l2_m2m_get_dst_vq(ctx->fh.m2m_ctx))) |
| return 0; |
| |
| ret = v4l2_m2m_ioctl_encoder_cmd(file, &ctx->fh, cmd); |
| if (ret < 0) |
| return ret; |
| |
| if (cmd->cmd == V4L2_ENC_CMD_STOP && |
| v4l2_m2m_has_stopped(ctx->fh.m2m_ctx)) |
| v4l2_event_queue_fh(&ctx->fh, &e5010_eos_event); |
| |
| if (cmd->cmd == V4L2_ENC_CMD_START && |
| v4l2_m2m_has_stopped(ctx->fh.m2m_ctx)) |
| vb2_clear_last_buffer_dequeued(cap_vq); |
| |
| return 0; |
| } |
| |
| static int e5010_start_streaming(struct vb2_queue *q, unsigned int count) |
| { |
| struct e5010_context *ctx = vb2_get_drv_priv(q); |
| int ret; |
| |
| struct e5010_q_data *queue = get_queue(ctx, q->type); |
| |
| v4l2_m2m_update_start_streaming_state(ctx->fh.m2m_ctx, q); |
| queue->sequence = 0; |
| |
| ret = pm_runtime_resume_and_get(ctx->e5010->dev); |
| if (ret < 0) { |
| v4l2_err(&ctx->e5010->v4l2_dev, "failed to power up jpeg\n"); |
| goto fail; |
| } |
| |
| ret = e5010_init_device(ctx->e5010); |
| if (ret) { |
| v4l2_err(&ctx->e5010->v4l2_dev, "failed to Enable e5010 device\n"); |
| goto fail; |
| } |
| |
| return 0; |
| |
| fail: |
| e5010_vb2_buffers_return(q, VB2_BUF_STATE_QUEUED); |
| |
| return ret; |
| } |
| |
| static void e5010_stop_streaming(struct vb2_queue *q) |
| { |
| struct e5010_context *ctx = vb2_get_drv_priv(q); |
| |
| e5010_vb2_buffers_return(q, VB2_BUF_STATE_ERROR); |
| |
| if (V4L2_TYPE_IS_OUTPUT(q->type)) |
| v4l2_m2m_update_stop_streaming_state(ctx->fh.m2m_ctx, q); |
| |
| if (V4L2_TYPE_IS_OUTPUT(q->type) && |
| v4l2_m2m_has_stopped(ctx->fh.m2m_ctx)) { |
| v4l2_event_queue_fh(&ctx->fh, &e5010_eos_event); |
| } |
| |
| pm_runtime_put_sync(ctx->e5010->dev); |
| } |
| |
| static void e5010_device_run(void *priv) |
| { |
| struct e5010_context *ctx = priv; |
| struct e5010_dev *e5010 = ctx->e5010; |
| struct vb2_v4l2_buffer *s_vb, *d_vb; |
| u32 reg = 0; |
| int ret = 0, luma_crop_offset = 0, chroma_crop_offset = 0; |
| unsigned long flags; |
| int num_planes = ctx->out_queue.fmt->num_planes; |
| |
| spin_lock_irqsave(&e5010->hw_lock, flags); |
| s_vb = v4l2_m2m_next_src_buf(ctx->fh.m2m_ctx); |
| WARN_ON(!s_vb); |
| d_vb = v4l2_m2m_next_dst_buf(ctx->fh.m2m_ctx); |
| WARN_ON(!d_vb); |
| if (!s_vb || !d_vb) |
| goto no_ready_buf_err; |
| |
| s_vb->sequence = ctx->out_queue.sequence++; |
| d_vb->sequence = ctx->cap_queue.sequence++; |
| |
| v4l2_m2m_buf_copy_metadata(s_vb, d_vb, false); |
| |
| if (ctx != e5010->last_context_run || ctx->update_qp) { |
| dprintk(e5010, 1, "ctx updated: 0x%p -> 0x%p, updating qp tables\n", |
| e5010->last_context_run, ctx); |
| ret = update_qp_tables(ctx); |
| } |
| |
| if (ret) { |
| ctx->update_qp = true; |
| v4l2_err(&e5010->v4l2_dev, "failed to update QP tables\n"); |
| goto device_busy_err; |
| } else { |
| e5010->last_context_run = ctx; |
| ctx->update_qp = false; |
| } |
| |
| /* Set I/O Buffer addresses */ |
| reg = (u32)vb2_dma_contig_plane_dma_addr(&s_vb->vb2_buf, 0); |
| |
| if (ctx->out_queue.crop_set) { |
| luma_crop_offset = ctx->out_queue.bytesperline[0] * ctx->out_queue.crop.top + |
| ctx->out_queue.crop.left; |
| |
| if (ctx->out_queue.fmt->subsampling == V4L2_JPEG_CHROMA_SUBSAMPLING_422) { |
| chroma_crop_offset = |
| ctx->out_queue.bytesperline[0] * ctx->out_queue.crop.top |
| + ctx->out_queue.crop.left; |
| } else { |
| chroma_crop_offset = |
| ctx->out_queue.bytesperline[0] * ctx->out_queue.crop.top / 2 |
| + ctx->out_queue.crop.left; |
| } |
| |
| dprintk(e5010, 1, "Luma crop offset : %x, chroma crop offset : %x\n", |
| luma_crop_offset, chroma_crop_offset); |
| } |
| |
| ret = e5010_hw_set_input_luma_addr(e5010->core_base, reg + luma_crop_offset); |
| if (ret || !reg) { |
| v4l2_err(&e5010->v4l2_dev, "failed to set input luma address\n"); |
| goto device_busy_err; |
| } |
| |
| if (num_planes == 1) |
| reg += (ctx->out_queue.bytesperline[0]) * (ctx->out_queue.height); |
| else |
| reg = (u32)vb2_dma_contig_plane_dma_addr(&s_vb->vb2_buf, 1); |
| |
| dprintk(e5010, 3, |
| "ctx: 0x%p, luma_addr: 0x%x, chroma_addr: 0x%x, out_addr: 0x%x\n", |
| ctx, (u32)vb2_dma_contig_plane_dma_addr(&s_vb->vb2_buf, 0) + luma_crop_offset, |
| reg + chroma_crop_offset, (u32)vb2_dma_contig_plane_dma_addr(&d_vb->vb2_buf, 0)); |
| |
| dprintk(e5010, 3, |
| "ctx: 0x%p, buf indices: src_index: %d, dst_index: %d\n", |
| ctx, s_vb->vb2_buf.index, d_vb->vb2_buf.index); |
| |
| ret = e5010_hw_set_input_chroma_addr(e5010->core_base, reg + chroma_crop_offset); |
| if (ret || !reg) { |
| v4l2_err(&e5010->v4l2_dev, "failed to set input chroma address\n"); |
| goto device_busy_err; |
| } |
| |
| reg = (u32)vb2_dma_contig_plane_dma_addr(&d_vb->vb2_buf, 0); |
| reg += HEADER_SIZE; |
| ret = e5010_hw_set_output_base_addr(e5010->core_base, reg); |
| if (ret || !reg) { |
| v4l2_err(&e5010->v4l2_dev, "failed to set output base address\n"); |
| goto device_busy_err; |
| } |
| |
| /* Set input settings */ |
| ret = e5010_hw_set_horizontal_size(e5010->core_base, ctx->out_queue.crop.width - 1); |
| if (ret) { |
| v4l2_err(&e5010->v4l2_dev, "failed to set input width\n"); |
| goto device_busy_err; |
| } |
| |
| ret = e5010_hw_set_vertical_size(e5010->core_base, ctx->out_queue.crop.height - 1); |
| if (ret) { |
| v4l2_err(&e5010->v4l2_dev, "failed to set input width\n"); |
| goto device_busy_err; |
| } |
| |
| ret = e5010_hw_set_luma_stride(e5010->core_base, ctx->out_queue.bytesperline[0]); |
| if (ret) { |
| v4l2_err(&e5010->v4l2_dev, "failed to set luma stride\n"); |
| goto device_busy_err; |
| } |
| |
| ret = e5010_hw_set_chroma_stride(e5010->core_base, ctx->out_queue.bytesperline[0]); |
| if (ret) { |
| v4l2_err(&e5010->v4l2_dev, "failed to set chroma stride\n"); |
| goto device_busy_err; |
| } |
| |
| ret = e5010_set_input_subsampling(e5010->core_base, ctx->out_queue.fmt->subsampling); |
| if (ret) { |
| v4l2_err(&e5010->v4l2_dev, "failed to set input subsampling\n"); |
| goto device_busy_err; |
| } |
| |
| ret = e5010_hw_set_chroma_order(e5010->core_base, ctx->out_queue.fmt->chroma_order); |
| if (ret) { |
| v4l2_err(&e5010->v4l2_dev, "failed to set chroma order\n"); |
| goto device_busy_err; |
| } |
| |
| e5010_hw_set_output_max_size(e5010->core_base, d_vb->planes[0].length); |
| e5010_hw_encode_start(e5010->core_base, 1); |
| |
| spin_unlock_irqrestore(&e5010->hw_lock, flags); |
| |
| return; |
| |
| device_busy_err: |
| e5010_reset(e5010->dev, e5010->core_base, e5010->mmu_base); |
| |
| no_ready_buf_err: |
| if (s_vb) { |
| v4l2_m2m_src_buf_remove_by_buf(ctx->fh.m2m_ctx, s_vb); |
| v4l2_m2m_buf_done(s_vb, VB2_BUF_STATE_ERROR); |
| } |
| |
| if (d_vb) { |
| v4l2_m2m_dst_buf_remove_by_buf(ctx->fh.m2m_ctx, d_vb); |
| /* Payload set to 1 since 0 payload can trigger EOS */ |
| vb2_set_plane_payload(&d_vb->vb2_buf, 0, 1); |
| v4l2_m2m_buf_done(d_vb, VB2_BUF_STATE_ERROR); |
| } |
| v4l2_m2m_job_finish(e5010->m2m_dev, ctx->fh.m2m_ctx); |
| spin_unlock_irqrestore(&e5010->hw_lock, flags); |
| } |
| |
| #ifdef CONFIG_PM |
| static int e5010_runtime_resume(struct device *dev) |
| { |
| struct e5010_dev *e5010 = dev_get_drvdata(dev); |
| int ret; |
| |
| ret = clk_prepare_enable(e5010->clk); |
| if (ret < 0) { |
| v4l2_err(&e5010->v4l2_dev, "failed to enable clock\n"); |
| return ret; |
| } |
| |
| return 0; |
| } |
| |
| static int e5010_runtime_suspend(struct device *dev) |
| { |
| struct e5010_dev *e5010 = dev_get_drvdata(dev); |
| |
| clk_disable_unprepare(e5010->clk); |
| |
| return 0; |
| } |
| #endif |
| |
| #ifdef CONFIG_PM_SLEEP |
| static int e5010_suspend(struct device *dev) |
| { |
| struct e5010_dev *e5010 = dev_get_drvdata(dev); |
| |
| v4l2_m2m_suspend(e5010->m2m_dev); |
| |
| return pm_runtime_force_suspend(dev); |
| } |
| |
| static int e5010_resume(struct device *dev) |
| { |
| struct e5010_dev *e5010 = dev_get_drvdata(dev); |
| int ret; |
| |
| ret = pm_runtime_force_resume(dev); |
| if (ret < 0) |
| return ret; |
| |
| ret = e5010_init_device(e5010); |
| if (ret) { |
| dev_err(dev, "Failed to re-enable e5010 device\n"); |
| return ret; |
| } |
| |
| v4l2_m2m_resume(e5010->m2m_dev); |
| |
| return ret; |
| } |
| #endif |
| |
| static const struct dev_pm_ops e5010_pm_ops = { |
| SET_RUNTIME_PM_OPS(e5010_runtime_suspend, |
| e5010_runtime_resume, NULL) |
| SET_SYSTEM_SLEEP_PM_OPS(e5010_suspend, e5010_resume) |
| }; |
| |
| static const struct v4l2_ioctl_ops e5010_ioctl_ops = { |
| .vidioc_querycap = e5010_querycap, |
| |
| .vidioc_enum_fmt_vid_cap = e5010_enum_fmt, |
| .vidioc_g_fmt_vid_cap_mplane = e5010_g_fmt, |
| .vidioc_try_fmt_vid_cap_mplane = e5010_try_fmt, |
| .vidioc_s_fmt_vid_cap_mplane = e5010_s_fmt, |
| |
| .vidioc_enum_fmt_vid_out = e5010_enum_fmt, |
| .vidioc_g_fmt_vid_out_mplane = e5010_g_fmt, |
| .vidioc_try_fmt_vid_out_mplane = e5010_try_fmt, |
| .vidioc_s_fmt_vid_out_mplane = e5010_s_fmt, |
| |
| .vidioc_g_selection = e5010_g_selection, |
| .vidioc_s_selection = e5010_s_selection, |
| |
| .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_expbuf = v4l2_m2m_ioctl_expbuf, |
| .vidioc_create_bufs = v4l2_m2m_ioctl_create_bufs, |
| .vidioc_prepare_buf = v4l2_m2m_ioctl_prepare_buf, |
| |
| .vidioc_streamon = v4l2_m2m_ioctl_streamon, |
| .vidioc_streamoff = v4l2_m2m_ioctl_streamoff, |
| .vidioc_log_status = v4l2_ctrl_log_status, |
| |
| .vidioc_subscribe_event = e5010_subscribe_event, |
| .vidioc_unsubscribe_event = v4l2_event_unsubscribe, |
| .vidioc_try_encoder_cmd = v4l2_m2m_ioctl_try_encoder_cmd, |
| .vidioc_encoder_cmd = e5010_encoder_cmd, |
| |
| .vidioc_enum_framesizes = e5010_enum_framesizes, |
| }; |
| |
| static const struct vb2_ops e5010_video_ops = { |
| .queue_setup = e5010_queue_setup, |
| .buf_queue = e5010_buf_queue, |
| .buf_finish = e5010_buf_finish, |
| .buf_prepare = e5010_buf_prepare, |
| .buf_out_validate = e5010_buf_out_validate, |
| .wait_prepare = vb2_ops_wait_prepare, |
| .wait_finish = vb2_ops_wait_finish, |
| .start_streaming = e5010_start_streaming, |
| .stop_streaming = e5010_stop_streaming, |
| }; |
| |
| static const struct v4l2_file_operations e5010_fops = { |
| .owner = THIS_MODULE, |
| .open = e5010_open, |
| .release = e5010_release, |
| .poll = v4l2_m2m_fop_poll, |
| .unlocked_ioctl = video_ioctl2, |
| .mmap = v4l2_m2m_fop_mmap, |
| }; |
| |
| static const struct v4l2_m2m_ops e5010_m2m_ops = { |
| .device_run = e5010_device_run, |
| }; |
| |
| static const struct of_device_id e5010_of_match[] = { |
| {.compatible = "img,e5010-jpeg-enc"}, { /* end */}, |
| }; |
| MODULE_DEVICE_TABLE(of, e5010_of_match); |
| |
| static struct platform_driver e5010_driver = { |
| .probe = e5010_probe, |
| .remove_new = e5010_remove, |
| .driver = { |
| .name = E5010_MODULE_NAME, |
| .of_match_table = e5010_of_match, |
| .pm = &e5010_pm_ops, |
| }, |
| }; |
| module_platform_driver(e5010_driver); |
| |
| MODULE_LICENSE("GPL"); |
| MODULE_DESCRIPTION("Imagination E5010 JPEG encoder driver"); |