Commit a4e6f95a authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'pinctrl-v5.15-2' of git://git.kernel.org/pub/scm/linux/kernel/git/linusw/linux-pinctrl

Pull pin control fixes from Linus Walleij:
 "Some few pin control fixes for the v5.15 kernel cycle. The most
  critical is the AMD fixes.

   - Fix wakeup interrupts in the AMD driver affecting AMD laptops.

   - Fix parent irqspec translation in the Qualcomm SPMI GPIO driver.

   - Fix deferred probe handling in the Rockchip driver, this is a
     stopgap solution while we look for something more elegant.

   - Add PM suspend callbacks to the Qualcomm SC7280 driver.

   - Some minor doc fix (should have come in earlier, sorry)"

* tag 'pinctrl-v5.15-2' of git://git.kernel.org/pub/scm/linux/kernel/git/linusw/linux-pinctrl:
  pinctrl: qcom: sc7280: Add PM suspend callbacks
  gpio/rockchip: fetch deferred output settings on probe
  pinctrl/rockchip: add a queue for deferred pin output settings on probe
  pinctrl: qcom: spmi-gpio: correct parent irqspec translation
  pinctrl: amd: Handle wake-up interrupt
  pinctrl: amd: Add irq field data
  pinctrl: core: Remove duplicated word from devm_pinctrl_unregister()
parents 62da74a7 28406a21
...@@ -689,6 +689,7 @@ static int rockchip_gpio_probe(struct platform_device *pdev) ...@@ -689,6 +689,7 @@ static int rockchip_gpio_probe(struct platform_device *pdev)
struct device_node *pctlnp = of_get_parent(np); struct device_node *pctlnp = of_get_parent(np);
struct pinctrl_dev *pctldev = NULL; struct pinctrl_dev *pctldev = NULL;
struct rockchip_pin_bank *bank = NULL; struct rockchip_pin_bank *bank = NULL;
struct rockchip_pin_output_deferred *cfg;
static int gpio; static int gpio;
int id, ret; int id, ret;
...@@ -716,12 +717,33 @@ static int rockchip_gpio_probe(struct platform_device *pdev) ...@@ -716,12 +717,33 @@ static int rockchip_gpio_probe(struct platform_device *pdev)
if (ret) if (ret)
return ret; return ret;
/*
* Prevent clashes with a deferred output setting
* being added right at this moment.
*/
mutex_lock(&bank->deferred_lock);
ret = rockchip_gpiolib_register(bank); ret = rockchip_gpiolib_register(bank);
if (ret) { if (ret) {
clk_disable_unprepare(bank->clk); clk_disable_unprepare(bank->clk);
mutex_unlock(&bank->deferred_lock);
return ret; return ret;
} }
while (!list_empty(&bank->deferred_output)) {
cfg = list_first_entry(&bank->deferred_output,
struct rockchip_pin_output_deferred, head);
list_del(&cfg->head);
ret = rockchip_gpio_direction_output(&bank->gpio_chip, cfg->pin, cfg->arg);
if (ret)
dev_warn(dev, "setting output pin %u to %u failed\n", cfg->pin, cfg->arg);
kfree(cfg);
}
mutex_unlock(&bank->deferred_lock);
platform_set_drvdata(pdev, bank); platform_set_drvdata(pdev, bank);
dev_info(dev, "probed %pOF\n", np); dev_info(dev, "probed %pOF\n", np);
......
...@@ -2306,7 +2306,7 @@ EXPORT_SYMBOL_GPL(devm_pinctrl_register_and_init); ...@@ -2306,7 +2306,7 @@ EXPORT_SYMBOL_GPL(devm_pinctrl_register_and_init);
/** /**
* devm_pinctrl_unregister() - Resource managed version of pinctrl_unregister(). * devm_pinctrl_unregister() - Resource managed version of pinctrl_unregister().
* @dev: device for which which resource was allocated * @dev: device for which resource was allocated
* @pctldev: the pinctrl device to unregister. * @pctldev: the pinctrl device to unregister.
*/ */
void devm_pinctrl_unregister(struct device *dev, struct pinctrl_dev *pctldev) void devm_pinctrl_unregister(struct device *dev, struct pinctrl_dev *pctldev)
......
...@@ -445,6 +445,7 @@ static int amd_gpio_irq_set_wake(struct irq_data *d, unsigned int on) ...@@ -445,6 +445,7 @@ static int amd_gpio_irq_set_wake(struct irq_data *d, unsigned int on)
struct gpio_chip *gc = irq_data_get_irq_chip_data(d); struct gpio_chip *gc = irq_data_get_irq_chip_data(d);
struct amd_gpio *gpio_dev = gpiochip_get_data(gc); struct amd_gpio *gpio_dev = gpiochip_get_data(gc);
u32 wake_mask = BIT(WAKE_CNTRL_OFF_S0I3) | BIT(WAKE_CNTRL_OFF_S3); u32 wake_mask = BIT(WAKE_CNTRL_OFF_S0I3) | BIT(WAKE_CNTRL_OFF_S3);
int err;
raw_spin_lock_irqsave(&gpio_dev->lock, flags); raw_spin_lock_irqsave(&gpio_dev->lock, flags);
pin_reg = readl(gpio_dev->base + (d->hwirq)*4); pin_reg = readl(gpio_dev->base + (d->hwirq)*4);
...@@ -457,6 +458,15 @@ static int amd_gpio_irq_set_wake(struct irq_data *d, unsigned int on) ...@@ -457,6 +458,15 @@ static int amd_gpio_irq_set_wake(struct irq_data *d, unsigned int on)
writel(pin_reg, gpio_dev->base + (d->hwirq)*4); writel(pin_reg, gpio_dev->base + (d->hwirq)*4);
raw_spin_unlock_irqrestore(&gpio_dev->lock, flags); raw_spin_unlock_irqrestore(&gpio_dev->lock, flags);
if (on)
err = enable_irq_wake(gpio_dev->irq);
else
err = disable_irq_wake(gpio_dev->irq);
if (err)
dev_err(&gpio_dev->pdev->dev, "failed to %s wake-up interrupt\n",
on ? "enable" : "disable");
return 0; return 0;
} }
...@@ -902,7 +912,6 @@ static struct pinctrl_desc amd_pinctrl_desc = { ...@@ -902,7 +912,6 @@ static struct pinctrl_desc amd_pinctrl_desc = {
static int amd_gpio_probe(struct platform_device *pdev) static int amd_gpio_probe(struct platform_device *pdev)
{ {
int ret = 0; int ret = 0;
int irq_base;
struct resource *res; struct resource *res;
struct amd_gpio *gpio_dev; struct amd_gpio *gpio_dev;
struct gpio_irq_chip *girq; struct gpio_irq_chip *girq;
...@@ -925,9 +934,9 @@ static int amd_gpio_probe(struct platform_device *pdev) ...@@ -925,9 +934,9 @@ static int amd_gpio_probe(struct platform_device *pdev)
if (!gpio_dev->base) if (!gpio_dev->base)
return -ENOMEM; return -ENOMEM;
irq_base = platform_get_irq(pdev, 0); gpio_dev->irq = platform_get_irq(pdev, 0);
if (irq_base < 0) if (gpio_dev->irq < 0)
return irq_base; return gpio_dev->irq;
#ifdef CONFIG_PM_SLEEP #ifdef CONFIG_PM_SLEEP
gpio_dev->saved_regs = devm_kcalloc(&pdev->dev, amd_pinctrl_desc.npins, gpio_dev->saved_regs = devm_kcalloc(&pdev->dev, amd_pinctrl_desc.npins,
...@@ -987,7 +996,7 @@ static int amd_gpio_probe(struct platform_device *pdev) ...@@ -987,7 +996,7 @@ static int amd_gpio_probe(struct platform_device *pdev)
goto out2; goto out2;
} }
ret = devm_request_irq(&pdev->dev, irq_base, amd_gpio_irq_handler, ret = devm_request_irq(&pdev->dev, gpio_dev->irq, amd_gpio_irq_handler,
IRQF_SHARED, KBUILD_MODNAME, gpio_dev); IRQF_SHARED, KBUILD_MODNAME, gpio_dev);
if (ret) if (ret)
goto out2; goto out2;
......
...@@ -98,6 +98,7 @@ struct amd_gpio { ...@@ -98,6 +98,7 @@ struct amd_gpio {
struct resource *res; struct resource *res;
struct platform_device *pdev; struct platform_device *pdev;
u32 *saved_regs; u32 *saved_regs;
int irq;
}; };
/* KERNCZ configuration*/ /* KERNCZ configuration*/
......
...@@ -2092,6 +2092,23 @@ static bool rockchip_pinconf_pull_valid(struct rockchip_pin_ctrl *ctrl, ...@@ -2092,6 +2092,23 @@ static bool rockchip_pinconf_pull_valid(struct rockchip_pin_ctrl *ctrl,
return false; return false;
} }
static int rockchip_pinconf_defer_output(struct rockchip_pin_bank *bank,
unsigned int pin, u32 arg)
{
struct rockchip_pin_output_deferred *cfg;
cfg = kzalloc(sizeof(*cfg), GFP_KERNEL);
if (!cfg)
return -ENOMEM;
cfg->pin = pin;
cfg->arg = arg;
list_add_tail(&cfg->head, &bank->deferred_output);
return 0;
}
/* set the pin config settings for a specified pin */ /* set the pin config settings for a specified pin */
static int rockchip_pinconf_set(struct pinctrl_dev *pctldev, unsigned int pin, static int rockchip_pinconf_set(struct pinctrl_dev *pctldev, unsigned int pin,
unsigned long *configs, unsigned num_configs) unsigned long *configs, unsigned num_configs)
...@@ -2136,6 +2153,22 @@ static int rockchip_pinconf_set(struct pinctrl_dev *pctldev, unsigned int pin, ...@@ -2136,6 +2153,22 @@ static int rockchip_pinconf_set(struct pinctrl_dev *pctldev, unsigned int pin,
if (rc != RK_FUNC_GPIO) if (rc != RK_FUNC_GPIO)
return -EINVAL; return -EINVAL;
/*
* Check for gpio driver not being probed yet.
* The lock makes sure that either gpio-probe has completed
* or the gpio driver hasn't probed yet.
*/
mutex_lock(&bank->deferred_lock);
if (!gpio || !gpio->direction_output) {
rc = rockchip_pinconf_defer_output(bank, pin - bank->pin_base, arg);
mutex_unlock(&bank->deferred_lock);
if (rc)
return rc;
break;
}
mutex_unlock(&bank->deferred_lock);
rc = gpio->direction_output(gpio, pin - bank->pin_base, rc = gpio->direction_output(gpio, pin - bank->pin_base,
arg); arg);
if (rc) if (rc)
...@@ -2204,6 +2237,11 @@ static int rockchip_pinconf_get(struct pinctrl_dev *pctldev, unsigned int pin, ...@@ -2204,6 +2237,11 @@ static int rockchip_pinconf_get(struct pinctrl_dev *pctldev, unsigned int pin,
if (rc != RK_FUNC_GPIO) if (rc != RK_FUNC_GPIO)
return -EINVAL; return -EINVAL;
if (!gpio || !gpio->get) {
arg = 0;
break;
}
rc = gpio->get(gpio, pin - bank->pin_base); rc = gpio->get(gpio, pin - bank->pin_base);
if (rc < 0) if (rc < 0)
return rc; return rc;
...@@ -2450,6 +2488,9 @@ static int rockchip_pinctrl_register(struct platform_device *pdev, ...@@ -2450,6 +2488,9 @@ static int rockchip_pinctrl_register(struct platform_device *pdev,
pin_bank->name, pin); pin_bank->name, pin);
pdesc++; pdesc++;
} }
INIT_LIST_HEAD(&pin_bank->deferred_output);
mutex_init(&pin_bank->deferred_lock);
} }
ret = rockchip_pinctrl_parse_dt(pdev, info); ret = rockchip_pinctrl_parse_dt(pdev, info);
...@@ -2716,6 +2757,31 @@ static int rockchip_pinctrl_probe(struct platform_device *pdev) ...@@ -2716,6 +2757,31 @@ static int rockchip_pinctrl_probe(struct platform_device *pdev)
return 0; return 0;
} }
static int rockchip_pinctrl_remove(struct platform_device *pdev)
{
struct rockchip_pinctrl *info = platform_get_drvdata(pdev);
struct rockchip_pin_bank *bank;
struct rockchip_pin_output_deferred *cfg;
int i;
of_platform_depopulate(&pdev->dev);
for (i = 0; i < info->ctrl->nr_banks; i++) {
bank = &info->ctrl->pin_banks[i];
mutex_lock(&bank->deferred_lock);
while (!list_empty(&bank->deferred_output)) {
cfg = list_first_entry(&bank->deferred_output,
struct rockchip_pin_output_deferred, head);
list_del(&cfg->head);
kfree(cfg);
}
mutex_unlock(&bank->deferred_lock);
}
return 0;
}
static struct rockchip_pin_bank px30_pin_banks[] = { static struct rockchip_pin_bank px30_pin_banks[] = {
PIN_BANK_IOMUX_FLAGS(0, 32, "gpio0", IOMUX_SOURCE_PMU, PIN_BANK_IOMUX_FLAGS(0, 32, "gpio0", IOMUX_SOURCE_PMU,
IOMUX_SOURCE_PMU, IOMUX_SOURCE_PMU,
...@@ -3175,6 +3241,7 @@ static const struct of_device_id rockchip_pinctrl_dt_match[] = { ...@@ -3175,6 +3241,7 @@ static const struct of_device_id rockchip_pinctrl_dt_match[] = {
static struct platform_driver rockchip_pinctrl_driver = { static struct platform_driver rockchip_pinctrl_driver = {
.probe = rockchip_pinctrl_probe, .probe = rockchip_pinctrl_probe,
.remove = rockchip_pinctrl_remove,
.driver = { .driver = {
.name = "rockchip-pinctrl", .name = "rockchip-pinctrl",
.pm = &rockchip_pinctrl_dev_pm_ops, .pm = &rockchip_pinctrl_dev_pm_ops,
......
...@@ -141,6 +141,8 @@ struct rockchip_drv { ...@@ -141,6 +141,8 @@ struct rockchip_drv {
* @toggle_edge_mode: bit mask to toggle (falling/rising) edge mode * @toggle_edge_mode: bit mask to toggle (falling/rising) edge mode
* @recalced_mask: bit mask to indicate a need to recalulate the mask * @recalced_mask: bit mask to indicate a need to recalulate the mask
* @route_mask: bits describing the routing pins of per bank * @route_mask: bits describing the routing pins of per bank
* @deferred_output: gpio output settings to be done after gpio bank probed
* @deferred_lock: mutex for the deferred_output shared btw gpio and pinctrl
*/ */
struct rockchip_pin_bank { struct rockchip_pin_bank {
struct device *dev; struct device *dev;
...@@ -169,6 +171,8 @@ struct rockchip_pin_bank { ...@@ -169,6 +171,8 @@ struct rockchip_pin_bank {
u32 toggle_edge_mode; u32 toggle_edge_mode;
u32 recalced_mask; u32 recalced_mask;
u32 route_mask; u32 route_mask;
struct list_head deferred_output;
struct mutex deferred_lock;
}; };
/** /**
...@@ -243,6 +247,12 @@ struct rockchip_pin_config { ...@@ -243,6 +247,12 @@ struct rockchip_pin_config {
unsigned int nconfigs; unsigned int nconfigs;
}; };
struct rockchip_pin_output_deferred {
struct list_head head;
unsigned int pin;
u32 arg;
};
/** /**
* struct rockchip_pin_group: represent group of pins of a pinmux function. * struct rockchip_pin_group: represent group of pins of a pinmux function.
* @name: name of the pin group, used to lookup the group. * @name: name of the pin group, used to lookup the group.
......
...@@ -1496,6 +1496,7 @@ static const struct of_device_id sc7280_pinctrl_of_match[] = { ...@@ -1496,6 +1496,7 @@ static const struct of_device_id sc7280_pinctrl_of_match[] = {
static struct platform_driver sc7280_pinctrl_driver = { static struct platform_driver sc7280_pinctrl_driver = {
.driver = { .driver = {
.name = "sc7280-pinctrl", .name = "sc7280-pinctrl",
.pm = &msm_pinctrl_dev_pm_ops,
.of_match_table = sc7280_pinctrl_of_match, .of_match_table = sc7280_pinctrl_of_match,
}, },
.probe = sc7280_pinctrl_probe, .probe = sc7280_pinctrl_probe,
......
// SPDX-License-Identifier: GPL-2.0-only // SPDX-License-Identifier: GPL-2.0-only
/* /*
* Copyright (c) 2012-2014, The Linux Foundation. All rights reserved. * Copyright (c) 2012-2014, 2016-2021 The Linux Foundation. All rights reserved.
*/ */
#include <linux/gpio/driver.h> #include <linux/gpio/driver.h>
...@@ -14,6 +14,7 @@ ...@@ -14,6 +14,7 @@
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include <linux/regmap.h> #include <linux/regmap.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/spmi.h>
#include <linux/types.h> #include <linux/types.h>
#include <dt-bindings/pinctrl/qcom,pmic-gpio.h> #include <dt-bindings/pinctrl/qcom,pmic-gpio.h>
...@@ -171,6 +172,8 @@ struct pmic_gpio_state { ...@@ -171,6 +172,8 @@ struct pmic_gpio_state {
struct pinctrl_dev *ctrl; struct pinctrl_dev *ctrl;
struct gpio_chip chip; struct gpio_chip chip;
struct irq_chip irq; struct irq_chip irq;
u8 usid;
u8 pid_base;
}; };
static const struct pinconf_generic_params pmic_gpio_bindings[] = { static const struct pinconf_generic_params pmic_gpio_bindings[] = {
...@@ -949,12 +952,36 @@ static int pmic_gpio_child_to_parent_hwirq(struct gpio_chip *chip, ...@@ -949,12 +952,36 @@ static int pmic_gpio_child_to_parent_hwirq(struct gpio_chip *chip,
unsigned int *parent_hwirq, unsigned int *parent_hwirq,
unsigned int *parent_type) unsigned int *parent_type)
{ {
*parent_hwirq = child_hwirq + 0xc0; struct pmic_gpio_state *state = gpiochip_get_data(chip);
*parent_hwirq = child_hwirq + state->pid_base;
*parent_type = child_type; *parent_type = child_type;
return 0; return 0;
} }
static void *pmic_gpio_populate_parent_fwspec(struct gpio_chip *chip,
unsigned int parent_hwirq,
unsigned int parent_type)
{
struct pmic_gpio_state *state = gpiochip_get_data(chip);
struct irq_fwspec *fwspec;
fwspec = kzalloc(sizeof(*fwspec), GFP_KERNEL);
if (!fwspec)
return NULL;
fwspec->fwnode = chip->irq.parent_domain->fwnode;
fwspec->param_count = 4;
fwspec->param[0] = state->usid;
fwspec->param[1] = parent_hwirq;
/* param[2] must be left as 0 */
fwspec->param[3] = parent_type;
return fwspec;
}
static int pmic_gpio_probe(struct platform_device *pdev) static int pmic_gpio_probe(struct platform_device *pdev)
{ {
struct irq_domain *parent_domain; struct irq_domain *parent_domain;
...@@ -965,6 +992,7 @@ static int pmic_gpio_probe(struct platform_device *pdev) ...@@ -965,6 +992,7 @@ static int pmic_gpio_probe(struct platform_device *pdev)
struct pmic_gpio_pad *pad, *pads; struct pmic_gpio_pad *pad, *pads;
struct pmic_gpio_state *state; struct pmic_gpio_state *state;
struct gpio_irq_chip *girq; struct gpio_irq_chip *girq;
const struct spmi_device *parent_spmi_dev;
int ret, npins, i; int ret, npins, i;
u32 reg; u32 reg;
...@@ -984,6 +1012,9 @@ static int pmic_gpio_probe(struct platform_device *pdev) ...@@ -984,6 +1012,9 @@ static int pmic_gpio_probe(struct platform_device *pdev)
state->dev = &pdev->dev; state->dev = &pdev->dev;
state->map = dev_get_regmap(dev->parent, NULL); state->map = dev_get_regmap(dev->parent, NULL);
parent_spmi_dev = to_spmi_device(dev->parent);
state->usid = parent_spmi_dev->usid;
state->pid_base = reg >> 8;
pindesc = devm_kcalloc(dev, npins, sizeof(*pindesc), GFP_KERNEL); pindesc = devm_kcalloc(dev, npins, sizeof(*pindesc), GFP_KERNEL);
if (!pindesc) if (!pindesc)
...@@ -1059,7 +1090,7 @@ static int pmic_gpio_probe(struct platform_device *pdev) ...@@ -1059,7 +1090,7 @@ static int pmic_gpio_probe(struct platform_device *pdev)
girq->fwnode = of_node_to_fwnode(state->dev->of_node); girq->fwnode = of_node_to_fwnode(state->dev->of_node);
girq->parent_domain = parent_domain; girq->parent_domain = parent_domain;
girq->child_to_parent_hwirq = pmic_gpio_child_to_parent_hwirq; girq->child_to_parent_hwirq = pmic_gpio_child_to_parent_hwirq;
girq->populate_parent_alloc_arg = gpiochip_populate_parent_fwspec_fourcell; girq->populate_parent_alloc_arg = pmic_gpio_populate_parent_fwspec;
girq->child_offset_to_irq = pmic_gpio_child_offset_to_irq; girq->child_offset_to_irq = pmic_gpio_child_offset_to_irq;
girq->child_irq_domain_ops.translate = pmic_gpio_domain_translate; girq->child_irq_domain_ops.translate = pmic_gpio_domain_translate;
......
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