at master 344 lines 7.7 kB view raw
1// SPDX-License-Identifier: GPL-2.0 2#include <linux/clk.h> 3#include <linux/device.h> 4#include <linux/export.h> 5#include <linux/gfp.h> 6 7struct devm_clk_state { 8 struct clk *clk; 9 void (*exit)(struct clk *clk); 10}; 11 12static void devm_clk_release(struct device *dev, void *res) 13{ 14 struct devm_clk_state *state = res; 15 16 if (state->exit) 17 state->exit(state->clk); 18 19 clk_put(state->clk); 20} 21 22static struct clk *__devm_clk_get(struct device *dev, const char *id, 23 struct clk *(*get)(struct device *dev, const char *id), 24 int (*init)(struct clk *clk), 25 void (*exit)(struct clk *clk)) 26{ 27 struct devm_clk_state *state; 28 struct clk *clk; 29 int ret; 30 31 state = devres_alloc(devm_clk_release, sizeof(*state), GFP_KERNEL); 32 if (!state) 33 return ERR_PTR(-ENOMEM); 34 35 clk = get(dev, id); 36 if (IS_ERR(clk)) { 37 ret = PTR_ERR(clk); 38 goto err_clk_get; 39 } 40 41 if (init) { 42 ret = init(clk); 43 if (ret) 44 goto err_clk_init; 45 } 46 47 state->clk = clk; 48 state->exit = exit; 49 50 devres_add(dev, state); 51 52 return clk; 53 54err_clk_init: 55 56 clk_put(clk); 57err_clk_get: 58 59 devres_free(state); 60 return ERR_PTR(ret); 61} 62 63struct clk *devm_clk_get(struct device *dev, const char *id) 64{ 65 return __devm_clk_get(dev, id, clk_get, NULL, NULL); 66} 67EXPORT_SYMBOL(devm_clk_get); 68 69struct clk *devm_clk_get_prepared(struct device *dev, const char *id) 70{ 71 return __devm_clk_get(dev, id, clk_get, clk_prepare, clk_unprepare); 72} 73EXPORT_SYMBOL_GPL(devm_clk_get_prepared); 74 75struct clk *devm_clk_get_enabled(struct device *dev, const char *id) 76{ 77 return __devm_clk_get(dev, id, clk_get, 78 clk_prepare_enable, clk_disable_unprepare); 79} 80EXPORT_SYMBOL_GPL(devm_clk_get_enabled); 81 82struct clk *devm_clk_get_optional(struct device *dev, const char *id) 83{ 84 return __devm_clk_get(dev, id, clk_get_optional, NULL, NULL); 85} 86EXPORT_SYMBOL(devm_clk_get_optional); 87 88struct clk *devm_clk_get_optional_prepared(struct device *dev, const char *id) 89{ 90 return __devm_clk_get(dev, id, clk_get_optional, 91 clk_prepare, clk_unprepare); 92} 93EXPORT_SYMBOL_GPL(devm_clk_get_optional_prepared); 94 95struct clk *devm_clk_get_optional_enabled(struct device *dev, const char *id) 96{ 97 return __devm_clk_get(dev, id, clk_get_optional, 98 clk_prepare_enable, clk_disable_unprepare); 99} 100EXPORT_SYMBOL_GPL(devm_clk_get_optional_enabled); 101 102struct clk *devm_clk_get_optional_enabled_with_rate(struct device *dev, 103 const char *id, 104 unsigned long rate) 105{ 106 struct clk *clk; 107 int ret; 108 109 clk = __devm_clk_get(dev, id, clk_get_optional, NULL, 110 clk_disable_unprepare); 111 if (IS_ERR(clk)) 112 return ERR_CAST(clk); 113 114 ret = clk_set_rate(clk, rate); 115 if (ret) 116 goto out_put_clk; 117 118 ret = clk_prepare_enable(clk); 119 if (ret) 120 goto out_put_clk; 121 122 return clk; 123 124out_put_clk: 125 devm_clk_put(dev, clk); 126 return ERR_PTR(ret); 127} 128EXPORT_SYMBOL_GPL(devm_clk_get_optional_enabled_with_rate); 129 130struct clk_bulk_devres { 131 struct clk_bulk_data *clks; 132 int num_clks; 133}; 134 135static void devm_clk_bulk_release(struct device *dev, void *res) 136{ 137 struct clk_bulk_devres *devres = res; 138 139 clk_bulk_put(devres->num_clks, devres->clks); 140} 141 142static int __devm_clk_bulk_get(struct device *dev, int num_clks, 143 struct clk_bulk_data *clks, bool optional) 144{ 145 struct clk_bulk_devres *devres; 146 int ret; 147 148 devres = devres_alloc(devm_clk_bulk_release, 149 sizeof(*devres), GFP_KERNEL); 150 if (!devres) 151 return -ENOMEM; 152 153 if (optional) 154 ret = clk_bulk_get_optional(dev, num_clks, clks); 155 else 156 ret = clk_bulk_get(dev, num_clks, clks); 157 if (!ret) { 158 devres->clks = clks; 159 devres->num_clks = num_clks; 160 devres_add(dev, devres); 161 } else { 162 devres_free(devres); 163 } 164 165 return ret; 166} 167 168int __must_check devm_clk_bulk_get(struct device *dev, int num_clks, 169 struct clk_bulk_data *clks) 170{ 171 return __devm_clk_bulk_get(dev, num_clks, clks, false); 172} 173EXPORT_SYMBOL_GPL(devm_clk_bulk_get); 174 175int __must_check devm_clk_bulk_get_optional(struct device *dev, int num_clks, 176 struct clk_bulk_data *clks) 177{ 178 return __devm_clk_bulk_get(dev, num_clks, clks, true); 179} 180EXPORT_SYMBOL_GPL(devm_clk_bulk_get_optional); 181 182static void devm_clk_bulk_release_enable(struct device *dev, void *res) 183{ 184 struct clk_bulk_devres *devres = res; 185 186 clk_bulk_disable_unprepare(devres->num_clks, devres->clks); 187 clk_bulk_put(devres->num_clks, devres->clks); 188} 189 190static int __devm_clk_bulk_get_enable(struct device *dev, int num_clks, 191 struct clk_bulk_data *clks, bool optional) 192{ 193 struct clk_bulk_devres *devres; 194 int ret; 195 196 devres = devres_alloc(devm_clk_bulk_release_enable, 197 sizeof(*devres), GFP_KERNEL); 198 if (!devres) 199 return -ENOMEM; 200 201 if (optional) 202 ret = clk_bulk_get_optional(dev, num_clks, clks); 203 else 204 ret = clk_bulk_get(dev, num_clks, clks); 205 if (ret) 206 goto err_clk_get; 207 208 ret = clk_bulk_prepare_enable(num_clks, clks); 209 if (ret) 210 goto err_clk_prepare; 211 212 devres->clks = clks; 213 devres->num_clks = num_clks; 214 devres_add(dev, devres); 215 216 return 0; 217 218err_clk_prepare: 219 clk_bulk_put(num_clks, clks); 220err_clk_get: 221 devres_free(devres); 222 return ret; 223} 224 225int __must_check devm_clk_bulk_get_optional_enable(struct device *dev, int num_clks, 226 struct clk_bulk_data *clks) 227{ 228 return __devm_clk_bulk_get_enable(dev, num_clks, clks, true); 229} 230EXPORT_SYMBOL_GPL(devm_clk_bulk_get_optional_enable); 231 232static void devm_clk_bulk_release_all(struct device *dev, void *res) 233{ 234 struct clk_bulk_devres *devres = res; 235 236 clk_bulk_put_all(devres->num_clks, devres->clks); 237} 238 239int __must_check devm_clk_bulk_get_all(struct device *dev, 240 struct clk_bulk_data **clks) 241{ 242 struct clk_bulk_devres *devres; 243 int ret; 244 245 devres = devres_alloc(devm_clk_bulk_release_all, 246 sizeof(*devres), GFP_KERNEL); 247 if (!devres) 248 return -ENOMEM; 249 250 ret = clk_bulk_get_all(dev, &devres->clks); 251 if (ret > 0) { 252 *clks = devres->clks; 253 devres->num_clks = ret; 254 devres_add(dev, devres); 255 } else { 256 devres_free(devres); 257 } 258 259 return ret; 260} 261EXPORT_SYMBOL_GPL(devm_clk_bulk_get_all); 262 263static void devm_clk_bulk_release_all_enable(struct device *dev, void *res) 264{ 265 struct clk_bulk_devres *devres = res; 266 267 clk_bulk_disable_unprepare(devres->num_clks, devres->clks); 268 clk_bulk_put_all(devres->num_clks, devres->clks); 269} 270 271int __must_check devm_clk_bulk_get_all_enabled(struct device *dev, 272 struct clk_bulk_data **clks) 273{ 274 struct clk_bulk_devres *devres; 275 int ret; 276 277 devres = devres_alloc(devm_clk_bulk_release_all_enable, 278 sizeof(*devres), GFP_KERNEL); 279 if (!devres) 280 return -ENOMEM; 281 282 ret = clk_bulk_get_all(dev, &devres->clks); 283 if (ret > 0) { 284 *clks = devres->clks; 285 devres->num_clks = ret; 286 } else { 287 devres_free(devres); 288 return ret; 289 } 290 291 ret = clk_bulk_prepare_enable(devres->num_clks, *clks); 292 if (!ret) { 293 devres_add(dev, devres); 294 } else { 295 clk_bulk_put_all(devres->num_clks, devres->clks); 296 devres_free(devres); 297 return ret; 298 } 299 300 return devres->num_clks; 301} 302EXPORT_SYMBOL_GPL(devm_clk_bulk_get_all_enabled); 303 304static int devm_clk_match(struct device *dev, void *res, void *data) 305{ 306 struct clk **c = res; 307 if (!c || !*c) { 308 WARN_ON(!c || !*c); 309 return 0; 310 } 311 return *c == data; 312} 313 314void devm_clk_put(struct device *dev, struct clk *clk) 315{ 316 int ret; 317 318 ret = devres_release(dev, devm_clk_release, devm_clk_match, clk); 319 320 WARN_ON(ret); 321} 322EXPORT_SYMBOL(devm_clk_put); 323 324struct clk *devm_get_clk_from_child(struct device *dev, 325 struct device_node *np, const char *con_id) 326{ 327 struct devm_clk_state *state; 328 struct clk *clk; 329 330 state = devres_alloc(devm_clk_release, sizeof(*state), GFP_KERNEL); 331 if (!state) 332 return ERR_PTR(-ENOMEM); 333 334 clk = of_clk_get_by_name(np, con_id); 335 if (!IS_ERR(clk)) { 336 state->clk = clk; 337 devres_add(dev, state); 338 } else { 339 devres_free(state); 340 } 341 342 return clk; 343} 344EXPORT_SYMBOL(devm_get_clk_from_child);