Hi Laurent On Wed, 23 May 2012, Laurent Pinchart wrote: > Instead of forcing all soc-camera drivers to go through the mid-layer to > handle power management, create soc_camera_power_[on|off]() functions > that can be called from the subdev .s_power() operation to manage > regulators and platform-specific power handling. This allows non > soc-camera hosts to use soc-camera-aware clients. > > Signed-off-by: Laurent Pinchart <laurent.pinchart@xxxxxxxxxxxxxxxx> > --- > drivers/media/video/imx074.c | 12 ++++ > drivers/media/video/mt9m001.c | 12 ++++ > drivers/media/video/mt9m111.c | 50 +++++++++++----- > drivers/media/video/mt9t031.c | 11 +++- > drivers/media/video/mt9t112.c | 12 ++++ > drivers/media/video/mt9v022.c | 12 ++++ > drivers/media/video/ov2640.c | 12 ++++ > drivers/media/video/ov5642.c | 11 +++- > drivers/media/video/ov6650.c | 12 ++++ > drivers/media/video/ov772x.c | 12 ++++ > drivers/media/video/ov9640.c | 13 ++++- > drivers/media/video/ov9740.c | 15 +++++- > drivers/media/video/rj54n1cb0c.c | 12 ++++ > drivers/media/video/sh_mobile_csi2.c | 10 +++- > drivers/media/video/soc_camera.c | 90 +++++++++++++++------------- > drivers/media/video/soc_camera_platform.c | 15 +++++- > drivers/media/video/tw9910.c | 12 ++++ > include/media/soc_camera.h | 2 + > 18 files changed, 260 insertions(+), 65 deletions(-) > > diff --git a/drivers/media/video/imx074.c b/drivers/media/video/imx074.c > index 351e9ba..1166c89 100644 > --- a/drivers/media/video/imx074.c > +++ b/drivers/media/video/imx074.c > @@ -268,6 +268,17 @@ static int imx074_g_chip_ident(struct v4l2_subdev *sd, > return 0; > } > > +static int imx074_s_power(struct v4l2_subdev *sd, int on) > +{ > + struct i2c_client *client = v4l2_get_subdevdata(sd); > + struct soc_camera_link *icl = soc_camera_i2c_to_link(client); > + > + if (on) > + return soc_camera_power_on(&client->dev, icl); > + else > + return soc_camera_power_off(&client->dev, icl); I think an inline function would be nice for these to just write in most of your converted drivers return soc_camera_s_power(&client->dev, icl, on); > +} > + > static int imx074_g_mbus_config(struct v4l2_subdev *sd, > struct v4l2_mbus_config *cfg) > { > @@ -292,6 +303,7 @@ static struct v4l2_subdev_video_ops imx074_subdev_video_ops = { > > static struct v4l2_subdev_core_ops imx074_subdev_core_ops = { > .g_chip_ident = imx074_g_chip_ident, > + .s_power = imx074_s_power, > }; > > static struct v4l2_subdev_ops imx074_subdev_ops = { > diff --git a/drivers/media/video/mt9m001.c b/drivers/media/video/mt9m001.c > index 7e64818..cf2aa00 100644 > --- a/drivers/media/video/mt9m001.c > +++ b/drivers/media/video/mt9m001.c > @@ -377,6 +377,17 @@ static int mt9m001_s_register(struct v4l2_subdev *sd, > } > #endif > > +static int mt9m001_s_power(struct v4l2_subdev *sd, int on) > +{ > + struct i2c_client *client = v4l2_get_subdevdata(sd); > + struct soc_camera_link *icl = soc_camera_i2c_to_link(client); > + > + if (on) > + return soc_camera_power_on(&client->dev, icl); > + else > + return soc_camera_power_off(&client->dev, icl); > +} > + > static int mt9m001_g_volatile_ctrl(struct v4l2_ctrl *ctrl) > { > struct mt9m001 *mt9m001 = container_of(ctrl->handler, > @@ -566,6 +577,7 @@ static struct v4l2_subdev_core_ops mt9m001_subdev_core_ops = { > .g_register = mt9m001_g_register, > .s_register = mt9m001_s_register, > #endif > + .s_power = mt9m001_s_power, > }; > > static int mt9m001_enum_fmt(struct v4l2_subdev *sd, unsigned int index, > diff --git a/drivers/media/video/mt9m111.c b/drivers/media/video/mt9m111.c > index b0c5299..b9bfb4f 100644 > --- a/drivers/media/video/mt9m111.c > +++ b/drivers/media/video/mt9m111.c > @@ -832,10 +832,35 @@ static int mt9m111_video_probe(struct i2c_client *client) > return v4l2_ctrl_handler_setup(&mt9m111->hdl); > } > > +static int mt9m111_power_on(struct mt9m111 *mt9m111) > +{ > + struct i2c_client *client = v4l2_get_subdevdata(&mt9m111->subdev); > + struct soc_camera_link *icl = soc_camera_i2c_to_link(client); > + int ret; > + > + ret = soc_camera_power_on(&client->dev, icl); > + if (ret < 0) > + return ret; > + > + ret = mt9m111_resume(mt9m111); > + if (ret < 0) > + dev_err(&client->dev, "Failed to resume the sensor: %d\n", ret); What do you think, don't we have to soc_camera_power_off() if the above resume fails? At least I was doing that in the original soc_camera_power_on() and you also preserve it that way. > + > + return ret; > +} > + > +static void mt9m111_power_off(struct mt9m111 *mt9m111) > +{ > + struct i2c_client *client = v4l2_get_subdevdata(&mt9m111->subdev); > + struct soc_camera_link *icl = soc_camera_i2c_to_link(client); > + > + mt9m111_suspend(mt9m111); > + soc_camera_power_off(&client->dev, icl); > +} > + > static int mt9m111_s_power(struct v4l2_subdev *sd, int on) > { > struct mt9m111 *mt9m111 = container_of(sd, struct mt9m111, subdev); > - struct i2c_client *client = v4l2_get_subdevdata(sd); > int ret = 0; > > mutex_lock(&mt9m111->power_lock); > @@ -845,23 +870,18 @@ static int mt9m111_s_power(struct v4l2_subdev *sd, int on) > * update the power state. > */ > if (mt9m111->power_count == !on) { > - if (on) { > - ret = mt9m111_resume(mt9m111); > - if (ret) { > - dev_err(&client->dev, > - "Failed to resume the sensor: %d\n", ret); > - goto out; > - } > - } else { > - mt9m111_suspend(mt9m111); > - } > + if (on) > + ret = mt9m111_power_on(mt9m111); > + else > + mt9m111_power_off(mt9m111); > } > > - /* Update the power count. */ > - mt9m111->power_count += on ? 1 : -1; > - WARN_ON(mt9m111->power_count < 0); > + if (!ret) { > + /* Update the power count. */ > + mt9m111->power_count += on ? 1 : -1; > + WARN_ON(mt9m111->power_count < 0); > + } > > -out: > mutex_unlock(&mt9m111->power_lock); > return ret; > } > diff --git a/drivers/media/video/mt9t031.c b/drivers/media/video/mt9t031.c > index 1415074..9666e20 100644 > --- a/drivers/media/video/mt9t031.c > +++ b/drivers/media/video/mt9t031.c > @@ -616,12 +616,19 @@ static struct device_type mt9t031_dev_type = { > static int mt9t031_s_power(struct v4l2_subdev *sd, int on) > { > struct i2c_client *client = v4l2_get_subdevdata(sd); > + struct soc_camera_link *icl = soc_camera_i2c_to_link(client); > struct video_device *vdev = soc_camera_i2c_to_vdev(client); > + int ret; > > - if (on) > + if (on) { > + ret = soc_camera_power_on(&client->dev, icl); > + if (ret < 0) > + return ret; > vdev->dev.type = &mt9t031_dev_type; > - else > + } else { > vdev->dev.type = NULL; > + soc_camera_power_off(&client->dev, icl); > + } > > return 0; > } > diff --git a/drivers/media/video/mt9t112.c b/drivers/media/video/mt9t112.c > index e1ae46a..a78242a 100644 > --- a/drivers/media/video/mt9t112.c > +++ b/drivers/media/video/mt9t112.c > @@ -776,12 +776,24 @@ static int mt9t112_s_register(struct v4l2_subdev *sd, > } > #endif > > +static int mt9t112_s_power(struct v4l2_subdev *sd, int on) > +{ > + struct i2c_client *client = v4l2_get_subdevdata(sd); > + struct soc_camera_link *icl = soc_camera_i2c_to_link(client); > + > + if (on) > + return soc_camera_power_on(&client->dev, icl); > + else > + return soc_camera_power_off(&client->dev, icl); > +} > + > static struct v4l2_subdev_core_ops mt9t112_subdev_core_ops = { > .g_chip_ident = mt9t112_g_chip_ident, > #ifdef CONFIG_VIDEO_ADV_DEBUG > .g_register = mt9t112_g_register, > .s_register = mt9t112_s_register, > #endif > + .s_power = mt9t112_s_power, > }; > > > diff --git a/drivers/media/video/mt9v022.c b/drivers/media/video/mt9v022.c > index bf63417..5715231 100644 > --- a/drivers/media/video/mt9v022.c > +++ b/drivers/media/video/mt9v022.c > @@ -445,6 +445,17 @@ static int mt9v022_s_register(struct v4l2_subdev *sd, > } > #endif > > +static int mt9v022_s_power(struct v4l2_subdev *sd, int on) > +{ > + struct i2c_client *client = v4l2_get_subdevdata(sd); > + struct soc_camera_link *icl = soc_camera_i2c_to_link(client); > + > + if (on) > + return soc_camera_power_on(&client->dev, icl); > + else > + return soc_camera_power_off(&client->dev, icl); > +} > + > static int mt9v022_g_volatile_ctrl(struct v4l2_ctrl *ctrl) > { > struct mt9v022 *mt9v022 = container_of(ctrl->handler, > @@ -664,6 +675,7 @@ static struct v4l2_subdev_core_ops mt9v022_subdev_core_ops = { > .g_register = mt9v022_g_register, > .s_register = mt9v022_s_register, > #endif > + .s_power = mt9v022_s_power, > }; > > static int mt9v022_enum_fmt(struct v4l2_subdev *sd, unsigned int index, > diff --git a/drivers/media/video/ov2640.c b/drivers/media/video/ov2640.c > index d9a427c..cc3da57 100644 > --- a/drivers/media/video/ov2640.c > +++ b/drivers/media/video/ov2640.c > @@ -742,6 +742,17 @@ static int ov2640_s_register(struct v4l2_subdev *sd, > } > #endif > > +static int ov2640_s_power(struct v4l2_subdev *sd, int on) > +{ > + struct i2c_client *client = v4l2_get_subdevdata(sd); > + struct soc_camera_link *icl = soc_camera_i2c_to_link(client); > + > + if (on) > + return soc_camera_power_on(&client->dev, icl); > + else > + return soc_camera_power_off(&client->dev, icl); > +} > + > /* Select the nearest higher resolution for capture */ > static const struct ov2640_win_size *ov2640_select_win(u32 *width, u32 *height) > { > @@ -987,6 +998,7 @@ static struct v4l2_subdev_core_ops ov2640_subdev_core_ops = { > .g_register = ov2640_g_register, > .s_register = ov2640_s_register, > #endif > + .s_power = ov2640_s_power, > }; > > static int ov2640_g_mbus_config(struct v4l2_subdev *sd, > diff --git a/drivers/media/video/ov5642.c b/drivers/media/video/ov5642.c > index 0bc9331..98de102 100644 > --- a/drivers/media/video/ov5642.c > +++ b/drivers/media/video/ov5642.c > @@ -933,13 +933,20 @@ static int ov5642_g_mbus_config(struct v4l2_subdev *sd, > > static int ov5642_s_power(struct v4l2_subdev *sd, int on) > { > - struct i2c_client *client; > + struct i2c_client *client = v4l2_get_subdevdata(sd); > + struct soc_camera_link *icl = soc_camera_i2c_to_link(client); > int ret; > > + if (on) > + ret = soc_camera_power_on(&client->dev, icl); > + else > + ret = soc_camera_power_off(&client->dev, icl); > + if (ret < 0) > + return ret; > + > if (!on) > return 0; How about if (!on) return soc_camera_power_off(&client->dev, icl); ret = soc_camera_power_on(&client->dev, icl); if (ret < 0) ... > > - client = v4l2_get_subdevdata(sd); > ret = ov5642_write_array(client, ov5642_default_regs_init); > if (!ret) > ret = ov5642_set_resolution(sd); > diff --git a/drivers/media/video/ov6650.c b/drivers/media/video/ov6650.c > index 3e028b1..2264c8f 100644 > --- a/drivers/media/video/ov6650.c > +++ b/drivers/media/video/ov6650.c > @@ -432,6 +432,17 @@ static int ov6650_set_register(struct v4l2_subdev *sd, > } > #endif > > +static int ov6650_s_power(struct v4l2_subdev *sd, int on) > +{ > + struct i2c_client *client = v4l2_get_subdevdata(sd); > + struct soc_camera_link *icl = soc_camera_i2c_to_link(client); > + > + if (on) > + return soc_camera_power_on(&client->dev, icl); > + else > + return soc_camera_power_off(&client->dev, icl); > +} > + > static int ov6650_g_crop(struct v4l2_subdev *sd, struct v4l2_crop *a) > { > struct i2c_client *client = v4l2_get_subdevdata(sd); > @@ -866,6 +877,7 @@ static struct v4l2_subdev_core_ops ov6650_core_ops = { > .g_register = ov6650_get_register, > .s_register = ov6650_set_register, > #endif > + .s_power = ov6650_s_power, > }; > > /* Request bus settings on camera side */ > diff --git a/drivers/media/video/ov772x.c b/drivers/media/video/ov772x.c > index 6d79b89..3f37369 100644 > --- a/drivers/media/video/ov772x.c > +++ b/drivers/media/video/ov772x.c > @@ -683,6 +683,17 @@ static int ov772x_s_register(struct v4l2_subdev *sd, > } > #endif > > +static int ov772x_s_power(struct v4l2_subdev *sd, int on) > +{ > + struct i2c_client *client = v4l2_get_subdevdata(sd); > + struct soc_camera_link *icl = soc_camera_i2c_to_link(client); > + > + if (on) > + return soc_camera_power_on(&client->dev, icl); > + else > + return soc_camera_power_off(&client->dev, icl); > +} > + > static const struct ov772x_win_size *ov772x_select_win(u32 width, u32 height) > { > __u32 diff; > @@ -996,6 +1007,7 @@ static struct v4l2_subdev_core_ops ov772x_subdev_core_ops = { > .g_register = ov772x_g_register, > .s_register = ov772x_s_register, > #endif > + .s_power = ov772x_s_power, > }; > > static int ov772x_enum_fmt(struct v4l2_subdev *sd, unsigned int index, > diff --git a/drivers/media/video/ov9640.c b/drivers/media/video/ov9640.c > index 23412de..712cf6f 100644 > --- a/drivers/media/video/ov9640.c > +++ b/drivers/media/video/ov9640.c > @@ -333,6 +333,17 @@ static int ov9640_set_register(struct v4l2_subdev *sd, > } > #endif > > +static int ov9640_s_power(struct v4l2_subdev *sd, int on) > +{ > + struct i2c_client *client = v4l2_get_subdevdata(sd); > + struct soc_camera_link *icl = soc_camera_i2c_to_link(client); > + > + if (on) > + return soc_camera_power_on(&client->dev, icl); > + else > + return soc_camera_power_off(&client->dev, icl); > +} > + > /* select nearest higher resolution for capture */ > static void ov9640_res_roundup(u32 *width, u32 *height) > { > @@ -631,7 +642,7 @@ static struct v4l2_subdev_core_ops ov9640_core_ops = { > .g_register = ov9640_get_register, > .s_register = ov9640_set_register, > #endif > - > + .s_power = ov9640_s_power, > }; > > /* Request bus settings on camera side */ > diff --git a/drivers/media/video/ov9740.c b/drivers/media/video/ov9740.c > index 3eb07c2..effd0f1 100644 > --- a/drivers/media/video/ov9740.c > +++ b/drivers/media/video/ov9740.c > @@ -786,16 +786,29 @@ static int ov9740_g_chip_ident(struct v4l2_subdev *sd, > > static int ov9740_s_power(struct v4l2_subdev *sd, int on) > { > + struct i2c_client *client = v4l2_get_subdevdata(sd); > + struct soc_camera_link *icl = soc_camera_i2c_to_link(client); > struct ov9740_priv *priv = to_ov9740(sd); > + int ret; > > - if (!priv->current_enable) > + if (on) { > + ret = soc_camera_power_on(&client->dev, icl); > + if (ret < 0) > + return ret; > + } > + > + if (!priv->current_enable) { > + if (!on) > + soc_camera_power_off(&client->dev, icl); > return 0; > + } > > if (on) { > ov9740_s_fmt(sd, &priv->current_mf); > ov9740_s_stream(sd, priv->current_enable); > } else { > ov9740_s_stream(sd, 0); > + soc_camera_power_off(&client->dev, icl); > priv->current_enable = true; > } > > diff --git a/drivers/media/video/rj54n1cb0c.c b/drivers/media/video/rj54n1cb0c.c > index f6419b2..f0c3c64 100644 > --- a/drivers/media/video/rj54n1cb0c.c > +++ b/drivers/media/video/rj54n1cb0c.c > @@ -1180,6 +1180,17 @@ static int rj54n1_s_register(struct v4l2_subdev *sd, > } > #endif > > +static int rj54n1_s_power(struct v4l2_subdev *sd, int on) > +{ > + struct i2c_client *client = v4l2_get_subdevdata(sd); > + struct soc_camera_link *icl = soc_camera_i2c_to_link(client); > + > + if (on) > + return soc_camera_power_on(&client->dev, icl); > + else > + return soc_camera_power_off(&client->dev, icl); > +} > + > static int rj54n1_s_ctrl(struct v4l2_ctrl *ctrl) > { > struct rj54n1 *rj54n1 = container_of(ctrl->handler, struct rj54n1, hdl); > @@ -1230,6 +1241,7 @@ static struct v4l2_subdev_core_ops rj54n1_subdev_core_ops = { > .g_register = rj54n1_g_register, > .s_register = rj54n1_s_register, > #endif > + .s_power = rj54n1_s_power, > }; > > static int rj54n1_g_mbus_config(struct v4l2_subdev *sd, > diff --git a/drivers/media/video/sh_mobile_csi2.c b/drivers/media/video/sh_mobile_csi2.c > index 0528650..88c6911 100644 > --- a/drivers/media/video/sh_mobile_csi2.c > +++ b/drivers/media/video/sh_mobile_csi2.c > @@ -276,12 +276,20 @@ static void sh_csi2_client_disconnect(struct sh_csi2 *priv) > > static int sh_csi2_s_power(struct v4l2_subdev *sd, int on) > { > + struct i2c_client *client = v4l2_get_subdevdata(sd); > + struct soc_camera_link *icl = soc_camera_i2c_to_link(client); > struct sh_csi2 *priv = container_of(sd, struct sh_csi2, subdev); > + int ret; > > - if (on) > + if (on) { > + ret = soc_camera_power_on(&client->dev, icl); > + if (ret < 0) > + return ret; > return sh_csi2_client_connect(priv); > + } > > sh_csi2_client_disconnect(priv); > + soc_camera_power_off(&client->dev, icl); > return 0; Actually, I might have confused you on this one, sorry. This is not an external device, and as such it cannot have platform .s_power() callbacks or regulators to switch its power. So, maybe we don't have to patch this one at all? > } > > diff --git a/drivers/media/video/soc_camera.c b/drivers/media/video/soc_camera.c > index 55b981f..6c50032 100644 > --- a/drivers/media/video/soc_camera.c > +++ b/drivers/media/video/soc_camera.c > @@ -50,66 +50,74 @@ static LIST_HEAD(hosts); > static LIST_HEAD(devices); > static DEFINE_MUTEX(list_lock); /* Protects the list of hosts */ > > -static int soc_camera_power_on(struct soc_camera_device *icd, > - struct soc_camera_link *icl) > +int soc_camera_power_on(struct device *dev, struct soc_camera_link *icl) > { > - struct v4l2_subdev *sd = soc_camera_to_subdev(icd); > - int ret = regulator_bulk_enable(icl->num_regulators, > - icl->regulators); > + int ret; > + > + ret = regulator_bulk_enable(icl->num_regulators, > + icl->regulators); Let's avoid purely stylistic changes ;-) > if (ret < 0) { > - dev_err(icd->pdev, "Cannot enable regulators\n"); > + dev_err(dev, "Cannot enable regulators\n"); > return ret; > } > > if (icl->power) { > - ret = icl->power(icd->control, 1); > + ret = icl->power(dev, 1); > if (ret < 0) { > - dev_err(icd->pdev, > + dev_err(dev, > "Platform failed to power-on the camera.\n"); > - goto elinkpwr; > + regulator_bulk_disable(icl->num_regulators, > + icl->regulators); > } > } > > - ret = v4l2_subdev_call(sd, core, s_power, 1); > - if (ret < 0 && ret != -ENOIOCTLCMD && ret != -ENODEV) > - goto esdpwr; > - > - return 0; > - > -esdpwr: > - if (icl->power) > - icl->power(icd->control, 0); > -elinkpwr: > - regulator_bulk_disable(icl->num_regulators, > - icl->regulators); > return ret; > } > +EXPORT_SYMBOL(soc_camera_power_on); > > -static int soc_camera_power_off(struct soc_camera_device *icd, > - struct soc_camera_link *icl) > +int soc_camera_power_off(struct device *dev, struct soc_camera_link *icl) > { > - struct v4l2_subdev *sd = soc_camera_to_subdev(icd); > - int ret = v4l2_subdev_call(sd, core, s_power, 0); > - > - if (ret < 0 && ret != -ENOIOCTLCMD && ret != -ENODEV) > - return ret; > + int ret; > > if (icl->power) { > - ret = icl->power(icd->control, 0); > - if (ret < 0) { > - dev_err(icd->pdev, > + ret = icl->power(dev, 0); > + if (ret < 0) > + dev_err(dev, > "Platform failed to power-off the camera.\n"); > - return ret; > - } > } > > ret = regulator_bulk_disable(icl->num_regulators, > icl->regulators); This is also a change in behaviour: currently if any of power-off stages fails we bail out. With this patch you change it to continue with the next stage. I'm not sure which one is more correct, but at least I wouldn't silently change it under the counter ;-) > if (ret < 0) > - dev_err(icd->pdev, "Cannot disable regulators\n"); > + dev_err(dev, "Cannot disable regulators\n"); > > return ret; > } > +EXPORT_SYMBOL(soc_camera_power_off); > + > +static int __soc_camera_power_on(struct soc_camera_device *icd) > +{ > + struct v4l2_subdev *sd = soc_camera_to_subdev(icd); > + int ret; > + > + ret = v4l2_subdev_call(sd, core, s_power, 1); > + if (ret < 0 && ret != -ENOIOCTLCMD && ret != -ENODEV) > + return ret; > + > + return 0; > +} > + > +static int __soc_camera_power_off(struct soc_camera_device *icd) > +{ > + struct v4l2_subdev *sd = soc_camera_to_subdev(icd); > + int ret; > + > + ret = v4l2_subdev_call(sd, core, s_power, 0); > + if (ret < 0 && ret != -ENOIOCTLCMD && ret != -ENODEV) > + return ret; > + > + return 0; > +} > > const struct soc_camera_format_xlate *soc_camera_xlate_by_fourcc( > struct soc_camera_device *icd, unsigned int fourcc) > @@ -542,7 +550,7 @@ static int soc_camera_open(struct file *file) > goto eiciadd; > } > > - ret = soc_camera_power_on(icd, icl); > + ret = __soc_camera_power_on(icd); > if (ret < 0) > goto epower; > > @@ -584,7 +592,7 @@ einitvb: > esfmt: > pm_runtime_disable(&icd->vdev->dev); > eresume: > - soc_camera_power_off(icd, icl); > + __soc_camera_power_off(icd); > epower: > ici->ops->remove(icd); > eiciadd: > @@ -601,8 +609,6 @@ static int soc_camera_close(struct file *file) > > icd->use_count--; > if (!icd->use_count) { > - struct soc_camera_link *icl = to_soc_camera_link(icd); > - > pm_runtime_suspend(&icd->vdev->dev); > pm_runtime_disable(&icd->vdev->dev); > > @@ -610,7 +616,7 @@ static int soc_camera_close(struct file *file) > vb2_queue_release(&icd->vb2_vidq); > ici->ops->remove(icd); > > - soc_camera_power_off(icd, icl); > + __soc_camera_power_off(icd); > } > > if (icd->streamer == file) > @@ -1070,7 +1076,7 @@ static int soc_camera_probe(struct soc_camera_device *icd) > * again after initialisation, even though it shouldn't be needed, we > * don't do any IO here. > */ > - ret = soc_camera_power_on(icd, icl); > + ret = soc_camera_power_on(NULL, icl); Oops, no good... I think, at least dev_err() et al. don't like being called with NULL dev... But even without that, I think, this looks so bad, that it defeats the whole conversion... IIRC, you didn't like keeping the original soc-camera platform device pointer for icl->power(), because non soc-camera hosts and platforms don't have that device and would have to either pass NULL or some other device pointer. Now, we're switching all icl->power() calls to point to the physical device, which is largely useless also for soc-camera platforms (they'd have to go back to the original platform device if they wish to use that pointer at all), all - except one - in which we pass NULL... So, what's the point? :) I really would keep the soc-camera platform device pointer for icl->power(). There would be a slight difficulty to get to that pointer in soc_camera_power_*(). Just passing a subdevice pointer to it and using v4l2_get_subdev_hostdata(sd) isn't sufficient, because that would only work in the soc-camera environment. If the same client driver is running outside of it, v4l2_get_subdev_hostdata(sd) might well point to something completely different. So, we need a way to distinguish, whether this client is running on an soc-camera host or not. It is possible to have a system with two cameras - one connected to the SoC interface and another one to USB with both sensors using soc-camera services. Only one of them is associated with an soc-camera device, and another one is not. One way to distinguish this would be to scan the devices list in soc_camera.c to see, whether any icd->link == icl. If the client is found - we pass its platform device pointer to icl->power(). If not - pass NULL. The wrapper inline function would then become return soc_camera_s_power(icl, on); Do you see any problems with this approach? > if (ret < 0) > goto epower; > > @@ -1143,7 +1149,7 @@ static int soc_camera_probe(struct soc_camera_device *icd) > > ici->ops->remove(icd); > > - soc_camera_power_off(icd, icl); > + __soc_camera_power_off(icd); > > mutex_unlock(&icd->video_lock); > > @@ -1165,7 +1171,7 @@ eadddev: > video_device_release(icd->vdev); > icd->vdev = NULL; > evdc: > - soc_camera_power_off(icd, icl); > + __soc_camera_power_off(icd); > epower: > ici->ops->remove(icd); > eadd: > diff --git a/drivers/media/video/soc_camera_platform.c b/drivers/media/video/soc_camera_platform.c > index f59ccad..efd85e7 100644 > --- a/drivers/media/video/soc_camera_platform.c > +++ b/drivers/media/video/soc_camera_platform.c > @@ -50,7 +50,20 @@ static int soc_camera_platform_fill_fmt(struct v4l2_subdev *sd, > return 0; > } > > -static struct v4l2_subdev_core_ops platform_subdev_core_ops; > +static int soc_camera_platform_s_power(struct v4l2_subdev *sd, int on) > +{ > + struct i2c_client *client = v4l2_get_subdevdata(sd); Nnnnoo... soc_camera_platform is special - it doesn't (have to) have an i2c device associated with it. You're only guaranteed to have a subdevice and an struct soc_camera_platform_priv instance (the latter actually contains the former as its only member). But you have the advantage, that this driver always only runs under soc-camera. At least I hope noone ever comes up with an idea to use it outside of soc-camera ;-) So, if we accept my above proposal to use the platform device for soc_camera_power_*(), then you can use something like struct soc_camera_platform_info *p = v4l2_get_subdevdata(sd); soc_camera_s_power(p->icd->link, on); > + struct soc_camera_link *icl = soc_camera_i2c_to_link(client); > + > + if (on) > + return soc_camera_power_on(&client->dev, icl); > + else > + return soc_camera_power_off(&client->dev, icl); > +} > + > +static struct v4l2_subdev_core_ops platform_subdev_core_ops = { > + .s_power = soc_camera_platform_s_power, > +}; > > static int soc_camera_platform_enum_fmt(struct v4l2_subdev *sd, unsigned int index, > enum v4l2_mbus_pixelcode *code) > diff --git a/drivers/media/video/tw9910.c b/drivers/media/video/tw9910.c > index 9f53eac..61a445f 100644 > --- a/drivers/media/video/tw9910.c > +++ b/drivers/media/video/tw9910.c > @@ -566,6 +566,17 @@ static int tw9910_s_register(struct v4l2_subdev *sd, > } > #endif > > +static int tw9910_s_power(struct v4l2_subdev *sd, int on) > +{ > + struct i2c_client *client = v4l2_get_subdevdata(sd); > + struct soc_camera_link *icl = soc_camera_i2c_to_link(client); > + > + if (on) > + return soc_camera_power_on(&client->dev, icl); > + else > + return soc_camera_power_off(&client->dev, icl); > +} > + > static int tw9910_set_frame(struct v4l2_subdev *sd, u32 *width, u32 *height) > { > struct i2c_client *client = v4l2_get_subdevdata(sd); > @@ -814,6 +825,7 @@ static struct v4l2_subdev_core_ops tw9910_subdev_core_ops = { > .g_register = tw9910_g_register, > .s_register = tw9910_s_register, > #endif > + .s_power = tw9910_s_power, > }; > > static int tw9910_enum_fmt(struct v4l2_subdev *sd, unsigned int index, > diff --git a/include/media/soc_camera.h b/include/media/soc_camera.h > index d865dcf..0b61d85 100644 > --- a/include/media/soc_camera.h > +++ b/include/media/soc_camera.h > @@ -253,6 +253,8 @@ unsigned long soc_camera_apply_sensor_flags(struct soc_camera_link *icl, > unsigned long flags); > unsigned long soc_camera_apply_board_flags(struct soc_camera_link *icl, > const struct v4l2_mbus_config *cfg); > +int soc_camera_power_on(struct device *dev, struct soc_camera_link *icl); > +int soc_camera_power_off(struct device *dev, struct soc_camera_link *icl); > > /* This is only temporary here - until v4l2-subdev begins to link to video_device */ > #include <linux/i2c.h> > -- > 1.7.3.4 Let me know if you find any loose ends in my proposals. Thanks Guennadi --- Guennadi Liakhovetski, Ph.D. Freelance Open-Source Software Developer http://www.open-technology.de/ -- To unsubscribe from this list: send the line "unsubscribe linux-media" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html