On 04/08/2017 12:37 AM, Helen Koike wrote:
> Allow user space to change the image format as the frame size, the
> media bus pixel format, colorspace, quantization, field YCbCr encoding
> and the transfer function
> 
> Signed-off-by: Helen Koike <helen.ko...@collabora.com>
> 
> ---
> 
> Changes in v2:
> [media] vimc: sen: Support several image formats
>       - this is a new commit in the serie (the old one was splitted in two)
>       - add init_cfg to initialize try_fmt
>       - reorder code in vimc_sen_set_fmt
>       - allow user space to change all fields from struct v4l2_mbus_framefmt
>         (e.g. colospace, quantization, field, xfer_func, ycbcr_enc)
>       - merge with patch for the enum_mbus_code and enum_frame_size
>       - change commit message
>       - add vimc_pix_map_by_index
>       - rename MIN/MAX macros
>       - check set_fmt default parameters for quantization, colorspace ...
> 
> 
> ---
>  drivers/media/platform/vimc/vimc-core.c   |   8 ++
>  drivers/media/platform/vimc/vimc-core.h   |  12 +++
>  drivers/media/platform/vimc/vimc-sensor.c | 143 
> ++++++++++++++++++++++++------
>  3 files changed, 134 insertions(+), 29 deletions(-)
> 
> diff --git a/drivers/media/platform/vimc/vimc-core.c 
> b/drivers/media/platform/vimc/vimc-core.c
> index 7c23735..bc4b1bb 100644
> --- a/drivers/media/platform/vimc/vimc-core.c
> +++ b/drivers/media/platform/vimc/vimc-core.c
> @@ -324,6 +324,14 @@ static const struct vimc_pix_map vimc_pix_map_list[] = {
>       },
>  };
>  
> +const struct vimc_pix_map *vimc_pix_map_by_index(unsigned int i)
> +{
> +     if (i >= ARRAY_SIZE(vimc_pix_map_list))
> +             return NULL;
> +
> +     return &vimc_pix_map_list[i];
> +}
> +
>  const struct vimc_pix_map *vimc_pix_map_by_code(u32 code)
>  {
>       unsigned int i;
> diff --git a/drivers/media/platform/vimc/vimc-core.h 
> b/drivers/media/platform/vimc/vimc-core.h
> index 8c3d401..2146672 100644
> --- a/drivers/media/platform/vimc/vimc-core.h
> +++ b/drivers/media/platform/vimc/vimc-core.h
> @@ -21,6 +21,11 @@
>  #include <linux/slab.h>
>  #include <media/v4l2-device.h>
>  
> +#define VIMC_FRAME_MAX_WIDTH 4096
> +#define VIMC_FRAME_MAX_HEIGHT 2160
> +#define VIMC_FRAME_MIN_WIDTH 16
> +#define VIMC_FRAME_MIN_HEIGHT 16
> +
>  /**
>   * struct vimc_pix_map - maps media bus code with v4l2 pixel format
>   *
> @@ -107,6 +112,13 @@ static inline void vimc_pads_cleanup(struct media_pad 
> *pads)
>  int vimc_pipeline_s_stream(struct media_entity *ent, int enable);
>  
>  /**
> + * vimc_pix_map_by_index - get vimc_pix_map struct by its index
> + *
> + * @i:                       index of the vimc_pix_map struct in 
> vimc_pix_map_list
> + */
> +const struct vimc_pix_map *vimc_pix_map_by_index(unsigned int i);
> +
> +/**
>   * vimc_pix_map_by_code - get vimc_pix_map struct by media bus code
>   *
>   * @code:            media bus format code defined by MEDIA_BUS_FMT_* macros
> diff --git a/drivers/media/platform/vimc/vimc-sensor.c 
> b/drivers/media/platform/vimc/vimc-sensor.c
> index abb2172..c86b4e6 100644
> --- a/drivers/media/platform/vimc/vimc-sensor.c
> +++ b/drivers/media/platform/vimc/vimc-sensor.c
> @@ -24,8 +24,6 @@
>  
>  #include "vimc-sensor.h"
>  
> -#define VIMC_SEN_FRAME_MAX_WIDTH 4096
> -
>  struct vimc_sen_device {
>       struct vimc_ent_device ved;
>       struct v4l2_subdev sd;
> @@ -37,18 +35,41 @@ struct vimc_sen_device {
>       int frame_size;
>  };
>  
> +static const struct v4l2_mbus_framefmt fmt_default = {
> +     .width = 640,
> +     .height = 480,
> +     .code = MEDIA_BUS_FMT_RGB888_1X24,
> +     .field = V4L2_FIELD_NONE,
> +     .colorspace = V4L2_COLORSPACE_SRGB,
> +     .quantization = V4L2_QUANTIZATION_FULL_RANGE,
> +     .xfer_func = V4L2_XFER_FUNC_SRGB,
> +};
> +
> +static int vimc_sen_init_cfg(struct v4l2_subdev *sd,
> +                          struct v4l2_subdev_pad_config *cfg)
> +{
> +     unsigned int i;
> +
> +     for (i = 0; i < sd->entity.num_pads; i++) {
> +             struct v4l2_mbus_framefmt *mf;
> +
> +             mf = v4l2_subdev_get_try_format(sd, cfg, i);
> +             *mf = fmt_default;
> +     }
> +
> +     return 0;
> +}
> +
>  static int vimc_sen_enum_mbus_code(struct v4l2_subdev *sd,
>                                  struct v4l2_subdev_pad_config *cfg,
>                                  struct v4l2_subdev_mbus_code_enum *code)
>  {
> -     struct vimc_sen_device *vsen =
> -                             container_of(sd, struct vimc_sen_device, sd);
> +     const struct vimc_pix_map *vpix = vimc_pix_map_by_index(code->index);
>  
> -     /* TODO: Add support for other codes */
> -     if (code->index)
> +     if (!vpix)
>               return -EINVAL;
>  
> -     code->code = vsen->mbus_format.code;
> +     code->code = vpix->code;
>  
>       return 0;
>  }
> @@ -57,33 +78,34 @@ static int vimc_sen_enum_frame_size(struct v4l2_subdev 
> *sd,
>                                   struct v4l2_subdev_pad_config *cfg,
>                                   struct v4l2_subdev_frame_size_enum *fse)
>  {
> -     struct vimc_sen_device *vsen =
> -                             container_of(sd, struct vimc_sen_device, sd);
> +     const struct vimc_pix_map *vpix;
>  
> -     /* TODO: Add support to other formats */
>       if (fse->index)
>               return -EINVAL;
>  
> -     /* TODO: Add support for other codes */
> -     if (fse->code != vsen->mbus_format.code)
> +     /* Only accept code in the pix map table */
> +     vpix = vimc_pix_map_by_code(fse->code);
> +     if (!vpix)
>               return -EINVAL;
>  
> -     fse->min_width = vsen->mbus_format.width;
> -     fse->max_width = vsen->mbus_format.width;
> -     fse->min_height = vsen->mbus_format.height;
> -     fse->max_height = vsen->mbus_format.height;
> +     fse->min_width = VIMC_FRAME_MIN_WIDTH;
> +     fse->max_width = VIMC_FRAME_MAX_WIDTH;
> +     fse->min_height = VIMC_FRAME_MIN_HEIGHT;
> +     fse->max_height = VIMC_FRAME_MAX_HEIGHT;
>  
>       return 0;
>  }
>  
>  static int vimc_sen_get_fmt(struct v4l2_subdev *sd,
>                           struct v4l2_subdev_pad_config *cfg,
> -                         struct v4l2_subdev_format *format)
> +                         struct v4l2_subdev_format *fmt)
>  {
>       struct vimc_sen_device *vsen =
>                               container_of(sd, struct vimc_sen_device, sd);
>  
> -     format->format = vsen->mbus_format;
> +     fmt->format = fmt->which == V4L2_SUBDEV_FORMAT_TRY ?
> +                   *v4l2_subdev_get_try_format(sd, cfg, fmt->pad) :
> +                   vsen->mbus_format;
>  
>       return 0;
>  }
> @@ -110,12 +132,81 @@ static void vimc_sen_tpg_s_format(struct 
> vimc_sen_device *vsen)
>       tpg_s_xfer_func(&vsen->tpg, vsen->mbus_format.xfer_func);
>  }
>  
> +static void vimc_sen_adjust_fmt(struct v4l2_mbus_framefmt *fmt)
> +{
> +     const struct vimc_pix_map *vpix;
> +
> +     /* Only accept code in the pix map table */
> +     vpix = vimc_pix_map_by_code(fmt->code);
> +     if (!vpix)
> +             fmt->code = fmt_default.code;
> +
> +     fmt->width = clamp_t(u32, fmt->width, VIMC_FRAME_MIN_WIDTH,
> +                          VIMC_FRAME_MAX_WIDTH);
> +     fmt->height = clamp_t(u32, fmt->height, VIMC_FRAME_MIN_HEIGHT,
> +                           VIMC_FRAME_MAX_HEIGHT);

I would expect that width and height need to be multiple of some value.
Height needs to be a multiple of 2 (otherwise you can never support
interlaced formats). Width needs to be a multiple of 2 at minimum as
well.

> +
> +     if (fmt->field == V4L2_FIELD_ANY)
> +             fmt->field = fmt_default.field;
> +
> +     /* Check if values are out of range */
> +     if (fmt->colorspace == V4L2_COLORSPACE_DEFAULT
> +         || fmt->colorspace > V4L2_COLORSPACE_DCI_P3)
> +             fmt->colorspace = fmt_default.colorspace;

If the colorspace is invalid, then set all four fields to their defaults.

> +     if (fmt->ycbcr_enc == V4L2_YCBCR_ENC_DEFAULT
> +         || fmt->ycbcr_enc > V4L2_YCBCR_ENC_SMPTE240M)
> +             fmt->ycbcr_enc = fmt_default.ycbcr_enc;
> +     if (fmt->quantization == V4L2_QUANTIZATION_DEFAULT
> +         || fmt->quantization > V4L2_QUANTIZATION_LIM_RANGE)
> +             fmt->quantization = fmt_default.quantization;
> +     if (fmt->xfer_func == V4L2_XFER_FUNC_DEFAULT
> +         || fmt->xfer_func > V4L2_XFER_FUNC_SMPTE2084)
> +             fmt->xfer_func = fmt_default.xfer_func;

The _DEFAULT values are all valid for these three fields. If they
have an unknown value, then just reset them to the _DEFAULT value.

> +}
> +
> +static int vimc_sen_set_fmt(struct v4l2_subdev *sd,
> +                         struct v4l2_subdev_pad_config *cfg,
> +                         struct v4l2_subdev_format *fmt)
> +{
> +     struct vimc_sen_device *vsen = v4l2_get_subdevdata(sd);
> +     struct v4l2_mbus_framefmt *mf;
> +
> +     if (fmt->which == V4L2_SUBDEV_FORMAT_ACTIVE) {
> +             /* Do not change the format while stream is on */
> +             if (vsen->frame)
> +                     return -EBUSY;
> +
> +             mf = &vsen->mbus_format;
> +     } else {
> +             mf = v4l2_subdev_get_try_format(sd, cfg, fmt->pad);
> +     }
> +
> +     /* Set the new format */
> +     vimc_sen_adjust_fmt(&fmt->format);
> +
> +     dev_dbg(vsen->sd.v4l2_dev->mdev->dev, "%s: format update: "
> +             "old:%dx%d (0x%x, %d, %d, %d, %d) "
> +             "new:%dx%d (0x%x, %d, %d, %d, %d)\n", vsen->sd.name,
> +             /* old */
> +             mf->width, mf->height, mf->code,
> +             mf->colorspace, mf->quantization,
> +             mf->xfer_func, mf->ycbcr_enc,
> +             /* new */
> +             fmt->format.width, fmt->format.height, fmt->format.code,
> +             fmt->format.colorspace, fmt->format.quantization,
> +             fmt->format.xfer_func, fmt->format.ycbcr_enc);
> +
> +     *mf = fmt->format;
> +
> +     return 0;
> +}
> +
>  static const struct v4l2_subdev_pad_ops vimc_sen_pad_ops = {
> +     .init_cfg               = vimc_sen_init_cfg,
>       .enum_mbus_code         = vimc_sen_enum_mbus_code,
>       .enum_frame_size        = vimc_sen_enum_frame_size,
>       .get_fmt                = vimc_sen_get_fmt,
> -     /* TODO: Add support to other formats */
> -     .set_fmt                = vimc_sen_get_fmt,
> +     .set_fmt                = vimc_sen_set_fmt,
>  };
>  
>  static int vimc_thread_sen(void *data)
> @@ -248,19 +339,13 @@ struct vimc_ent_device *vimc_sen_create(struct 
> v4l2_device *v4l2_dev,
>       if (ret)
>               goto err_free_vsen;
>  
> -     /* Set the active frame format (this is hardcoded for now) */
> -     vsen->mbus_format.width = 640;
> -     vsen->mbus_format.height = 480;
> -     vsen->mbus_format.code = MEDIA_BUS_FMT_RGB888_1X24;
> -     vsen->mbus_format.field = V4L2_FIELD_NONE;
> -     vsen->mbus_format.colorspace = V4L2_COLORSPACE_SRGB;
> -     vsen->mbus_format.quantization = V4L2_QUANTIZATION_FULL_RANGE;
> -     vsen->mbus_format.xfer_func = V4L2_XFER_FUNC_SRGB;
> +     /* Initialize the frame format */
> +     vsen->mbus_format = fmt_default;
>  
>       /* Initialize the test pattern generator */
>       tpg_init(&vsen->tpg, vsen->mbus_format.width,
>                vsen->mbus_format.height);
> -     ret = tpg_alloc(&vsen->tpg, VIMC_SEN_FRAME_MAX_WIDTH);
> +     ret = tpg_alloc(&vsen->tpg, VIMC_FRAME_MAX_WIDTH);
>       if (ret)
>               goto err_unregister_ent_sd;
>  
> 

Regards,

        Hans

Reply via email to