mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2024-12-22 18:44:44 +08:00
media: atomisp: ov2680: Drop custom ATOMISP_IOC_S_EXPOSURE support
Exposure and gain control should use standard v4l2 controls, not a custom ioctl. The next patch in this series will re-add support as standard controls, this is split into 2 patches for easier reviewing. Reviewed-by: Andy Shevchenko <andy@kernel.org> Signed-off-by: Hans de Goede <hdegoede@redhat.com> Signed-off-by: Mauro Carvalho Chehab <mchehab@kernel.org>
This commit is contained in:
parent
91caf79758
commit
8eb47aa3a1
@ -60,181 +60,6 @@ static int ov2680_write_reg_array(struct i2c_client *client,
|
||||
return 0;
|
||||
}
|
||||
|
||||
static long __ov2680_set_exposure(struct v4l2_subdev *sd, int coarse_itg,
|
||||
int gain, int digitgain)
|
||||
|
||||
{
|
||||
struct i2c_client *client = v4l2_get_subdevdata(sd);
|
||||
struct ov2680_device *dev = to_ov2680_sensor(sd);
|
||||
u16 vts;
|
||||
int ret, exp_val;
|
||||
|
||||
dev_dbg(&client->dev,
|
||||
"+++++++__ov2680_set_exposure coarse_itg %d, gain %d, digitgain %d++\n",
|
||||
coarse_itg, gain, digitgain);
|
||||
|
||||
vts = dev->res->lines_per_frame;
|
||||
|
||||
/* group hold */
|
||||
ret = ov_write_reg8(client, OV2680_GROUP_ACCESS, 0x00);
|
||||
if (ret) {
|
||||
dev_err(&client->dev, "%s: write 0x%02x: error, aborted\n",
|
||||
__func__, OV2680_GROUP_ACCESS);
|
||||
return ret;
|
||||
}
|
||||
|
||||
/* Increase the VTS to match exposure + MARGIN */
|
||||
if (coarse_itg > vts - OV2680_INTEGRATION_TIME_MARGIN)
|
||||
vts = (u16)coarse_itg + OV2680_INTEGRATION_TIME_MARGIN;
|
||||
|
||||
ret = ov_write_reg16(client, OV2680_TIMING_VTS_H, vts);
|
||||
if (ret) {
|
||||
dev_err(&client->dev, "%s: write 0x%02x: error, aborted\n",
|
||||
__func__, OV2680_TIMING_VTS_H);
|
||||
return ret;
|
||||
}
|
||||
|
||||
/* set exposure */
|
||||
|
||||
/* Lower four bit should be 0*/
|
||||
exp_val = coarse_itg << 4;
|
||||
ret = ov_write_reg8(client, OV2680_EXPOSURE_L, exp_val & 0xFF);
|
||||
if (ret) {
|
||||
dev_err(&client->dev, "%s: write 0x%02x: error, aborted\n",
|
||||
__func__, OV2680_EXPOSURE_L);
|
||||
return ret;
|
||||
}
|
||||
|
||||
ret = ov_write_reg8(client, OV2680_EXPOSURE_M, (exp_val >> 8) & 0xFF);
|
||||
if (ret) {
|
||||
dev_err(&client->dev, "%s: write 0x%02x: error, aborted\n",
|
||||
__func__, OV2680_EXPOSURE_M);
|
||||
return ret;
|
||||
}
|
||||
|
||||
ret = ov_write_reg8(client, OV2680_EXPOSURE_H, (exp_val >> 16) & 0x0F);
|
||||
if (ret) {
|
||||
dev_err(&client->dev, "%s: write 0x%02x: error, aborted\n",
|
||||
__func__, OV2680_EXPOSURE_H);
|
||||
return ret;
|
||||
}
|
||||
|
||||
/* Analog gain */
|
||||
ret = ov_write_reg16(client, OV2680_AGC_H, gain);
|
||||
if (ret) {
|
||||
dev_err(&client->dev, "%s: write 0x%02x: error, aborted\n",
|
||||
__func__, OV2680_AGC_H);
|
||||
return ret;
|
||||
}
|
||||
/* Digital gain */
|
||||
if (digitgain) {
|
||||
ret = ov_write_reg16(client, OV2680_MWB_RED_GAIN_H, digitgain);
|
||||
if (ret) {
|
||||
dev_err(&client->dev,
|
||||
"%s: write 0x%02x: error, aborted\n",
|
||||
__func__, OV2680_MWB_RED_GAIN_H);
|
||||
return ret;
|
||||
}
|
||||
|
||||
ret = ov_write_reg16(client, OV2680_MWB_GREEN_GAIN_H, digitgain);
|
||||
if (ret) {
|
||||
dev_err(&client->dev,
|
||||
"%s: write 0x%02x: error, aborted\n",
|
||||
__func__, OV2680_MWB_RED_GAIN_H);
|
||||
return ret;
|
||||
}
|
||||
|
||||
ret = ov_write_reg16(client, OV2680_MWB_BLUE_GAIN_H, digitgain);
|
||||
if (ret) {
|
||||
dev_err(&client->dev,
|
||||
"%s: write 0x%02x: error, aborted\n",
|
||||
__func__, OV2680_MWB_RED_GAIN_H);
|
||||
return ret;
|
||||
}
|
||||
}
|
||||
|
||||
/* End group */
|
||||
ret = ov_write_reg8(client, OV2680_GROUP_ACCESS, 0x10);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
/* Delay launch group */
|
||||
ret = ov_write_reg8(client, OV2680_GROUP_ACCESS, 0xa0);
|
||||
if (ret)
|
||||
return ret;
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int ov2680_set_exposure(struct v4l2_subdev *sd, int exposure,
|
||||
int gain, int digitgain)
|
||||
{
|
||||
struct ov2680_device *dev = to_ov2680_sensor(sd);
|
||||
int ret = 0;
|
||||
|
||||
mutex_lock(&dev->input_lock);
|
||||
|
||||
dev->exposure = exposure;
|
||||
dev->gain = gain;
|
||||
dev->digitgain = digitgain;
|
||||
|
||||
if (dev->power_on)
|
||||
ret = __ov2680_set_exposure(sd, exposure, gain, digitgain);
|
||||
|
||||
mutex_unlock(&dev->input_lock);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static long ov2680_s_exposure(struct v4l2_subdev *sd,
|
||||
struct atomisp_exposure *exposure)
|
||||
{
|
||||
u16 coarse_itg = exposure->integration_time[0];
|
||||
u16 analog_gain = exposure->gain[0];
|
||||
u16 digital_gain = exposure->gain[1];
|
||||
|
||||
/* we should not accept the invalid value below */
|
||||
if (analog_gain == 0) {
|
||||
struct i2c_client *client = v4l2_get_subdevdata(sd);
|
||||
|
||||
v4l2_err(client, "%s: invalid value\n", __func__);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
return ov2680_set_exposure(sd, coarse_itg, analog_gain, digital_gain);
|
||||
}
|
||||
|
||||
static long ov2680_ioctl(struct v4l2_subdev *sd, unsigned int cmd, void *arg)
|
||||
{
|
||||
switch (cmd) {
|
||||
case ATOMISP_IOC_S_EXPOSURE:
|
||||
return ov2680_s_exposure(sd, arg);
|
||||
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* This returns the exposure time being used. This should only be used
|
||||
* for filling in EXIF data, not for actual image processing.
|
||||
*/
|
||||
static int ov2680_q_exposure(struct v4l2_subdev *sd, s32 *value)
|
||||
{
|
||||
struct i2c_client *client = v4l2_get_subdevdata(sd);
|
||||
u32 reg_val;
|
||||
int ret;
|
||||
|
||||
/* get exposure */
|
||||
ret = ov_read_reg24(client, OV2680_EXPOSURE_H, ®_val);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
/* Lower four bits are not part of the exposure val (always 0) */
|
||||
*value = reg_val >> 4;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void ov2680_set_bayer_order(struct ov2680_device *sensor, struct v4l2_mbus_framefmt *fmt)
|
||||
{
|
||||
static const int ov2680_hv_flip_bayer_order[] = {
|
||||
@ -314,25 +139,8 @@ static int ov2680_s_ctrl(struct v4l2_ctrl *ctrl)
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int ov2680_g_volatile_ctrl(struct v4l2_ctrl *ctrl)
|
||||
{
|
||||
struct v4l2_subdev *sd = ctrl_to_sd(ctrl);
|
||||
int ret = 0;
|
||||
|
||||
switch (ctrl->id) {
|
||||
case V4L2_CID_EXPOSURE_ABSOLUTE:
|
||||
ret = ov2680_q_exposure(sd, &ctrl->val);
|
||||
break;
|
||||
default:
|
||||
ret = -EINVAL;
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static const struct v4l2_ctrl_ops ov2680_ctrl_ops = {
|
||||
.s_ctrl = ov2680_s_ctrl,
|
||||
.g_volatile_ctrl = ov2680_g_volatile_ctrl
|
||||
};
|
||||
|
||||
static int ov2680_init_registers(struct v4l2_subdev *sd)
|
||||
@ -441,10 +249,6 @@ static int power_up(struct v4l2_subdev *sd)
|
||||
if (ret)
|
||||
goto fail_init_registers;
|
||||
|
||||
ret = __ov2680_set_exposure(sd, dev->exposure, dev->gain, dev->digitgain);
|
||||
if (ret)
|
||||
goto fail_init_registers;
|
||||
|
||||
dev->power_on = true;
|
||||
return 0;
|
||||
|
||||
@ -545,7 +349,7 @@ static int ov2680_set_fmt(struct v4l2_subdev *sd,
|
||||
struct i2c_client *client = v4l2_get_subdevdata(sd);
|
||||
struct v4l2_mbus_framefmt *fmt;
|
||||
struct ov2680_resolution *res;
|
||||
int vts, ret = 0;
|
||||
int ret = 0;
|
||||
|
||||
res = v4l2_find_nearest_size(ov2680_res_preview, ARRAY_SIZE(ov2680_res_preview),
|
||||
width, height,
|
||||
@ -575,13 +379,7 @@ static int ov2680_set_fmt(struct v4l2_subdev *sd,
|
||||
goto err;
|
||||
}
|
||||
|
||||
vts = dev->res->lines_per_frame;
|
||||
|
||||
/* If necessary increase the VTS to match exposure + MARGIN */
|
||||
if (dev->exposure > vts - OV2680_INTEGRATION_TIME_MARGIN)
|
||||
vts = dev->exposure + OV2680_INTEGRATION_TIME_MARGIN;
|
||||
|
||||
ret = ov_write_reg16(client, OV2680_TIMING_VTS_H, vts);
|
||||
ret = ov_write_reg16(client, OV2680_TIMING_VTS_H, dev->res->lines_per_frame);
|
||||
if (ret) {
|
||||
dev_err(&client->dev, "ov2680 write vts err: %d\n", ret);
|
||||
goto err;
|
||||
@ -807,7 +605,6 @@ static const struct v4l2_subdev_sensor_ops ov2680_sensor_ops = {
|
||||
|
||||
static const struct v4l2_subdev_core_ops ov2680_core_ops = {
|
||||
.s_power = ov2680_s_power,
|
||||
.ioctl = ov2680_ioctl,
|
||||
};
|
||||
|
||||
static const struct v4l2_subdev_pad_ops ov2680_pad_ops = {
|
||||
@ -877,8 +674,6 @@ static int ov2680_probe(struct i2c_client *client)
|
||||
|
||||
dev->client = client;
|
||||
dev->res = &ov2680_res_preview[0];
|
||||
dev->exposure = dev->res->lines_per_frame - OV2680_INTEGRATION_TIME_MARGIN;
|
||||
dev->gain = 250; /* 0-2047 */
|
||||
v4l2_i2c_subdev_init(&dev->sd, client, &ov2680_ops);
|
||||
|
||||
pdata = gmin_camera_platform_data(&dev->sd,
|
||||
|
@ -163,9 +163,6 @@ struct ov2680_device {
|
||||
struct camera_sensor_platform_data *platform_data;
|
||||
bool power_on;
|
||||
bool is_streaming;
|
||||
u16 exposure;
|
||||
u16 gain;
|
||||
u16 digitgain;
|
||||
|
||||
struct ov2680_mode {
|
||||
struct v4l2_mbus_framefmt fmt;
|
||||
|
Loading…
Reference in New Issue
Block a user