Re: [PATCH v3 2/4] media: ov5675: add device-tree support and support runtime PM

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Hi all,

On 5/10/22 11:46, Jacopo Mondi wrote:
Hi Quentin,

On Mon, May 09, 2022 at 04:32:24PM +0200, Quentin Schulz wrote:
From: Quentin Schulz <quentin.schulz@xxxxxxxxxxxxxxxxxxxxx>

Until now, this driver only supported ACPI. This adds support for
Device Tree too while enabling clock and regulators in runtime PM.

Signed-off-by: Quentin Schulz <quentin.schulz@xxxxxxxxxxxxxxxxxxxxx>

Thanks for addressing all comments on the previous version.

Looks good to me!

Unfortunately the sensor disagrees :/

For some reasons, the first three power-on + power-off are successful (sometimes only the first two) and then the sensor is not working until next cold boot. I got lucky when I tested the patch before sending, much less now.

I'm looking into it, don't know how long it will take to get the runtime PM fixed. Please hold onto those patches (well.. technically patches 3/4 and 4/4 don't depend on anything from this patch, so feel free to merge those once reviewed).

Cheers,
Quentin

Reviewed-by: Jacopo Mondi <jacopo@xxxxxxxxxx>

Thanks
   j

---

v3:
  - added linux/mod_devicetable.h include,
  - moved delay for reset pulse right after the regulators are enabled,
  - removed check on is_acpi_node in favor of checks on presence of OF
  properties (e.g. devm_clk_get_optional returns NULL),
  - moved power management out of system suspend/resume into runtime PM
  callbacks,
  - removed ACPI specific comment since it's not specific to this driver,
  - changed devm_clk_get to devm_clk_get_optional,
  - remove OF use of clock-frequency (handled by devm_clk_get_optional
  directly),
  - removed name of clock (only one, so no need for anything explicit)
  when requesting a clock from OF,
  - wrapped lines to 80 chars,

v2:
  - fixed unused-const-variable warning by removing of_match_ptr in
  of_match_table, reported by kernel test robot,

  drivers/media/i2c/ov5675.c | 143 +++++++++++++++++++++++++++++++------
  1 file changed, 122 insertions(+), 21 deletions(-)

diff --git a/drivers/media/i2c/ov5675.c b/drivers/media/i2c/ov5675.c
index 82ba9f56baec..cee380196774 100644
--- a/drivers/media/i2c/ov5675.c
+++ b/drivers/media/i2c/ov5675.c
@@ -3,10 +3,14 @@

  #include <asm/unaligned.h>
  #include <linux/acpi.h>
+#include <linux/clk.h>
  #include <linux/delay.h>
+#include <linux/gpio/consumer.h>
  #include <linux/i2c.h>
+#include <linux/mod_devicetable.h>
  #include <linux/module.h>
  #include <linux/pm_runtime.h>
+#include <linux/regulator/consumer.h>
  #include <media/v4l2-ctrls.h>
  #include <media/v4l2-device.h>
  #include <media/v4l2-fwnode.h>
@@ -17,7 +21,7 @@

  #define OV5675_LINK_FREQ_450MHZ		450000000ULL
  #define OV5675_SCLK			90000000LL
-#define OV5675_MCLK			19200000
+#define OV5675_XVCLK_19_2		19200000
  #define OV5675_DATA_LANES		2
  #define OV5675_RGB_DEPTH		10

@@ -76,6 +80,14 @@

  #define to_ov5675(_sd)			container_of(_sd, struct ov5675, sd)

+static const char * const ov5675_supply_names[] = {
+	"avdd",		/* Analog power */
+	"dovdd",	/* Digital I/O power */
+	"dvdd",		/* Digital core power */
+};
+
+#define OV5675_NUM_SUPPLIES	ARRAY_SIZE(ov5675_supply_names)
+
  enum {
  	OV5675_LINK_FREQ_900MBPS,
  };
@@ -484,6 +496,9 @@ struct ov5675 {
  	struct v4l2_subdev sd;
  	struct media_pad pad;
  	struct v4l2_ctrl_handler ctrl_handler;
+	struct clk		*xvclk;
+	struct gpio_desc	*reset_gpio;
+	struct regulator_bulk_data supplies[OV5675_NUM_SUPPLIES];

  	/* V4L2 Controls */
  	struct v4l2_ctrl *link_freq;
@@ -944,6 +959,50 @@ static int ov5675_set_stream(struct v4l2_subdev *sd, int enable)
  	return ret;
  }

+static int ov5675_power_off(struct device *dev)
+{
+	struct v4l2_subdev *sd = dev_get_drvdata(dev);
+	struct ov5675 *ov5675 = to_ov5675(sd);
+
+	gpiod_set_value_cansleep(ov5675->reset_gpio, 1);
+	usleep_range(1000, 1200);
+
+	regulator_bulk_disable(OV5675_NUM_SUPPLIES, ov5675->supplies);
+	clk_disable_unprepare(ov5675->xvclk);
+
+	return 0;
+}
+
+static int ov5675_power_on(struct device *dev)
+{
+	struct v4l2_subdev *sd = dev_get_drvdata(dev);
+	struct ov5675 *ov5675 = to_ov5675(sd);
+	int ret;
+
+	ret = clk_prepare_enable(ov5675->xvclk);
+	if (ret < 0) {
+		dev_err(dev, "failed to enable xvclk: %d\n", ret);
+		return ret;
+	}
+
+	gpiod_set_value_cansleep(ov5675->reset_gpio, 1);
+
+	ret = regulator_bulk_enable(OV5675_NUM_SUPPLIES, ov5675->supplies);
+	if (ret) {
+		clk_disable_unprepare(ov5675->xvclk);
+		return ret;
+	}
+
+	/* Reset pulse should be at least 2ms */
+	usleep_range(2000, 2200);
+
+	gpiod_set_value_cansleep(ov5675->reset_gpio, 0);
+
+	usleep_range(1000, 1200);
+
+	return 0;
+}
+
  static int __maybe_unused ov5675_suspend(struct device *dev)
  {
  	struct v4l2_subdev *sd = dev_get_drvdata(dev);
@@ -1106,32 +1165,60 @@ static const struct v4l2_subdev_internal_ops ov5675_internal_ops = {
  	.open = ov5675_open,
  };

-static int ov5675_check_hwcfg(struct device *dev)
+static int ov5675_get_hwcfg(struct ov5675 *ov5675, struct device *dev)
  {
  	struct fwnode_handle *ep;
  	struct fwnode_handle *fwnode = dev_fwnode(dev);
  	struct v4l2_fwnode_endpoint bus_cfg = {
  		.bus_type = V4L2_MBUS_CSI2_DPHY
  	};
-	u32 mclk;
+	u32 xvclk_rate;
  	int ret;
  	unsigned int i, j;

  	if (!fwnode)
  		return -ENXIO;

-	ret = fwnode_property_read_u32(fwnode, "clock-frequency", &mclk);
+	ov5675->xvclk = devm_clk_get_optional(dev, NULL);
+	if (IS_ERR(ov5675->xvclk))
+		return dev_err_probe(dev, PTR_ERR(ov5675->xvclk),
+				     "failed to get xvclk: %ld\n",
+				     PTR_ERR(ov5675->xvclk));

-	if (ret) {
-		dev_err(dev, "can't get clock frequency");
-		return ret;
+	if (ov5675->xvclk) {
+		xvclk_rate = clk_get_rate(ov5675->xvclk);
+	} else {
+		ret = fwnode_property_read_u32(fwnode, "clock-frequency",
+					       &xvclk_rate);
+
+		if (ret) {
+			dev_err(dev, "can't get clock frequency");
+			return ret;
+		}
  	}

-	if (mclk != OV5675_MCLK) {
-		dev_err(dev, "external clock %d is not supported", mclk);
+	if (xvclk_rate != OV5675_XVCLK_19_2) {
+		dev_err(dev, "external clock rate %u is unsupported",
+			xvclk_rate);
  		return -EINVAL;
  	}

+	ov5675->reset_gpio = devm_gpiod_get_optional(dev, "reset",
+						     GPIOD_OUT_HIGH);
+	if (IS_ERR(ov5675->reset_gpio)) {
+		ret = PTR_ERR(ov5675->reset_gpio);
+		dev_err(dev, "failed to get reset-gpios: %d\n", ret);
+		return ret;
+	}
+
+	for (i = 0; i < OV5675_NUM_SUPPLIES; i++)
+		ov5675->supplies[i].supply = ov5675_supply_names[i];
+
+	ret = devm_regulator_bulk_get(dev, OV5675_NUM_SUPPLIES,
+				      ov5675->supplies);
+	if (ret)
+		return ret;
+
  	ep = fwnode_graph_get_next_endpoint(fwnode, NULL);
  	if (!ep)
  		return -ENXIO;
@@ -1186,6 +1273,9 @@ static int ov5675_remove(struct i2c_client *client)
  	pm_runtime_disable(&client->dev);
  	mutex_destroy(&ov5675->mutex);

+	if (!pm_runtime_status_suspended(&client->dev))
+		ov5675_power_off(&client->dev);
+
  	return 0;
  }

@@ -1195,25 +1285,31 @@ static int ov5675_probe(struct i2c_client *client)
  	bool full_power;
  	int ret;

-	ret = ov5675_check_hwcfg(&client->dev);
+	ov5675 = devm_kzalloc(&client->dev, sizeof(*ov5675), GFP_KERNEL);
+	if (!ov5675)
+		return -ENOMEM;
+
+	ret = ov5675_get_hwcfg(ov5675, &client->dev);
  	if (ret) {
-		dev_err(&client->dev, "failed to check HW configuration: %d",
+		dev_err(&client->dev, "failed to get HW configuration: %d",
  			ret);
  		return ret;
  	}

-	ov5675 = devm_kzalloc(&client->dev, sizeof(*ov5675), GFP_KERNEL);
-	if (!ov5675)
-		return -ENOMEM;
-
  	v4l2_i2c_subdev_init(&ov5675->sd, client, &ov5675_subdev_ops);

+	ret = ov5675_power_on(&client->dev);
+	if (ret) {
+		dev_err(&client->dev, "failed to power on: %d\n", ret);
+		return ret;
+	}
+
  	full_power = acpi_dev_state_d0(&client->dev);
  	if (full_power) {
  		ret = ov5675_identify_module(ov5675);
  		if (ret) {
  			dev_err(&client->dev, "failed to find sensor: %d", ret);
-			return ret;
+			goto probe_power_off;
  		}
  	}

@@ -1243,11 +1339,6 @@ static int ov5675_probe(struct i2c_client *client)
  		goto probe_error_media_entity_cleanup;
  	}

-	/*
-	 * Device is already turned on by i2c-core with ACPI domain PM.
-	 * Enable runtime PM and turn off the device.
-	 */
-
  	/* Set the device's state to active if it's in D0 state. */
  	if (full_power)
  		pm_runtime_set_active(&client->dev);
@@ -1262,12 +1353,15 @@ static int ov5675_probe(struct i2c_client *client)
  probe_error_v4l2_ctrl_handler_free:
  	v4l2_ctrl_handler_free(ov5675->sd.ctrl_handler);
  	mutex_destroy(&ov5675->mutex);
+probe_power_off:
+	ov5675_power_off(&client->dev);

  	return ret;
  }

  static const struct dev_pm_ops ov5675_pm_ops = {
  	SET_SYSTEM_SLEEP_PM_OPS(ov5675_suspend, ov5675_resume)
+	SET_RUNTIME_PM_OPS(ov5675_power_off, ov5675_power_on, NULL)
  };

  #ifdef CONFIG_ACPI
@@ -1279,11 +1373,18 @@ static const struct acpi_device_id ov5675_acpi_ids[] = {
  MODULE_DEVICE_TABLE(acpi, ov5675_acpi_ids);
  #endif

+static const struct of_device_id ov5675_of_match[] = {
+	{ .compatible = "ovti,ov5675", },
+	{ /* sentinel */ },
+};
+MODULE_DEVICE_TABLE(of, ov5675_of_match);
+
  static struct i2c_driver ov5675_i2c_driver = {
  	.driver = {
  		.name = "ov5675",
  		.pm = &ov5675_pm_ops,
  		.acpi_match_table = ACPI_PTR(ov5675_acpi_ids),
+		.of_match_table = ov5675_of_match,
  	},
  	.probe_new = ov5675_probe,
  	.remove = ov5675_remove,
--
2.35.3




[Index of Archives]     [Device Tree Compilter]     [Device Tree Spec]     [Linux Driver Backports]     [Video for Linux]     [Linux USB Devel]     [Linux PCI Devel]     [Linux Audio Users]     [Linux Kernel]     [Linux SCSI]     [XFree86]     [Yosemite Backpacking]


  Powered by Linux