Use new struct v4l2_mbus_framefmt framesamples field to return maximum size of data transmitted per single frame. The framesamples value is adjusted according to pixel format and configured in the ISP registers. Except the pixel width and height, the frame size can also be made dependent on JPEG quality ratio, when corresponding control is available at the sub-device interface. To ensure the pixel format is not changed after streaming is activated the media entity 'stream_count' is checked before the format change is attempted. Signed-off-by: Sylwester Nawrocki <s.nawrocki@xxxxxxxxxxx> Signed-off-by: Kyungmin Park <kyungmin.park@xxxxxxxxxxx> --- drivers/media/video/m5mols/m5mols.h | 8 ++++++++ drivers/media/video/m5mols/m5mols_capture.c | 4 ++++ drivers/media/video/m5mols/m5mols_core.c | 18 ++++++++++++++++++ drivers/media/video/m5mols/m5mols_reg.h | 2 ++ 4 files changed, 32 insertions(+), 0 deletions(-) diff --git a/drivers/media/video/m5mols/m5mols.h b/drivers/media/video/m5mols/m5mols.h index 82c8817..d729812 100644 --- a/drivers/media/video/m5mols/m5mols.h +++ b/drivers/media/video/m5mols/m5mols.h @@ -16,9 +16,17 @@ #ifndef M5MOLS_H #define M5MOLS_H +#include <asm/sizes.h> #include <media/v4l2-subdev.h> #include "m5mols_reg.h" +/* + * This is an amount of data transmitted in addition to the configured + * JPEG_SIZE_MAX value. + */ +#define M5MOLS_JPEG_TAGS_SIZE 0x20000 +#define M5MOLS_MAIN_JPEG_SIZE_MAX (5 * SZ_1M) + extern int m5mols_debug; #define to_m5mols(__sd) container_of(__sd, struct m5mols_info, sd) diff --git a/drivers/media/video/m5mols/m5mols_capture.c b/drivers/media/video/m5mols/m5mols_capture.c index 3248ac8..c8da22f 100644 --- a/drivers/media/video/m5mols/m5mols_capture.c +++ b/drivers/media/video/m5mols/m5mols_capture.c @@ -119,6 +119,7 @@ static int m5mols_capture_info(struct m5mols_info *info) int m5mols_start_capture(struct m5mols_info *info) { + struct v4l2_mbus_framefmt *mf = &info->ffmt[info->res_type]; struct v4l2_subdev *sd = &info->sd; u8 resolution = info->resolution; int timeout; @@ -170,6 +171,9 @@ int m5mols_start_capture(struct m5mols_info *info) if (!ret) ret = m5mols_write(sd, CAPP_MAIN_IMAGE_SIZE, resolution); if (!ret) + ret = m5mols_write(sd, CAPP_JPEG_SIZE_MAX, + mf->framesamples - M5MOLS_JPEG_TAGS_SIZE); + if (!ret) ret = m5mols_enable_interrupt(sd, REG_INT_CAPTURE); if (!ret) ret = m5mols_write(sd, CAPC_START, REG_CAP_START_MAIN); diff --git a/drivers/media/video/m5mols/m5mols_core.c b/drivers/media/video/m5mols/m5mols_core.c index b5957d7..995165c 100644 --- a/drivers/media/video/m5mols/m5mols_core.c +++ b/drivers/media/video/m5mols/m5mols_core.c @@ -505,6 +505,16 @@ static struct v4l2_mbus_framefmt *__find_format(struct m5mols_info *info, return &info->ffmt[type]; } +static inline void m5mols_get_buffer_size(struct m5mols_info *info, + struct v4l2_mbus_framefmt *mf) +{ + u32 min_sz = mf->width * mf->height + M5MOLS_JPEG_TAGS_SIZE; + + /* Clamp provided value to the maximum needed */ + mf->framesamples = clamp_t(u32, mf->framesamples, min_sz, + M5MOLS_MAIN_JPEG_SIZE_MAX); +} + static int m5mols_get_fmt(struct v4l2_subdev *sd, struct v4l2_subdev_fh *fh, struct v4l2_subdev_format *fmt) { @@ -537,6 +547,14 @@ static int m5mols_set_fmt(struct v4l2_subdev *sd, struct v4l2_subdev_fh *fh, if (!sfmt) return 0; + if (format->code == V4L2_MBUS_FMT_JPEG_1X8) { + mutex_lock(&sd->entity.parent->graph_mutex); + ret = sd->entity.stream_count; + mutex_unlock(&sd->entity.parent->graph_mutex); + if (ret > 0) + return -EBUSY; + m5mols_get_buffer_size(info, format); + } format->code = m5mols_default_ffmt[type].code; format->colorspace = V4L2_COLORSPACE_JPEG; diff --git a/drivers/media/video/m5mols/m5mols_reg.h b/drivers/media/video/m5mols/m5mols_reg.h index c755bd6..d44f5cc 100644 --- a/drivers/media/video/m5mols/m5mols_reg.h +++ b/drivers/media/video/m5mols/m5mols_reg.h @@ -342,6 +342,7 @@ */ #define CATB_YUVOUT_MAIN 0x00 #define CATB_MAIN_IMAGE_SIZE 0x01 +#define CATB_JPEG_SIZE_MAX 0x0f #define CATB_MCC_MODE 0x1d #define CATB_WDR_EN 0x2c #define CATB_LIGHT_CTRL 0x40 @@ -354,6 +355,7 @@ #define REG_JPEG 0x10 #define CAPP_MAIN_IMAGE_SIZE I2C_REG(CAT_CAPT_PARM, CATB_MAIN_IMAGE_SIZE, 1) +#define CAPP_JPEG_SIZE_MAX I2C_REG(CAT_CAPT_PARM, CATB_JPEG_SIZE_MAX, 4) #define CAPP_MCC_MODE I2C_REG(CAT_CAPT_PARM, CATB_MCC_MODE, 1) #define REG_MCC_OFF 0x00 -- 1.7.7.2 -- To unsubscribe from this list: send the line "unsubscribe linux-media" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html