Commit cf52df8a authored by Sylwester Nawrocki's avatar Sylwester Nawrocki Committed by Mauro Carvalho Chehab

[media] s5p-fimc: Correct color format enumeration

Replace fimc_find_format() and find_mbus_format() with single function
that can return a pointer to the private format description based
on fourcc, media bus code or index in the table.

Create separate VIDIOC_ENUM_FMT ioctl handlers for video capture
and mem-to-mem video node. This is needed because some formats are
valid only for the video capture and some only for the mem-to-mem
video node.
Signed-off-by: default avatarSylwester Nawrocki <s.nawrocki@samsung.com>
Signed-off-by: default avatarKyungmin Park <kyungmin.park@samsung.com>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab@redhat.com>
parent 9e803a04
...@@ -389,6 +389,22 @@ static int fimc_vidioc_querycap_capture(struct file *file, void *priv, ...@@ -389,6 +389,22 @@ static int fimc_vidioc_querycap_capture(struct file *file, void *priv,
return 0; return 0;
} }
static int fimc_cap_enum_fmt_mplane(struct file *file, void *priv,
struct v4l2_fmtdesc *f)
{
struct fimc_fmt *fmt;
fmt = fimc_find_format(NULL, NULL, FMT_FLAGS_CAM | FMT_FLAGS_M2M,
f->index);
if (!fmt)
return -EINVAL;
strncpy(f->description, fmt->name, sizeof(f->description) - 1);
f->pixelformat = fmt->fourcc;
if (fmt->fourcc == V4L2_MBUS_FMT_JPEG_1X8)
f->flags |= V4L2_FMT_FLAG_COMPRESSED;
return 0;
}
/* Synchronize formats of the camera interface input and attached sensor. */ /* Synchronize formats of the camera interface input and attached sensor. */
static int sync_capture_fmt(struct fimc_ctx *ctx) static int sync_capture_fmt(struct fimc_ctx *ctx)
{ {
...@@ -407,7 +423,7 @@ static int sync_capture_fmt(struct fimc_ctx *ctx) ...@@ -407,7 +423,7 @@ static int sync_capture_fmt(struct fimc_ctx *ctx)
} }
dbg("w: %d, h: %d, code= %d", fmt->width, fmt->height, fmt->code); dbg("w: %d, h: %d, code= %d", fmt->width, fmt->height, fmt->code);
frame->fmt = find_mbus_format(fmt, FMT_FLAGS_CAM); frame->fmt = fimc_find_format(NULL, &fmt->code, FMT_FLAGS_CAM, -1);
if (!frame->fmt) { if (!frame->fmt) {
err("fimc source format not found\n"); err("fimc source format not found\n");
return -EINVAL; return -EINVAL;
...@@ -469,12 +485,10 @@ static int fimc_cap_s_fmt_mplane(struct file *file, void *priv, ...@@ -469,12 +485,10 @@ static int fimc_cap_s_fmt_mplane(struct file *file, void *priv,
frame = &ctx->d_frame; frame = &ctx->d_frame;
pix = &f->fmt.pix_mp; pix = &f->fmt.pix_mp;
frame->fmt = find_format(f, FMT_FLAGS_M2M | FMT_FLAGS_CAM); frame->fmt = fimc_find_format(&pix->pixelformat, NULL,
if (!frame->fmt) { FMT_FLAGS_M2M | FMT_FLAGS_CAM, 0);
v4l2_err(fimc->vid_cap.vfd, if (WARN(frame->fmt == NULL, "Pixel format lookup failed\n"))
"Not supported capture (FIMC target) color format\n");
return -EINVAL; return -EINVAL;
}
for (i = 0; i < frame->fmt->colplanes; i++) { for (i = 0; i < frame->fmt->colplanes; i++) {
frame->payload[i] = frame->payload[i] =
...@@ -654,7 +668,7 @@ static int fimc_cap_s_crop(struct file *file, void *fh, struct v4l2_crop *cr) ...@@ -654,7 +668,7 @@ static int fimc_cap_s_crop(struct file *file, void *fh, struct v4l2_crop *cr)
static const struct v4l2_ioctl_ops fimc_capture_ioctl_ops = { static const struct v4l2_ioctl_ops fimc_capture_ioctl_ops = {
.vidioc_querycap = fimc_vidioc_querycap_capture, .vidioc_querycap = fimc_vidioc_querycap_capture,
.vidioc_enum_fmt_vid_cap_mplane = fimc_vidioc_enum_fmt_mplane, .vidioc_enum_fmt_vid_cap_mplane = fimc_cap_enum_fmt_mplane,
.vidioc_try_fmt_vid_cap_mplane = fimc_cap_try_fmt_mplane, .vidioc_try_fmt_vid_cap_mplane = fimc_cap_try_fmt_mplane,
.vidioc_s_fmt_vid_cap_mplane = fimc_cap_s_fmt_mplane, .vidioc_s_fmt_vid_cap_mplane = fimc_cap_s_fmt_mplane,
.vidioc_g_fmt_vid_cap_mplane = fimc_cap_g_fmt_mplane, .vidioc_g_fmt_vid_cap_mplane = fimc_cap_g_fmt_mplane,
...@@ -715,7 +729,6 @@ int fimc_register_capture_device(struct fimc_dev *fimc, ...@@ -715,7 +729,6 @@ int fimc_register_capture_device(struct fimc_dev *fimc,
struct video_device *vfd; struct video_device *vfd;
struct fimc_vid_cap *vid_cap; struct fimc_vid_cap *vid_cap;
struct fimc_ctx *ctx; struct fimc_ctx *ctx;
struct v4l2_format f;
struct fimc_frame *fr; struct fimc_frame *fr;
struct vb2_queue *q; struct vb2_queue *q;
int ret = -ENOMEM; int ret = -ENOMEM;
...@@ -730,9 +743,8 @@ int fimc_register_capture_device(struct fimc_dev *fimc, ...@@ -730,9 +743,8 @@ int fimc_register_capture_device(struct fimc_dev *fimc,
ctx->state = FIMC_CTX_CAP; ctx->state = FIMC_CTX_CAP;
/* Default format of the output frames */ /* Default format of the output frames */
f.fmt.pix.pixelformat = V4L2_PIX_FMT_RGB32;
fr = &ctx->d_frame; fr = &ctx->d_frame;
fr->fmt = find_format(&f, FMT_FLAGS_M2M); fr->fmt = fimc_find_format(NULL, NULL, FMT_FLAGS_CAM, 0);
fr->width = fr->f_width = fr->o_width = 640; fr->width = fr->f_width = fr->o_width = 640;
fr->height = fr->f_height = fr->o_height = 480; fr->height = fr->f_height = fr->o_height = 480;
......
...@@ -866,18 +866,17 @@ static int fimc_m2m_querycap(struct file *file, void *fh, ...@@ -866,18 +866,17 @@ static int fimc_m2m_querycap(struct file *file, void *fh,
return 0; return 0;
} }
int fimc_vidioc_enum_fmt_mplane(struct file *file, void *priv, static int fimc_m2m_enum_fmt_mplane(struct file *file, void *priv,
struct v4l2_fmtdesc *f) struct v4l2_fmtdesc *f)
{ {
struct fimc_fmt *fmt; struct fimc_fmt *fmt;
if (f->index >= ARRAY_SIZE(fimc_formats)) fmt = fimc_find_format(NULL, NULL, FMT_FLAGS_M2M, f->index);
if (!fmt)
return -EINVAL; return -EINVAL;
fmt = &fimc_formats[f->index];
strncpy(f->description, fmt->name, sizeof(f->description) - 1); strncpy(f->description, fmt->name, sizeof(f->description) - 1);
f->pixelformat = fmt->fourcc; f->pixelformat = fmt->fourcc;
return 0; return 0;
} }
...@@ -916,34 +915,36 @@ static int fimc_m2m_g_fmt_mplane(struct file *file, void *fh, ...@@ -916,34 +915,36 @@ static int fimc_m2m_g_fmt_mplane(struct file *file, void *fh,
return fimc_fill_format(frame, f); return fimc_fill_format(frame, f);
} }
struct fimc_fmt *find_format(struct v4l2_format *f, unsigned int mask) /**
* fimc_find_format - lookup fimc color format by fourcc or media bus format
* @pixelformat: fourcc to match, ignored if null
* @mbus_code: media bus code to match, ignored if null
* @mask: the color flags to match
* @index: offset in the fimc_formats array, ignored if negative
*/
struct fimc_fmt *fimc_find_format(u32 *pixelformat, u32 *mbus_code,
unsigned int mask, int index)
{ {
struct fimc_fmt *fmt; struct fimc_fmt *fmt, *def_fmt = NULL;
unsigned int i; unsigned int i;
int id = 0;
for (i = 0; i < ARRAY_SIZE(fimc_formats); ++i) { if (index >= ARRAY_SIZE(fimc_formats))
fmt = &fimc_formats[i]; return NULL;
if (fmt->fourcc == f->fmt.pix_mp.pixelformat &&
(fmt->flags & mask))
break;
}
return (i == ARRAY_SIZE(fimc_formats)) ? NULL : fmt;
}
struct fimc_fmt *find_mbus_format(struct v4l2_mbus_framefmt *f,
unsigned int mask)
{
struct fimc_fmt *fmt;
unsigned int i;
for (i = 0; i < ARRAY_SIZE(fimc_formats); ++i) { for (i = 0; i < ARRAY_SIZE(fimc_formats); ++i) {
fmt = &fimc_formats[i]; fmt = &fimc_formats[i];
if (fmt->mbus_code == f->code && (fmt->flags & mask)) if (!(fmt->flags & mask))
break; continue;
if (pixelformat && fmt->fourcc == *pixelformat)
return fmt;
if (mbus_code && fmt->mbus_code == *mbus_code)
return fmt;
if (index == id)
def_fmt = fmt;
id++;
} }
return def_fmt;
return (i == ARRAY_SIZE(fimc_formats)) ? NULL : fmt;
} }
int fimc_try_fmt_mplane(struct fimc_ctx *ctx, struct v4l2_format *f) int fimc_try_fmt_mplane(struct fimc_ctx *ctx, struct v4l2_format *f)
...@@ -966,7 +967,7 @@ int fimc_try_fmt_mplane(struct fimc_ctx *ctx, struct v4l2_format *f) ...@@ -966,7 +967,7 @@ int fimc_try_fmt_mplane(struct fimc_ctx *ctx, struct v4l2_format *f)
dbg("w: %d, h: %d", pix->width, pix->height); dbg("w: %d, h: %d", pix->width, pix->height);
mask = is_output ? FMT_FLAGS_M2M : FMT_FLAGS_M2M | FMT_FLAGS_CAM; mask = is_output ? FMT_FLAGS_M2M : FMT_FLAGS_M2M | FMT_FLAGS_CAM;
fmt = find_format(f, mask); fmt = fimc_find_format(&pix->pixelformat, NULL, mask, -1);
if (!fmt) { if (!fmt) {
v4l2_err(fimc->v4l2_dev, "Fourcc format (0x%X) invalid.\n", v4l2_err(fimc->v4l2_dev, "Fourcc format (0x%X) invalid.\n",
pix->pixelformat); pix->pixelformat);
...@@ -1061,7 +1062,8 @@ static int fimc_m2m_s_fmt_mplane(struct file *file, void *fh, ...@@ -1061,7 +1062,8 @@ static int fimc_m2m_s_fmt_mplane(struct file *file, void *fh,
frame = &ctx->d_frame; frame = &ctx->d_frame;
pix = &f->fmt.pix_mp; pix = &f->fmt.pix_mp;
frame->fmt = find_format(f, FMT_FLAGS_M2M); frame->fmt = fimc_find_format(&pix->pixelformat, NULL,
FMT_FLAGS_M2M, 0);
if (!frame->fmt) if (!frame->fmt)
return -EINVAL; return -EINVAL;
...@@ -1290,8 +1292,8 @@ static int fimc_m2m_s_crop(struct file *file, void *fh, struct v4l2_crop *cr) ...@@ -1290,8 +1292,8 @@ static int fimc_m2m_s_crop(struct file *file, void *fh, struct v4l2_crop *cr)
static const struct v4l2_ioctl_ops fimc_m2m_ioctl_ops = { static const struct v4l2_ioctl_ops fimc_m2m_ioctl_ops = {
.vidioc_querycap = fimc_m2m_querycap, .vidioc_querycap = fimc_m2m_querycap,
.vidioc_enum_fmt_vid_cap_mplane = fimc_vidioc_enum_fmt_mplane, .vidioc_enum_fmt_vid_cap_mplane = fimc_m2m_enum_fmt_mplane,
.vidioc_enum_fmt_vid_out_mplane = fimc_vidioc_enum_fmt_mplane, .vidioc_enum_fmt_vid_out_mplane = fimc_m2m_enum_fmt_mplane,
.vidioc_g_fmt_vid_cap_mplane = fimc_m2m_g_fmt_mplane, .vidioc_g_fmt_vid_cap_mplane = fimc_m2m_g_fmt_mplane,
.vidioc_g_fmt_vid_out_mplane = fimc_m2m_g_fmt_mplane, .vidioc_g_fmt_vid_out_mplane = fimc_m2m_g_fmt_mplane,
......
...@@ -654,9 +654,8 @@ void fimc_ctrls_delete(struct fimc_ctx *ctx); ...@@ -654,9 +654,8 @@ void fimc_ctrls_delete(struct fimc_ctx *ctx);
void fimc_ctrls_activate(struct fimc_ctx *ctx, bool active); void fimc_ctrls_activate(struct fimc_ctx *ctx, bool active);
int fimc_fill_format(struct fimc_frame *frame, struct v4l2_format *f); int fimc_fill_format(struct fimc_frame *frame, struct v4l2_format *f);
struct fimc_fmt *find_format(struct v4l2_format *f, unsigned int mask); struct fimc_fmt *fimc_find_format(u32 *pixelformat, u32 *mbus_code,
struct fimc_fmt *find_mbus_format(struct v4l2_mbus_framefmt *f, unsigned int mask, int index);
unsigned int mask);
int fimc_check_scaler_ratio(int sw, int sh, int dw, int dh, int rot); int fimc_check_scaler_ratio(int sw, int sh, int dw, int dh, int rot);
int fimc_set_scaler_info(struct fimc_ctx *ctx); int fimc_set_scaler_info(struct fimc_ctx *ctx);
......
...@@ -194,7 +194,7 @@ int fimc_pipeline_s_stream(struct fimc_dev *fimc, int on) ...@@ -194,7 +194,7 @@ int fimc_pipeline_s_stream(struct fimc_dev *fimc, int on)
if ((on && p->csis) || !on) if ((on && p->csis) || !on)
ret = v4l2_subdev_call(on ? p->csis : p->sensor, ret = v4l2_subdev_call(on ? p->csis : p->sensor,
video, s_stream, on); video, s_stream, on);
if (ret && ret != -ENOIOCTLCMD) if (ret < 0 && ret != -ENOIOCTLCMD)
return ret; return ret;
if ((!on && p->csis) || on) if ((!on && p->csis) || on)
ret = v4l2_subdev_call(on ? p->sensor : p->csis, ret = v4l2_subdev_call(on ? p->sensor : p->csis,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment