Author | Tokens | Token Proportion | Commits | Commit Proportion |
---|---|---|---|---|
Maciej Purski | 441 | 13.77% | 2 | 2.38% |
Rajendra Nayak | 331 | 10.33% | 2 | 2.38% |
Corentin Labbe | 309 | 9.65% | 1 | 1.19% |
Matti Vaittinen | 300 | 9.37% | 1 | 1.19% |
Thierry Reding | 203 | 6.34% | 1 | 1.19% |
Charles Keepax | 202 | 6.31% | 5 | 5.95% |
Mark Brown | 166 | 5.18% | 12 | 14.29% |
Laxman Dewangan | 155 | 4.84% | 5 | 5.95% |
Chanwoo Choi | 143 | 4.46% | 1 | 1.19% |
Javier Martinez Canillas | 135 | 4.21% | 4 | 4.76% |
Dmitry Osipenko | 123 | 3.84% | 2 | 2.38% |
David Collins | 118 | 3.68% | 1 | 1.19% |
Sergei Shtylyov | 79 | 2.47% | 1 | 1.19% |
Matthias Kaehlcke | 74 | 2.31% | 1 | 1.19% |
Chunyan Zhang | 52 | 1.62% | 2 | 2.38% |
Stephen Boyd | 43 | 1.34% | 6 | 7.14% |
Yadwinder Singh Brar | 39 | 1.22% | 2 | 2.38% |
Marco Felsch | 35 | 1.09% | 2 | 2.38% |
Doug Anderson | 33 | 1.03% | 2 | 2.38% |
Oleksij Rempel | 32 | 1.00% | 2 | 2.38% |
Krzysztof Kozlowski | 29 | 0.91% | 3 | 3.57% |
Stephen Warren | 21 | 0.66% | 2 | 2.38% |
Liam Girdwood | 18 | 0.56% | 2 | 2.38% |
Cristian Marussi | 16 | 0.50% | 1 | 1.19% |
Shawn Guo | 14 | 0.44% | 1 | 1.19% |
J Keerthy | 12 | 0.37% | 1 | 1.19% |
Andrei.Stefanescu | 12 | 0.37% | 1 | 1.19% |
Axel Lin | 12 | 0.37% | 3 | 3.57% |
Rob Herring | 12 | 0.37% | 1 | 1.19% |
Kishon Vijay Abraham I | 11 | 0.34% | 1 | 1.19% |
Liang He | 6 | 0.19% | 1 | 1.19% |
Christophe Jaillet | 6 | 0.19% | 2 | 2.38% |
Björn Andersson | 5 | 0.16% | 1 | 1.19% |
Peng Wu | 3 | 0.09% | 1 | 1.19% |
Frank Rowand | 3 | 0.09% | 1 | 1.19% |
Nishka Dasgupta | 2 | 0.06% | 1 | 1.19% |
ChiYuan Huang | 2 | 0.06% | 1 | 1.19% |
Thomas Gleixner | 2 | 0.06% | 1 | 1.19% |
Yue haibing | 1 | 0.03% | 1 | 1.19% |
Lee Jones | 1 | 0.03% | 1 | 1.19% |
Suzuki K. Poulose | 1 | 0.03% | 1 | 1.19% |
Jiapeng Chong | 1 | 0.03% | 1 | 1.19% |
Total | 3203 | 84 |
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804
// SPDX-License-Identifier: GPL-2.0-or-later /* * OF helpers for regulator framework * * Copyright (C) 2011 Texas Instruments, Inc. * Rajendra Nayak <rnayak@ti.com> */ #include <linux/module.h> #include <linux/slab.h> #include <linux/of.h> #include <linux/regulator/machine.h> #include <linux/regulator/driver.h> #include <linux/regulator/of_regulator.h> #include "internal.h" static const char *const regulator_states[PM_SUSPEND_MAX + 1] = { [PM_SUSPEND_STANDBY] = "regulator-state-standby", [PM_SUSPEND_MEM] = "regulator-state-mem", [PM_SUSPEND_MAX] = "regulator-state-disk", }; static void fill_limit(int *limit, int val) { if (val) if (val == 1) *limit = REGULATOR_NOTIF_LIMIT_ENABLE; else *limit = val; else *limit = REGULATOR_NOTIF_LIMIT_DISABLE; } static void of_get_regulator_prot_limits(struct device_node *np, struct regulation_constraints *constraints) { u32 pval; int i; static const char *const props[] = { "regulator-oc-%s-microamp", "regulator-ov-%s-microvolt", "regulator-temp-%s-kelvin", "regulator-uv-%s-microvolt", }; struct notification_limit *limits[] = { &constraints->over_curr_limits, &constraints->over_voltage_limits, &constraints->temp_limits, &constraints->under_voltage_limits, }; bool set[4] = {0}; /* Protection limits: */ for (i = 0; i < ARRAY_SIZE(props); i++) { char prop[255]; bool found; int j; static const char *const lvl[] = { "protection", "error", "warn" }; int *l[] = { &limits[i]->prot, &limits[i]->err, &limits[i]->warn, }; for (j = 0; j < ARRAY_SIZE(lvl); j++) { snprintf(prop, 255, props[i], lvl[j]); found = !of_property_read_u32(np, prop, &pval); if (found) fill_limit(l[j], pval); set[i] |= found; } } constraints->over_current_detection = set[0]; constraints->over_voltage_detection = set[1]; constraints->over_temp_detection = set[2]; constraints->under_voltage_detection = set[3]; } static int of_get_regulation_constraints(struct device *dev, struct device_node *np, struct regulator_init_data **init_data, const struct regulator_desc *desc) { struct regulation_constraints *constraints = &(*init_data)->constraints; struct regulator_state *suspend_state; struct device_node *suspend_np; unsigned int mode; int ret, i, len; int n_phandles; u32 pval; n_phandles = of_count_phandle_with_args(np, "regulator-coupled-with", NULL); n_phandles = max(n_phandles, 0); constraints->name = of_get_property(np, "regulator-name", NULL); if (!of_property_read_u32(np, "regulator-min-microvolt", &pval)) constraints->min_uV = pval; if (!of_property_read_u32(np, "regulator-max-microvolt", &pval)) constraints->max_uV = pval; /* Voltage change possible? */ if (constraints->min_uV != constraints->max_uV) constraints->valid_ops_mask |= REGULATOR_CHANGE_VOLTAGE; /* Do we have a voltage range, if so try to apply it? */ if (constraints->min_uV && constraints->max_uV) constraints->apply_uV = true; if (!of_property_read_u32(np, "regulator-microvolt-offset", &pval)) constraints->uV_offset = pval; if (!of_property_read_u32(np, "regulator-min-microamp", &pval)) constraints->min_uA = pval; if (!of_property_read_u32(np, "regulator-max-microamp", &pval)) constraints->max_uA = pval; if (!of_property_read_u32(np, "regulator-input-current-limit-microamp", &pval)) constraints->ilim_uA = pval; /* Current change possible? */ if (constraints->min_uA != constraints->max_uA) constraints->valid_ops_mask |= REGULATOR_CHANGE_CURRENT; constraints->boot_on = of_property_read_bool(np, "regulator-boot-on"); constraints->always_on = of_property_read_bool(np, "regulator-always-on"); if (!constraints->always_on) /* status change should be possible. */ constraints->valid_ops_mask |= REGULATOR_CHANGE_STATUS; constraints->pull_down = of_property_read_bool(np, "regulator-pull-down"); constraints->system_critical = of_property_read_bool(np, "system-critical-regulator"); if (of_property_read_bool(np, "regulator-allow-bypass")) constraints->valid_ops_mask |= REGULATOR_CHANGE_BYPASS; if (of_property_read_bool(np, "regulator-allow-set-load")) constraints->valid_ops_mask |= REGULATOR_CHANGE_DRMS; ret = of_property_read_u32(np, "regulator-ramp-delay", &pval); if (!ret) { if (pval) constraints->ramp_delay = pval; else constraints->ramp_disable = true; } ret = of_property_read_u32(np, "regulator-settling-time-us", &pval); if (!ret) constraints->settling_time = pval; ret = of_property_read_u32(np, "regulator-settling-time-up-us", &pval); if (!ret) constraints->settling_time_up = pval; if (constraints->settling_time_up && constraints->settling_time) { pr_warn("%pOFn: ambiguous configuration for settling time, ignoring 'regulator-settling-time-up-us'\n", np); constraints->settling_time_up = 0; } ret = of_property_read_u32(np, "regulator-settling-time-down-us", &pval); if (!ret) constraints->settling_time_down = pval; if (constraints->settling_time_down && constraints->settling_time) { pr_warn("%pOFn: ambiguous configuration for settling time, ignoring 'regulator-settling-time-down-us'\n", np); constraints->settling_time_down = 0; } ret = of_property_read_u32(np, "regulator-enable-ramp-delay", &pval); if (!ret) constraints->enable_time = pval; ret = of_property_read_u32(np, "regulator-uv-survival-time-ms", &pval); if (!ret) constraints->uv_less_critical_window_ms = pval; else constraints->uv_less_critical_window_ms = REGULATOR_DEF_UV_LESS_CRITICAL_WINDOW_MS; constraints->soft_start = of_property_read_bool(np, "regulator-soft-start"); ret = of_property_read_u32(np, "regulator-active-discharge", &pval); if (!ret) { constraints->active_discharge = (pval) ? REGULATOR_ACTIVE_DISCHARGE_ENABLE : REGULATOR_ACTIVE_DISCHARGE_DISABLE; } if (!of_property_read_u32(np, "regulator-initial-mode", &pval)) { if (desc && desc->of_map_mode) { mode = desc->of_map_mode(pval); if (mode == REGULATOR_MODE_INVALID) pr_err("%pOFn: invalid mode %u\n", np, pval); else constraints->initial_mode = mode; } else { pr_warn("%pOFn: mapping for mode %d not defined\n", np, pval); } } len = of_property_count_elems_of_size(np, "regulator-allowed-modes", sizeof(u32)); if (len > 0) { if (desc && desc->of_map_mode) { for (i = 0; i < len; i++) { ret = of_property_read_u32_index(np, "regulator-allowed-modes", i, &pval); if (ret) { pr_err("%pOFn: couldn't read allowed modes index %d, ret=%d\n", np, i, ret); break; } mode = desc->of_map_mode(pval); if (mode == REGULATOR_MODE_INVALID) pr_err("%pOFn: invalid regulator-allowed-modes element %u\n", np, pval); else constraints->valid_modes_mask |= mode; } if (constraints->valid_modes_mask) constraints->valid_ops_mask |= REGULATOR_CHANGE_MODE; } else { pr_warn("%pOFn: mode mapping not defined\n", np); } } if (!of_property_read_u32(np, "regulator-system-load", &pval)) constraints->system_load = pval; if (n_phandles) { constraints->max_spread = devm_kzalloc(dev, sizeof(*constraints->max_spread) * n_phandles, GFP_KERNEL); if (!constraints->max_spread) return -ENOMEM; of_property_read_u32_array(np, "regulator-coupled-max-spread", constraints->max_spread, n_phandles); } if (!of_property_read_u32(np, "regulator-max-step-microvolt", &pval)) constraints->max_uV_step = pval; constraints->over_current_protection = of_property_read_bool(np, "regulator-over-current-protection"); of_get_regulator_prot_limits(np, constraints); for (i = 0; i < ARRAY_SIZE(regulator_states); i++) { switch (i) { case PM_SUSPEND_MEM: suspend_state = &constraints->state_mem; break; case PM_SUSPEND_MAX: suspend_state = &constraints->state_disk; break; case PM_SUSPEND_STANDBY: suspend_state = &constraints->state_standby; break; case PM_SUSPEND_ON: case PM_SUSPEND_TO_IDLE: default: continue; } suspend_np = of_get_child_by_name(np, regulator_states[i]); if (!suspend_np) continue; if (!suspend_state) { of_node_put(suspend_np); continue; } if (!of_property_read_u32(suspend_np, "regulator-mode", &pval)) { if (desc && desc->of_map_mode) { mode = desc->of_map_mode(pval); if (mode == REGULATOR_MODE_INVALID) pr_err("%pOFn: invalid mode %u\n", np, pval); else suspend_state->mode = mode; } else { pr_warn("%pOFn: mapping for mode %d not defined\n", np, pval); } } if (of_property_read_bool(suspend_np, "regulator-on-in-suspend")) suspend_state->enabled = ENABLE_IN_SUSPEND; else if (of_property_read_bool(suspend_np, "regulator-off-in-suspend")) suspend_state->enabled = DISABLE_IN_SUSPEND; if (!of_property_read_u32(suspend_np, "regulator-suspend-min-microvolt", &pval)) suspend_state->min_uV = pval; if (!of_property_read_u32(suspend_np, "regulator-suspend-max-microvolt", &pval)) suspend_state->max_uV = pval; if (!of_property_read_u32(suspend_np, "regulator-suspend-microvolt", &pval)) suspend_state->uV = pval; else /* otherwise use min_uV as default suspend voltage */ suspend_state->uV = suspend_state->min_uV; if (of_property_read_bool(suspend_np, "regulator-changeable-in-suspend")) suspend_state->changeable = true; if (i == PM_SUSPEND_MEM) constraints->initial_state = PM_SUSPEND_MEM; of_node_put(suspend_np); suspend_state = NULL; suspend_np = NULL; } return 0; } /** * of_get_regulator_init_data - extract regulator_init_data structure info * @dev: device requesting for regulator_init_data * @node: regulator device node * @desc: regulator description * * Populates regulator_init_data structure by extracting data from device * tree node, returns a pointer to the populated structure or NULL if memory * alloc fails. */ struct regulator_init_data *of_get_regulator_init_data(struct device *dev, struct device_node *node, const struct regulator_desc *desc) { struct regulator_init_data *init_data; if (!node) return NULL; init_data = devm_kzalloc(dev, sizeof(*init_data), GFP_KERNEL); if (!init_data) return NULL; /* Out of memory? */ if (of_get_regulation_constraints(dev, node, &init_data, desc)) return NULL; return init_data; } EXPORT_SYMBOL_GPL(of_get_regulator_init_data); struct devm_of_regulator_matches { struct of_regulator_match *matches; unsigned int num_matches; }; static void devm_of_regulator_put_matches(struct device *dev, void *res) { struct devm_of_regulator_matches *devm_matches = res; int i; for (i = 0; i < devm_matches->num_matches; i++) of_node_put(devm_matches->matches[i].of_node); } /** * of_regulator_match - extract multiple regulator init data from device tree. * @dev: device requesting the data * @node: parent device node of the regulators * @matches: match table for the regulators * @num_matches: number of entries in match table * * This function uses a match table specified by the regulator driver to * parse regulator init data from the device tree. @node is expected to * contain a set of child nodes, each providing the init data for one * regulator. The data parsed from a child node will be matched to a regulator * based on either the deprecated property regulator-compatible if present, * or otherwise the child node's name. Note that the match table is modified * in place and an additional of_node reference is taken for each matched * regulator. * * Returns the number of matches found or a negative error code on failure. */ int of_regulator_match(struct device *dev, struct device_node *node, struct of_regulator_match *matches, unsigned int num_matches) { unsigned int count = 0; unsigned int i; const char *name; struct device_node *child; struct devm_of_regulator_matches *devm_matches; if (!dev || !node) return -EINVAL; devm_matches = devres_alloc(devm_of_regulator_put_matches, sizeof(struct devm_of_regulator_matches), GFP_KERNEL); if (!devm_matches) return -ENOMEM; devm_matches->matches = matches; devm_matches->num_matches = num_matches; devres_add(dev, devm_matches); for (i = 0; i < num_matches; i++) { struct of_regulator_match *match = &matches[i]; match->init_data = NULL; match->of_node = NULL; } for_each_child_of_node(node, child) { name = of_get_property(child, "regulator-compatible", NULL); if (!name) name = child->name; for (i = 0; i < num_matches; i++) { struct of_regulator_match *match = &matches[i]; if (match->of_node) continue; if (strcmp(match->name, name)) continue; match->init_data = of_get_regulator_init_data(dev, child, match->desc); if (!match->init_data) { dev_err(dev, "failed to parse DT for regulator %pOFn\n", child); of_node_put(child); return -EINVAL; } match->of_node = of_node_get(child); count++; break; } } return count; } EXPORT_SYMBOL_GPL(of_regulator_match); static struct device_node *regulator_of_get_init_node(struct device *dev, const struct regulator_desc *desc) { struct device_node *search, *child; const char *name; if (!dev->of_node || !desc->of_match) return NULL; if (desc->regulators_node) { search = of_get_child_by_name(dev->of_node, desc->regulators_node); } else { search = of_node_get(dev->of_node); if (!strcmp(desc->of_match, search->name)) return search; } if (!search) { dev_dbg(dev, "Failed to find regulator container node '%s'\n", desc->regulators_node); return NULL; } for_each_available_child_of_node(search, child) { name = of_get_property(child, "regulator-compatible", NULL); if (!name) { if (!desc->of_match_full_name) name = child->name; else name = child->full_name; } if (!strcmp(desc->of_match, name)) { of_node_put(search); /* * 'of_node_get(child)' is already performed by the * for_each loop. */ return child; } } of_node_put(search); return NULL; } struct regulator_init_data *regulator_of_get_init_data(struct device *dev, const struct regulator_desc *desc, struct regulator_config *config, struct device_node **node) { struct device_node *child; struct regulator_init_data *init_data = NULL; child = regulator_of_get_init_node(config->dev, desc); if (!child) return NULL; init_data = of_get_regulator_init_data(dev, child, desc); if (!init_data) { dev_err(dev, "failed to parse DT for regulator %pOFn\n", child); goto error; } if (desc->of_parse_cb) { int ret; ret = desc->of_parse_cb(child, desc, config); if (ret) { if (ret == -EPROBE_DEFER) { of_node_put(child); return ERR_PTR(-EPROBE_DEFER); } dev_err(dev, "driver callback failed to parse DT for regulator %pOFn\n", child); goto error; } } *node = child; return init_data; error: of_node_put(child); return NULL; } struct regulator_dev *of_find_regulator_by_node(struct device_node *np) { struct device *dev; dev = class_find_device_by_of_node(®ulator_class, np); return dev ? dev_to_rdev(dev) : NULL; } /* * Returns number of regulators coupled with rdev. */ int of_get_n_coupled(struct regulator_dev *rdev) { struct device_node *node = rdev->dev.of_node; int n_phandles; n_phandles = of_count_phandle_with_args(node, "regulator-coupled-with", NULL); return (n_phandles > 0) ? n_phandles : 0; } /* Looks for "to_find" device_node in src's "regulator-coupled-with" property */ static bool of_coupling_find_node(struct device_node *src, struct device_node *to_find, int *index) { int n_phandles, i; bool found = false; n_phandles = of_count_phandle_with_args(src, "regulator-coupled-with", NULL); for (i = 0; i < n_phandles; i++) { struct device_node *tmp = of_parse_phandle(src, "regulator-coupled-with", i); if (!tmp) break; /* found */ if (tmp == to_find) found = true; of_node_put(tmp); if (found) { *index = i; break; } } return found; } /** * of_check_coupling_data - Parse rdev's coupling properties and check data * consistency * @rdev: pointer to regulator_dev whose data is checked * * Function checks if all the following conditions are met: * - rdev's max_spread is greater than 0 * - all coupled regulators have the same max_spread * - all coupled regulators have the same number of regulator_dev phandles * - all regulators are linked to each other * * Returns true if all conditions are met. */ bool of_check_coupling_data(struct regulator_dev *rdev) { struct device_node *node = rdev->dev.of_node; int n_phandles = of_get_n_coupled(rdev); struct device_node *c_node; int index; int i; bool ret = true; /* iterate over rdev's phandles */ for (i = 0; i < n_phandles; i++) { int max_spread = rdev->constraints->max_spread[i]; int c_max_spread, c_n_phandles; if (max_spread <= 0) { dev_err(&rdev->dev, "max_spread value invalid\n"); return false; } c_node = of_parse_phandle(node, "regulator-coupled-with", i); if (!c_node) ret = false; c_n_phandles = of_count_phandle_with_args(c_node, "regulator-coupled-with", NULL); if (c_n_phandles != n_phandles) { dev_err(&rdev->dev, "number of coupled reg phandles mismatch\n"); ret = false; goto clean; } if (!of_coupling_find_node(c_node, node, &index)) { dev_err(&rdev->dev, "missing 2-way linking for coupled regulators\n"); ret = false; goto clean; } if (of_property_read_u32_index(c_node, "regulator-coupled-max-spread", index, &c_max_spread)) { ret = false; goto clean; } if (c_max_spread != max_spread) { dev_err(&rdev->dev, "coupled regulators max_spread mismatch\n"); ret = false; goto clean; } clean: of_node_put(c_node); if (!ret) break; } return ret; } /** * of_parse_coupled_regulator() - Get regulator_dev pointer from rdev's property * @rdev: Pointer to regulator_dev, whose DTS is used as a source to parse * "regulator-coupled-with" property * @index: Index in phandles array * * Returns the regulator_dev pointer parsed from DTS. If it has not been yet * registered, returns NULL */ struct regulator_dev *of_parse_coupled_regulator(struct regulator_dev *rdev, int index) { struct device_node *node = rdev->dev.of_node; struct device_node *c_node; struct regulator_dev *c_rdev; c_node = of_parse_phandle(node, "regulator-coupled-with", index); if (!c_node) return NULL; c_rdev = of_find_regulator_by_node(c_node); of_node_put(c_node); return c_rdev; } /* * Check if name is a supply name according to the '*-supply' pattern * return 0 if false * return length of supply name without the -supply */ static int is_supply_name(const char *name) { int strs, i; strs = strlen(name); /* string need to be at minimum len(x-supply) */ if (strs < 8) return 0; for (i = strs - 6; i > 0; i--) { /* find first '-' and check if right part is supply */ if (name[i] != '-') continue; if (strcmp(name + i + 1, "supply") != 0) return 0; return i; } return 0; } /* * of_regulator_bulk_get_all - get multiple regulator consumers * * @dev: Device to supply * @np: device node to search for consumers * @consumers: Configuration of consumers; clients are stored here. * * @return number of regulators on success, an errno on failure. * * This helper function allows drivers to get several regulator * consumers in one operation. If any of the regulators cannot be * acquired then any regulators that were allocated will be freed * before returning to the caller. */ int of_regulator_bulk_get_all(struct device *dev, struct device_node *np, struct regulator_bulk_data **consumers) { int num_consumers = 0; struct regulator *tmp; struct property *prop; int i, n = 0, ret; char name[64]; *consumers = NULL; /* * first pass: get numbers of xxx-supply * second pass: fill consumers */ restart: for_each_property_of_node(np, prop) { i = is_supply_name(prop->name); if (i == 0) continue; if (!*consumers) { num_consumers++; continue; } else { memcpy(name, prop->name, i); name[i] = '\0'; tmp = regulator_get(dev, name); if (IS_ERR(tmp)) { ret = -EINVAL; goto error; } (*consumers)[n].consumer = tmp; n++; continue; } } if (*consumers) return num_consumers; if (num_consumers == 0) return 0; *consumers = kmalloc_array(num_consumers, sizeof(struct regulator_bulk_data), GFP_KERNEL); if (!*consumers) return -ENOMEM; goto restart; error: while (--n >= 0) regulator_put(consumers[n]->consumer); return ret; } EXPORT_SYMBOL_GPL(of_regulator_bulk_get_all);
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with Cregit http://github.com/cregit/cregit
Version 2.0-RC1