Commit a4ab1086 authored by Tony Lindgren's avatar Tony Lindgren Committed by Linus Walleij

pinctrl: single: Fix group and function selector use

We must use a mutex around the generic_add functions and save the
function and group selector in case we need to remove them. Otherwise
the selector use will be racy for deferred probe at least.

Note that struct device_node *np is unused in pcs_add_function() we
remove that too and fix a checkpatch warning for bare unsigned while
at it.

Fixes: 571aec4d ("pinctrl: single: Use generic pinmux helpers for
managing functions")
Reported-by: default avatarH. Nikolaus Schaller <hns@goldelico.com>
Cc: Christ van Willegen <cvwillegen@gmail.com>
Cc: Haojian Zhuang <haojian.zhuang@linaro.org>
Cc: Jacopo Mondi <jacopo+renesas@jmondi.org>
Cc: Paul Cercueil <paul@crapouillou.net>
Cc: Sean Wang <sean.wang@mediatek.com>
Signed-off-by: default avatarTony Lindgren <tony@atomide.com>
Tested-By: default avatarH. Nikolaus Schaller <hns@goldelico.com>
Reviewed-by: default avatarAndy Shevchenko <andy.shevchenko@gmail.com>
Signed-off-by: default avatarLinus Walleij <linus.walleij@linaro.org>
parent f913cfce
...@@ -747,38 +747,44 @@ static int pcs_allocate_pin_table(struct pcs_device *pcs) ...@@ -747,38 +747,44 @@ static int pcs_allocate_pin_table(struct pcs_device *pcs)
/** /**
* pcs_add_function() - adds a new function to the function list * pcs_add_function() - adds a new function to the function list
* @pcs: pcs driver instance * @pcs: pcs driver instance
* @np: device node of the mux entry * @fcn: new function allocated
* @name: name of the function * @name: name of the function
* @vals: array of mux register value pairs used by the function * @vals: array of mux register value pairs used by the function
* @nvals: number of mux register value pairs * @nvals: number of mux register value pairs
* @pgnames: array of pingroup names for the function * @pgnames: array of pingroup names for the function
* @npgnames: number of pingroup names * @npgnames: number of pingroup names
*
* Caller must take care of locking.
*/ */
static struct pcs_function *pcs_add_function(struct pcs_device *pcs, static int pcs_add_function(struct pcs_device *pcs,
struct device_node *np, struct pcs_function **fcn,
const char *name, const char *name,
struct pcs_func_vals *vals, struct pcs_func_vals *vals,
unsigned nvals, unsigned int nvals,
const char **pgnames, const char **pgnames,
unsigned npgnames) unsigned int npgnames)
{ {
struct pcs_function *function; struct pcs_function *function;
int res; int selector;
function = devm_kzalloc(pcs->dev, sizeof(*function), GFP_KERNEL); function = devm_kzalloc(pcs->dev, sizeof(*function), GFP_KERNEL);
if (!function) if (!function)
return NULL; return -ENOMEM;
function->vals = vals; function->vals = vals;
function->nvals = nvals; function->nvals = nvals;
res = pinmux_generic_add_function(pcs->pctl, name, selector = pinmux_generic_add_function(pcs->pctl, name,
pgnames, npgnames, pgnames, npgnames,
function); function);
if (res) if (selector < 0) {
return NULL; devm_kfree(pcs->dev, function);
*fcn = NULL;
} else {
*fcn = function;
}
return function; return selector;
} }
/** /**
...@@ -979,8 +985,8 @@ static int pcs_parse_one_pinctrl_entry(struct pcs_device *pcs, ...@@ -979,8 +985,8 @@ static int pcs_parse_one_pinctrl_entry(struct pcs_device *pcs,
{ {
const char *name = "pinctrl-single,pins"; const char *name = "pinctrl-single,pins";
struct pcs_func_vals *vals; struct pcs_func_vals *vals;
int rows, *pins, found = 0, res = -ENOMEM, i; int rows, *pins, found = 0, res = -ENOMEM, i, fsel, gsel;
struct pcs_function *function; struct pcs_function *function = NULL;
rows = pinctrl_count_index_with_args(np, name); rows = pinctrl_count_index_with_args(np, name);
if (rows <= 0) { if (rows <= 0) {
...@@ -1030,21 +1036,25 @@ static int pcs_parse_one_pinctrl_entry(struct pcs_device *pcs, ...@@ -1030,21 +1036,25 @@ static int pcs_parse_one_pinctrl_entry(struct pcs_device *pcs,
} }
pgnames[0] = np->name; pgnames[0] = np->name;
function = pcs_add_function(pcs, np, np->name, vals, found, pgnames, 1); mutex_lock(&pcs->mutex);
if (!function) { fsel = pcs_add_function(pcs, &function, np->name, vals, found,
res = -ENOMEM; pgnames, 1);
if (fsel < 0) {
res = fsel;
goto free_pins; goto free_pins;
} }
res = pinctrl_generic_add_group(pcs->pctl, np->name, pins, found, pcs); gsel = pinctrl_generic_add_group(pcs->pctl, np->name, pins, found, pcs);
if (res < 0) if (gsel < 0) {
res = gsel;
goto free_function; goto free_function;
}
(*map)->type = PIN_MAP_TYPE_MUX_GROUP; (*map)->type = PIN_MAP_TYPE_MUX_GROUP;
(*map)->data.mux.group = np->name; (*map)->data.mux.group = np->name;
(*map)->data.mux.function = np->name; (*map)->data.mux.function = np->name;
if (PCS_HAS_PINCONF) { if (PCS_HAS_PINCONF && function) {
res = pcs_parse_pinconf(pcs, np, function, map); res = pcs_parse_pinconf(pcs, np, function, map);
if (res) if (res)
goto free_pingroups; goto free_pingroups;
...@@ -1052,14 +1062,16 @@ static int pcs_parse_one_pinctrl_entry(struct pcs_device *pcs, ...@@ -1052,14 +1062,16 @@ static int pcs_parse_one_pinctrl_entry(struct pcs_device *pcs,
} else { } else {
*num_maps = 1; *num_maps = 1;
} }
mutex_unlock(&pcs->mutex);
return 0; return 0;
free_pingroups: free_pingroups:
pinctrl_generic_remove_last_group(pcs->pctl); pinctrl_generic_remove_group(pcs->pctl, gsel);
*num_maps = 1; *num_maps = 1;
free_function: free_function:
pinmux_generic_remove_last_function(pcs->pctl); pinmux_generic_remove_function(pcs->pctl, fsel);
mutex_unlock(&pcs->mutex);
free_pins: free_pins:
devm_kfree(pcs->dev, pins); devm_kfree(pcs->dev, pins);
...@@ -1077,9 +1089,9 @@ static int pcs_parse_bits_in_pinctrl_entry(struct pcs_device *pcs, ...@@ -1077,9 +1089,9 @@ static int pcs_parse_bits_in_pinctrl_entry(struct pcs_device *pcs,
{ {
const char *name = "pinctrl-single,bits"; const char *name = "pinctrl-single,bits";
struct pcs_func_vals *vals; struct pcs_func_vals *vals;
int rows, *pins, found = 0, res = -ENOMEM, i; int rows, *pins, found = 0, res = -ENOMEM, i, fsel, gsel;
int npins_in_row; int npins_in_row;
struct pcs_function *function; struct pcs_function *function = NULL;
rows = pinctrl_count_index_with_args(np, name); rows = pinctrl_count_index_with_args(np, name);
if (rows <= 0) { if (rows <= 0) {
...@@ -1166,15 +1178,19 @@ static int pcs_parse_bits_in_pinctrl_entry(struct pcs_device *pcs, ...@@ -1166,15 +1178,19 @@ static int pcs_parse_bits_in_pinctrl_entry(struct pcs_device *pcs,
} }
pgnames[0] = np->name; pgnames[0] = np->name;
function = pcs_add_function(pcs, np, np->name, vals, found, pgnames, 1); mutex_lock(&pcs->mutex);
if (!function) { fsel = pcs_add_function(pcs, &function, np->name, vals, found,
res = -ENOMEM; pgnames, 1);
if (fsel < 0) {
res = fsel;
goto free_pins; goto free_pins;
} }
res = pinctrl_generic_add_group(pcs->pctl, np->name, pins, found, pcs); gsel = pinctrl_generic_add_group(pcs->pctl, np->name, pins, found, pcs);
if (res < 0) if (gsel < 0) {
res = gsel;
goto free_function; goto free_function;
}
(*map)->type = PIN_MAP_TYPE_MUX_GROUP; (*map)->type = PIN_MAP_TYPE_MUX_GROUP;
(*map)->data.mux.group = np->name; (*map)->data.mux.group = np->name;
...@@ -1186,13 +1202,16 @@ static int pcs_parse_bits_in_pinctrl_entry(struct pcs_device *pcs, ...@@ -1186,13 +1202,16 @@ static int pcs_parse_bits_in_pinctrl_entry(struct pcs_device *pcs,
} }
*num_maps = 1; *num_maps = 1;
mutex_unlock(&pcs->mutex);
return 0; return 0;
free_pingroups: free_pingroups:
pinctrl_generic_remove_last_group(pcs->pctl); pinctrl_generic_remove_group(pcs->pctl, gsel);
*num_maps = 1; *num_maps = 1;
free_function: free_function:
pinmux_generic_remove_last_function(pcs->pctl); pinmux_generic_remove_function(pcs->pctl, fsel);
mutex_unlock(&pcs->mutex);
free_pins: free_pins:
devm_kfree(pcs->dev, pins); devm_kfree(pcs->dev, pins);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment