/**************************************************************************** * boards/arm/cxd56xx/drivers/camera/isx012.c * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. The * ASF licenses this file to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance with the * License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations * under the License. * ****************************************************************************/ /**************************************************************************** * Included Files ****************************************************************************/ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include /**************************************************************************** * Pre-processor Definitions ****************************************************************************/ /* The following macro is enabled because * it is to make stable startup. (other case) */ /* #define ISX012_NOT_USE_NSTBY */ /* The following macro is disabled because it is to see detailed control. */ /* #define ISX012_CHECK_IN_DETAIL */ /* Skip invalid frame because it occurs first due to the spec of isx012. */ #define ISX012_FRAME_SKIP_EN #define OUT_HSIZE_QVGA (320) #define OUT_VSIZE_QVGA (240) #define OUT_HSIZE_VGA (640) #define OUT_VSIZE_VGA (480) #define OUT_HSIZE_HD (1280) #define OUT_VSIZE_HD (720) #define OUT_HSIZE_QUADVGA (1280) #define OUT_VSIZE_QUADVGA (960) #define OUT_HSIZE_FULLHD (1920) #define OUT_VSIZE_FULLHD (1080) #define OUT_HSIZE_3M (2048) #define OUT_VSIZE_3M (1536) #define OUT_HSIZE_5M (2560) #define OUT_VSIZE_5M (1920) #define OUT_YUV_VSIZE_MIN (64) #define OUT_YUV_HSIZE_MIN (96) #define OUT_JPG_VSIZE_MIN (64) #define OUT_JPG_HSIZE_MIN (96) #define OUT_YUV_15FPS_VSIZE_MAX (360) #define OUT_YUV_15FPS_HSIZE_MAX (480) #define OUT_YUV_30FPS_VSIZE_MAX (360) #define OUT_YUV_30FPS_HSIZE_MAX (480) #define OUT_YUV_60FPS_VSIZE_MAX (360) #define OUT_YUV_60FPS_HSIZE_MAX (480) #define OUT_YUV_120FPS_VSIZE_MAX (240) #define OUT_YUV_120FPS_HSIZE_MAX (320) #define OUT_JPG_15FPS_VSIZE_MAX (1944) #define OUT_JPG_15FPS_HSIZE_MAX (2592) #define OUT_JPG_30FPS_VSIZE_MAX (960) #define OUT_JPG_30FPS_HSIZE_MAX (1280) #define OUT_JPG_60FPS_VSIZE_MAX (480) #define OUT_JPG_60FPS_HSIZE_MAX (640) #define OUT_JPG_120FPS_VSIZE_MAX (240) #define OUT_JPG_120FPS_HSIZE_MAX (320) #define OUT_YUVINT_30FPS_VSIZE_MAX (240) #define OUT_YUVINT_30FPS_HSIZE_MAX (400) #define OUT_JPGINT_30FPS_VSIZE_MAX (960) #define OUT_JPGINT_30FPS_HSIZE_MAX (1280) #define OUT_JPGINT_15FPS_VSIZE_MAX (1224) #define OUT_JPGINT_15FPS_HSIZE_MAX (1632) #define VINT_TIMEOUT (400) /* ms */ #define VINT_WAIT_TIME (5) /* ms */ #define VINT_DELAY_TIME (0) /* ms */ #define CAMERA_MODE_TIMEOUT (800) /* ms */ #define CAMERA_MODE_WAIT_TIME (10) /* ms */ #define CAMERA_MODE_DELAY_TIME (0) /* ms */ #define DEVICE_STATE_TIMEOUT (100) /* ms */ #define DEVICE_STATE_WAIT_TIME (1) /* ms */ #define DEVICE_STATE_DELAY_TIME (2) /* ms */ #define I2CFREQ_STANDARD (100000) /* Standard mode : 100kHz */ #define I2CFREQ_FAST (400000) /* Fast mode : 400kHz */ #define ISX012_SIZE_STEP (2) #define CXC_RGB_DATA_UNIT_NUM (27) #define CXC_RGB_DATA_UNIT_SIZE (7) #define CXC_GRB_DATA_UNIT_NUM (27) #define CXC_GRB_DATA_UNIT_SIZE (7) #define SHD_RGB_DATA_UNIT_NUM (27) #define SHD_RGB_DATA_UNIT_SIZE (11) #define SHD_GRB_DATA_UNIT_NUM (27) #define SHD_GRB_DATA_UNIT_SIZE (11) #define SHD_R1_DATA_UNIT_NUM (14) #define SHD_R1_DATA_UNIT_SIZE (11) #define SHD_R2_DATA_UNIT_NUM (14) #define SHD_R2_DATA_UNIT_SIZE (11) #define SHD_B2_DATA_UNIT_NUM (14) #define SHD_B2_DATA_UNIT_SIZE (11) #ifdef CONFIG_DEBUG_IMAGER_ERROR #define imagererr(format, ...) _err(format, ##__VA_ARGS__) #else #define imagererr(x...) #endif #ifdef CONFIG_DEBUG_IMAGER_WARN #define imagerwarn(format, ...) _warn(format, ##__VA_ARGS__) #else #define imagerwarn(x...) #endif #ifdef CONFIG_DEBUG_IMAGER_INFO #define imagerinfo(format, ...) _info(format, ##__VA_ARGS__) #else #define imagerinfo(x...) #endif #define CHECK_RANGE(value,min,max,step) do { \ if ((value < min) || \ (value > max) || \ ((value - min) % step != 0)) \ { \ return -EINVAL;\ } \ } while (0) /**************************************************************************** * Private Types ****************************************************************************/ enum isx012_state_e { STATE_ISX012_PRESLEEP, STATE_ISX012_SLEEP, STATE_ISX012_ACTIVE, STATE_ISX012_POWEROFF, }; typedef enum isx012_state_e isx012_state_t; struct isx012_reg_s { uint16_t regaddr; uint16_t regval; uint8_t regsize; }; typedef struct isx012_reg_s isx012_reg_t; struct isx012_conv_v4l2_to_regval_s { int32_t v4l2; int16_t regval; }; typedef struct isx012_conv_v4l2_to_regval_s isx012_conv_v4l2_to_regval_t; struct isx012_modeparam_s { uint8_t fps; /* use ISX012 register setting value */ uint32_t format; /* use V4L2 definition */ uint16_t hsize; uint16_t vsize; uint16_t int_hsize; uint16_t int_vsize; }; typedef struct isx012_modeparam_s isx012_modeparam_t; struct isx012_param_s { isx012_modeparam_t video; /* Parameter for video capture mode */ isx012_modeparam_t still; /* Parameter for still capture mode */ }; typedef struct isx012_param_s isx012_param_t; struct isx012_dev_s { FAR struct i2c_master_s *i2c; /* I2C interface */ uint8_t i2c_addr; /* I2C address */ int i2c_freq; /* Frequency */ isx012_state_t state; /* ISX012 status */ bool dma_state; /* true means "in DMA" */ uint8_t mode; /* ISX012 mode */ isx012_param_t param; /* ISX012 parameters */ void *video_priv; /* pointer to video private data */ }; typedef struct isx012_dev_s isx012_dev_t; #define ARRAY_NENTRIES(a) (sizeof(a)/sizeof(isx012_reg_t)) /**************************************************************************** * Private Function Prototypes ****************************************************************************/ /* register operations */ static uint16_t isx012_getreg(isx012_dev_t *priv, uint16_t regaddr, uint16_t regsize); static int isx012_putreg(isx012_dev_t *priv, uint16_t regaddr, uint16_t regval, uint16_t regsize); static int isx012_putreglist(isx012_dev_t *priv, FAR const isx012_reg_t *reglist, size_t nentries); #ifdef ISX012_CHECK_IN_DETAIL static int isx012_putregs(isx012_dev_t *priv, uint16_t regaddr, uint8_t *regvals, uint8_t regsize); static int isx012_chipid(FAR struct i2c_master_s *i2c); #endif static int isx012_chk_int_state(isx012_dev_t *priv, uint8_t sts, uint32_t delay_time, uint32_t wait_time, uint32_t timeout); static int isx012_set_mode_param(isx012_dev_t *priv, enum v4l2_buf_type type, isx012_modeparam_t *param); static int isx012_change_camera_mode(isx012_dev_t *priv, uint8_t mode); static int isx012_change_device_state(isx012_dev_t *priv, isx012_state_t state); static int isx012_set_supported_frminterval(uint32_t fps_index, FAR struct v4l2_fract *interval); static int8_t isx012_get_maximum_fps(FAR struct v4l2_frmivalenum *frmival); static int isx012_set_shd(FAR isx012_dev_t *priv); static bool is_movie_needed(isx012_modeparam_t *param); /* video driver HAL infterface */ static int isx012_open(FAR void *video_private); static int isx012_close(void); static int isx012_do_halfpush(bool enable); static int isx012_set_buftype(enum v4l2_buf_type type); static int isx012_set_buf(uint32_t bufaddr, uint32_t bufsize); static int isx012_cancel_dma(void); static int isx012_check_fmt(enum v4l2_buf_type buf_type, uint32_t pixel_format); static int isx012_get_range_of_fmt(FAR struct v4l2_fmtdesc *format); static int isx012_get_range_of_framesize(FAR struct v4l2_frmsizeenum *frmsize); static int isx012_try_format(FAR struct v4l2_format *format); static int isx012_set_format(FAR struct v4l2_format *format); static int isx012_get_range_of_frameinterval(FAR struct v4l2_frmivalenum *frmival); static int isx012_set_frameinterval(FAR struct v4l2_streamparm *parm); static int isx012_get_range_of_ctrlval(FAR struct v4l2_query_ext_ctrl *range); static int isx012_get_menu_of_ctrlval(FAR struct v4l2_querymenu *menu); static int isx012_get_ctrlval(uint16_t ctrl_class, FAR struct v4l2_ext_control *control); static int isx012_set_ctrlval(uint16_t ctrl_class, FAR struct v4l2_ext_control *control); static int isx012_refresh(void); /**************************************************************************** * Private Data ****************************************************************************/ static isx012_dev_t g_isx012_private; #ifndef ISX012_NOT_USE_NSTBY static const isx012_reg_t g_isx012_presleep[] = { {PLL_CKSEL, 0x00, 0x01}, /* PLL_CKSEL */ {SRCCK_DIV, 0x00, 0x01}, /* SRCCK_DIV */ {INCK_SET, 0x17, 0x01}, /* INCK_SET */ }; #define ISX012_PRESLEEP_NENTRIES ARRAY_NENTRIES(g_isx012_presleep) #endif static const isx012_reg_t g_isx012_def_init[] = { #ifdef ISX012_NOT_USE_NSTBY {PLL_CKSEL, 0x00, 0x01}, {SRCCK_DIV, 0x00, 0x01}, #endif {DRIVABILITY, 0xaa, 0x01}, {VIFCONFIG, 0x0200, 0x02}, {YUVCONFIG_TN, 0xff0a, 0x02}, {ILCODELEN, 0x00, 0x01}, {AFMODE_MONI, 0x01, 0x01}, {YUVCONFIG, 0xff6a, 0x02}, {VIF_REC601_Y, 0x10fe, 0x02}, {VIF_REC601_C, 0x10f0, 0x02}, {HSENS_MODE_SEL, 0x11, 0x01}, {VIF_CLKCONFIG1, 0x30, 0x01}, {VIF_CLKCONFIG2, 0x30, 0x01}, {VIF_CLKCONFIG3, 0x30, 0x01}, {VIF_CLKCONFIG4, 0x30, 0x01}, {VIF_CLKCONFIG5, 0x30, 0x01}, {VIF_CLKCONFIG6, 0x30, 0x01}, {VIF_CLKCONFIG7, 0x30, 0x01}, {VIF_CLKCONFIG8, 0x30, 0x01}, {VIF_CLKCONFIG9, 0x30, 0x01}, {VIF_CLKCONFIG10, 0x30, 0x01}, {VIF_CLKCONFIG11, 0x30, 0x01}, {VIF_CLKCONFIG12, 0x30, 0x01}, {VIF_CLKCONFIG13, 0x11, 0x01}, {VIF_CLKCONFIG14, 0x11, 0x01}, {VIF_CLKCONFIG15, 0x11, 0x01}, {VIF_CLKCONFIG16, 0x11, 0x01}, #ifdef ISX012_NOT_USE_NSTBY {INCK_SET, 0x17, 0x01}, /* INCK_SET */ #endif {FRM_FIX_SN1_2, 0xff, 0x01}, /* Fix framerate */ {FAST_MODECHG_EN, 0x01, 0x01}, {FAST_SHT_MODE_SEL, 0x01, 0x01}, {CAP_HALF_AE_CTRL, 0x07, 0x01}, /* HAFREL=HIGHSPEED, CAP=Auto */ {HALF_AWB_CTRL, 0x01, 0x01}, {AESPEED_FAST, 0x0f, 0x01}, {FASTMOVE_TIMEOUT, 0x2d, 0x01}, {YGAMMA_MODE, 0x01, 0x01}, {INT_QLTY2, 0x50, 0x01}, }; #define ISX012_RESET_NENTRIES ARRAY_NENTRIES(g_isx012_def_init) static const uint8_t g_isx012_cxc_rgb_data[CXC_RGB_DATA_UNIT_NUM] [CXC_RGB_DATA_UNIT_SIZE] = { {0x01, 0x43, 0xc0, 0xf0, 0x4f, 0xfc, 0x13}, /* CXC_RGB_UNIT0 */ {0x80, 0x44, 0x20, 0x21, 0x48, 0x04, 0x0e}, /* CXC_RGB_UNIT1 */ {0x81, 0x43, 0xc0, 0x10, 0x30, 0xfc, 0x13}, /* CXC_RGB_UNIT2 */ {0xff, 0x04, 0x20, 0x11, 0x48, 0x08, 0x12}, /* CXC_RGB_UNIT3 */ {0x81, 0x43, 0xe0, 0x20, 0x48, 0x08, 0x12}, /* CXC_RGB_UNIT4 */ {0x80, 0x03, 0xe0, 0x00, 0x38, 0x04, 0x10}, /* CXC_RGB_UNIT5 */ {0x01, 0x84, 0x20, 0x21, 0x48, 0x04, 0x10}, /* CXC_RGB_UNIT6 */ {0x01, 0x04, 0xc0, 0x10, 0x20, 0x00, 0x08}, /* CXC_RGB_UNIT7 */ {0x81, 0x82, 0xc0, 0x20, 0x38, 0x08, 0x0e}, /* CXC_RGB_UNIT8 */ {0x01, 0x43, 0xc0, 0x10, 0x20, 0x04, 0x04}, /* CXC_RGB_UNIT9 */ {0x01, 0x41, 0x40, 0x10, 0x20, 0x08, 0x0a}, /* CXC_RGB_UNIT10 */ {0x82, 0x82, 0x80, 0x20, 0x20, 0x04, 0x04}, /* CXC_RGB_UNIT11 */ {0x82, 0x80, 0x20, 0x20, 0x08, 0x04, 0x06}, /* CXC_RGB_UNIT12 */ {0x81, 0x42, 0xa0, 0x10, 0x20, 0x04, 0x08}, /* CXC_RGB_UNIT13 */ {0x81, 0x80, 0x00, 0x00, 0x00, 0x04, 0x00}, /* CXC_RGB_UNIT14 */ {0x01, 0x41, 0x80, 0x10, 0x20, 0x00, 0x08}, /* CXC_RGB_UNIT15 */ {0x00, 0x42, 0x20, 0x20, 0x08, 0x08, 0x00}, /* CXC_RGB_UNIT16 */ {0x82, 0xc0, 0x40, 0x20, 0x20, 0x08, 0x08}, /* CXC_RGB_UNIT17 */ {0x80, 0x02, 0xa0, 0x10, 0x20, 0x08, 0x04}, /* CXC_RGB_UNIT18 */ {0x02, 0x81, 0x60, 0x30, 0x20, 0x08, 0x0a}, /* CXC_RGB_UNIT19 */ {0x82, 0x42, 0xc0, 0x10, 0x30, 0x04, 0x0a}, /* CXC_RGB_UNIT20 */ {0x03, 0xc3, 0xa0, 0x40, 0x28, 0x0c, 0x0a}, /* CXC_RGB_UNIT21 */ {0x03, 0xc3, 0xc0, 0x20, 0x20, 0x08, 0x08}, /* CXC_RGB_UNIT22 */ {0x82, 0xc2, 0xc0, 0x30, 0x40, 0x10, 0x0e}, /* CXC_RGB_UNIT23 */ {0x84, 0x03, 0xa1, 0x40, 0x28, 0x08, 0x08}, /* CXC_RGB_UNIT24 */ {0x02, 0x82, 0xa0, 0x30, 0x30, 0x0c, 0x10}, /* CXC_RGB_UNIT25 */ {0x84, 0x03, 0xe1, 0x40, 0x28, 0x10, 0x0a}, /* CXC_RGB_UNIT26 */ }; static const uint8_t g_isx012_cxc_grb_data[CXC_GRB_DATA_UNIT_NUM] [CXC_GRB_DATA_UNIT_SIZE] = { {0x00, 0x3d, 0x40, 0x0f, 0xc0, 0x03, 0xf2}, /* CXC_GRB_UNIT0 */ {0x80, 0x7c, 0x80, 0x1f, 0xd8, 0x03, 0xf0}, /* CXC_GRB_UNIT1 */ {0x00, 0x3c, 0x40, 0x0f, 0xd0, 0x03, 0xf0}, /* CXC_GRB_UNIT2 */ {0x80, 0x3c, 0x20, 0x1f, 0xe0, 0x07, 0xf6}, /* CXC_GRB_UNIT3 */ {0x00, 0x3c, 0x00, 0x1f, 0xd0, 0x07, 0xf4}, /* CXC_GRB_UNIT4 */ {0x00, 0x3d, 0x40, 0x0f, 0xc8, 0x03, 0xf2}, /* CXC_GRB_UNIT5 */ {0x80, 0xfc, 0x5f, 0xff, 0xd7, 0x07, 0xf4}, /* CXC_GRB_UNIT6 */ {0x01, 0x7d, 0x40, 0x0f, 0xd0, 0xff, 0xf3}, /* CXC_GRB_UNIT7 */ {0x7f, 0xfd, 0x3f, 0x0f, 0xc8, 0x03, 0xf2}, /* CXC_GRB_UNIT8 */ {0x81, 0x7c, 0x20, 0x0f, 0xd0, 0xff, 0xf7}, /* CXC_GRB_UNIT9 */ {0x7e, 0xfe, 0x5f, 0x0f, 0xd8, 0x03, 0xf6}, /* CXC_GRB_UNIT10 */ {0x80, 0xbd, 0xa0, 0x2f, 0xe8, 0x07, 0xfa}, /* CXC_GRB_UNIT11 */ {0x80, 0xfe, 0xbf, 0x0f, 0xe8, 0xff, 0xf9}, /* CXC_GRB_UNIT12 */ {0x00, 0x3e, 0x80, 0x3f, 0xe8, 0x0f, 0xfa}, /* CXC_GRB_UNIT13 */ {0x02, 0x40, 0xe0, 0x0f, 0xf8, 0x03, 0xfe}, /* CXC_GRB_UNIT14 */ {0x80, 0x7f, 0xe0, 0x1f, 0xf8, 0x17, 0xfe}, /* CXC_GRB_UNIT15 */ {0x85, 0xff, 0xe0, 0x2f, 0x08, 0x04, 0x04}, /* CXC_GRB_UNIT16 */ {0x81, 0x40, 0x20, 0x20, 0x00, 0x08, 0x00}, /* CXC_GRB_UNIT17 */ {0x84, 0x00, 0x21, 0x30, 0x10, 0x0c, 0x06}, /* CXC_GRB_UNIT18 */ {0x02, 0x82, 0x40, 0x20, 0x10, 0x0c, 0x02}, /* CXC_GRB_UNIT19 */ {0x83, 0x00, 0x21, 0x40, 0x08, 0x10, 0x06}, /* CXC_GRB_UNIT20 */ {0x83, 0x82, 0xa0, 0x20, 0x20, 0x08, 0x08}, /* CXC_GRB_UNIT21 */ {0x02, 0x81, 0x40, 0x30, 0x18, 0x0c, 0x06}, /* CXC_GRB_UNIT22 */ {0x03, 0x81, 0x80, 0x10, 0x20, 0x04, 0x08}, /* CXC_GRB_UNIT23 */ {0x82, 0x82, 0x80, 0x20, 0x20, 0x0c, 0x06}, /* CXC_GRB_UNIT24 */ {0x83, 0xc1, 0x40, 0x20, 0x20, 0x04, 0x08}, /* CXC_GRB_UNIT25 */ {0x01, 0x82, 0xa0, 0x20, 0x20, 0x08, 0x08}, /* CXC_GRB_UNIT26 */ }; static const uint8_t g_isx012_shd_rgb_data[SHD_RGB_DATA_UNIT_NUM] [SHD_RGB_DATA_UNIT_SIZE] = { {0xf1, 0x59, 0x52, 0x7b, 0x98, 0xc4, 0x9d, 0x23, 0x29, 0x87, 0x46}, /* SHD_RGB_UNIT0 */ {0xc6, 0x81, 0xd1, 0x70, 0x56, 0xe4, 0x9c, 0x1b, 0x6d, 0x07, 0x48}, /* SHD_RGB_UNIT1 */ {0xdd, 0xf1, 0x51, 0x7d, 0xa8, 0xb4, 0x1e, 0x25, 0x49, 0xc7, 0x46}, /* SHD_RGB_UNIT2 */ {0xbd, 0xf1, 0x50, 0x6d, 0x2a, 0x44, 0x1b, 0x0a, 0x01, 0x87, 0x44}, /* SHD_RGB_UNIT3 */ {0xd0, 0xa9, 0x51, 0x77, 0x84, 0xd4, 0x9d, 0x1f, 0x2d, 0xc7, 0x44}, /* SHD_RGB_UNIT4 */ {0xa8, 0xa9, 0xcf, 0x62, 0x98, 0xa3, 0x17, 0xdb, 0xfc, 0x05, 0x38}, /* SHD_RGB_UNIT5 */ {0x90, 0xe1, 0x8e, 0x6a, 0x08, 0xc4, 0x9b, 0x0e, 0x11, 0x07, 0x43}, /* SHD_RGB_UNIT6 */ {0xac, 0xa9, 0x4f, 0x5d, 0x4e, 0x13, 0x15, 0xb9, 0xf8, 0x44, 0x2b}, /* SHD_RGB_UNIT7 */ {0x44, 0x21, 0xcb, 0x56, 0x0e, 0x63, 0x98, 0xe3, 0x78, 0x86, 0x3d}, /* SHD_RGB_UNIT8 */ {0xab, 0x81, 0x4f, 0x62, 0x7c, 0xc3, 0x94, 0xb4, 0x98, 0x84, 0x26}, /* SHD_RGB_UNIT9 */ {0x14, 0xe9, 0x48, 0x46, 0x4a, 0x12, 0x93, 0xa4, 0x84, 0xc5, 0x31}, /* SHD_RGB_UNIT10 */ {0x81, 0xe9, 0x4d, 0x67, 0xac, 0x73, 0x17, 0xd0, 0xdc, 0x24, 0x29}, /* SHD_RGB_UNIT11 */ {0x12, 0xb9, 0x08, 0x40, 0x02, 0x52, 0x10, 0x84, 0x6c, 0x64, 0x25}, /* SHD_RGB_UNIT12 */ {0x4c, 0x91, 0xcb, 0x5b, 0x4c, 0xe3, 0x19, 0xec, 0xdc, 0x05, 0x34}, /* SHD_RGB_UNIT13 */ {0x37, 0x39, 0x8a, 0x44, 0x2a, 0x02, 0x10, 0x80, 0x14, 0xe4, 0x20}, /* SHD_RGB_UNIT14 */ {0x1c, 0x51, 0x49, 0x53, 0xe4, 0x02, 0x17, 0xd3, 0xb8, 0xe6, 0x3d}, /* SHD_RGB_UNIT15 */ {0x8b, 0xd9, 0x8d, 0x53, 0xc8, 0x72, 0x12, 0x98, 0x50, 0x24, 0x23}, /* SHD_RGB_UNIT16 */ {0x19, 0x11, 0x89, 0x4c, 0x8c, 0x32, 0x16, 0xc7, 0x14, 0x06, 0x38}, /* SHD_RGB_UNIT17 */ {0xca, 0xc1, 0x10, 0x6c, 0xe0, 0x83, 0x97, 0xd0, 0x4c, 0xa5, 0x2d}, /* SHD_RGB_UNIT18 */ {0x3e, 0x99, 0x0a, 0x51, 0xbc, 0xc2, 0x15, 0xc2, 0x28, 0x26, 0x39}, /* SHD_RGB_UNIT19 */ {0xa5, 0x89, 0x0f, 0x7b, 0x8c, 0x64, 0x9d, 0x14, 0xb9, 0x46, 0x3e}, /* SHD_RGB_UNIT20 */ {0x8f, 0x41, 0xce, 0x5e, 0x5e, 0x03, 0x98, 0xdc, 0x50, 0xe6, 0x3a}, /* SHD_RGB_UNIT21 */ {0xb4, 0x49, 0x90, 0x72, 0x50, 0x74, 0xa1, 0x3a, 0x05, 0x88, 0x4b}, /* SHD_RGB_UNIT22 */ {0xe1, 0xd1, 0x91, 0x71, 0x38, 0xc4, 0x1b, 0x0a, 0xed, 0x86, 0x42}, /* SHD_RGB_UNIT23 */ {0xcb, 0x49, 0xd1, 0x78, 0x86, 0x74, 0x9f, 0x2d, 0xb9, 0x88, 0x51}, /* SHD_RGB_UNIT24 */ {0x11, 0x62, 0x93, 0x7c, 0x9c, 0x94, 0x1d, 0x1b, 0x41, 0x67, 0x46}, /* SHD_RGB_UNIT25 */ {0xcf, 0x81, 0x91, 0x77, 0x82, 0x54, 0x9f, 0x2a, 0x21, 0xa8, 0x4d}, /* SHD_RGB_UNIT26 */ }; static const uint8_t g_isx012_shd_grb_data[SHD_GRB_DATA_UNIT_NUM] [SHD_GRB_DATA_UNIT_SIZE] = { {0xe8, 0xa9, 0x0f, 0x78, 0xe4, 0x13, 0x9d, 0xf0, 0x04, 0xe7, 0x39}, /* SHD_GRB_UNIT0 */ {0xbd, 0x51, 0x0e, 0x6f, 0x94, 0x63, 0x1c, 0xea, 0x4c, 0x27, 0x3c}, /* SHD_GRB_UNIT1 */ {0xd7, 0x19, 0x4f, 0x7a, 0xf4, 0xd3, 0x1d, 0xf7, 0x20, 0xe7, 0x3a}, /* SHD_GRB_UNIT2 */ {0xb6, 0x11, 0x0e, 0x6c, 0x76, 0x03, 0x9b, 0xdd, 0xf0, 0x06, 0x39}, /* SHD_GRB_UNIT3 */ {0xc9, 0xc1, 0x8e, 0x75, 0xc8, 0xe3, 0x9c, 0xef, 0xf8, 0xa6, 0x39}, /* SHD_GRB_UNIT4 */ {0xa0, 0x69, 0x0d, 0x62, 0x20, 0x93, 0x97, 0xbf, 0xf4, 0xa5, 0x30}, /* SHD_GRB_UNIT5 */ {0x8c, 0xb1, 0x8c, 0x68, 0x60, 0x13, 0x9b, 0xe0, 0xcc, 0xa6, 0x38}, /* SHD_GRB_UNIT6 */ {0x9f, 0x71, 0x0d, 0x5c, 0xf4, 0x12, 0x15, 0xab, 0x00, 0x65, 0x28}, /* SHD_GRB_UNIT7 */ {0x44, 0x41, 0x0a, 0x56, 0xbc, 0x02, 0x98, 0xc4, 0x50, 0x26, 0x34}, /* SHD_GRB_UNIT8 */ {0x9a, 0x59, 0x4d, 0x5f, 0x16, 0x83, 0x14, 0xa8, 0x9c, 0x64, 0x25}, /* SHD_GRB_UNIT9 */ {0x15, 0xc1, 0xc8, 0x46, 0x38, 0x22, 0x13, 0x9a, 0x74, 0x65, 0x2c}, /* SHD_GRB_UNIT10 */ {0x78, 0x11, 0x4c, 0x63, 0x36, 0xb3, 0x96, 0xbb, 0xcc, 0x44, 0x27}, /* SHD_GRB_UNIT11 */ {0x11, 0xa1, 0x48, 0x40, 0x04, 0x72, 0x10, 0x83, 0x70, 0x84, 0x23}, /* SHD_GRB_UNIT12 */ {0x4a, 0x69, 0xca, 0x59, 0xe0, 0xf2, 0x98, 0xcc, 0xb0, 0xa5, 0x2e}, /* SHD_GRB_UNIT13 */ {0x33, 0xc1, 0x09, 0x44, 0x24, 0x02, 0x10, 0x80, 0x14, 0x84, 0x20}, /* SHD_GRB_UNIT14 */ {0x1b, 0xd1, 0x48, 0x52, 0x98, 0x72, 0x96, 0xb7, 0x8c, 0x06, 0x35}, /* SHD_GRB_UNIT15 */ {0x81, 0x39, 0xcc, 0x51, 0x96, 0x32, 0x92, 0x92, 0x48, 0x44, 0x22}, /* SHD_GRB_UNIT16 */ {0x17, 0xb9, 0x48, 0x4b, 0x5e, 0xa2, 0x15, 0xb0, 0xd8, 0x45, 0x30}, /* SHD_GRB_UNIT17 */ {0xc0, 0x19, 0xce, 0x69, 0x56, 0x23, 0x97, 0xba, 0x38, 0x05, 0x2a}, /* SHD_GRB_UNIT18 */ {0x3b, 0xe1, 0x09, 0x50, 0x82, 0x42, 0x95, 0xac, 0xf8, 0x05, 0x31}, /* SHD_GRB_UNIT19 */ {0x94, 0x19, 0x4d, 0x78, 0xca, 0xe3, 0x9c, 0xe8, 0xa8, 0xa6, 0x35}, /* SHD_GRB_UNIT20 */ {0x8b, 0x71, 0xcc, 0x5d, 0xf8, 0xa2, 0x97, 0xc0, 0x24, 0xa6, 0x32}, /* SHD_GRB_UNIT21 */ {0xa4, 0xb1, 0x8d, 0x6d, 0x96, 0xd3, 0xa0, 0x09, 0xe1, 0xa7, 0x3f}, /* SHD_GRB_UNIT22 */ {0xde, 0x09, 0xcf, 0x70, 0x92, 0x73, 0x9b, 0xe0, 0xcc, 0x06, 0x38}, /* SHD_GRB_UNIT23 */ {0xc0, 0x89, 0x4e, 0x74, 0xcc, 0x13, 0x1e, 0xfc, 0x84, 0x48, 0x45}, /* SHD_GRB_UNIT24 */ {0x06, 0x7a, 0xd0, 0x7a, 0xe6, 0x33, 0x1d, 0xef, 0x24, 0x07, 0x3b}, /* SHD_GRB_UNIT25 */ {0xc4, 0xb1, 0x0e, 0x74, 0xca, 0x33, 0x1e, 0xfc, 0xc4, 0x07, 0x41}, /* SHD_GRB_UNIT26 */ }; static const uint8_t g_isx012_shd_r1_data[SHD_R1_DATA_UNIT_NUM] [SHD_R1_DATA_UNIT_SIZE] = { {0x10, 0x92, 0x10, 0x82, 0xf8, 0x43, 0x1f, 0xfb, 0xf0, 0xe7, 0x40}, /* SHD_R1_UNIT0 */ {0x07, 0x92, 0xd0, 0x82, 0xec, 0x33, 0x9e, 0xed, 0x68, 0xe7, 0x3c}, /* SHD_R1_UNIT1 */ {0xfa, 0x21, 0xd0, 0x7e, 0xce, 0xa3, 0x1b, 0xcd, 0x20, 0xe6, 0x31}, /* SHD_R1_UNIT2 */ {0xa6, 0x69, 0xce, 0x78, 0xbc, 0xa3, 0x1b, 0xbe, 0x44, 0x25, 0x28}, /* SHD_R1_UNIT3 */ {0x45, 0x19, 0xcb, 0x65, 0x78, 0xe3, 0x1b, 0xc8, 0x3c, 0xa5, 0x24}, /* SHD_R1_UNIT4 */ {0x15, 0xc1, 0x48, 0x4d, 0xd6, 0x72, 0x99, 0xd3, 0xdc, 0x25, 0x27}, /* SHD_R1_UNIT5 */ {0x11, 0x01, 0x08, 0x41, 0x42, 0x42, 0x15, 0xc1, 0xa4, 0x06, 0x2f}, /* SHD_R1_UNIT6 */ {0x39, 0x89, 0x08, 0x40, 0x0a, 0x22, 0x12, 0xab, 0x0c, 0x26, 0x38}, /* SHD_R1_UNIT7 */ {0x91, 0x71, 0x4a, 0x49, 0x2c, 0xa2, 0x11, 0x9c, 0xc4, 0xa5, 0x33}, /* SHD_R1_UNIT8 */ {0xe2, 0xe1, 0x4d, 0x5f, 0xa4, 0x22, 0x94, 0xa3, 0xa0, 0x05, 0x34}, /* SHD_R1_UNIT9 */ {0xc7, 0x41, 0x50, 0x7c, 0x7e, 0xd3, 0x19, 0xc5, 0x48, 0x86, 0x35}, /* SHD_R1_UNIT10 */ {0xda, 0xa9, 0xcf, 0x8c, 0x42, 0x24, 0x20, 0xf5, 0x8c, 0x67, 0x3c}, /* SHD_R1_UNIT11 */ {0xf6, 0x89, 0xd0, 0x88, 0x90, 0x34, 0x23, 0x0b, 0x15, 0xa8, 0x3f}, /* SHD_R1_UNIT12 */ {0x00, 0x72, 0x10, 0x89, 0x68, 0x04, 0x69, 0x00, 0x00, 0x19, 0x26}, /* SHD_R1_UNIT13 */ }; static const uint8_t g_isx012_shd_r2_data[SHD_R2_DATA_UNIT_NUM] [SHD_R2_DATA_UNIT_SIZE] = { {0x3a, 0xe2, 0x11, 0x8c, 0x42, 0x74, 0xa1, 0x0c, 0x89, 0x08, 0x46}, /* SHD_R2_UNIT0 */ {0x30, 0xe2, 0xd1, 0x8c, 0x36, 0x54, 0x20, 0xfe, 0xec, 0x47, 0x41}, /* SHD_R2_UNIT1 */ {0x20, 0x5a, 0x91, 0x88, 0x16, 0x94, 0x1d, 0xda, 0x80, 0x26, 0x35}, /* SHD_R2_UNIT2 */ {0xc2, 0x69, 0x0f, 0x81, 0x00, 0x94, 0x9d, 0xc9, 0x84, 0xe5, 0x29}, /* SHD_R2_UNIT3 */ {0x54, 0xb1, 0x0b, 0x6c, 0xb2, 0xb3, 0x9d, 0xd4, 0x74, 0x85, 0x25}, /* SHD_R2_UNIT4 */ {0x1a, 0xf1, 0x08, 0x50, 0xfc, 0xe2, 0x1a, 0xe0, 0x2c, 0x66, 0x28}, /* SHD_R2_UNIT5 */ {0x14, 0x01, 0x88, 0x41, 0x4e, 0x32, 0x16, 0xcb, 0x08, 0x87, 0x31}, /* SHD_R2_UNIT6 */ {0x42, 0x99, 0x08, 0x40, 0x0c, 0x72, 0x92, 0xb1, 0x58, 0x86, 0x3b}, /* SHD_R2_UNIT7 */ {0xa8, 0xd9, 0xca, 0x4a, 0x32, 0xe2, 0x91, 0xa0, 0x04, 0x66, 0x36}, /* SHD_R2_UNIT8 */ {0x02, 0xc2, 0x4e, 0x64, 0xbe, 0xd2, 0x94, 0xa9, 0xe0, 0xc5, 0x36}, /* SHD_R2_UNIT9 */ {0xe1, 0x61, 0x91, 0x84, 0xb6, 0x43, 0x9b, 0xcf, 0x9c, 0x66, 0x38}, /* SHD_R2_UNIT10 */ {0xf6, 0xa1, 0x50, 0x97, 0x8e, 0x34, 0x22, 0x04, 0x01, 0x08, 0x40}, /* SHD_R2_UNIT11 */ {0x15, 0x9a, 0x51, 0x92, 0xf2, 0xd4, 0xa5, 0x1d, 0x99, 0xa8, 0x43}, /* SHD_R2_UNIT12 */ {0x21, 0x82, 0x91, 0x92, 0xbe, 0xf4, 0x9e, 0xf3, 0x4c, 0x87, 0x38}, /* SHD_R2_UNIT13 */ }; static const uint8_t g_isx012_shd_b2_data[SHD_B2_DATA_UNIT_NUM] [SHD_B2_DATA_UNIT_SIZE] = { {0xef, 0x39, 0xcf, 0x74, 0x88, 0xb3, 0x1b, 0xdf, 0x20, 0x47, 0x3b}, /* SHD_B2_UNIT0 */ {0xdf, 0x59, 0xcf, 0x77, 0x8c, 0x43, 0x1b, 0xd7, 0xb8, 0x46, 0x37}, /* SHD_B2_UNIT1 */ {0xcc, 0xc1, 0x0e, 0x73, 0x78, 0xa3, 0x99, 0xc1, 0xd0, 0x25, 0x2f}, /* SHD_B2_UNIT2 */ {0x87, 0x09, 0x0d, 0x6c, 0x64, 0x93, 0x99, 0xb6, 0x30, 0xc5, 0x27}, /* SHD_B2_UNIT3 */ {0x3f, 0xb1, 0x0a, 0x5f, 0x2a, 0x93, 0x99, 0xbc, 0x1c, 0x85, 0x24}, /* SHD_B2_UNIT4 */ {0x16, 0xc9, 0x48, 0x4c, 0xb6, 0x92, 0x17, 0xc4, 0x94, 0x85, 0x26}, /* SHD_B2_UNIT5 */ {0x10, 0x09, 0x88, 0x41, 0x3a, 0x52, 0x94, 0xb2, 0x2c, 0xc6, 0x2c}, /* SHD_B2_UNIT6 */ {0x33, 0x79, 0x08, 0x40, 0x08, 0xc2, 0x11, 0xa2, 0x94, 0x65, 0x34}, /* SHD_B2_UNIT7 */ {0x7e, 0x39, 0x4a, 0x48, 0x26, 0x52, 0x91, 0x96, 0x64, 0x05, 0x2f}, /* SHD_B2_UNIT8 */ {0xbf, 0x09, 0x8d, 0x5b, 0x92, 0xa2, 0x93, 0x9d, 0x4c, 0x65, 0x2f}, /* SHD_B2_UNIT9 */ {0x95, 0xf9, 0x0e, 0x73, 0x48, 0x63, 0x98, 0xb9, 0xd8, 0xa5, 0x30}, /* SHD_B2_UNIT10 */ {0xa5, 0xb1, 0x8d, 0x83, 0xf4, 0xa3, 0x1d, 0xe0, 0xd0, 0x06, 0x36}, /* SHD_B2_UNIT11 */ {0xbe, 0xa9, 0x4e, 0x79, 0x50, 0xd4, 0x20, 0xf6, 0x54, 0xa7, 0x38}, /* SHD_B2_UNIT12 */ {0xc5, 0x91, 0xce, 0x7a, 0xf4, 0x03, 0x44, 0x00, 0x60, 0x60, 0x00}, /* SHD_B2_UNIT13 */ }; static const isx012_reg_t g_isx012_shd_thresholds[] = { {SHD_INP_TH_HB_H_R2, 0x1478, 2}, {SHD_INP_TH_HB_L_R2, 0x1380, 2}, {SHD_INP_TH_LB_H_R2, 0x10cc, 2}, {SHD_INP_TH_LB_L_R2, 0x1004, 2}, {SHD_INP_TH_HB_H_RB, 0x10cc, 2}, {SHD_INP_TH_HB_L_RB, 0x1004, 2}, {SHD_INP_TH_LB_H_RB, 0x0000, 2}, {SHD_INP_TH_LB_L_RB, 0x0000, 2}, }; #define ISX012_SHD_THRESHOLDS_NENTRIES ARRAY_NENTRIES(g_isx012_shd_thresholds) static const isx012_reg_t g_isx012_shd_wb[] = { {NORMR, 0x1101, 2}, {NORMB, 0x0f7b, 2}, {AWBPRER, 0x0147, 2}, {AWBPREB, 0x022a, 2}, {SHD_PRER_OFFSET_R2, 0x001b, 2}, {SHD_PRER_OFFSET_RB, 0x000b, 2}, {SHD_PREB_OFFSET_RB, 0x0003, 2}, }; #define ISX012_SHD_WB_NENTRIES ARRAY_NENTRIES(g_isx012_shd_wb) static isx012_conv_v4l2_to_regval_t g_isx012_supported_colorfx[ISX012_MAX_COLOREFFECT + 1] = { {V4L2_COLORFX_NONE, REGVAL_EFFECT_NONE}, {V4L2_COLORFX_BW, REGVAL_EFFECT_MONOTONE}, {V4L2_COLORFX_SEPIA, REGVAL_EFFECT_SEPIA}, {V4L2_COLORFX_NEGATIVE, REGVAL_EFFECT_NEGPOS}, {V4L2_COLORFX_SKETCH, REGVAL_EFFECT_SKETCH}, {V4L2_COLORFX_SOLARIZATION, REGVAL_EFFECT_SOLARIZATION}, {V4L2_COLORFX_PASTEL, REGVAL_EFFECT_PASTEL}, }; static isx012_conv_v4l2_to_regval_t g_isx012_supported_presetwb[ISX012_MAX_PRESETWB + 1] = { {V4L2_WHITE_BALANCE_AUTO, REGVAL_AWB_ATM}, {V4L2_WHITE_BALANCE_INCANDESCENT, REGVAL_AWB_LIGHTBULB}, {V4L2_WHITE_BALANCE_FLUORESCENT, REGVAL_AWB_FLUORESCENTLIGHT}, {V4L2_WHITE_BALANCE_DAYLIGHT, REGVAL_AWB_CLEARWEATHER}, {V4L2_WHITE_BALANCE_CLOUDY, REGVAL_AWB_CLOUDYWEATHER}, {V4L2_WHITE_BALANCE_SHADE, REGVAL_AWB_SHADE}, }; static isx012_conv_v4l2_to_regval_t g_isx012_supported_photometry[ISX012_MAX_PHOTOMETRY + 1] = { {V4L2_EXPOSURE_METERING_AVERAGE, REGVAL_PHOTOMETRY_AVERAGE}, {V4L2_EXPOSURE_METERING_CENTER_WEIGHTED, REGVAL_PHOTOMETRY_CENTERWEIGHT}, {V4L2_EXPOSURE_METERING_SPOT, REGVAL_PHOTOMETRY_SPOT}, {V4L2_EXPOSURE_METERING_MATRIX, REGVAL_PHOTOMETRY_MULTIPATTERN}, }; static isx012_conv_v4l2_to_regval_t g_isx012_supported_iso[ISX012_MAX_ISO + 1] = { {25 * 1000, REGVAL_ISO_25}, {32 * 1000, REGVAL_ISO_32}, {40 * 1000, REGVAL_ISO_40}, {50 * 1000, REGVAL_ISO_50}, {64 * 1000, REGVAL_ISO_64}, {80 * 1000, REGVAL_ISO_80}, {100 * 1000, REGVAL_ISO_100}, {125 * 1000, REGVAL_ISO_125}, {160 * 1000, REGVAL_ISO_160}, {200 * 1000, REGVAL_ISO_200}, {250 * 1000, REGVAL_ISO_250}, {320 * 1000, REGVAL_ISO_320}, {400 * 1000, REGVAL_ISO_400}, {500 * 1000, REGVAL_ISO_500}, {640 * 1000, REGVAL_ISO_640}, {800 * 1000, REGVAL_ISO_800}, {1000 * 1000, REGVAL_ISO_1000}, {1250 * 1000, REGVAL_ISO_1250}, {1600 * 1000, REGVAL_ISO_1600}, }; static struct video_devops_s g_isx012_video_devops = { .open = isx012_open, .close = isx012_close, .do_halfpush = isx012_do_halfpush, .set_buftype = isx012_set_buftype, .set_buf = isx012_set_buf, .cancel_dma = isx012_cancel_dma, .get_range_of_fmt = isx012_get_range_of_fmt, .get_range_of_framesize = isx012_get_range_of_framesize, .try_format = isx012_try_format, .set_format = isx012_set_format, .get_range_of_frameinterval = isx012_get_range_of_frameinterval, .set_frameinterval = isx012_set_frameinterval, .get_range_of_ctrlvalue = isx012_get_range_of_ctrlval, .get_menu_of_ctrlvalue = isx012_get_menu_of_ctrlval, .get_ctrlvalue = isx012_get_ctrlval, .set_ctrlvalue = isx012_set_ctrlval, .refresh = isx012_refresh, }; /**************************************************************************** * Public Data ****************************************************************************/ /**************************************************************************** * Private Functions ****************************************************************************/ static uint16_t isx012_getreg(isx012_dev_t *priv, uint16_t regaddr, uint16_t regsize) { struct i2c_config_s config; volatile uint16_t regval = 0; volatile uint8_t buffer[2]; int ret; /* Set up the I2C configuration */ config.frequency = priv->i2c_freq; config.address = priv->i2c_addr; config.addrlen = 7; buffer[0] = regaddr >> 8; buffer[1] = regaddr & 0xff; /* Write the register address */ ret = i2c_write(priv->i2c, &config, (uint8_t *)buffer, 2); if (ret < 0) { imagererr("i2c_write failed: %d\n", ret); return 0; } /* Restart and read 16bits from the register */ ret = i2c_read(priv->i2c, &config, (uint8_t *)buffer, regsize); if (ret < 0) { imagererr("i2c_read failed: %d\n", ret); return 0; } memcpy((uint8_t *)®val, (uint8_t *)buffer, regsize); return regval; } static int isx012_putreg(isx012_dev_t *priv, uint16_t regaddr, uint16_t regval, uint16_t regsize) { struct i2c_config_s config; volatile uint8_t buffer[4]; int ret; /* Set up the I2C configuration */ config.frequency = priv->i2c_freq; config.address = priv->i2c_addr; config.addrlen = 7; /* Set up for the transfer */ buffer[0] = regaddr >> 8; /* RegAddr Hi */ buffer[1] = regaddr & 0xff; /* RegAddr Low */ memcpy((uint8_t *)&buffer[2], (uint8_t *)®val, regsize); /* And do it */ ret = i2c_write(priv->i2c, &config, (uint8_t *)buffer, regsize + 2); if (ret < 0) { imagererr("i2c_write failed: %d\n", ret); } return ret; } static int isx012_putreglist(isx012_dev_t *priv, FAR const isx012_reg_t *reglist, size_t nentries) { FAR const isx012_reg_t *entry; int ret = OK; for (entry = reglist; nentries > 0; nentries--, entry++) { ret = isx012_putreg(priv, entry->regaddr, entry->regval, entry->regsize); if (ret < 0) { imagererr("isx012_putreg failed: %d\n", ret); return ret; } } return ret; } static int isx012_chk_int_state(isx012_dev_t *priv, uint8_t sts, uint32_t delay_time, uint32_t wait_time, uint32_t timeout) { int ret = 0; volatile uint8_t data; uint32_t time = 0; nxsig_usleep(delay_time * 1000); while (time < timeout) { data = isx012_getreg(priv, INTSTS0, sizeof(data)); data = data & sts; if (data != 0) { ret = isx012_putreg(priv, INTCLR0, data, sizeof(data)); return ret; } nxsig_usleep(wait_time * 1000); time += wait_time; } return ERROR; } static int isx012_replace_fmt_v4l2val_to_regval(uint32_t v4l2val, uint8_t *regval) { if (regval == NULL) { return -EINVAL; } switch (v4l2val) { case V4L2_PIX_FMT_UYVY: *regval = REGVAL_OUTFMT_YUV; break; case V4L2_PIX_FMT_JPEG: *regval = REGVAL_OUTFMT_JPEG; break; case V4L2_PIX_FMT_JPEG_WITH_SUBIMG: *regval = REGVAL_OUTFMT_INTERLEAVE; break; default: /* Unsupported format */ return -EINVAL; } return OK; } static bool is_movie_needed(isx012_modeparam_t *param) { bool need = true; if (param->format == V4L2_PIX_FMT_UYVY) { if (param->fps >= REGVAL_FPSTYPE_30FPS) /* This means fps <= 30 */ { need = false; } } return need; } static int isx012_set_mode_param(isx012_dev_t *priv, enum v4l2_buf_type type, isx012_modeparam_t *param) { int ret = 0; uint8_t format; uint16_t fps_regaddr; uint16_t fmt_regaddr; uint16_t sensmode_regaddr; uint16_t hsize_regaddr; uint16_t vsize_regaddr; uint8_t sensmode; /* Get register address for type */ if (type == V4L2_BUF_TYPE_VIDEO_CAPTURE) { if (is_movie_needed(param)) { if (priv->mode == REGVAL_MODESEL_HREL) { /* In Half release state, * the setting which need movie mode is prohibited. */ return -EPERM; } fps_regaddr = FPSTYPE_MOVIE; fmt_regaddr = OUTFMT_MOVIE; sensmode_regaddr = SENSMODE_MOVIE; hsize_regaddr = HSIZE_MOVIE; vsize_regaddr = VSIZE_MOVIE; } else { fps_regaddr = FPSTYPE_MONI; fmt_regaddr = OUTFMT_MONI; sensmode_regaddr = SENSMODE_MONI; hsize_regaddr = HSIZE_MONI; vsize_regaddr = VSIZE_MONI; } } else { fps_regaddr = FPSTYPE_CAP; fmt_regaddr = OUTFMT_CAP; sensmode_regaddr = SENSMODE_CAP; hsize_regaddr = HSIZE_CAP; vsize_regaddr = VSIZE_CAP; } ret = isx012_putreg(priv, fps_regaddr, param->fps, sizeof(uint8_t)); if (ret < 0) { return ret; } ret = isx012_replace_fmt_v4l2val_to_regval(param->format, &format); if (ret < 0) { return ret; } ret = isx012_putreg(priv, fmt_regaddr, format, sizeof(uint8_t)); if (ret < 0) { return ret; } switch (param->fps) { case REGVAL_FPSTYPE_120FPS: sensmode = REGVAL_SENSMODE_1_8; break; case REGVAL_FPSTYPE_60FPS: sensmode = REGVAL_SENSMODE_1_4; break; case REGVAL_FPSTYPE_30FPS: sensmode = REGVAL_SENSMODE_1_2; break; default: sensmode = REGVAL_SENSMODE_ALLPIX; break; } ret = isx012_putreg(priv, sensmode_regaddr, sensmode, sizeof(uint8_t)); if (ret < 0) { return ret; } ret = isx012_putreg(priv, hsize_regaddr, param->hsize, sizeof(uint16_t)); if (ret < 0) { return ret; } ret = isx012_putreg(priv, vsize_regaddr, param->vsize, sizeof(uint16_t)); if (ret < 0) { return ret; } if (format == REGVAL_OUTFMT_INTERLEAVE) { ret = isx012_putreg(priv, HSIZE_TN, param->int_hsize, sizeof(uint16_t)); if (ret < 0) { return ret; } ret = isx012_putreg(priv, VSIZE_TN, param->int_vsize, sizeof(uint16_t)); if (ret < 0) { return ret; } } return ret; } void isx012_callback(uint8_t code, uint32_t size, uint32_t addr) { enum v4l2_buf_type type; FAR struct isx012_dev_s *priv = &g_isx012_private; if (priv->mode == REGVAL_MODESEL_CAP) { /* ISX012 capture mode => still capture */ type = V4L2_BUF_TYPE_STILL_CAPTURE; } else { /* ISX012 monitor/halfrelease/movie mode => video capture */ type = V4L2_BUF_TYPE_VIDEO_CAPTURE; } video_common_notify_dma_done(code, type, size, priv->video_priv); return; } /**************************************************************************** * isx012_change_camera_mode ****************************************************************************/ static int isx012_change_camera_mode(isx012_dev_t *priv, uint8_t mode) { int ret = 0; uint16_t format_addr; uint8_t format_data; uint32_t vifmode; #ifdef ISX012_FRAME_SKIP_EN uint8_t mask_num; int i; #endif /* ISX012_FRAME_SKIP_EN */ if (priv->state != STATE_ISX012_ACTIVE) { return -EPERM; } switch (mode) { case REGVAL_MODESEL_MON: case REGVAL_MODESEL_HREL: format_addr = OUTFMT_MONI; break; case REGVAL_MODESEL_MOV: format_addr = OUTFMT_MOVIE; break; case REGVAL_MODESEL_CAP: format_addr = OUTFMT_CAP; break; default: return -EPERM; } format_data = isx012_getreg(priv, format_addr, 1); switch (format_data) /* mode parallel */ { case REGVAL_OUTFMT_YUV: vifmode = REGVAL_VIFMODE_YUV_PARALLEL; break; case REGVAL_OUTFMT_JPEG: vifmode = REGVAL_VIFMODE_JPEG_PARALLEL; break; case REGVAL_OUTFMT_INTERLEAVE: vifmode = REGVAL_VIFMODE_INTERLEAVE_PARALLEL; break; case REGVAL_OUTFMT_RGB: vifmode = REGVAL_VIFMODE_RGB_PARALLEL; break; default: vifmode = REGVAL_VIFMODE_YUV_PARALLEL; break; } ret = isx012_putreg(priv, VIFMODE, vifmode, sizeof(vifmode)); if (ret < 0) { return ret; } isx012_putreg(priv, INTCLR0, CM_CHANGED_STS, 1); ret = isx012_putreg(priv, MODESEL, mode, sizeof(mode)); if (ret < 0) { return ret; } /* Wait CM_CHANGED */ ret = isx012_chk_int_state(priv, CM_CHANGED_STS, CAMERA_MODE_DELAY_TIME, CAMERA_MODE_WAIT_TIME, CAMERA_MODE_TIMEOUT); if (ret != 0) { return ret; } #ifdef ISX012_FRAME_SKIP_EN if (mode != REGVAL_MODESEL_HREL) { isx012_putreg(priv, INTCLR0, VINT_STS, 1); mask_num = isx012_getreg(priv, RO_MASK_NUM, sizeof(mask_num)); for (i = 0; i < mask_num; i++) { /* Wait Next VINT */ ret = isx012_chk_int_state(priv, VINT_STS, VINT_DELAY_TIME, VINT_WAIT_TIME, VINT_TIMEOUT); if (ret != 0) { return ret; } } } #endif /* ISX012_FRAME_SKIP_EN */ return OK; } /**************************************************************************** * isx012_change_device_state ****************************************************************************/ static int isx012_change_device_state(isx012_dev_t *priv, isx012_state_t state) { int ret = 0; #ifdef ISX012_FRAME_SKIP_EN int i; uint8_t mute_cnt; #endif /* ISX012_FRAME_SKIP_EN */ if (priv->state == STATE_ISX012_PRESLEEP || priv->state == state) { return -EPERM; } switch (state) { case STATE_ISX012_SLEEP: isx012_putreg(priv, INTCLR0, OM_CHANGED_STS, 1); board_isx012_set_sleep(1); break; case STATE_ISX012_ACTIVE: isx012_putreg(priv, INTCLR0, OM_CHANGED_STS | CM_CHANGED_STS, 1); board_isx012_release_sleep(); break; case STATE_ISX012_PRESLEEP: return -EPERM; default: return -EPERM; } /* Wait OM_CHANGED */ ret = isx012_chk_int_state(priv, OM_CHANGED_STS, DEVICE_STATE_DELAY_TIME, DEVICE_STATE_WAIT_TIME, DEVICE_STATE_TIMEOUT); if (ret != 0) { return ret; } priv->state = state; if (state == STATE_ISX012_ACTIVE) { /* Wait CM_CHANGED -> Monitoring */ ret = isx012_chk_int_state(priv, CM_CHANGED_STS, CAMERA_MODE_DELAY_TIME, CAMERA_MODE_WAIT_TIME, CAMERA_MODE_TIMEOUT); if (ret != 0) { return ret; } #ifdef ISX012_FRAME_SKIP_EN mute_cnt = isx012_getreg(priv, MUTECNT, sizeof(mute_cnt)); isx012_putreg(priv, INTCLR0, VINT_STS, 1); for (i = 0; i < mute_cnt; i++) { /* Wait Next VINT */ ret = isx012_chk_int_state(priv, VINT_STS, VINT_DELAY_TIME, VINT_WAIT_TIME, VINT_TIMEOUT); if (ret != 0) { return ret; } } #endif /* ISX012_FRAME_SKIP_EN */ } priv->mode = REGVAL_MODESEL_MON; return OK; } int init_isx012(FAR struct isx012_dev_s *priv) { int ret; #ifdef ISX012_NOT_USE_NSTBY board_isx012_release_sleep(); board_isx012_release_reset(); nxsig_usleep(6000); #else board_isx012_release_reset(); nxsig_usleep(6000); #endif #ifdef ISX012_CHECK_IN_DETAIL /* check the chip id */ ret = isx012_chipid(priv); if (ret < 0) { imagererr("isx012_chipid failed: %d\n", ret); board_isx012_set_reset(); return ret; } #endif /* Wait OM_CHANGED Power OFF -> PreSleep */ ret = isx012_chk_int_state(priv, OM_CHANGED_STS, DEVICE_STATE_DELAY_TIME, DEVICE_STATE_WAIT_TIME, DEVICE_STATE_TIMEOUT); if (ret != OK) { imagererr("OM_CHANGED_STS(PreSleep) is Not occurred: %d\n", ret); return ret; } priv->state = STATE_ISX012_PRESLEEP; #ifndef ISX012_NOT_USE_NSTBY /* set the isx012 clock */ /* Write INCK_SET register ISX012 change state PreSleep -> Sleep */ ret = isx012_putreglist(priv, g_isx012_presleep, ISX012_PRESLEEP_NENTRIES); if (ret != OK) { imagererr("isx012_putreglist(INCK_SET) failed: %d\n", ret); return ret; } /* Wait OM_CHANGED PreSleep -> Sleep */ ret = isx012_chk_int_state(priv, OM_CHANGED_STS, DEVICE_STATE_DELAY_TIME, DEVICE_STATE_WAIT_TIME, DEVICE_STATE_TIMEOUT); if (ret != OK) { imagererr("OM_CHANGED_STS(Sleep) is Not occurred: %d\n", ret); return ret; } #endif priv->state = STATE_ISX012_SLEEP; priv->i2c_freq = I2CFREQ_FAST; /* initialize the isx012 hardware */ ret = isx012_putreglist(priv, g_isx012_def_init, ISX012_RESET_NENTRIES); if (ret < 0) { imagererr("isx012_putreglist failed: %d\n", ret); board_isx012_set_reset(); return ret; } /* Set shading adjustment */ ret = isx012_set_shd(priv); if (ret < 0) { imagererr("isx012_set_shd failed: %d\n", ret); board_isx012_set_reset(); return ret; } /* monitor mode default format: YUV4:2:2 QVGA */ priv->param.video.fps = REGVAL_FPSTYPE_30FPS; priv->param.video.format = V4L2_PIX_FMT_UYVY; priv->param.video.hsize = VIDEO_HSIZE_QVGA; priv->param.video.vsize = VIDEO_VSIZE_QVGA; priv->param.video.int_hsize = 0; priv->param.video.int_vsize = 0; ret = isx012_set_mode_param(priv, V4L2_BUF_TYPE_VIDEO_CAPTURE, &priv->param.video); if (ret < 0) { board_isx012_set_reset(); return ret; } /* capture mode default format: JPEG FULLHD */ priv->param.still.fps = REGVAL_FPSTYPE_15FPS; priv->param.still.format = V4L2_PIX_FMT_JPEG; priv->param.still.hsize = VIDEO_HSIZE_FULLHD; priv->param.still.vsize = VIDEO_VSIZE_FULLHD; priv->param.still.int_hsize = 0; priv->param.still.int_vsize = 0; ret = isx012_set_mode_param(priv, V4L2_BUF_TYPE_STILL_CAPTURE, &priv->param.still); if (ret < 0) { board_isx012_set_reset(); return ret; } return ret; } static int isx012_open(FAR void *video_private) { FAR struct isx012_dev_s *priv = &g_isx012_private; int ret = 0; ret = board_isx012_power_on(); if (ret < 0) { imagererr("Failed to power on %d\n", ret); return ret; } ret = init_isx012(priv); if (ret < 0) { imagererr("Failed to init_isx012 %d\n", ret); board_isx012_set_reset(); board_isx012_power_off(); return ret; } ret = cxd56_cisifinit(); if (ret < 0) { imagererr("Fail cxd56_cisifinit %d\n", ret); return ret; } /* Save video private information address */ g_isx012_private.video_priv = video_private; return ret; } static int isx012_close(void) { FAR struct isx012_dev_s *priv = &g_isx012_private; g_isx012_private.video_priv = NULL; int ret = 0; if (priv->state == STATE_ISX012_ACTIVE) { board_isx012_set_sleep(1); } board_isx012_set_reset(); ret = board_isx012_power_off(); if (ret < 0) { imagererr("Failed to power off %d\n", ret); return ret; } ret = cxd56_cisifstopcapture(); if (ret < 0) { imagererr("Fail cxd56_cisifstopcapture %d\n", ret); return ret; } ret = cxd56_cisiffinalize(); if (ret < 0) { imagererr("Fail cxd56_cisiffinalize %d\n", ret); return ret; } priv->i2c_freq = I2CFREQ_STANDARD; priv->state = STATE_ISX012_POWEROFF; return ret; } static int isx012_do_halfpush(bool enable) { FAR struct isx012_dev_s *priv = &g_isx012_private; uint8_t mode; int ret = -EPERM; if (enable) { /* state transition : MONITORING -> HALFRELEASE */ if (priv->mode == REGVAL_MODESEL_MON) { mode = REGVAL_MODESEL_HREL; ret = OK; } } else { /* state transition : HALFRELEASE -> MONITORING */ if (priv->mode == REGVAL_MODESEL_HREL) { mode = REGVAL_MODESEL_MON; ret = OK; } } if (ret == OK) { ret = isx012_change_camera_mode(priv, mode); if (ret == OK) { priv->mode = mode; } } return ret; } static int isx012_set_buftype(enum v4l2_buf_type type) { FAR struct isx012_dev_s *priv = &g_isx012_private; uint8_t mode; int ret = OK; if (type == V4L2_BUF_TYPE_VIDEO_CAPTURE) { if (priv->mode == REGVAL_MODESEL_HREL) { /* state transition : HALFRELEASE -> HALFRELEASE */ mode = REGVAL_MODESEL_HREL; } else { /* state transition : CAPTURE -> MONITORING * or MONITORING -> MONITORING */ if (is_movie_needed(&priv->param.video)) { mode = REGVAL_MODESEL_MOV; } else { mode = REGVAL_MODESEL_MON; } } } else { /* state transition : any -> CAPTURE */ mode = REGVAL_MODESEL_CAP; } /* In no active case, activate */ if (priv->state != STATE_ISX012_ACTIVE) { isx012_change_device_state(priv, STATE_ISX012_ACTIVE); } if (mode != priv->mode) { ret = isx012_change_camera_mode(priv, mode); if (ret == OK) { priv->mode = mode; } } return ret; } static int isx012_set_buf(uint32_t bufaddr, uint32_t bufsize) { int ret; FAR struct isx012_dev_s *priv = &g_isx012_private; isx012_modeparam_t *mode_param = NULL; cisif_param_t cis_param = { 0 }; cisif_sarea_t sarea = { 0 }; sarea.strg_addr = (uint8_t *)bufaddr; sarea.strg_size = bufsize; if (priv->dma_state) { ret = cxd56_cisifsetdmabuf(&sarea); } else { if (priv->mode == REGVAL_MODESEL_CAP) { mode_param = &priv->param.still; } else { mode_param = &priv->param.video; } switch (mode_param->format) { case V4L2_PIX_FMT_UYVY: /* Set YUV 4:2:2 information */ cis_param.yuv_param.hsize = mode_param->hsize; cis_param.yuv_param.vsize = mode_param->vsize; break; case V4L2_PIX_FMT_JPEG: /* Set JPEG information */ /* no setting */ break; case V4L2_PIX_FMT_JPEG_WITH_SUBIMG: /* Set JPEG + YUV 4:2:2 information */ cis_param.yuv_param.hsize = mode_param->int_hsize; cis_param.yuv_param.vsize = mode_param->int_vsize; break; default: /* Unsupported format */ return -EINVAL; } cis_param.format = mode_param->format; cis_param.comp_func = isx012_callback; ret = cxd56_cisifstartcapture(&cis_param, &sarea); if (ret != OK) { return ret; } priv->dma_state = true; } return ret; } static int isx012_cancel_dma(void) { int ret; FAR struct isx012_dev_s *priv = &g_isx012_private; ret = cxd56_cisifstopcapture(); if (ret != OK) { return ret; } priv->dma_state = false; return ret; } static int isx012_check_fmt(enum v4l2_buf_type buf_type, uint32_t pixel_format) { switch (buf_type) { case V4L2_BUF_TYPE_VIDEO_CAPTURE: case V4L2_BUF_TYPE_STILL_CAPTURE: if ((pixel_format != V4L2_PIX_FMT_JPEG) && (pixel_format != V4L2_PIX_FMT_JPEG_WITH_SUBIMG) && (pixel_format != V4L2_PIX_FMT_UYVY)) { /* Unsupported format */ return -EINVAL; } break; default: /* Unsupported type */ return -EINVAL; } return OK; } static int isx012_get_range_of_fmt(FAR struct v4l2_fmtdesc *format) { if (format == NULL) { return -EINVAL; } switch (format->type) { case V4L2_BUF_TYPE_VIDEO_CAPTURE: case V4L2_BUF_TYPE_STILL_CAPTURE: switch (format->index) { case 0: /* JPEG */ strncpy(format->description, "JPEG", V4L2_FMT_DSC_MAX); format->pixelformat = V4L2_PIX_FMT_JPEG; break; case 1: /* JPEG + YUV 4:2:2 */ strncpy(format->description, "JPEG + YUV 4:2:2", V4L2_FMT_DSC_MAX); format->pixelformat = V4L2_PIX_FMT_JPEG_WITH_SUBIMG; format->subimg_pixelformat = V4L2_PIX_FMT_UYVY; break; case 2: /* YUV 4:2:2 */ strncpy(format->description, "YUV 4:2:2", V4L2_FMT_DSC_MAX); format->pixelformat = V4L2_PIX_FMT_UYVY; break; default: /* 3, 4, ... */ return -EINVAL; } break; default: /* Unsupported type */ return -EINVAL; } return OK; } static int isx012_get_range_of_framesize( FAR struct v4l2_frmsizeenum *frmsize) { int ret; if (frmsize == NULL) { return -EINVAL; } if (frmsize->index != 0) { return -EINVAL; } ret = isx012_check_fmt(frmsize->buf_type, frmsize->pixel_format); if (ret != OK) { return ret; } switch (frmsize->pixel_format) { case V4L2_PIX_FMT_UYVY: /* YUV 4:2:2 */ frmsize->type = V4L2_FRMSIZE_TYPE_STEPWISE; frmsize->stepwise.min_width = OUT_YUV_HSIZE_MIN; frmsize->stepwise.max_width = OUT_YUV_15FPS_HSIZE_MAX; frmsize->stepwise.step_width = ISX012_SIZE_STEP; frmsize->stepwise.min_height = OUT_YUV_VSIZE_MIN; frmsize->stepwise.max_height = OUT_YUV_15FPS_VSIZE_MAX; frmsize->stepwise.step_height = ISX012_SIZE_STEP; break; case V4L2_PIX_FMT_JPEG: /* JPEG */ frmsize->type = V4L2_FRMSIZE_TYPE_STEPWISE; frmsize->stepwise.min_width = OUT_JPG_HSIZE_MIN; frmsize->stepwise.max_width = OUT_JPG_15FPS_HSIZE_MAX; frmsize->stepwise.step_width = ISX012_SIZE_STEP; frmsize->stepwise.min_height = OUT_JPG_VSIZE_MIN; frmsize->stepwise.max_height = OUT_JPG_15FPS_VSIZE_MAX; frmsize->stepwise.step_height = ISX012_SIZE_STEP; break; case V4L2_PIX_FMT_JPEG_WITH_SUBIMG: /* JPEG + YUV 4:2:2 */ if (frmsize->subimg_pixel_format != V4L2_PIX_FMT_UYVY) { /* Unsupported pixel format */ return -EINVAL; } frmsize->type = V4L2_FRMSIZE_TYPE_STEPWISE; frmsize->stepwise.min_width = OUT_JPG_HSIZE_MIN; frmsize->stepwise.max_width = OUT_JPGINT_15FPS_HSIZE_MAX; frmsize->stepwise.step_width = ISX012_SIZE_STEP; frmsize->stepwise.min_height = OUT_JPG_VSIZE_MIN; frmsize->stepwise.max_height = OUT_JPGINT_15FPS_VSIZE_MAX; frmsize->stepwise.step_height = ISX012_SIZE_STEP; frmsize->subimg_type = V4L2_FRMSIZE_TYPE_STEPWISE; frmsize->subimg.stepwise.min_width = OUT_YUV_HSIZE_MIN; frmsize->subimg.stepwise.max_width = OUT_YUVINT_30FPS_HSIZE_MAX; frmsize->subimg.stepwise.step_width = ISX012_SIZE_STEP; frmsize->subimg.stepwise.min_height = OUT_YUV_VSIZE_MIN; frmsize->subimg.stepwise.max_height = OUT_YUVINT_30FPS_VSIZE_MAX; frmsize->subimg.stepwise.step_height = ISX012_SIZE_STEP; break; default: /* Unsupported pixel format */ return -EINVAL; } return OK; } static int isx012_try_format(FAR struct v4l2_format *format) { int ret; FAR struct v4l2_frmsizeenum support; if (format == NULL) { return -EINVAL; } /* Get supported frame size information */ support.index = 0; support.buf_type = format->type; support.pixel_format = format->fmt.pix.pixelformat; support.subimg_pixel_format = format->fmt.pix.subimg_pixelformat; ret = isx012_get_range_of_framesize(&support); if (ret != OK) { return ret; } CHECK_RANGE(format->fmt.pix.width, support.stepwise.min_width, support.stepwise.max_width, support.stepwise.step_width); CHECK_RANGE(format->fmt.pix.height, support.stepwise.min_height, support.stepwise.max_height, support.stepwise.step_height); if (support.pixel_format == V4L2_PIX_FMT_JPEG_WITH_SUBIMG) { CHECK_RANGE(format->fmt.pix.subimg_width, support.subimg.stepwise.min_width, support.subimg.stepwise.max_width, support.subimg.stepwise.step_width); CHECK_RANGE(format->fmt.pix.subimg_height, support.subimg.stepwise.min_height, support.subimg.stepwise.max_height, support.subimg.stepwise.step_height); } return OK; } static int isx012_set_format(FAR struct v4l2_format *format) { int ret; int8_t max_fps; struct v4l2_frmivalenum frmival; isx012_modeparam_t mode_param; FAR isx012_modeparam_t *current_param; FAR struct isx012_dev_s *priv = &g_isx012_private; ret = isx012_try_format(format); if (ret < 0) { return ret; } frmival.index = 0; frmival.buf_type = format->type; frmival.pixel_format = format->fmt.pix.pixelformat; frmival.width = format->fmt.pix.width; frmival.height = format->fmt.pix.height; frmival.subimg_pixel_format = format->fmt.pix.subimg_pixelformat; frmival.subimg_width = format->fmt.pix.subimg_width; frmival.subimg_height = format->fmt.pix.subimg_height; max_fps = isx012_get_maximum_fps(&frmival); if (max_fps < 0) { return max_fps; } switch (format->type) { case V4L2_BUF_TYPE_VIDEO_CAPTURE: current_param = &priv->param.video; break; case V4L2_BUF_TYPE_STILL_CAPTURE: current_param = &priv->param.still; break; default: return -EINVAL; } memcpy(&mode_param, current_param, sizeof(mode_param)); mode_param.format = format->fmt.pix.pixelformat; mode_param.hsize = format->fmt.pix.width; mode_param.vsize = format->fmt.pix.height; mode_param.int_hsize = format->fmt.pix.subimg_width; mode_param.int_vsize = format->fmt.pix.subimg_height; if (mode_param.fps < max_fps) { mode_param.fps = max_fps; } ret = isx012_set_mode_param(priv, format->type, &mode_param); if (ret != OK) { return ret; } memcpy(current_param, &mode_param, sizeof(mode_param)); return OK; } static int isx012_set_supported_frminterval(uint32_t fps_index, FAR struct v4l2_fract *interval) { switch (fps_index) { case REGVAL_FPSTYPE_120FPS: interval->numerator = 1; interval->denominator = 120; break; case REGVAL_FPSTYPE_60FPS: interval->numerator = 1; interval->denominator = 60; break; case REGVAL_FPSTYPE_30FPS: interval->numerator = 1; interval->denominator = 30; break; case REGVAL_FPSTYPE_15FPS: interval->numerator = 1; interval->denominator = 15; break; case REGVAL_FPSTYPE_7_5FPS: interval->numerator = 2; interval->denominator = 15; break; case REGVAL_FPSTYPE_6FPS: interval->numerator = 1; interval->denominator = 6; break; case REGVAL_FPSTYPE_5FPS: interval->numerator = 1; interval->denominator = 5; break; default: return -EINVAL; } return OK; } static int8_t isx012_get_maximum_fps(FAR struct v4l2_frmivalenum *frmival) { int ret; uint8_t max_fps = REGVAL_FPSTYPE_120FPS; if (frmival == NULL) { return -EINVAL; } ret = isx012_check_fmt(frmival->buf_type, frmival->pixel_format); if (ret != OK) { return ret; } switch (frmival->pixel_format) { case V4L2_PIX_FMT_UYVY: /* YUV 4:2:2 */ if ((frmival->width < OUT_YUV_HSIZE_MIN) || (frmival->height < OUT_YUV_VSIZE_MIN) || (frmival->width > OUT_YUV_15FPS_HSIZE_MAX) || (frmival->height > OUT_YUV_15FPS_VSIZE_MAX)) { /* IN frame size is out of range */ return -EINVAL; } else if ((frmival->width <= OUT_YUV_120FPS_HSIZE_MAX) && (frmival->height <= OUT_YUV_120FPS_VSIZE_MAX)) { /* support 120FPS, 60FPS, 30FPS, 15FPS, 7.5FPS, 6FPS, and 5FPS */ max_fps = REGVAL_FPSTYPE_120FPS; } else { /* support 60FPS, 30FPS, 15FPS, 7.5FPS, 6FPS, and 5FPS */ max_fps = REGVAL_FPSTYPE_60FPS; } break; case V4L2_PIX_FMT_JPEG: /* JPEG */ if ((frmival->width < OUT_JPG_HSIZE_MIN) || (frmival->height < OUT_JPG_VSIZE_MIN) || (frmival->width > OUT_JPG_15FPS_HSIZE_MAX) || (frmival->height > OUT_JPG_15FPS_VSIZE_MAX)) { /* IN frame size is out of range */ return -EINVAL; } else if ((frmival->width <= OUT_JPG_120FPS_HSIZE_MAX) && (frmival->height <= OUT_JPG_120FPS_VSIZE_MAX)) { /* support 120FPS, 60FPS, 30FPS, 15FPS, 7.5FPS, 6FPS, and 5FPS */ max_fps = REGVAL_FPSTYPE_120FPS; } else if ((frmival->width <= OUT_JPG_60FPS_HSIZE_MAX) && (frmival->height <= OUT_JPG_60FPS_VSIZE_MAX)) { /* support 60FPS, 30FPS, 15FPS, 7.5FPS, 6FPS, and 5FPS */ max_fps = REGVAL_FPSTYPE_60FPS; } else if ((frmival->width <= OUT_JPG_30FPS_HSIZE_MAX) && (frmival->height <= OUT_JPG_30FPS_VSIZE_MAX)) { /* support 30FPS, 15FPS, 7.5FPS, 6FPS, and 5FPS */ max_fps = REGVAL_FPSTYPE_30FPS; } else { /* support 15FPS, 7.5FPS, 6FPS, and 5FPS */ max_fps = REGVAL_FPSTYPE_15FPS; } break; case V4L2_PIX_FMT_JPEG_WITH_SUBIMG: /* JPEG + YUV 4:2:2 */ if (frmival->subimg_pixel_format != V4L2_PIX_FMT_UYVY) { /* Unsupported pixel format */ return -EINVAL; } if ((frmival->width < OUT_JPG_HSIZE_MIN) || (frmival->height < OUT_JPG_VSIZE_MIN) || (frmival->width > OUT_JPGINT_15FPS_HSIZE_MAX) || (frmival->height > OUT_JPGINT_15FPS_VSIZE_MAX) || (frmival->subimg_width < OUT_YUV_HSIZE_MIN) || (frmival->subimg_height < OUT_YUV_VSIZE_MIN) || (frmival->subimg_width > OUT_YUVINT_30FPS_HSIZE_MAX) || (frmival->subimg_height > OUT_YUVINT_30FPS_VSIZE_MAX)) { /* IN frame size is out of range */ return -EINVAL; } else if ((frmival->width <= OUT_JPGINT_30FPS_HSIZE_MAX) && (frmival->height <= OUT_JPGINT_30FPS_VSIZE_MAX)) { /* support 30FPS, 15FPS, 7.5FPS, 6FPS, 5FPS */ max_fps = REGVAL_FPSTYPE_30FPS; } else { /* support 15FPS, 7.5FPS, 6FPS, 5FPS */ max_fps = REGVAL_FPSTYPE_15FPS; } break; default: return -EINVAL; } return (int8_t)max_fps; } static int isx012_get_range_of_frameinterval (FAR struct v4l2_frmivalenum *frmival) { int ret; int8_t max_fps; max_fps = isx012_get_maximum_fps(frmival); if (max_fps < 0) { return max_fps; } frmival->type = V4L2_FRMIVAL_TYPE_DISCRETE; ret = isx012_set_supported_frminterval(frmival->index + max_fps, &frmival->discrete); return ret; } static int isx012_change_fraction_to_fps(FAR struct v4l2_fract *interval) { if (interval->denominator == interval->numerator * 120) /* 120FPS */ { return REGVAL_FPSTYPE_120FPS; } else if(interval->denominator == interval->numerator * 60) /* 60FPS */ { return REGVAL_FPSTYPE_60FPS; } else if(interval->denominator == interval->numerator * 30) /* 30FPS */ { return REGVAL_FPSTYPE_30FPS; } else if(interval->denominator == interval->numerator * 15) /* 15FPS */ { return REGVAL_FPSTYPE_15FPS; } else if(interval->denominator * 10 == interval->numerator * 75) /* 7.5FPS */ { return REGVAL_FPSTYPE_7_5FPS; } else if(interval->denominator == interval->numerator * 6) /* 6FPS */ { return REGVAL_FPSTYPE_6FPS; } else if(interval->denominator == interval->numerator * 5) /* 5FPS */ { return REGVAL_FPSTYPE_5FPS; } else { return -EINVAL; } } static int isx012_set_frameinterval(FAR struct v4l2_streamparm *parm) { int ret; int8_t fps; int8_t max_fps; isx012_modeparam_t mode_param; FAR isx012_modeparam_t *current_param; struct v4l2_frmivalenum frmival; FAR struct isx012_dev_s *priv = &g_isx012_private; fps = isx012_change_fraction_to_fps(&parm->parm.capture.timeperframe); if (fps < 0) { return fps; } frmival.buf_type = parm->type; switch (frmival.buf_type) { case V4L2_BUF_TYPE_VIDEO_CAPTURE: current_param = &priv->param.video; break; case V4L2_BUF_TYPE_STILL_CAPTURE: current_param = &priv->param.still; break; default: return -EINVAL; } memcpy(&mode_param, current_param, sizeof(mode_param)); /* Get maximum fps settable value in current image format */ frmival.pixel_format = mode_param.format; frmival.height = mode_param.vsize; frmival.width = mode_param.hsize; frmival.subimg_pixel_format = V4L2_PIX_FMT_UYVY; frmival.subimg_height = mode_param.int_vsize; frmival.subimg_width = mode_param.int_hsize; max_fps = isx012_get_maximum_fps(&frmival); if (max_fps < 0) { return fps; } if (fps < max_fps) { return -EINVAL; } mode_param.fps = fps; ret = isx012_set_mode_param(priv, parm->type, &mode_param); if (ret != OK) { return ret; } memcpy(current_param, &mode_param, sizeof(mode_param)); return OK; } static int isx012_get_range_of_ctrlval(FAR struct v4l2_query_ext_ctrl *range) { if (range == NULL) { return -EINVAL; } switch (range->ctrl_class) { case V4L2_CTRL_CLASS_USER: switch (range->id) { case V4L2_CID_BRIGHTNESS: range->type = ISX012_TYPE_BRIGHTNESS; range->minimum = ISX012_MIN_BRIGHTNESS; range->maximum = ISX012_MAX_BRIGHTNESS; range->step = ISX012_STEP_BRIGHTNESS; range->default_value = ISX012_DEF_BRIGHTNESS; strncpy(range->name, ISX012_NAME_BRIGHTNESS, sizeof(range->name)); break; case V4L2_CID_CONTRAST: range->type = ISX012_TYPE_CONTRAST; range->minimum = ISX012_MIN_CONTRAST; range->maximum = ISX012_MAX_CONTRAST; range->step = ISX012_STEP_CONTRAST; range->default_value = ISX012_DEF_CONTRAST; strncpy(range->name, ISX012_NAME_CONTRAST, sizeof(range->name)); break; case V4L2_CID_SATURATION: range->type = ISX012_TYPE_SATURATION; range->minimum = ISX012_MIN_SATURATION; range->maximum = ISX012_MAX_SATURATION; range->step = ISX012_STEP_SATURATION; range->default_value = ISX012_DEF_SATURATION; strncpy(range->name, ISX012_NAME_SATURATION, sizeof(range->name)); break; case V4L2_CID_HUE: range->type = ISX012_TYPE_HUE; range->minimum = ISX012_MIN_HUE; range->maximum = ISX012_MAX_HUE; range->step = ISX012_STEP_HUE; range->default_value = ISX012_DEF_HUE; strncpy(range->name, ISX012_NAME_HUE, sizeof(range->name)); break; case V4L2_CID_AUTO_WHITE_BALANCE: range->type = ISX012_TYPE_AUTOWB; range->minimum = ISX012_MIN_AUTOWB; range->maximum = ISX012_MAX_AUTOWB; range->step = ISX012_STEP_AUTOWB; range->default_value = ISX012_DEF_AUTOWB; strncpy(range->name, ISX012_NAME_AUTOWB, sizeof(range->name)); break; case V4L2_CID_GAMMA_CURVE: range->type = ISX012_TYPE_GAMMACURVE; range->minimum = ISX012_MIN_GAMMACURVE; range->maximum = ISX012_MAX_GAMMACURVE; range->step = ISX012_STEP_GAMMACURVE; range->default_value = ISX012_DEF_GAMMACURVE; strncpy(range->name, ISX012_NAME_GAMMACURVE, sizeof(range->name)); break; case V4L2_CID_EXPOSURE: range->type = ISX012_TYPE_EXPOSURE; range->minimum = ISX012_MIN_EXPOSURE; range->maximum = ISX012_MAX_EXPOSURE; range->step = ISX012_STEP_EXPOSURE; range->default_value = ISX012_DEF_EXPOSURE; strncpy(range->name, ISX012_NAME_EXPOSURE, sizeof(range->name)); break; case V4L2_CID_HFLIP: range->type = ISX012_TYPE_HFLIP; range->minimum = ISX012_MIN_HFLIP; range->maximum = ISX012_MAX_HFLIP; range->step = ISX012_STEP_HFLIP; range->default_value = ISX012_DEF_HFLIP; strncpy(range->name, ISX012_NAME_HFLIP, sizeof(range->name)); break; case V4L2_CID_VFLIP: range->type = ISX012_TYPE_VFLIP; range->minimum = ISX012_MIN_VFLIP; range->maximum = ISX012_MAX_VFLIP; range->step = ISX012_STEP_VFLIP; range->default_value = ISX012_DEF_VFLIP; strncpy(range->name, ISX012_NAME_VFLIP, sizeof(range->name)); break; case V4L2_CID_HFLIP_STILL: range->type = ISX012_TYPE_HFLIP_STILL; range->minimum = ISX012_MIN_HFLIP_STILL; range->maximum = ISX012_MAX_HFLIP_STILL; range->step = ISX012_STEP_HFLIP_STILL; range->default_value = ISX012_DEF_HFLIP_STILL; strncpy(range->name, ISX012_NAME_HFLIP_STILL, sizeof(range->name)); break; case V4L2_CID_VFLIP_STILL: range->type = ISX012_TYPE_VFLIP_STILL; range->minimum = ISX012_MIN_VFLIP_STILL; range->maximum = ISX012_MAX_VFLIP_STILL; range->step = ISX012_STEP_VFLIP_STILL; range->default_value = ISX012_DEF_VFLIP_STILL; strncpy(range->name, ISX012_NAME_VFLIP_STILL, sizeof(range->name)); break; case V4L2_CID_SHARPNESS: range->type = ISX012_TYPE_SHARPNESS; range->minimum = ISX012_MIN_SHARPNESS; range->maximum = ISX012_MAX_SHARPNESS; range->step = ISX012_STEP_SHARPNESS; range->default_value = ISX012_DEF_SHARPNESS; strncpy(range->name, ISX012_NAME_SHARPNESS, sizeof(range->name)); break; case V4L2_CID_COLOR_KILLER: range->type = ISX012_TYPE_COLORKILLER; range->minimum = ISX012_MIN_COLORKILLER; range->maximum = ISX012_MAX_COLORKILLER; range->step = ISX012_STEP_COLORKILLER; range->default_value = ISX012_DEF_COLORKILLER; strncpy(range->name, ISX012_NAME_COLORKILLER, sizeof(range->name)); break; case V4L2_CID_COLORFX: range->type = ISX012_TYPE_COLOREFFECT; range->minimum = ISX012_MIN_COLOREFFECT; range->maximum = ISX012_MAX_COLOREFFECT; range->step = ISX012_STEP_COLOREFFECT; range->default_value = ISX012_DEF_COLOREFFECT; strncpy(range->name, ISX012_NAME_COLOREFFECT, sizeof(range->name)); break; default: /* Unsupported control id */ return -EINVAL; } break; case V4L2_CTRL_CLASS_CAMERA: switch (range->id) { case V4L2_CID_EXPOSURE_AUTO: range->type = ISX012_TYPE_EXPOSUREAUTO; range->minimum = ISX012_MIN_EXPOSUREAUTO; range->maximum = ISX012_MAX_EXPOSUREAUTO; range->step = ISX012_STEP_EXPOSUREAUTO; range->default_value = ISX012_DEF_EXPOSUREAUTO; strncpy(range->name, ISX012_NAME_EXPOSUREAUTO, sizeof(range->name)); break; case V4L2_CID_EXPOSURE_ABSOLUTE: range->type = ISX012_TYPE_EXPOSURETIME; range->minimum = ISX012_MIN_EXPOSURETIME; range->maximum = ISX012_MAX_EXPOSURETIME; range->step = ISX012_STEP_EXPOSURETIME; range->default_value = ISX012_DEF_EXPOSURETIME; strncpy(range->name, ISX012_NAME_EXPOSURETIME, sizeof(range->name)); break; case V4L2_CID_EXPOSURE_METERING: range->type = ISX012_TYPE_PHOTOMETRY; range->minimum = ISX012_MIN_PHOTOMETRY; range->maximum = ISX012_MAX_PHOTOMETRY; range->step = ISX012_STEP_PHOTOMETRY; range->default_value = ISX012_DEF_PHOTOMETRY; strncpy(range->name, ISX012_NAME_PHOTOMETRY, sizeof(range->name)); break; case V4L2_CID_ZOOM_ABSOLUTE: range->type = ISX012_TYPE_ZOOM; range->minimum = ISX012_MIN_ZOOM; range->maximum = ISX012_MAX_ZOOM; range->step = ISX012_STEP_ZOOM; range->default_value = ISX012_DEF_ZOOM; strncpy(range->name, ISX012_NAME_ZOOM, sizeof(range->name)); break; case V4L2_CID_AUTO_N_PRESET_WHITE_BALANCE: range->type = ISX012_TYPE_PRESETWB; range->minimum = ISX012_MIN_PRESETWB; range->maximum = ISX012_MAX_PRESETWB; range->step = ISX012_STEP_PRESETWB; range->default_value = ISX012_DEF_PRESETWB; strncpy(range->name, ISX012_NAME_PRESETWB, sizeof(range->name)); break; case V4L2_CID_WIDE_DYNAMIC_RANGE: range->type = ISX012_TYPE_YGAMMA; range->minimum = ISX012_MIN_YGAMMA; range->maximum = ISX012_MAX_YGAMMA; range->step = ISX012_STEP_YGAMMA; range->default_value = ISX012_DEF_YGAMMA; strncpy(range->name, ISX012_NAME_YGAMMA, sizeof(range->name)); break; case V4L2_CID_ISO_SENSITIVITY: range->type = ISX012_TYPE_ISO; range->minimum = ISX012_MIN_ISO; range->maximum = ISX012_MAX_ISO; range->step = ISX012_STEP_ISO; range->default_value = ISX012_DEF_ISO; strncpy(range->name, ISX012_NAME_ISO, sizeof(range->name)); break; case V4L2_CID_ISO_SENSITIVITY_AUTO: range->type = ISX012_TYPE_ISOAUTO; range->minimum = ISX012_MIN_ISOAUTO; range->maximum = ISX012_MAX_ISOAUTO; range->step = ISX012_STEP_ISOAUTO; range->default_value = ISX012_DEF_ISOAUTO; strncpy(range->name, ISX012_NAME_ISOAUTO, sizeof(range->name)); break; case V4L2_CID_3A_LOCK: range->type = ISX012_TYPE_3ALOCK; range->minimum = ISX012_MIN_3ALOCK; range->maximum = ISX012_MAX_3ALOCK; range->step = ISX012_STEP_3ALOCK; range->default_value = ISX012_DEF_3ALOCK; strncpy(range->name, ISX012_NAME_3ALOCK, sizeof(range->name)); break; case V4L2_CID_3A_PARAMETER: range->type = V4L2_CTRL_TYPE_U16; range->minimum = 0; range->maximum = 65535; range->step = 1; range->elems = 3; strncpy(range->name, "AWB/AE parameter", sizeof(range->name)); break; case V4L2_CID_3A_STATUS: range->type = V4L2_CTRL_TYPE_INTEGER; range->minimum = 0; range->maximum = 3; range->step = 1; strncpy(range->name, "AWB/AE status", sizeof(range->name)); break; default: /* Unsupported control id */ return -EINVAL; } break; case V4L2_CTRL_CLASS_JPEG: switch (range->id) { case V4L2_CID_JPEG_COMPRESSION_QUALITY: range->type = ISX012_TYPE_JPGQUALITY; range->minimum = ISX012_MIN_JPGQUALITY; range->maximum = ISX012_MAX_JPGQUALITY; range->step = ISX012_STEP_JPGQUALITY; range->default_value = ISX012_DEF_JPGQUALITY; strncpy(range->name, ISX012_NAME_JPGQUALITY, sizeof(range->name)); break; default: /* Unsupported control id */ return -EINVAL; } break; default: /* Unsupported control class */ return -EINVAL; } return OK; } static int isx012_get_menu_of_ctrlval(FAR struct v4l2_querymenu *menu) { if (menu == NULL) { return -EINVAL; } switch (menu->ctrl_class) { case V4L2_CTRL_CLASS_USER: switch (menu->id) { case V4L2_CID_COLORFX: if (menu->index > ISX012_MAX_COLOREFFECT) { return -EINVAL; } menu->value = g_isx012_supported_colorfx[menu->index].v4l2; break; default: /* Unsupported control id */ return -EINVAL; } break; case V4L2_CTRL_CLASS_CAMERA: switch (menu->id) { case V4L2_CID_EXPOSURE_METERING: if (menu->index > ISX012_MAX_PHOTOMETRY) { return -EINVAL; } menu->value = g_isx012_supported_photometry[menu->index].v4l2; break; case V4L2_CID_AUTO_N_PRESET_WHITE_BALANCE: if (menu->index > ISX012_MAX_PRESETWB) { return -EINVAL; } menu->value = g_isx012_supported_presetwb[menu->index].v4l2; break; case V4L2_CID_ISO_SENSITIVITY: if (menu->index > ISX012_MAX_ISO) { return -EINVAL; } menu->value = g_isx012_supported_iso[menu->index].v4l2; break; default: /* Unsupported control id */ return -EINVAL; } break; default: /* Unsupported control class */ return -EINVAL; } return OK; } static int isx012_get_ctrlval(uint16_t ctrl_class, FAR struct v4l2_ext_control *control) { FAR struct isx012_dev_s *priv = &g_isx012_private; int16_t readvalue; uint8_t cnt; uint8_t threea_enable; uint16_t read_src; uint16_t *read_dst; int ret = -EINVAL; if (control == NULL) { return -EINVAL; } switch (ctrl_class) { case V4L2_CTRL_CLASS_USER: switch (control->id) { case V4L2_CID_BRIGHTNESS: control->value = isx012_getreg(priv, ISX012_REG_BRIGHTNESS, ISX012_SIZE_BRIGHTNESS); break; case V4L2_CID_CONTRAST: control->value = isx012_getreg(priv, ISX012_REG_CONTRAST, ISX012_SIZE_CONTRAST); break; case V4L2_CID_SATURATION: control->value = isx012_getreg(priv, ISX012_REG_SATURATION, ISX012_SIZE_SATURATION); break; case V4L2_CID_HUE: control->value = isx012_getreg(priv, ISX012_REG_HUE, ISX012_SIZE_HUE); break; case V4L2_CID_AUTO_WHITE_BALANCE: readvalue = isx012_getreg(priv, ISX012_REG_AUTOWB, ISX012_SIZE_AUTOWB); /* Convert to V4L2 value */ if (readvalue & REGVAL_CPUEXT_BIT_AWBSTOP) { control->value = false; } else { control->value = true; } break; case V4L2_CID_GAMMA_CURVE: if (control->p_u16 == NULL) { return -EINVAL; } read_src = ISX012_REG_GAMMACURVE; read_dst = control->p_u16; for (cnt = 0; cnt < ISX012_ELEMS_GAMMACURVE; cnt++) { *read_dst = isx012_getreg(priv, read_src, ISX012_SIZE_GAMMACURVE); read_src += ISX012_SIZE_GAMMACURVE; read_dst++; } break; case V4L2_CID_EXPOSURE: control->value = isx012_getreg(priv, ISX012_REG_EXPOSURE, ISX012_SIZE_EXPOSURE); break; case V4L2_CID_HFLIP: readvalue = isx012_getreg(priv, ISX012_REG_HFLIP, ISX012_SIZE_HFLIP); if (readvalue & REGVAL_READVECT_BIT_H) { control->value = true; } else { control->value = false; } break; case V4L2_CID_VFLIP: readvalue = isx012_getreg(priv, ISX012_REG_VFLIP, ISX012_SIZE_VFLIP); if (readvalue & REGVAL_READVECT_BIT_V) { control->value = true; } else { control->value = false; } break; case V4L2_CID_HFLIP_STILL: readvalue = isx012_getreg(priv, ISX012_REG_HFLIP_STILL, ISX012_SIZE_HFLIP_STILL); if (readvalue & REGVAL_READVECT_BIT_H) { control->value = true; } else { control->value = false; } break; case V4L2_CID_VFLIP_STILL: readvalue = isx012_getreg(priv, ISX012_REG_VFLIP_STILL, ISX012_SIZE_VFLIP_STILL); if (readvalue & REGVAL_READVECT_BIT_V) { control->value = true; } else { control->value = false; } break; case V4L2_CID_SHARPNESS: control->value = isx012_getreg(priv, ISX012_REG_SHARPNESS, ISX012_SIZE_SHARPNESS); break; case V4L2_CID_COLOR_KILLER: readvalue = isx012_getreg(priv, ISX012_REG_COLORKILLER, ISX012_SIZE_COLORKILLER); if (readvalue == REGVAL_EFFECT_MONOTONE) { control->value = true; } else { control->value = false; } break; case V4L2_CID_COLORFX: readvalue = isx012_getreg(priv, ISX012_REG_COLOREFFECT, ISX012_SIZE_COLOREFFECT); for (cnt = 0; cnt <= ISX012_MAX_COLOREFFECT; cnt++) { if (g_isx012_supported_colorfx[cnt].regval == readvalue) { ret = OK; break; } } if (ret != OK) { return ret; } control->value = g_isx012_supported_colorfx[cnt].v4l2; break; default: /* Unsupported control id */ return -EINVAL; } break; case V4L2_CTRL_CLASS_CAMERA: switch (control->id) { case V4L2_CID_EXPOSURE_AUTO: readvalue = isx012_getreg(priv, ISX012_REG_EXPOSURETIME, ISX012_SIZE_EXPOSURETIME); if (readvalue) { control->value = V4L2_EXPOSURE_MANUAL; } else { control->value = V4L2_EXPOSURE_AUTO; } break; case V4L2_CID_EXPOSURE_ABSOLUTE: control->value = isx012_getreg(priv, ISX012_REG_EXPOSURETIME, ISX012_SIZE_EXPOSURETIME); break; case V4L2_CID_AUTO_N_PRESET_WHITE_BALANCE: readvalue = isx012_getreg(priv, ISX012_REG_PRESETWB, ISX012_SIZE_PRESETWB); for (cnt = 0; cnt <= ISX012_MAX_PRESETWB; cnt++) { if (g_isx012_supported_presetwb[cnt].regval == readvalue) { ret = OK; break; } } if (ret != OK) { return ret; } control->value = g_isx012_supported_presetwb[cnt].v4l2; break; case V4L2_CID_WIDE_DYNAMIC_RANGE: readvalue = isx012_getreg(priv, ISX012_REG_YGAMMA, ISX012_SIZE_YGAMMA); if (readvalue) { control->value = false; } else { control->value = true; } break; case V4L2_CID_ISO_SENSITIVITY: readvalue = isx012_getreg(priv, ISX012_REG_ISO, ISX012_SIZE_ISO); for (cnt = 0; cnt <= ISX012_MAX_ISO; cnt++) { if (g_isx012_supported_iso[cnt].regval == readvalue) { ret = OK; break; } } if (ret != OK) { return ret; } control->value = g_isx012_supported_iso[cnt].v4l2; break; case V4L2_CID_ISO_SENSITIVITY_AUTO: readvalue = isx012_getreg(priv, ISX012_REG_ISOAUTO, ISX012_SIZE_ISOAUTO); if (readvalue == REGVAL_ISO_AUTO) { control->value = V4L2_ISO_SENSITIVITY_AUTO; } else { control->value = V4L2_ISO_SENSITIVITY_MANUAL; } break; case V4L2_CID_EXPOSURE_METERING: readvalue = isx012_getreg(priv, ISX012_REG_PHOTOMETRY, ISX012_SIZE_PHOTOMETRY); for (cnt = 0; cnt <= ISX012_MAX_PHOTOMETRY; cnt++) { if (g_isx012_supported_photometry[cnt].regval == readvalue) { ret = OK; break; } } if (ret != OK) { return ret; } control->value = g_isx012_supported_photometry[cnt].v4l2; break; case V4L2_CID_3A_PARAMETER: if (control->p_u16 == NULL) { return -EINVAL; } /* Get AWB parameter */ control->p_u16[0] = isx012_getreg(priv, RATIO_R, 2); control->p_u16[1] = isx012_getreg(priv, RATIO_B, 2); /* Get AE parameter */ control->p_u16[2] = isx012_getreg(priv, AELEVEL, 2); break; case V4L2_CID_3A_STATUS: /* Initialize returned status */ control->value = V4L2_3A_STATUS_STABLE; /* Get AWB/AE enable or not */ threea_enable = isx012_getreg(priv, CPUEXT, 1); /* Check AWB */ if ((threea_enable & REGVAL_CPUEXT_BIT_AWBSTOP) != REGVAL_CPUEXT_BIT_AWBSTOP) { /* Check AWB status */ readvalue = isx012_getreg(priv, AWBSTS, 1); if (readvalue != REGVAL_AWBSTS_STOP) /* AWB is not stopped */ { control->value |= V4L2_3A_STATUS_AWB_OPERATING; } } /* Check AE */ if ((threea_enable & REGVAL_CPUEXT_BIT_AESTOP) != REGVAL_CPUEXT_BIT_AESTOP) { /* Check AE status */ readvalue = isx012_getreg(priv, AESTS, 1); if (readvalue != REGVAL_AESTS_STOP) /* AE is not stopped */ { control->value |= V4L2_3A_STATUS_AE_OPERATING; } } break; default: /* Unsupported control id */ return -EINVAL; } break; case V4L2_CTRL_CLASS_JPEG: switch (control->id) { case V4L2_CID_JPEG_COMPRESSION_QUALITY: control->value = isx012_getreg(priv, ISX012_REG_JPGQUALITY, ISX012_SIZE_JPGQUALITY); break; default: /* Unsupported control id */ return -EINVAL; } break; default: /* Unsupported control class */ return -EINVAL; } return OK; } static int isx012_set_ctrlval(uint16_t ctrl_class, FAR struct v4l2_ext_control *control) { FAR struct isx012_dev_s *priv = &g_isx012_private; int ret = -EINVAL; uint8_t cnt; uint16_t *write_src; uint16_t write_dst; uint16_t regval; uint16_t exposure_time_lsb; uint16_t exposure_time_msb; if (control == NULL) { return -EINVAL; } switch (ctrl_class) { case V4L2_CTRL_CLASS_USER: switch (control->id) { case V4L2_CID_BRIGHTNESS: CHECK_RANGE(control->value, ISX012_MIN_BRIGHTNESS, ISX012_MAX_BRIGHTNESS, ISX012_STEP_BRIGHTNESS); ret = isx012_putreg(priv, ISX012_REG_BRIGHTNESS, control->value, ISX012_SIZE_BRIGHTNESS); break; case V4L2_CID_CONTRAST: CHECK_RANGE(control->value, ISX012_MIN_CONTRAST, ISX012_MAX_CONTRAST, ISX012_STEP_CONTRAST); ret = isx012_putreg(priv, ISX012_REG_CONTRAST, control->value, ISX012_SIZE_CONTRAST); break; case V4L2_CID_SATURATION: CHECK_RANGE(control->value, ISX012_MIN_SATURATION, ISX012_MAX_SATURATION, ISX012_STEP_SATURATION); ret = isx012_putreg(priv, ISX012_REG_SATURATION, control->value, ISX012_SIZE_SATURATION); break; case V4L2_CID_HUE: CHECK_RANGE(control->value, ISX012_MIN_HUE, ISX012_MAX_HUE, ISX012_STEP_HUE); ret = isx012_putreg(priv, ISX012_REG_HUE, control->value, ISX012_SIZE_HUE); break; case V4L2_CID_AUTO_WHITE_BALANCE: CHECK_RANGE(control->value, ISX012_MIN_AUTOWB, ISX012_MAX_AUTOWB, ISX012_STEP_AUTOWB); regval = isx012_getreg(priv, ISX012_REG_AUTOWB, ISX012_SIZE_AUTOWB); if (control->value) { /* Because true means setting auto white balance * turn off the stop bit */ regval &= ~REGVAL_CPUEXT_BIT_AWBSTOP; } else { /* Because false means stopping auto white balance, * turn on the stop bit. */ regval |= REGVAL_CPUEXT_BIT_AWBSTOP; } ret = isx012_putreg(priv, ISX012_REG_AUTOWB, regval, ISX012_SIZE_AUTOWB); break; case V4L2_CID_GAMMA_CURVE: if (control->p_u16 == NULL) { return -EINVAL; } write_src = control->p_u16; write_dst = ISX012_REG_GAMMACURVE; for (cnt = 0; cnt < ISX012_ELEMS_GAMMACURVE; cnt++) { CHECK_RANGE(*write_src, ISX012_MIN_GAMMACURVE, ISX012_MAX_GAMMACURVE, ISX012_STEP_GAMMACURVE); ret = isx012_putreg(priv, write_dst, *write_src, ISX012_SIZE_GAMMACURVE); write_src++; write_dst += ISX012_SIZE_GAMMACURVE; } break; case V4L2_CID_EXPOSURE: CHECK_RANGE(control->value, ISX012_MIN_EXPOSURE, ISX012_MAX_EXPOSURE, ISX012_STEP_EXPOSURE); ret = isx012_putreg(priv, ISX012_REG_EXPOSURE, control->value, ISX012_SIZE_EXPOSURE); break; case V4L2_CID_HFLIP: CHECK_RANGE(control->value, ISX012_MIN_HFLIP, ISX012_MAX_HFLIP, ISX012_STEP_HFLIP); regval = isx012_getreg(priv, ISX012_REG_HFLIP, ISX012_SIZE_HFLIP); if (control->value) { regval |= REGVAL_READVECT_BIT_H; } else { regval &= ~REGVAL_READVECT_BIT_H; } ret = isx012_putreg(priv, ISX012_REG_HFLIP, regval, ISX012_SIZE_HFLIP); break; case V4L2_CID_VFLIP: CHECK_RANGE(control->value, ISX012_MIN_VFLIP, ISX012_MAX_VFLIP, ISX012_STEP_VFLIP); regval = isx012_getreg(priv, ISX012_REG_VFLIP, ISX012_SIZE_VFLIP); if (control->value) { regval |= REGVAL_READVECT_BIT_V; } else { regval &= ~REGVAL_READVECT_BIT_V; } ret = isx012_putreg(priv, ISX012_REG_VFLIP, regval, ISX012_SIZE_VFLIP); break; case V4L2_CID_HFLIP_STILL: CHECK_RANGE(control->value, ISX012_MIN_HFLIP_STILL, ISX012_MAX_HFLIP_STILL, ISX012_STEP_HFLIP_STILL); regval = isx012_getreg(priv, ISX012_REG_HFLIP_STILL, ISX012_SIZE_HFLIP_STILL); if (control->value) { regval |= REGVAL_READVECT_BIT_H; } else { regval &= ~REGVAL_READVECT_BIT_H; } ret = isx012_putreg(priv, ISX012_REG_HFLIP_STILL, regval, ISX012_SIZE_HFLIP_STILL); break; case V4L2_CID_VFLIP_STILL: CHECK_RANGE(control->value, ISX012_MIN_VFLIP_STILL, ISX012_MAX_VFLIP_STILL, ISX012_STEP_VFLIP_STILL); regval = isx012_getreg(priv, ISX012_REG_VFLIP_STILL, ISX012_SIZE_VFLIP_STILL); if (control->value) { regval |= REGVAL_READVECT_BIT_V; } else { regval &= ~REGVAL_READVECT_BIT_V; } ret = isx012_putreg(priv, ISX012_REG_VFLIP_STILL, regval, ISX012_SIZE_VFLIP_STILL); break; case V4L2_CID_SHARPNESS: CHECK_RANGE(control->value, ISX012_MIN_SHARPNESS, ISX012_MAX_SHARPNESS, ISX012_STEP_SHARPNESS); ret = isx012_putreg(priv, ISX012_REG_SHARPNESS, control->value, ISX012_SIZE_SHARPNESS); break; case V4L2_CID_COLOR_KILLER: CHECK_RANGE(control->value, ISX012_MIN_COLORKILLER, ISX012_MAX_COLORKILLER, ISX012_STEP_COLORKILLER); if (control->value) { regval = REGVAL_EFFECT_MONOTONE; } else { regval = REGVAL_EFFECT_NONE; } ret = isx012_putreg(priv, ISX012_REG_COLORKILLER, regval, ISX012_SIZE_COLORKILLER); break; case V4L2_CID_COLORFX: for (cnt = 0; cnt <= ISX012_MAX_COLOREFFECT; cnt++) { if (g_isx012_supported_colorfx[cnt].v4l2 == control->value) { ret = OK; break; } } if (ret != OK) { return ret; } ret = isx012_putreg(priv, ISX012_REG_COLOREFFECT, g_isx012_supported_colorfx[cnt].regval, ISX012_SIZE_COLOREFFECT); break; default: /* Unsupported control id */ return -EINVAL; } break; case V4L2_CTRL_CLASS_CAMERA: switch (control->id) { case V4L2_CID_EXPOSURE_AUTO: CHECK_RANGE(control->value, ISX012_MIN_EXPOSUREAUTO, ISX012_MAX_EXPOSUREAUTO, ISX012_STEP_EXPOSUREAUTO); if (control->value == V4L2_EXPOSURE_AUTO) { /* Register is the same as V4L2_CID_EXPOSURE_ABSOLUTE. * If this register value = REGVAL_EXPOSURETIME_AUTO(=0), * it means auto. Otherwise, it means manual. */ ret = isx012_putreg(priv, ISX012_REG_EXPOSURETIME, REGVAL_EXPOSURETIME_AUTO, ISX012_SIZE_EXPOSURETIME); } else { /* In manual case, read current value of register which * value adjusted automatically by ISX012 HW is set to. * It has 32bits length which is composed of LSB 16bits * and MSB 16bits. */ exposure_time_lsb = isx012_getreg (priv, ISX012_REG_EXPOSUREAUTOVALUE_LSB, ISX012_SIZE_EXPOSUREAUTOVALUE); exposure_time_msb = isx012_getreg (priv, ISX012_REG_EXPOSUREAUTOVALUE_MSB, ISX012_SIZE_EXPOSUREAUTOVALUE); /* Register value adjusted automatically by ISX012 HW * has the different unit from manual value register. * automatic value register : 1 microsec unit * manual value register : 100 microsec unit */ regval = (uint16_t)(((exposure_time_msb << 16) | exposure_time_lsb) / ISX012_UNIT_EXPOSURETIME_US); ret = isx012_putreg(priv, ISX012_REG_EXPOSURETIME, regval, ISX012_SIZE_EXPOSURETIME); } break; case V4L2_CID_EXPOSURE_ABSOLUTE: CHECK_RANGE(control->value, ISX012_MIN_EXPOSURETIME, ISX012_MAX_EXPOSURETIME, ISX012_STEP_EXPOSURETIME); ret = isx012_putreg(priv, ISX012_REG_EXPOSURETIME, control->value, ISX012_SIZE_EXPOSURETIME); break; case V4L2_CID_WIDE_DYNAMIC_RANGE: CHECK_RANGE(control->value, ISX012_MIN_YGAMMA, ISX012_MAX_YGAMMA, ISX012_STEP_YGAMMA); if (control->value) { regval = REGVAL_YGAMMA_AUTO; } else { regval = REGVAL_YGAMMA_OFF; } ret = isx012_putreg(priv, ISX012_REG_YGAMMA, regval, ISX012_SIZE_YGAMMA); break; case V4L2_CID_ISO_SENSITIVITY: for (cnt = 0; cnt <= ISX012_MAX_ISO; cnt++) { if (g_isx012_supported_iso[cnt].v4l2 == control->value) { ret = OK; break; } } if (ret != OK) { return ret; } ret = isx012_putreg(priv, ISX012_REG_ISO, g_isx012_supported_iso[cnt].regval, ISX012_SIZE_ISO); break; case V4L2_CID_ISO_SENSITIVITY_AUTO: CHECK_RANGE(control->value, ISX012_MIN_ISOAUTO, ISX012_MAX_ISOAUTO, ISX012_STEP_ISOAUTO); if (control->value == V4L2_ISO_SENSITIVITY_AUTO) { ret = isx012_putreg(priv, ISX012_REG_ISOAUTO, REGVAL_ISO_AUTO, ISX012_SIZE_ISOAUTO); } else { /* In manual case, read auto adjust value and set it */ regval = isx012_getreg(priv, ISX012_REG_ISOAUTOVALUE, ISX012_SIZE_ISOAUTOVALUE); ret = isx012_putreg(priv, ISX012_REG_ISO, regval, ISX012_SIZE_ISO); } break; case V4L2_CID_EXPOSURE_METERING: for (cnt = 0; cnt <= ISX012_MAX_PHOTOMETRY; cnt++) { if (g_isx012_supported_photometry[cnt].v4l2 == control->value) { ret = OK; break; } } if (ret != OK) { return ret; } ret = isx012_putreg(priv, ISX012_REG_PHOTOMETRY, g_isx012_supported_photometry[cnt].regval, ISX012_SIZE_PHOTOMETRY); break; case V4L2_CID_AUTO_N_PRESET_WHITE_BALANCE: for (cnt = 0; cnt <= ISX012_MAX_PRESETWB; cnt++) { if (g_isx012_supported_presetwb[cnt].v4l2 == control->value) { ret = OK; break; } } if (ret != OK) { return ret; } ret = isx012_putreg(priv, ISX012_REG_PRESETWB, g_isx012_supported_presetwb[cnt].regval, ISX012_SIZE_PRESETWB); break; case V4L2_CID_3A_LOCK: CHECK_RANGE(control->value, ISX012_MIN_3ALOCK, ISX012_MAX_3ALOCK, ISX012_STEP_3ALOCK); regval = 0; if ((control->value & V4L2_LOCK_EXPOSURE) == V4L2_LOCK_EXPOSURE) { regval |= REGVAL_CPUEXT_BIT_AESTOP; } if ((control->value & V4L2_LOCK_WHITE_BALANCE) == V4L2_LOCK_WHITE_BALANCE) { regval |= REGVAL_CPUEXT_BIT_AWBSTOP; } ret = isx012_putreg(priv, ISX012_REG_3ALOCK, regval, ISX012_SIZE_3ALOCK); break; case V4L2_CID_3A_PARAMETER: /* AWB parameter : red */ ret = isx012_putreg(priv, INIT_CONT_INR, control->p_u16[0], 2); ret = isx012_putreg(priv, INIT_CONT_OUTR, control->p_u16[0], 2); /* AWB parameter : blue */ ret = isx012_putreg(priv, INIT_CONT_INB, control->p_u16[1], 2); ret = isx012_putreg(priv, INIT_CONT_OUTB, control->p_u16[1], 2); /* AE parameter */ ret = isx012_putreg(priv, AE_START_LEVEL, control->p_u16[2], 2); break; default: /* Unsupported control id */ return -EINVAL; } break; case V4L2_CTRL_CLASS_JPEG: switch (control->id) { case V4L2_CID_JPEG_COMPRESSION_QUALITY: CHECK_RANGE(control->value, ISX012_MIN_JPGQUALITY, ISX012_MAX_JPGQUALITY, ISX012_STEP_JPGQUALITY); ret = isx012_putreg(priv, ISX012_REG_JPGQUALITY, control->value, ISX012_SIZE_JPGQUALITY); break; default: /* Unsupported control id */ return -EINVAL; } break; default: /* Unsupported control class */ return -EINVAL; } return ret; } static int isx012_refresh(void) { int ret = 0; uint8_t mask_num; int i; FAR struct isx012_dev_s *priv = &g_isx012_private; if (priv->state != STATE_ISX012_ACTIVE) { /* In inactive state, setting is reflected in activated timing */ return OK; } if (priv->mode != REGVAL_MODESEL_MON) { return -EPERM; } /* Set MONI_REFRESH */ isx012_putreg(priv, INTCLR0, CM_CHANGED_STS, 1); ret = isx012_putreg(priv, MONI_REFRESH, 1, 1); if (ret < 0) { return ret; } /* Wait CM_CHANGED */ ret = isx012_chk_int_state(priv, CM_CHANGED_STS, CAMERA_MODE_DELAY_TIME, CAMERA_MODE_WAIT_TIME, CAMERA_MODE_TIMEOUT); if (ret != 0) { return ret; } /* Invalid frame skip */ isx012_putreg(priv, INTCLR0, VINT_STS, 1); mask_num = isx012_getreg(priv, RO_MASK_NUM, sizeof(mask_num)); for (i = 0; i < mask_num; i++) { /* Wait Next VINT */ ret = isx012_chk_int_state(priv, VINT_STS, VINT_DELAY_TIME, VINT_WAIT_TIME, VINT_TIMEOUT); if (ret != 0) { return ret; } } return OK; } static int isx012_set_shd(FAR isx012_dev_t *priv) { int ret; int unit_cnt; int size_cnt; /* At first, disable CXC and SHD */ ret = isx012_putreg(priv, SHD_EN, 0x50, 1); if (ret < 0) { imagererr("isx012_putreg(disable CXC/SHD) failed: %d\n", ret); return ret; } /* Set CXC Validity */ ret = isx012_putreg(priv, CXC_VALID, 0x8282, 2); if (ret < 0) { imagererr("isx012_putreg(CXC_VALID) failed: %d\n", ret); return ret; } /* Set CXC R Gb data */ for (unit_cnt = 0; unit_cnt < CXC_RGB_DATA_UNIT_NUM; unit_cnt++) { for (size_cnt = 0; size_cnt < CXC_RGB_DATA_UNIT_SIZE; size_cnt++) { ret = isx012_putreg(priv, CXC_RGB_UNIT(unit_cnt, size_cnt), g_isx012_cxc_rgb_data[unit_cnt][size_cnt], 1); if (ret < 0) { imagererr("isx012_putreg(CXC R Gb) failed: %d\n", ret); return ret; } } } /* Set CXC G Rb data */ for (unit_cnt = 0; unit_cnt < CXC_GRB_DATA_UNIT_NUM; unit_cnt++) { for (size_cnt = 0; size_cnt < CXC_GRB_DATA_UNIT_SIZE; size_cnt++) { ret = isx012_putreg(priv, CXC_GRB_UNIT(unit_cnt, size_cnt), g_isx012_cxc_grb_data[unit_cnt][size_cnt], 1); if (ret < 0) { imagererr("isx012_putreg(CXC G Rb) failed: %d\n", ret); return ret; } } } /* Set SHD Validity */ ret = isx012_putreg(priv, SHD_VALID, 0x9191, 2); if (ret < 0) { imagererr("isx012_putreg(SHD_VALID) failed: %d\n", ret); return ret; } /* Set SHD R Gb data */ for (unit_cnt = 0; unit_cnt < SHD_RGB_DATA_UNIT_NUM; unit_cnt++) { for (size_cnt = 0; size_cnt < SHD_RGB_DATA_UNIT_SIZE; size_cnt++) { ret = isx012_putreg(priv, SHD_RGB_UNIT(unit_cnt, size_cnt), g_isx012_shd_rgb_data[unit_cnt][size_cnt], 1); if (ret < 0) { imagererr("isx012_putreg(SHD R Gb) failed: %d\n", ret); return ret; } } } /* Set SHD G Rb data */ for (unit_cnt = 0; unit_cnt < SHD_GRB_DATA_UNIT_NUM; unit_cnt++) { for (size_cnt = 0; size_cnt < SHD_GRB_DATA_UNIT_SIZE; size_cnt++) { ret = isx012_putreg(priv, SHD_GRB_UNIT(unit_cnt, size_cnt), g_isx012_shd_grb_data[unit_cnt][size_cnt], 1); if (ret < 0) { imagererr("isx012_putreg(SHD G Rb) failed: %d\n", ret); return ret; } } } /* Set SHD R1 data */ for (unit_cnt = 0; unit_cnt < SHD_R1_DATA_UNIT_NUM; unit_cnt++) { for (size_cnt = 0; size_cnt < SHD_R1_DATA_UNIT_SIZE; size_cnt++) { ret = isx012_putreg(priv, SHD_R1_UNIT(unit_cnt, size_cnt), g_isx012_shd_r1_data[unit_cnt][size_cnt], 1); if (ret < 0) { imagererr("isx012_putreg(SHD R1) failed: %d\n", ret); return ret; } } } /* Set SHD R2 data */ for (unit_cnt = 0; unit_cnt < SHD_R2_DATA_UNIT_NUM; unit_cnt++) { for (size_cnt = 0; size_cnt < SHD_R2_DATA_UNIT_SIZE; size_cnt++) { ret = isx012_putreg(priv, SHD_R2_UNIT(unit_cnt, size_cnt), g_isx012_shd_r2_data[unit_cnt][size_cnt], 1); if (ret < 0) { imagererr("isx012_putreg(SHD R2) failed: %d\n", ret); return ret; } } } /* Set SHD B2 data */ for (unit_cnt = 0; unit_cnt < SHD_B2_DATA_UNIT_NUM; unit_cnt++) { for (size_cnt = 0; size_cnt < SHD_B2_DATA_UNIT_SIZE; size_cnt++) { ret = isx012_putreg(priv, SHD_B2_UNIT(unit_cnt, size_cnt), g_isx012_shd_b2_data[unit_cnt][size_cnt], 1); if (ret < 0) { imagererr("isx012_putreg(SHD B2) failed: %d\n", ret); return ret; } } } /* Set SHD thresholds data */ ret = isx012_putreglist(priv, g_isx012_shd_thresholds, ISX012_SHD_THRESHOLDS_NENTRIES); if (ret < 0) { imagererr("isx012_putreglist failed(SHD thresholds): %d\n", ret); board_isx012_set_reset(); return ret; } /* Set SHD white balance data */ ret = isx012_putreglist(priv, g_isx012_shd_wb, ISX012_SHD_WB_NENTRIES); if (ret < 0) { imagererr("isx012_putreglist(SHD white balance) failed: %d\n", ret); board_isx012_set_reset(); return ret; } /* Enable CXC and SHD */ ret = isx012_putreg(priv, SHD_EN, 0x57, 1); if (ret < 0) { imagererr("isx012_putreg(enable CXC/SHD) failed: %d\n", ret); return ret; } return OK; } /**************************************************************************** * Public Functions ****************************************************************************/ int isx012_register(FAR struct i2c_master_s *i2c) { FAR struct isx012_dev_s *priv = &g_isx012_private; /* Save i2c information */ priv->i2c = i2c; priv->i2c_addr = ISX012_I2C_SLV_ADDR; priv->i2c_freq = I2CFREQ_STANDARD; /* Initialize other information */ priv->state = STATE_ISX012_POWEROFF; return OK; } int isx012_unregister(void) { /* no procedure */ return OK; } FAR struct video_devops_s *isx012_initialize(void) { /* return address of video operations variable */ return &g_isx012_video_devops; } int isx012_uninitialize(void) { /* No procedure */ return OK; }