2
0
mirror of https://github.com/edk2-porting/linux-next.git synced 2025-01-16 09:34:22 +08:00

pinctrl: mediatek: Check gpio pin number and use binary search in mtk_hw_pin_field_lookup()

1. Check if gpio pin number is in valid range to prevent from get invalid
   pointer 'desc' in the following code:
	desc = (const struct mtk_pin_desc *)&hw->soc->pins[gpio];

2. Improve  mtk_hw_pin_field_lookup()
2.1 Modify mtk_hw_pin_field_lookup() to use binary search for accelerating
     search.
2.2 Correct message after the following check fail:
    if (hw->soc->reg_cal && hw->soc->reg_cal[field].range) {
		rc = &hw->soc->reg_cal[field];
    The original message is:
    	"Not support field %d for pin %d (%s)\n"
    However, the check is on soc chip level, not on pin level yet.
    So the message is corrected as:
    	"Not support field %d for this soc\n"

Signed-off-by: Light Hsieh <light.hsieh@mediatek.com>
Link: https://lore.kernel.org/r/1579675994-7001-1-git-send-email-light.hsieh@mediatek.com
Acked-by: Sean Wang <sean.wang@kernel.org>
Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
This commit is contained in:
Light Hsieh 2020-01-22 14:53:09 +08:00 committed by Linus Walleij
parent ef1ea54eab
commit 3de7deefce
2 changed files with 46 additions and 6 deletions

View File

@ -68,32 +68,44 @@ static int mtk_hw_pin_field_lookup(struct mtk_pinctrl *hw,
{ {
const struct mtk_pin_field_calc *c, *e; const struct mtk_pin_field_calc *c, *e;
const struct mtk_pin_reg_calc *rc; const struct mtk_pin_reg_calc *rc;
int start = 0, end, check;
bool found = false;
u32 bits; u32 bits;
if (hw->soc->reg_cal && hw->soc->reg_cal[field].range) { if (hw->soc->reg_cal && hw->soc->reg_cal[field].range) {
rc = &hw->soc->reg_cal[field]; rc = &hw->soc->reg_cal[field];
} else { } else {
dev_dbg(hw->dev, dev_dbg(hw->dev,
"Not support field %d for pin %d (%s)\n", "Not support field %d for this soc\n", field);
field, desc->number, desc->name);
return -ENOTSUPP; return -ENOTSUPP;
} }
end = rc->nranges - 1;
c = rc->range; c = rc->range;
e = c + rc->nranges; e = c + rc->nranges;
while (c < e) { while (start <= end) {
if (desc->number >= c->s_pin && desc->number <= c->e_pin) check = (start + end) >> 1;
if (desc->number >= rc->range[check].s_pin
&& desc->number <= rc->range[check].e_pin) {
found = true;
break; break;
c++; } else if (start == end)
break;
else if (desc->number < rc->range[check].s_pin)
end = check - 1;
else
start = check + 1;
} }
if (c >= e) { if (!found) {
dev_dbg(hw->dev, "Not support field %d for pin = %d (%s)\n", dev_dbg(hw->dev, "Not support field %d for pin = %d (%s)\n",
field, desc->number, desc->name); field, desc->number, desc->name);
return -ENOTSUPP; return -ENOTSUPP;
} }
c = rc->range + check;
if (c->i_base > hw->nbase - 1) { if (c->i_base > hw->nbase - 1) {
dev_err(hw->dev, dev_err(hw->dev,
"Invalid base for field %d for pin = %d (%s)\n", "Invalid base for field %d for pin = %d (%s)\n",
@ -182,6 +194,9 @@ int mtk_hw_set_value(struct mtk_pinctrl *hw, const struct mtk_pin_desc *desc,
if (err) if (err)
return err; return err;
if (value < 0 || value > pf.mask)
return -EINVAL;
if (!pf.next) if (!pf.next)
mtk_rmw(hw, pf.index, pf.offset, pf.mask << pf.bitpos, mtk_rmw(hw, pf.index, pf.offset, pf.mask << pf.bitpos,
(value & pf.mask) << pf.bitpos); (value & pf.mask) << pf.bitpos);

View File

@ -81,6 +81,8 @@ static int mtk_pinconf_get(struct pinctrl_dev *pctldev,
int val, val2, err, reg, ret = 1; int val, val2, err, reg, ret = 1;
const struct mtk_pin_desc *desc; const struct mtk_pin_desc *desc;
if (pin >= hw->soc->npins)
return -EINVAL;
desc = (const struct mtk_pin_desc *)&hw->soc->pins[pin]; desc = (const struct mtk_pin_desc *)&hw->soc->pins[pin];
switch (param) { switch (param) {
@ -206,6 +208,10 @@ static int mtk_pinconf_set(struct pinctrl_dev *pctldev, unsigned int pin,
int err = 0; int err = 0;
u32 reg; u32 reg;
if (pin >= hw->soc->npins) {
err = -EINVAL;
goto err;
}
desc = (const struct mtk_pin_desc *)&hw->soc->pins[pin]; desc = (const struct mtk_pin_desc *)&hw->soc->pins[pin];
switch ((u32)param) { switch ((u32)param) {
@ -693,6 +699,9 @@ static int mtk_gpio_get_direction(struct gpio_chip *chip, unsigned int gpio)
const struct mtk_pin_desc *desc; const struct mtk_pin_desc *desc;
int value, err; int value, err;
if (gpio > hw->soc->npins)
return -EINVAL;
desc = (const struct mtk_pin_desc *)&hw->soc->pins[gpio]; desc = (const struct mtk_pin_desc *)&hw->soc->pins[gpio];
err = mtk_hw_get_value(hw, desc, PINCTRL_PIN_REG_DIR, &value); err = mtk_hw_get_value(hw, desc, PINCTRL_PIN_REG_DIR, &value);
@ -708,6 +717,9 @@ static int mtk_gpio_get(struct gpio_chip *chip, unsigned int gpio)
const struct mtk_pin_desc *desc; const struct mtk_pin_desc *desc;
int value, err; int value, err;
if (gpio > hw->soc->npins)
return -EINVAL;
desc = (const struct mtk_pin_desc *)&hw->soc->pins[gpio]; desc = (const struct mtk_pin_desc *)&hw->soc->pins[gpio];
err = mtk_hw_get_value(hw, desc, PINCTRL_PIN_REG_DI, &value); err = mtk_hw_get_value(hw, desc, PINCTRL_PIN_REG_DI, &value);
@ -722,6 +734,9 @@ static void mtk_gpio_set(struct gpio_chip *chip, unsigned int gpio, int value)
struct mtk_pinctrl *hw = gpiochip_get_data(chip); struct mtk_pinctrl *hw = gpiochip_get_data(chip);
const struct mtk_pin_desc *desc; const struct mtk_pin_desc *desc;
if (gpio > hw->soc->npins)
return;
desc = (const struct mtk_pin_desc *)&hw->soc->pins[gpio]; desc = (const struct mtk_pin_desc *)&hw->soc->pins[gpio];
mtk_hw_set_value(hw, desc, PINCTRL_PIN_REG_DO, !!value); mtk_hw_set_value(hw, desc, PINCTRL_PIN_REG_DO, !!value);
@ -729,12 +744,22 @@ static void mtk_gpio_set(struct gpio_chip *chip, unsigned int gpio, int value)
static int mtk_gpio_direction_input(struct gpio_chip *chip, unsigned int gpio) static int mtk_gpio_direction_input(struct gpio_chip *chip, unsigned int gpio)
{ {
struct mtk_pinctrl *hw = gpiochip_get_data(chip);
if (gpio > hw->soc->npins)
return -EINVAL;
return pinctrl_gpio_direction_input(chip->base + gpio); return pinctrl_gpio_direction_input(chip->base + gpio);
} }
static int mtk_gpio_direction_output(struct gpio_chip *chip, unsigned int gpio, static int mtk_gpio_direction_output(struct gpio_chip *chip, unsigned int gpio,
int value) int value)
{ {
struct mtk_pinctrl *hw = gpiochip_get_data(chip);
if (gpio > hw->soc->npins)
return -EINVAL;
mtk_gpio_set(chip, gpio, value); mtk_gpio_set(chip, gpio, value);
return pinctrl_gpio_direction_output(chip->base + gpio); return pinctrl_gpio_direction_output(chip->base + gpio);