1 // SPDX-License-Identifier: GPL-2.0-only 2 /* 3 * TI SCI Generic Power Domain Driver 4 * 5 * Copyright (C) 2015-2017 Texas Instruments Incorporated - http://www.ti.com/ 6 * J Keerthy <j-keerthy@ti.com> 7 * Dave Gerlach <d-gerlach@ti.com> 8 */ 9 10 #include <linux/err.h> 11 #include <linux/module.h> 12 #include <linux/of.h> 13 #include <linux/platform_device.h> 14 #include <linux/pm_domain.h> 15 #include <linux/slab.h> 16 #include <linux/soc/ti/ti_sci_protocol.h> 17 #include <dt-bindings/soc/ti,sci_pm_domain.h> 18 19 /** 20 * struct ti_sci_genpd_provider: holds common TI SCI genpd provider data 21 * @ti_sci: handle to TI SCI protocol driver that provides ops to 22 * communicate with system control processor. 23 * @dev: pointer to dev for the driver for devm allocs 24 * @pd_list: list of all the power domains on the device 25 * @data: onecell data for genpd core 26 */ 27 struct ti_sci_genpd_provider { 28 const struct ti_sci_handle *ti_sci; 29 struct device *dev; 30 struct list_head pd_list; 31 struct genpd_onecell_data data; 32 }; 33 34 /** 35 * struct ti_sci_pm_domain: TI specific data needed for power domain 36 * @idx: index of the device that identifies it with the system 37 * control processor. 38 * @exclusive: Permissions for exclusive request or shared request of the 39 * device. 40 * @pd: generic_pm_domain for use with the genpd framework 41 * @node: link for the genpd list 42 * @parent: link to the parent TI SCI genpd provider 43 */ 44 struct ti_sci_pm_domain { 45 int idx; 46 u8 exclusive; 47 struct generic_pm_domain pd; 48 struct list_head node; 49 struct ti_sci_genpd_provider *parent; 50 }; 51 52 #define genpd_to_ti_sci_pd(gpd) container_of(gpd, struct ti_sci_pm_domain, pd) 53 54 /* 55 * ti_sci_pd_power_off(): genpd power down hook 56 * @domain: pointer to the powerdomain to power off 57 */ 58 static int ti_sci_pd_power_off(struct generic_pm_domain *domain) 59 { 60 struct ti_sci_pm_domain *pd = genpd_to_ti_sci_pd(domain); 61 const struct ti_sci_handle *ti_sci = pd->parent->ti_sci; 62 63 return ti_sci->ops.dev_ops.put_device(ti_sci, pd->idx); 64 } 65 66 /* 67 * ti_sci_pd_power_on(): genpd power up hook 68 * @domain: pointer to the powerdomain to power on 69 */ 70 static int ti_sci_pd_power_on(struct generic_pm_domain *domain) 71 { 72 struct ti_sci_pm_domain *pd = genpd_to_ti_sci_pd(domain); 73 const struct ti_sci_handle *ti_sci = pd->parent->ti_sci; 74 75 if (pd->exclusive) 76 return ti_sci->ops.dev_ops.get_device_exclusive(ti_sci, 77 pd->idx); 78 else 79 return ti_sci->ops.dev_ops.get_device(ti_sci, pd->idx); 80 } 81 82 /* 83 * ti_sci_pd_xlate(): translation service for TI SCI genpds 84 * @genpdspec: DT identification data for the genpd 85 * @data: genpd core data for all the powerdomains on the device 86 */ 87 static struct generic_pm_domain *ti_sci_pd_xlate( 88 const struct of_phandle_args *genpdspec, 89 void *data) 90 { 91 struct genpd_onecell_data *genpd_data = data; 92 unsigned int idx = genpdspec->args[0]; 93 94 if (genpdspec->args_count != 1 && genpdspec->args_count != 2) 95 return ERR_PTR(-EINVAL); 96 97 if (idx >= genpd_data->num_domains) { 98 pr_err("%s: invalid domain index %u\n", __func__, idx); 99 return ERR_PTR(-EINVAL); 100 } 101 102 if (!genpd_data->domains[idx]) 103 return ERR_PTR(-ENOENT); 104 105 genpd_to_ti_sci_pd(genpd_data->domains[idx])->exclusive = 106 genpdspec->args[1]; 107 108 return genpd_data->domains[idx]; 109 } 110 111 static const struct of_device_id ti_sci_pm_domain_matches[] = { 112 { .compatible = "ti,sci-pm-domain", }, 113 { }, 114 }; 115 MODULE_DEVICE_TABLE(of, ti_sci_pm_domain_matches); 116 117 static bool ti_sci_pm_idx_exists(struct ti_sci_genpd_provider *pd_provider, u32 idx) 118 { 119 struct ti_sci_pm_domain *pd; 120 121 list_for_each_entry(pd, &pd_provider->pd_list, node) { 122 if (pd->idx == idx) 123 return true; 124 } 125 126 return false; 127 } 128 129 static int ti_sci_pm_domain_probe(struct platform_device *pdev) 130 { 131 struct device *dev = &pdev->dev; 132 struct ti_sci_genpd_provider *pd_provider; 133 struct ti_sci_pm_domain *pd; 134 struct device_node *np __free(device_node) = NULL; 135 struct of_phandle_args args; 136 u32 max_id = 0; 137 int index; 138 139 pd_provider = devm_kzalloc(dev, sizeof(*pd_provider), GFP_KERNEL); 140 if (!pd_provider) 141 return -ENOMEM; 142 143 pd_provider->ti_sci = devm_ti_sci_get_handle(dev); 144 if (IS_ERR(pd_provider->ti_sci)) 145 return PTR_ERR(pd_provider->ti_sci); 146 147 pd_provider->dev = dev; 148 149 INIT_LIST_HEAD(&pd_provider->pd_list); 150 151 /* Find highest device ID used for power domains */ 152 for_each_node_with_property(np, "power-domains") { 153 index = 0; 154 155 while (!of_parse_phandle_with_args(np, "power-domains", 156 "#power-domain-cells", 157 index, &args)) { 158 159 if (args.args_count >= 1 && args.np == dev->of_node) { 160 of_node_put(args.np); 161 if (args.args[0] > max_id) { 162 max_id = args.args[0]; 163 } else { 164 if (ti_sci_pm_idx_exists(pd_provider, args.args[0])) { 165 index++; 166 continue; 167 } 168 } 169 170 pd = devm_kzalloc(dev, sizeof(*pd), GFP_KERNEL); 171 if (!pd) 172 return -ENOMEM; 173 174 pd->pd.name = devm_kasprintf(dev, GFP_KERNEL, 175 "pd:%d", 176 args.args[0]); 177 if (!pd->pd.name) 178 return -ENOMEM; 179 180 pd->pd.power_off = ti_sci_pd_power_off; 181 pd->pd.power_on = ti_sci_pd_power_on; 182 pd->pd.flags |= GENPD_FLAG_ACTIVE_WAKEUP; 183 pd->idx = args.args[0]; 184 pd->parent = pd_provider; 185 186 pm_genpd_init(&pd->pd, NULL, true); 187 188 list_add(&pd->node, &pd_provider->pd_list); 189 } else { 190 of_node_put(args.np); 191 } 192 193 index++; 194 } 195 } 196 197 pd_provider->data.domains = 198 devm_kcalloc(dev, max_id + 1, 199 sizeof(*pd_provider->data.domains), 200 GFP_KERNEL); 201 if (!pd_provider->data.domains) 202 return -ENOMEM; 203 204 pd_provider->data.num_domains = max_id + 1; 205 pd_provider->data.xlate = ti_sci_pd_xlate; 206 207 list_for_each_entry(pd, &pd_provider->pd_list, node) 208 pd_provider->data.domains[pd->idx] = &pd->pd; 209 210 return of_genpd_add_provider_onecell(dev->of_node, &pd_provider->data); 211 } 212 213 static struct platform_driver ti_sci_pm_domains_driver = { 214 .probe = ti_sci_pm_domain_probe, 215 .driver = { 216 .name = "ti_sci_pm_domains", 217 .of_match_table = ti_sci_pm_domain_matches, 218 }, 219 }; 220 module_platform_driver(ti_sci_pm_domains_driver); 221 MODULE_LICENSE("GPL v2"); 222 MODULE_DESCRIPTION("TI System Control Interface (SCI) Power Domain driver"); 223 MODULE_AUTHOR("Dave Gerlach"); 224