Commit ad97bc37 authored by Sowjanya Komatineni's avatar Sowjanya Komatineni Committed by Mauro Carvalho Chehab

media: i2c: imx274: Add IMX274 power on and off sequence

IMX274 has analog 2.8V supply, digital core 1.8V supply, and vddl digital
io 1.2V supply which are optional based on camera module design.

IMX274 also need external 24Mhz clock and is optional based on
camera module design.

This patch adds support for IMX274 power on and off to enable and
disable these supplies and external clock.
Reviewed-by: default avatarLuca Ceresoli <luca@lucaceresoli.net>
Signed-off-by: default avatarSowjanya Komatineni <skomatineni@nvidia.com>
Signed-off-by: default avatarSakari Ailus <sakari.ailus@linux.intel.com>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab+huawei@kernel.org>
parent f4fcf010
...@@ -18,7 +18,9 @@ ...@@ -18,7 +18,9 @@
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/of_gpio.h> #include <linux/of_gpio.h>
#include <linux/pm_runtime.h>
#include <linux/regmap.h> #include <linux/regmap.h>
#include <linux/regulator/consumer.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/v4l2-mediabus.h> #include <linux/v4l2-mediabus.h>
#include <linux/videodev2.h> #include <linux/videodev2.h>
...@@ -131,6 +133,15 @@ ...@@ -131,6 +133,15 @@
#define IMX274_TABLE_WAIT_MS 0 #define IMX274_TABLE_WAIT_MS 0
#define IMX274_TABLE_END 1 #define IMX274_TABLE_END 1
/* regulator supplies */
static const char * const imx274_supply_names[] = {
"vddl", /* IF (1.2V) supply */
"vdig", /* Digital Core (1.8V) supply */
"vana", /* Analog (2.8V) supply */
};
#define IMX274_NUM_SUPPLIES ARRAY_SIZE(imx274_supply_names)
/* /*
* imx274 I2C operation related structure * imx274 I2C operation related structure
*/ */
...@@ -501,6 +512,8 @@ struct imx274_ctrls { ...@@ -501,6 +512,8 @@ struct imx274_ctrls {
* @frame_rate: V4L2 frame rate structure * @frame_rate: V4L2 frame rate structure
* @regmap: Pointer to regmap structure * @regmap: Pointer to regmap structure
* @reset_gpio: Pointer to reset gpio * @reset_gpio: Pointer to reset gpio
* @supplies: List of analog and digital supply regulators
* @inck: Pointer to sensor input clock
* @lock: Mutex structure * @lock: Mutex structure
* @mode: Parameters for the selected readout mode * @mode: Parameters for the selected readout mode
*/ */
...@@ -514,6 +527,8 @@ struct stimx274 { ...@@ -514,6 +527,8 @@ struct stimx274 {
struct v4l2_fract frame_interval; struct v4l2_fract frame_interval;
struct regmap *regmap; struct regmap *regmap;
struct gpio_desc *reset_gpio; struct gpio_desc *reset_gpio;
struct regulator_bulk_data supplies[IMX274_NUM_SUPPLIES];
struct clk *inck;
struct mutex lock; /* mutex lock for operations */ struct mutex lock; /* mutex lock for operations */
const struct imx274_mode *mode; const struct imx274_mode *mode;
}; };
...@@ -726,6 +741,12 @@ static int imx274_start_stream(struct stimx274 *priv) ...@@ -726,6 +741,12 @@ static int imx274_start_stream(struct stimx274 *priv)
{ {
int err = 0; int err = 0;
err = __v4l2_ctrl_handler_setup(&priv->ctrls.handler);
if (err) {
dev_err(&priv->client->dev, "Error %d setup controls\n", err);
return err;
}
/* /*
* Refer to "Standby Cancel Sequence when using CSI-2" in * Refer to "Standby Cancel Sequence when using CSI-2" in
* imx274 datasheet, it should wait 10ms or more here. * imx274 datasheet, it should wait 10ms or more here.
...@@ -767,6 +788,66 @@ static void imx274_reset(struct stimx274 *priv, int rst) ...@@ -767,6 +788,66 @@ static void imx274_reset(struct stimx274 *priv, int rst)
usleep_range(IMX274_RESET_DELAY1, IMX274_RESET_DELAY2); usleep_range(IMX274_RESET_DELAY1, IMX274_RESET_DELAY2);
} }
static int imx274_power_on(struct device *dev)
{
struct i2c_client *client = to_i2c_client(dev);
struct v4l2_subdev *sd = i2c_get_clientdata(client);
struct stimx274 *imx274 = to_imx274(sd);
int ret;
/* keep sensor in reset before power on */
imx274_reset(imx274, 0);
ret = clk_prepare_enable(imx274->inck);
if (ret) {
dev_err(&imx274->client->dev,
"Failed to enable input clock: %d\n", ret);
return ret;
}
ret = regulator_bulk_enable(IMX274_NUM_SUPPLIES, imx274->supplies);
if (ret) {
dev_err(&imx274->client->dev,
"Failed to enable regulators: %d\n", ret);
goto fail_reg;
}
udelay(2);
imx274_reset(imx274, 1);
return 0;
fail_reg:
clk_disable_unprepare(imx274->inck);
return ret;
}
static int imx274_power_off(struct device *dev)
{
struct i2c_client *client = to_i2c_client(dev);
struct v4l2_subdev *sd = i2c_get_clientdata(client);
struct stimx274 *imx274 = to_imx274(sd);
imx274_reset(imx274, 0);
regulator_bulk_disable(IMX274_NUM_SUPPLIES, imx274->supplies);
clk_disable_unprepare(imx274->inck);
return 0;
}
static int imx274_regulators_get(struct device *dev, struct stimx274 *imx274)
{
unsigned int i;
for (i = 0; i < IMX274_NUM_SUPPLIES; i++)
imx274->supplies[i].supply = imx274_supply_names[i];
return devm_regulator_bulk_get(dev, IMX274_NUM_SUPPLIES,
imx274->supplies);
}
/** /**
* imx274_s_ctrl - This is used to set the imx274 V4L2 controls * imx274_s_ctrl - This is used to set the imx274 V4L2 controls
* @ctrl: V4L2 control to be set * @ctrl: V4L2 control to be set
...@@ -781,6 +862,9 @@ static int imx274_s_ctrl(struct v4l2_ctrl *ctrl) ...@@ -781,6 +862,9 @@ static int imx274_s_ctrl(struct v4l2_ctrl *ctrl)
struct stimx274 *imx274 = to_imx274(sd); struct stimx274 *imx274 = to_imx274(sd);
int ret = -EINVAL; int ret = -EINVAL;
if (!pm_runtime_get_if_in_use(&imx274->client->dev))
return 0;
dev_dbg(&imx274->client->dev, dev_dbg(&imx274->client->dev,
"%s : s_ctrl: %s, value: %d\n", __func__, "%s : s_ctrl: %s, value: %d\n", __func__,
ctrl->name, ctrl->val); ctrl->name, ctrl->val);
...@@ -811,6 +895,8 @@ static int imx274_s_ctrl(struct v4l2_ctrl *ctrl) ...@@ -811,6 +895,8 @@ static int imx274_s_ctrl(struct v4l2_ctrl *ctrl)
break; break;
} }
pm_runtime_put(&imx274->client->dev);
return ret; return ret;
} }
...@@ -1271,10 +1357,8 @@ static int imx274_s_frame_interval(struct v4l2_subdev *sd, ...@@ -1271,10 +1357,8 @@ static int imx274_s_frame_interval(struct v4l2_subdev *sd,
* *
* Return: 0 on success, errors otherwise * Return: 0 on success, errors otherwise
*/ */
static int imx274_load_default(struct stimx274 *priv) static void imx274_load_default(struct stimx274 *priv)
{ {
int ret;
/* load default control values */ /* load default control values */
priv->frame_interval.numerator = 1; priv->frame_interval.numerator = 1;
priv->frame_interval.denominator = IMX274_DEF_FRAME_RATE; priv->frame_interval.denominator = IMX274_DEF_FRAME_RATE;
...@@ -1282,29 +1366,6 @@ static int imx274_load_default(struct stimx274 *priv) ...@@ -1282,29 +1366,6 @@ static int imx274_load_default(struct stimx274 *priv)
priv->ctrls.gain->val = IMX274_DEF_GAIN; priv->ctrls.gain->val = IMX274_DEF_GAIN;
priv->ctrls.vflip->val = 0; priv->ctrls.vflip->val = 0;
priv->ctrls.test_pattern->val = TEST_PATTERN_DISABLED; priv->ctrls.test_pattern->val = TEST_PATTERN_DISABLED;
/* update frame rate */
ret = imx274_set_frame_interval(priv,
priv->frame_interval);
if (ret)
return ret;
/* update exposure time */
ret = v4l2_ctrl_s_ctrl(priv->ctrls.exposure, priv->ctrls.exposure->val);
if (ret)
return ret;
/* update gain */
ret = v4l2_ctrl_s_ctrl(priv->ctrls.gain, priv->ctrls.gain->val);
if (ret)
return ret;
/* update vflip */
ret = v4l2_ctrl_s_ctrl(priv->ctrls.vflip, priv->ctrls.vflip->val);
if (ret)
return ret;
return 0;
} }
/** /**
...@@ -1329,6 +1390,13 @@ static int imx274_s_stream(struct v4l2_subdev *sd, int on) ...@@ -1329,6 +1390,13 @@ static int imx274_s_stream(struct v4l2_subdev *sd, int on)
mutex_lock(&imx274->lock); mutex_lock(&imx274->lock);
if (on) { if (on) {
ret = pm_runtime_get_sync(&imx274->client->dev);
if (ret < 0) {
pm_runtime_put_noidle(&imx274->client->dev);
mutex_unlock(&imx274->lock);
return ret;
}
/* load mode registers */ /* load mode registers */
ret = imx274_mode_regs(imx274); ret = imx274_mode_regs(imx274);
if (ret) if (ret)
...@@ -1349,12 +1417,6 @@ static int imx274_s_stream(struct v4l2_subdev *sd, int on) ...@@ -1349,12 +1417,6 @@ static int imx274_s_stream(struct v4l2_subdev *sd, int on)
if (ret) if (ret)
goto fail; goto fail;
/* update exposure time */
ret = __v4l2_ctrl_s_ctrl(imx274->ctrls.exposure,
imx274->ctrls.exposure->val);
if (ret)
goto fail;
/* start stream */ /* start stream */
ret = imx274_start_stream(imx274); ret = imx274_start_stream(imx274);
if (ret) if (ret)
...@@ -1364,6 +1426,8 @@ static int imx274_s_stream(struct v4l2_subdev *sd, int on) ...@@ -1364,6 +1426,8 @@ static int imx274_s_stream(struct v4l2_subdev *sd, int on)
ret = imx274_write_table(imx274, imx274_stop); ret = imx274_write_table(imx274, imx274_stop);
if (ret) if (ret)
goto fail; goto fail;
pm_runtime_put(&imx274->client->dev);
} }
mutex_unlock(&imx274->lock); mutex_unlock(&imx274->lock);
...@@ -1371,6 +1435,7 @@ static int imx274_s_stream(struct v4l2_subdev *sd, int on) ...@@ -1371,6 +1435,7 @@ static int imx274_s_stream(struct v4l2_subdev *sd, int on)
return 0; return 0;
fail: fail:
pm_runtime_put(&imx274->client->dev);
mutex_unlock(&imx274->lock); mutex_unlock(&imx274->lock);
dev_err(&imx274->client->dev, "s_stream failed\n"); dev_err(&imx274->client->dev, "s_stream failed\n");
return ret; return ret;
...@@ -1836,6 +1901,17 @@ static int imx274_probe(struct i2c_client *client) ...@@ -1836,6 +1901,17 @@ static int imx274_probe(struct i2c_client *client)
mutex_init(&imx274->lock); mutex_init(&imx274->lock);
imx274->inck = devm_clk_get_optional(&client->dev, "inck");
if (IS_ERR(imx274->inck))
return PTR_ERR(imx274->inck);
ret = imx274_regulators_get(&client->dev, imx274);
if (ret) {
dev_err(&client->dev,
"Failed to get power regulators, err: %d\n", ret);
return ret;
}
/* initialize format */ /* initialize format */
imx274->mode = &imx274_modes[IMX274_DEFAULT_BINNING]; imx274->mode = &imx274_modes[IMX274_DEFAULT_BINNING];
imx274->crop.width = IMX274_MAX_WIDTH; imx274->crop.width = IMX274_MAX_WIDTH;
...@@ -1883,15 +1959,20 @@ static int imx274_probe(struct i2c_client *client) ...@@ -1883,15 +1959,20 @@ static int imx274_probe(struct i2c_client *client)
goto err_me; goto err_me;
} }
/* pull sensor out of reset */ /* power on the sensor */
imx274_reset(imx274, 1); ret = imx274_power_on(&client->dev);
if (ret < 0) {
dev_err(&client->dev,
"%s : imx274 power on failed\n", __func__);
goto err_me;
}
/* initialize controls */ /* initialize controls */
ret = v4l2_ctrl_handler_init(&imx274->ctrls.handler, 4); ret = v4l2_ctrl_handler_init(&imx274->ctrls.handler, 4);
if (ret < 0) { if (ret < 0) {
dev_err(&client->dev, dev_err(&client->dev,
"%s : ctrl handler init Failed\n", __func__); "%s : ctrl handler init Failed\n", __func__);
goto err_me; goto err_power_off;
} }
imx274->ctrls.handler.lock = &imx274->lock; imx274->ctrls.handler.lock = &imx274->lock;
...@@ -1927,22 +2008,8 @@ static int imx274_probe(struct i2c_client *client) ...@@ -1927,22 +2008,8 @@ static int imx274_probe(struct i2c_client *client)
goto err_ctrls; goto err_ctrls;
} }
/* setup default controls */
ret = v4l2_ctrl_handler_setup(&imx274->ctrls.handler);
if (ret) {
dev_err(&client->dev,
"Error %d setup default controls\n", ret);
goto err_ctrls;
}
/* load default control values */ /* load default control values */
ret = imx274_load_default(imx274); imx274_load_default(imx274);
if (ret) {
dev_err(&client->dev,
"%s : imx274_load_default failed %d\n",
__func__, ret);
goto err_ctrls;
}
/* register subdevice */ /* register subdevice */
ret = v4l2_async_register_subdev(sd); ret = v4l2_async_register_subdev(sd);
...@@ -1953,11 +2020,17 @@ static int imx274_probe(struct i2c_client *client) ...@@ -1953,11 +2020,17 @@ static int imx274_probe(struct i2c_client *client)
goto err_ctrls; goto err_ctrls;
} }
pm_runtime_set_active(&client->dev);
pm_runtime_enable(&client->dev);
pm_runtime_idle(&client->dev);
dev_info(&client->dev, "imx274 : imx274 probe success !\n"); dev_info(&client->dev, "imx274 : imx274 probe success !\n");
return 0; return 0;
err_ctrls: err_ctrls:
v4l2_ctrl_handler_free(&imx274->ctrls.handler); v4l2_ctrl_handler_free(&imx274->ctrls.handler);
err_power_off:
imx274_power_off(&client->dev);
err_me: err_me:
media_entity_cleanup(&sd->entity); media_entity_cleanup(&sd->entity);
err_regmap: err_regmap:
...@@ -1970,16 +2043,27 @@ static int imx274_remove(struct i2c_client *client) ...@@ -1970,16 +2043,27 @@ static int imx274_remove(struct i2c_client *client)
struct v4l2_subdev *sd = i2c_get_clientdata(client); struct v4l2_subdev *sd = i2c_get_clientdata(client);
struct stimx274 *imx274 = to_imx274(sd); struct stimx274 *imx274 = to_imx274(sd);
pm_runtime_disable(&client->dev);
if (!pm_runtime_status_suspended(&client->dev))
imx274_power_off(&client->dev);
pm_runtime_set_suspended(&client->dev);
v4l2_async_unregister_subdev(sd); v4l2_async_unregister_subdev(sd);
v4l2_ctrl_handler_free(&imx274->ctrls.handler); v4l2_ctrl_handler_free(&imx274->ctrls.handler);
media_entity_cleanup(&sd->entity); media_entity_cleanup(&sd->entity);
mutex_destroy(&imx274->lock); mutex_destroy(&imx274->lock);
return 0; return 0;
} }
static const struct dev_pm_ops imx274_pm_ops = {
SET_RUNTIME_PM_OPS(imx274_power_off, imx274_power_on, NULL)
};
static struct i2c_driver imx274_i2c_driver = { static struct i2c_driver imx274_i2c_driver = {
.driver = { .driver = {
.name = DRIVER_NAME, .name = DRIVER_NAME,
.pm = &imx274_pm_ops,
.of_match_table = imx274_of_id_table, .of_match_table = imx274_of_id_table,
}, },
.probe_new = imx274_probe, .probe_new = imx274_probe,
......
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