rk3368: io-domain: add io domain volt sel
authorDavid Wu <wdc@rock-chips.com>
Wed, 24 Dec 2014 08:31:09 +0000 (16:31 +0800)
committerHuang, Tao <huangtao@rock-chips.com>
Wed, 24 Dec 2014 09:25:17 +0000 (17:25 +0800)
Signed-off-by: David Wu <wdc@rock-chips.com>
arch/arm64/boot/dts/rk3368-tb_8846.dts
arch/arm64/configs/rockchip_defconfig
drivers/power/avs/rockchip-io-domain.c

index af0b307cd94055c524aabd2bf636d5027c12cac9..31c3cdb872e4c8af152df00d1c2676f0b1dd7d3c 100644 (file)
                rockchip,grf = <&grf>;
                rockchip,pmu = <&pmu_grf>;
 
-               grf {
-                       dvp-supply = "vccio";
-                       flash0-supply = "vccio";
-                       wifi-supply = "act_ldo6";
-                       audio-supply = "vccio";
-                       sdcard-supply = "vccio_sd";
-                       gpio30-supply = "vccio";
-                       gpio1830-supply = "vccio";
-               };
-
-               pmu_grf {
-                       pmu-supply = "act_ldo4";
-                       vop-supply = "act_ldo4";
-               };
+               /*GRF_IO_VSEL*/
+               dvp-supply = <&ldo7_reg>;      /*DVPIO_VDD*/
+               flash0-supply = <&dcdc2_reg>;  /*FLASH0_VDD*/
+               wifi-supply = <&ldo7_reg>;     /*APIO2_VDD*/
+               audio-supply = <&dcdc2_reg>;   /*APIO3_VDD*/
+               sdcard-supply = <&ldo1_reg>;   /*SDMMC0_VDD*/
+               gpio30-supply = <&dcdc2_reg>;  /*APIO1_VDD*/
+               gpio1830-supply = <&dcdc2_reg>;/*ADIO4_VDD*/
+
+               /*PMU_GRF_IO_VSEL*/
+               pmu-supply = <&ldo5_reg>;      /*PMUIO_VDD*/
+               vop-supply = <&ldo5_reg>;      /*LCDC_VDD*/
        };
 };
 
index 10309a76896e284e3f3cecd438d9f50dd4ba6f73..fece209c092f0a69b282dc1cd923e04a6e7ec12a 100644 (file)
@@ -346,6 +346,7 @@ CONFIG_BATTERY_BQ27320=y
 CONFIG_BATTERY_RK30_ADC_FAC=y
 CONFIG_CW2015_BATTERY=y
 CONFIG_POWER_RESET_GPIO=y
+CONFIG_POWER_AVS=y
 CONFIG_ROCKCHIP_IODOMAIN=y
 CONFIG_SENSORS_ROCKCHIP_TSADC=y
 CONFIG_THERMAL=y
index b16013202cf944d9e66a6ac5a20a205a5b6824f2..d0964572268bddfaf7926e6ffa4271b557dbb4b9 100755 (executable)
 #define RK3288_SOC_CON2_FLASH0         BIT(7)
 #define RK3288_SOC_FLASH_SUPPLY_NUM    2
 
-#define RK3368_GRF_SOC_CON15                   0x43c
-#define RK3368_GRF_SOC_CON15_FLASH0            BIT(14)
+#define RK3368_GRF_SOC_CON15           0x43c
+#define RK3368_GRF_SOC_CON15_FLASH0    BIT(14)
 #define RK3368_SOC_FLASH_SUPPLY_NUM    2
 
+#define MAX_ROCKCHIP_GRF_NUM        2
 
 struct rockchip_iodomain;
 
 /**
  * @supplies: voltage settings matching the register bits.
  */
+
+enum rockchip_iodomain_grf_type {
+       GRF,
+       PMU_GRF,
+};
+
 struct rockchip_iodomain_soc_data {
        int grf_offset;
-       const char *supply_names[MAX_SUPPLIES];
+       int pmugrf_offset;
+       const char *grf_supply_names[MAX_SUPPLIES];
+       const char *pmugrf_supply_names[MAX_SUPPLIES];
        void (*init)(struct rockchip_iodomain *iod);
 };
 
@@ -64,19 +73,24 @@ struct rockchip_iodomain_supply {
        struct regulator *reg;
        struct notifier_block nb;
        int idx;
+       enum rockchip_iodomain_grf_type type;
 };
 
 struct rockchip_iodomain {
        struct device *dev;
        struct regmap *grf;
+       struct regmap *pmu;
        struct rockchip_iodomain_soc_data *soc_data;
-       struct rockchip_iodomain_supply supplies[MAX_SUPPLIES];
+       struct rockchip_iodomain_supply grf_supplies[MAX_SUPPLIES];
+       struct rockchip_iodomain_supply pmugrf_supplies[MAX_SUPPLIES];
 };
 
 static int rockchip_iodomain_write(struct rockchip_iodomain_supply *supply,
                                   int uV)
 {
        struct rockchip_iodomain *iod = supply->iod;
+       struct regmap *reg;
+       int offset;
        u32 val;
        int ret;
 
@@ -87,7 +101,15 @@ static int rockchip_iodomain_write(struct rockchip_iodomain_supply *supply,
        /* apply hiword-mask */
        val |= (BIT(supply->idx) << 16);
 
-       ret = regmap_write(iod->grf, iod->soc_data->grf_offset, val);
+       if (supply->type == GRF) {
+               reg = iod->grf;
+               offset = iod->soc_data->grf_offset;
+       } else if (supply->type == PMU_GRF) {
+               reg = iod->pmu;
+               offset = iod->soc_data->pmugrf_offset;
+       }
+
+       ret = regmap_write(reg, offset, val);
        if (ret)
                dev_err(iod->dev, "Couldn't write to GRF\n");
 
@@ -150,7 +172,7 @@ static void rk3288_iodomain_init(struct rockchip_iodomain *iod)
        u32 val;
 
        /* if no flash supply we should leave things alone */
-       if (!iod->supplies[RK3288_SOC_FLASH_SUPPLY_NUM].reg)
+       if (!iod->grf_supplies[RK3288_SOC_FLASH_SUPPLY_NUM].reg)
                return;
 
        /*
@@ -169,7 +191,7 @@ static void rk3368_iodomain_init(struct rockchip_iodomain *iod)
        u32 val;
 
        /* if no flash supply we should leave things alone */
-       if (!iod->supplies[RK3368_SOC_FLASH_SUPPLY_NUM].reg)
+       if (!iod->grf_supplies[RK3368_SOC_FLASH_SUPPLY_NUM].reg)
                return;
 
        /*
@@ -189,7 +211,7 @@ static void rk3368_iodomain_init(struct rockchip_iodomain *iod)
  */
 static const struct rockchip_iodomain_soc_data soc_data_rk3188 = {
        .grf_offset = 0x104,
-       .supply_names = {
+       .grf_supply_names = {
                NULL,
                NULL,
                NULL,
@@ -211,7 +233,7 @@ static const struct rockchip_iodomain_soc_data soc_data_rk3188 = {
 
 static const struct rockchip_iodomain_soc_data soc_data_rk3288 = {
        .grf_offset = 0x380,
-       .supply_names = {
+       .grf_supply_names = {
                "lcdc",         /* LCDC_VDD */
                "dvp",          /* DVPIO_VDD */
                "flash0",       /* FLASH0_VDD (emmc) */
@@ -228,16 +250,25 @@ static const struct rockchip_iodomain_soc_data soc_data_rk3288 = {
 
 static const struct rockchip_iodomain_soc_data soc_data_rk3368 = {
        .grf_offset = 0x900,
-       .supply_names = {
+       .pmugrf_offset = 0x100,
+       .grf_supply_names = {
+               NULL,
+               "dvp",          /*DVP IO domain*/
+               "flash0",       /*FLASH0 IO domain*/
+               "wifi",         /*APIO2 IO domain*/
+               NULL,
+               "audio",        /*APIO3 IO domain*/
+               "sdcard",       /*SDCARD IO domain*/
+               "gpio30",       /*APIO1 IO domain*/
+               "gpio1830",     /*ADIO4 IO domain*/
+       },
+       .pmugrf_supply_names = {
+               NULL,
+               NULL,
                NULL,
-               "dvp_v18sel",           /*DVP IO domain*/
-               "flash0_v18sel",                /*FLASH0 IO domain*/
-               "wifi_v18sel",  /*WIFI IO domain*/
                NULL,
-               "audio_v18sel", /*AUDIO IO domain*/
-               "sdcard_v18sel",                /*SDCARD IO domain*/
-               "gpio30_v18sel",                /*GPIO30 IO domain*/
-               "gpio1830_v18sel",      /*GPIO1830 IO domain*/
+               "pmu",          /*PMU IO domain*/
+               "vop",          /*LCDC IO domain*/
        },
        .init = rk3368_iodomain_init,
 };
@@ -259,47 +290,34 @@ static const struct of_device_id rockchip_iodomain_match[] = {
        { /* sentinel */ },
 };
 
-static int rockchip_iodomain_probe(struct platform_device *pdev)
+static int rockchip_iodomain_parse_supply(struct rockchip_iodomain *iod,
+                                         struct device_node *np,
+                                         enum rockchip_iodomain_grf_type type)
 {
-       struct device_node *np = pdev->dev.of_node;
-       const struct of_device_id *match;
-       struct rockchip_iodomain *iod;
+       struct rockchip_iodomain_supply *group_supply;
+       const char **group_supply_names;
        int i, ret = 0;
 
-       if (!np)
-               return -ENODEV;
-
-       iod = devm_kzalloc(&pdev->dev, sizeof(*iod), GFP_KERNEL);
-       if (!iod)
-               return -ENOMEM;
-
-       iod->dev = &pdev->dev;
-       platform_set_drvdata(pdev, iod);
-
-       match = of_match_node(rockchip_iodomain_match, np);
-       iod->soc_data = (struct rockchip_iodomain_soc_data *)match->data;
-
-       iod->grf = syscon_regmap_lookup_by_phandle(np, "rockchip,grf");
-       if (IS_ERR(iod->grf)) {
-               dev_err(&pdev->dev, "couldn't find grf regmap\n");
-               return PTR_ERR(iod->grf);
+       if (type == GRF) {
+               group_supply_names =
+                       (const char **)iod->soc_data->grf_supply_names;
+               group_supply = iod->grf_supplies;
+       } else if (type == PMU_GRF) {
+               group_supply_names =
+                       (const char **)iod->soc_data->pmugrf_supply_names;
+               group_supply = iod->pmugrf_supplies;
        }
 
        for (i = 0; i < MAX_SUPPLIES; i++) {
-               const char *supply_name = iod->soc_data->supply_names[i];
-               struct rockchip_iodomain_supply *supply = &iod->supplies[i];
+               const char *supply_name = group_supply_names[i];
+               struct rockchip_iodomain_supply *supply = &group_supply[i];
                struct regulator *reg;
                int uV;
-               const char *regulator_name = NULL;
 
                if (!supply_name)
                        continue;
 
-               of_property_read_string(np, supply_name, &regulator_name);
-               if (!regulator_name)
-                       continue;
-
-               reg = regulator_get(NULL, regulator_name);
+               reg = devm_regulator_get_optional(iod->dev, supply_name);
                if (IS_ERR(reg)) {
                        ret = PTR_ERR(reg);
 
@@ -334,6 +352,7 @@ static int rockchip_iodomain_probe(struct platform_device *pdev)
                supply->idx = i;
                supply->iod = iod;
                supply->reg = reg;
+               supply->type = type;
                supply->nb.notifier_call = rockchip_iodomain_notify;
 
                ret = rockchip_iodomain_write(supply, uV);
@@ -345,21 +364,16 @@ static int rockchip_iodomain_probe(struct platform_device *pdev)
                /* register regulator notifier */
                ret = regulator_register_notifier(reg, &supply->nb);
                if (ret) {
-                       dev_err(&pdev->dev,
+                       dev_err(iod->dev,
                                "regulator notifier request failed\n");
                        supply->reg = NULL;
                        goto unreg_notify;
                }
        }
 
-       if (iod->soc_data->init)
-               iod->soc_data->init(iod);
-
-       return 0;
-
 unreg_notify:
        for (i = MAX_SUPPLIES - 1; i >= 0; i--) {
-               struct rockchip_iodomain_supply *io_supply = &iod->supplies[i];
+               struct rockchip_iodomain_supply *io_supply = &group_supply[i];
 
                if (io_supply->reg)
                        regulator_unregister_notifier(io_supply->reg,
@@ -369,13 +383,75 @@ unreg_notify:
        return ret;
 }
 
+static int rockchip_iodomain_probe(struct platform_device *pdev)
+{
+       struct device_node *np = pdev->dev.of_node, *node;
+       const struct of_device_id *match;
+       struct rockchip_iodomain *iod;
+       int ret = 0;
+
+       if (!np)
+               return -ENODEV;
+
+       iod = devm_kzalloc(&pdev->dev, sizeof(*iod), GFP_KERNEL);
+       if (!iod)
+               return -ENOMEM;
+
+       iod->dev = &pdev->dev;
+       platform_set_drvdata(pdev, iod);
+
+       match = of_match_node(rockchip_iodomain_match, np);
+       iod->soc_data = (struct rockchip_iodomain_soc_data *)match->data;
+
+       iod->grf = syscon_regmap_lookup_by_phandle(np, "rockchip,grf");
+       if (IS_ERR(iod->grf)) {
+               dev_err(&pdev->dev, "couldn't find grf regmap\n");
+               return PTR_ERR(iod->grf);
+       }
+       ret = rockchip_iodomain_parse_supply(iod, np, GRF);
+       if (ret) {
+               dev_err(iod->dev,
+                       "rockchip iodomain parse grf supply failed\n");
+               return ret;
+       }
+
+       /* try to find the optional reference to the pmu syscon */
+       node = of_parse_phandle(np, "rockchip,pmu", 0);
+       if (node) {
+               iod->pmu = syscon_node_to_regmap(node);
+               if (IS_ERR(iod->pmu))
+                       return PTR_ERR(iod->pmu);
+               ret = rockchip_iodomain_parse_supply(iod, np, PMU_GRF);
+               if (ret) {
+                       dev_err(iod->dev,
+                               "rockchip iodomain parse pmu_grf supply failed\n");
+                       return ret;
+               }
+       }
+
+       if (iod->soc_data->init)
+               iod->soc_data->init(iod);
+
+       return ret;
+}
+
 static int rockchip_iodomain_remove(struct platform_device *pdev)
 {
        struct rockchip_iodomain *iod = platform_get_drvdata(pdev);
        int i;
 
        for (i = MAX_SUPPLIES - 1; i >= 0; i--) {
-               struct rockchip_iodomain_supply *io_supply = &iod->supplies[i];
+               struct rockchip_iodomain_supply *io_supply
+                       = &iod->grf_supplies[i];
+
+               if (io_supply->reg)
+                       regulator_unregister_notifier(io_supply->reg,
+                                                     &io_supply->nb);
+       }
+
+       for (i = MAX_SUPPLIES - 1; i >= 0; i--) {
+               struct rockchip_iodomain_supply *io_supply =
+                       &iod->pmugrf_supplies[i];
 
                if (io_supply->reg)
                        regulator_unregister_notifier(io_supply->reg,