The current code expect the configuration of the backlight to stay
constant after initialization. This patch allows to move it around.

Signed-off-by: Pantelis Antoniou <pa...@antoniou-consulting.com>
---
 drivers/video/backlight/tps65217_bl.c | 103 ++++++++++++++++++++++++++++++----
 1 file changed, 92 insertions(+), 11 deletions(-)

diff --git a/drivers/video/backlight/tps65217_bl.c 
b/drivers/video/backlight/tps65217_bl.c
index 7088163..69c1dfe 100644
--- a/drivers/video/backlight/tps65217_bl.c
+++ b/drivers/video/backlight/tps65217_bl.c
@@ -24,8 +24,11 @@
 #include <linux/module.h>
 #include <linux/platform_device.h>
 #include <linux/slab.h>
+#include <linux/i2c.h>
+#include <linux/of_i2c.h>
 
 struct tps65217_bl {
+       struct i2c_client *i2c_client;
        struct tps65217 *tps;
        struct device *dev;
        struct backlight_device *bl;
@@ -98,8 +101,6 @@ static int tps65217_bl_update_status(struct backlight_device 
*bl)
                        return rc;
                }
 
-               dev_dbg(tps65217_bl->dev, "brightness set to %d\n", brightness);
-
                if (!tps65217_bl->is_enabled)
                        rc = tps65217_bl_enable(tps65217_bl);
        } else {
@@ -187,14 +188,69 @@ static int tps65217_bl_hw_init(struct tps65217_bl 
*tps65217_bl,
 
 #ifdef CONFIG_OF
 static struct tps65217_bl_pdata *
-tps65217_bl_parse_dt(struct platform_device *pdev)
+tps65217_bl_parse_dt(struct platform_device *pdev, struct tps65217 **tpsp,
+               int *brightnessp)
 {
-       struct tps65217 *tps = dev_get_drvdata(pdev->dev.parent);
-       struct device_node *node = of_node_get(tps->dev->of_node);
+       struct i2c_client *i2c_client;
+       struct tps65217 *tps;
+       struct device_node *node, *rnode, *pnode;
        struct tps65217_bl_pdata *pdata, *err;
+       u32 tps_handle;
        u32 val;
 
-       node = of_find_node_by_name(node, "backlight");
+       tps = NULL;
+       node = NULL;
+       *brightnessp = 0;
+
+       /* our node (compatible) */
+       pnode = pdev->dev.of_node;
+       if (pnode != NULL &&
+               of_property_read_u32(pnode, "tps", &tps_handle) == 0) {
+               /* we are not instantiated from the mfd */
+               node = of_find_node_by_phandle(tps_handle);
+               if (node == NULL) {
+                       dev_err(&pdev->dev, "failed to find the tps node\n");
+                       err = ERR_PTR(-EINVAL);
+                       goto err;
+               }
+               i2c_client = of_find_i2c_device_by_node(node);
+               if (i2c_client == NULL) {
+                       dev_err(&pdev->dev, "failed to find the i2c device "
+                                       "of tps node\n");
+                       err = ERR_PTR(-EINVAL);
+                       goto err;
+               }
+               /* yeah this is gross; the whole concept is */
+               tps = i2c_get_clientdata(i2c_client);
+               if (tps == NULL) {
+                       dev_err(&pdev->dev, "failed to get tps structure\n");
+                       err = ERR_PTR(-EINVAL);
+                       goto err;
+               }
+
+               /* read default brightness */
+               val = 0;
+               of_property_read_u32(pnode, "brightness", &val);
+               if (val >= 100)
+                       val = 100;
+
+               *brightnessp = val;
+
+               /* no need for this anymore */
+               of_node_put(node);
+
+               dev_info(&pdev->dev, "got tps=%p from handle 0x%x\n", tps, 
tps_handle);
+       }
+
+       if (tps == NULL)
+               tps = dev_get_drvdata(pdev->dev.parent);
+
+       rnode = of_node_get(tps->dev->of_node);
+
+       node = of_find_node_by_name(rnode, "backlight");
+       of_node_put(rnode);
+       rnode = NULL;
+
        if (!node)
                return ERR_PTR(-ENODEV);
 
@@ -247,6 +303,7 @@ tps65217_bl_parse_dt(struct platform_device *pdev)
 
        of_node_put(node);
 
+       *tpsp = tps;
        return pdata;
 
 err:
@@ -254,9 +311,16 @@ err:
 
        return err;
 }
+
+static struct of_device_id tps65217_backlight_of_match[] = {
+       { .compatible = "tps65217-backlight" },
+       { }
+};
+MODULE_DEVICE_TABLE(of, tps65217_backlight_of_match);
 #else
 static struct tps65217_bl_pdata *
-tps65217_bl_parse_dt(struct platform_device *pdev)
+tps65217_bl_parse_dt(struct platform_device *pdev, struct tps65217 **tpsp,
+               int *brightnessp)
 {
        return NULL;
 }
@@ -265,13 +329,16 @@ tps65217_bl_parse_dt(struct platform_device *pdev)
 static int tps65217_bl_probe(struct platform_device *pdev)
 {
        int rc;
-       struct tps65217 *tps = dev_get_drvdata(pdev->dev.parent);
+       struct tps65217 *tps;
        struct tps65217_bl *tps65217_bl;
        struct tps65217_bl_pdata *pdata;
        struct backlight_properties bl_props;
+       int brightness = 0;
 
-       if (tps->dev->of_node) {
-               pdata = tps65217_bl_parse_dt(pdev);
+       tps = NULL;
+
+       if (pdev->dev.of_node) {
+               pdata = tps65217_bl_parse_dt(pdev, &tps, &brightness);
                if (IS_ERR(pdata))
                        return PTR_ERR(pdata);
        } else {
@@ -281,6 +348,14 @@ static int tps65217_bl_probe(struct platform_device *pdev)
                }
 
                pdata = pdev->dev.platform_data;
+
+               /* get the parent device */
+               tps = dev_get_drvdata(pdev->dev.parent);
+       }
+
+       if (tps == NULL) {
+               dev_err(&pdev->dev, "failed to find tps\n");
+               return -EINVAL;
        }
 
        tps65217_bl = devm_kzalloc(&pdev->dev, sizeof(*tps65217_bl),
@@ -311,9 +386,14 @@ static int tps65217_bl_probe(struct platform_device *pdev)
                return PTR_ERR(tps65217_bl->bl);
        }
 
-       tps65217_bl->bl->props.brightness = 0;
+       tps65217_bl->bl->props.brightness = brightness;
        platform_set_drvdata(pdev, tps65217_bl);
 
+       /* update with initial settings */
+       tps65217_bl_update_status(tps65217_bl->bl);
+
+       dev_info(&pdev->dev, "OK.\n");
+
        return 0;
 }
 
@@ -332,6 +412,7 @@ static struct platform_driver tps65217_bl_driver = {
        .driver         = {
                .owner  = THIS_MODULE,
                .name   = "tps65217-bl",
+               .of_match_table = of_match_ptr(tps65217_backlight_of_match),
        },
 };
 
-- 
1.7.12

--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

Reply via email to