Commit 8eb47aa3 authored by Hans de Goede's avatar Hans de Goede Committed by Mauro Carvalho Chehab

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: default avatarAndy Shevchenko <andy@kernel.org>
Signed-off-by: default avatarHans de Goede <hdegoede@redhat.com>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab@kernel.org>
parent 91caf797
...@@ -60,181 +60,6 @@ static int ov2680_write_reg_array(struct i2c_client *client, ...@@ -60,181 +60,6 @@ static int ov2680_write_reg_array(struct i2c_client *client,
return 0; 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, &reg_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 void ov2680_set_bayer_order(struct ov2680_device *sensor, struct v4l2_mbus_framefmt *fmt)
{ {
static const int ov2680_hv_flip_bayer_order[] = { static const int ov2680_hv_flip_bayer_order[] = {
...@@ -314,25 +139,8 @@ static int ov2680_s_ctrl(struct v4l2_ctrl *ctrl) ...@@ -314,25 +139,8 @@ static int ov2680_s_ctrl(struct v4l2_ctrl *ctrl)
return ret; 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 = { static const struct v4l2_ctrl_ops ov2680_ctrl_ops = {
.s_ctrl = ov2680_s_ctrl, .s_ctrl = ov2680_s_ctrl,
.g_volatile_ctrl = ov2680_g_volatile_ctrl
}; };
static int ov2680_init_registers(struct v4l2_subdev *sd) static int ov2680_init_registers(struct v4l2_subdev *sd)
...@@ -441,10 +249,6 @@ static int power_up(struct v4l2_subdev *sd) ...@@ -441,10 +249,6 @@ static int power_up(struct v4l2_subdev *sd)
if (ret) if (ret)
goto fail_init_registers; 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; dev->power_on = true;
return 0; return 0;
...@@ -545,7 +349,7 @@ static int ov2680_set_fmt(struct v4l2_subdev *sd, ...@@ -545,7 +349,7 @@ static int ov2680_set_fmt(struct v4l2_subdev *sd,
struct i2c_client *client = v4l2_get_subdevdata(sd); struct i2c_client *client = v4l2_get_subdevdata(sd);
struct v4l2_mbus_framefmt *fmt; struct v4l2_mbus_framefmt *fmt;
struct ov2680_resolution *res; struct ov2680_resolution *res;
int vts, ret = 0; int ret = 0;
res = v4l2_find_nearest_size(ov2680_res_preview, ARRAY_SIZE(ov2680_res_preview), res = v4l2_find_nearest_size(ov2680_res_preview, ARRAY_SIZE(ov2680_res_preview),
width, height, width, height,
...@@ -575,13 +379,7 @@ static int ov2680_set_fmt(struct v4l2_subdev *sd, ...@@ -575,13 +379,7 @@ static int ov2680_set_fmt(struct v4l2_subdev *sd,
goto err; goto err;
} }
vts = dev->res->lines_per_frame; ret = ov_write_reg16(client, OV2680_TIMING_VTS_H, 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);
if (ret) { if (ret) {
dev_err(&client->dev, "ov2680 write vts err: %d\n", ret); dev_err(&client->dev, "ov2680 write vts err: %d\n", ret);
goto err; goto err;
...@@ -807,7 +605,6 @@ static const struct v4l2_subdev_sensor_ops ov2680_sensor_ops = { ...@@ -807,7 +605,6 @@ static const struct v4l2_subdev_sensor_ops ov2680_sensor_ops = {
static const struct v4l2_subdev_core_ops ov2680_core_ops = { static const struct v4l2_subdev_core_ops ov2680_core_ops = {
.s_power = ov2680_s_power, .s_power = ov2680_s_power,
.ioctl = ov2680_ioctl,
}; };
static const struct v4l2_subdev_pad_ops ov2680_pad_ops = { static const struct v4l2_subdev_pad_ops ov2680_pad_ops = {
...@@ -877,8 +674,6 @@ static int ov2680_probe(struct i2c_client *client) ...@@ -877,8 +674,6 @@ static int ov2680_probe(struct i2c_client *client)
dev->client = client; dev->client = client;
dev->res = &ov2680_res_preview[0]; 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); v4l2_i2c_subdev_init(&dev->sd, client, &ov2680_ops);
pdata = gmin_camera_platform_data(&dev->sd, pdata = gmin_camera_platform_data(&dev->sd,
......
...@@ -163,9 +163,6 @@ struct ov2680_device { ...@@ -163,9 +163,6 @@ struct ov2680_device {
struct camera_sensor_platform_data *platform_data; struct camera_sensor_platform_data *platform_data;
bool power_on; bool power_on;
bool is_streaming; bool is_streaming;
u16 exposure;
u16 gain;
u16 digitgain;
struct ov2680_mode { struct ov2680_mode {
struct v4l2_mbus_framefmt fmt; struct v4l2_mbus_framefmt fmt;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment