289 lines
8.7 KiB
Diff
289 lines
8.7 KiB
Diff
From 85d48072cfb2e491d10a235c4ade2460e3bbc93a Mon Sep 17 00:00:00 2001
|
|
From: Christophe Priouzeau <christophe.priouzeau@st.com>
|
|
Date: Mon, 26 Nov 2018 14:40:42 +0100
|
|
Subject: [PATCH 21/52] ARM-stm32mp1-r0-rc2-PINCTRL
|
|
|
|
---
|
|
.../bindings/pinctrl/st,stm32-pinctrl.txt | 1 +
|
|
drivers/pinctrl/core.c | 15 +----
|
|
drivers/pinctrl/pinctrl-stmfx.c | 8 +++
|
|
drivers/pinctrl/stm32/pinctrl-stm32.c | 71 +++++++++++++++++++++-
|
|
include/linux/pinctrl/pinctrl.h | 2 +
|
|
5 files changed, 83 insertions(+), 14 deletions(-)
|
|
|
|
diff --git a/Documentation/devicetree/bindings/pinctrl/st,stm32-pinctrl.txt b/Documentation/devicetree/bindings/pinctrl/st,stm32-pinctrl.txt
|
|
index 286c981..1a5d1e2 100644
|
|
--- a/Documentation/devicetree/bindings/pinctrl/st,stm32-pinctrl.txt
|
|
+++ b/Documentation/devicetree/bindings/pinctrl/st,stm32-pinctrl.txt
|
|
@@ -58,6 +58,7 @@ Optional properties:
|
|
used to select GPIOs as interrupts).
|
|
- st,package: Indicates the SOC package used.
|
|
More details in include/dt-bindings/pinctrl/stm32-pinfunc.h
|
|
+ - hwlocks: reference to a phandle of a hardware spinlock provider node.
|
|
|
|
Example 1:
|
|
#include <dt-bindings/pinctrl/stm32f429-pinfunc.h>
|
|
diff --git a/drivers/pinctrl/core.c b/drivers/pinctrl/core.c
|
|
index a3dd777..0220d43 100644
|
|
--- a/drivers/pinctrl/core.c
|
|
+++ b/drivers/pinctrl/core.c
|
|
@@ -1992,7 +1992,7 @@ pinctrl_init_controller(struct pinctrl_desc *pctldesc, struct device *dev,
|
|
return ERR_PTR(ret);
|
|
}
|
|
|
|
-static int pinctrl_claim_hogs(struct pinctrl_dev *pctldev)
|
|
+int pinctrl_claim_hogs(struct pinctrl_dev *pctldev)
|
|
{
|
|
pctldev->p = create_pinctrl(pctldev->dev, pctldev);
|
|
if (PTR_ERR(pctldev->p) == -ENODEV) {
|
|
@@ -2030,21 +2030,10 @@ static int pinctrl_claim_hogs(struct pinctrl_dev *pctldev)
|
|
|
|
return 0;
|
|
}
|
|
+EXPORT_SYMBOL_GPL(pinctrl_claim_hogs);
|
|
|
|
int pinctrl_enable(struct pinctrl_dev *pctldev)
|
|
{
|
|
- int error;
|
|
-
|
|
- error = pinctrl_claim_hogs(pctldev);
|
|
- if (error) {
|
|
- dev_err(pctldev->dev, "could not claim hogs: %i\n",
|
|
- error);
|
|
- mutex_destroy(&pctldev->mutex);
|
|
- kfree(pctldev);
|
|
-
|
|
- return error;
|
|
- }
|
|
-
|
|
mutex_lock(&pinctrldev_list_mutex);
|
|
list_add_tail(&pctldev->node, &pinctrldev_list);
|
|
mutex_unlock(&pinctrldev_list_mutex);
|
|
diff --git a/drivers/pinctrl/pinctrl-stmfx.c b/drivers/pinctrl/pinctrl-stmfx.c
|
|
index e253ed1..15d5757 100644
|
|
--- a/drivers/pinctrl/pinctrl-stmfx.c
|
|
+++ b/drivers/pinctrl/pinctrl-stmfx.c
|
|
@@ -663,6 +663,14 @@ static int stmfx_pinctrl_probe(struct platform_device *pdev)
|
|
if (ret)
|
|
return ret;
|
|
|
|
+ /*
|
|
+ * Claim hogs after enabling gpio function, otherwise pin
|
|
+ * configuration will not apply
|
|
+ */
|
|
+ ret = pinctrl_claim_hogs(pctl->pctl_dev);
|
|
+ if (ret)
|
|
+ return ret;
|
|
+
|
|
pctl->irq_chip.name = dev_name(pctl->dev);
|
|
pctl->irq_chip.irq_mask = stmfx_pinctrl_irq_mask;
|
|
pctl->irq_chip.irq_unmask = stmfx_pinctrl_irq_unmask;
|
|
diff --git a/drivers/pinctrl/stm32/pinctrl-stm32.c b/drivers/pinctrl/stm32/pinctrl-stm32.c
|
|
index e25917f..914bee4 100644
|
|
--- a/drivers/pinctrl/stm32/pinctrl-stm32.c
|
|
+++ b/drivers/pinctrl/stm32/pinctrl-stm32.c
|
|
@@ -8,6 +8,7 @@
|
|
*/
|
|
#include <linux/clk.h>
|
|
#include <linux/gpio/driver.h>
|
|
+#include <linux/hwspinlock.h>
|
|
#include <linux/io.h>
|
|
#include <linux/irq.h>
|
|
#include <linux/mfd/syscon.h>
|
|
@@ -64,6 +65,8 @@
|
|
#define gpio_range_to_bank(chip) \
|
|
container_of(chip, struct stm32_gpio_bank, range)
|
|
|
|
+#define HWSPINLOCK_TIMEOUT 5 /* msec */
|
|
+
|
|
static const char * const stm32_gpio_functions[] = {
|
|
"gpio", "af0", "af1",
|
|
"af2", "af3", "af4",
|
|
@@ -111,6 +114,7 @@ struct stm32_pinctrl {
|
|
u32 npins;
|
|
u32 pkg;
|
|
u32 pin_base_shift;
|
|
+ struct hwspinlock *hwlock;
|
|
};
|
|
|
|
static inline int stm32_gpio_pin(int gpio)
|
|
@@ -598,14 +602,24 @@ static int stm32_pmx_get_func_groups(struct pinctrl_dev *pctldev,
|
|
static void stm32_pmx_set_mode(struct stm32_gpio_bank *bank,
|
|
int pin, u32 mode, u32 alt)
|
|
{
|
|
+ struct stm32_pinctrl *pctl = dev_get_drvdata(bank->gpio_chip.parent);
|
|
u32 val;
|
|
int alt_shift = (pin % 8) * 4;
|
|
int alt_offset = STM32_GPIO_AFRL + (pin / 8) * 4;
|
|
unsigned long flags;
|
|
+ int err = 0;
|
|
|
|
clk_enable(bank->clk);
|
|
spin_lock_irqsave(&bank->lock, flags);
|
|
|
|
+ if (pctl->hwlock)
|
|
+ err = hwspin_lock_timeout(pctl->hwlock, HWSPINLOCK_TIMEOUT);
|
|
+
|
|
+ if (err) {
|
|
+ dev_err(pctl->dev, "Can't get hwspinlock\n");
|
|
+ goto unlock;
|
|
+ }
|
|
+
|
|
val = readl_relaxed(bank->base + alt_offset);
|
|
val &= ~GENMASK(alt_shift + 3, alt_shift);
|
|
val |= (alt << alt_shift);
|
|
@@ -618,6 +632,10 @@ static void stm32_pmx_set_mode(struct stm32_gpio_bank *bank,
|
|
|
|
stm32_gpio_backup_mode(bank, pin, mode, alt);
|
|
|
|
+ if (pctl->hwlock)
|
|
+ hwspin_unlock(pctl->hwlock);
|
|
+
|
|
+unlock:
|
|
spin_unlock_irqrestore(&bank->lock, flags);
|
|
clk_disable(bank->clk);
|
|
}
|
|
@@ -707,12 +725,22 @@ static const struct pinmux_ops stm32_pmx_ops = {
|
|
static void stm32_pconf_set_driving(struct stm32_gpio_bank *bank,
|
|
unsigned offset, u32 drive)
|
|
{
|
|
+ struct stm32_pinctrl *pctl = dev_get_drvdata(bank->gpio_chip.parent);
|
|
unsigned long flags;
|
|
u32 val;
|
|
+ int err = 0;
|
|
|
|
clk_enable(bank->clk);
|
|
spin_lock_irqsave(&bank->lock, flags);
|
|
|
|
+ if (pctl->hwlock)
|
|
+ err = hwspin_lock_timeout(pctl->hwlock, HWSPINLOCK_TIMEOUT);
|
|
+
|
|
+ if (err) {
|
|
+ dev_err(pctl->dev, "Can't get hwspinlock\n");
|
|
+ goto unlock;
|
|
+ }
|
|
+
|
|
val = readl_relaxed(bank->base + STM32_GPIO_TYPER);
|
|
val &= ~BIT(offset);
|
|
val |= drive << offset;
|
|
@@ -720,6 +748,10 @@ static void stm32_pconf_set_driving(struct stm32_gpio_bank *bank,
|
|
|
|
stm32_gpio_backup_driving(bank, offset, drive);
|
|
|
|
+ if (pctl->hwlock)
|
|
+ hwspin_unlock(pctl->hwlock);
|
|
+
|
|
+unlock:
|
|
spin_unlock_irqrestore(&bank->lock, flags);
|
|
clk_disable(bank->clk);
|
|
}
|
|
@@ -745,12 +777,22 @@ static u32 stm32_pconf_get_driving(struct stm32_gpio_bank *bank,
|
|
static void stm32_pconf_set_speed(struct stm32_gpio_bank *bank,
|
|
unsigned offset, u32 speed)
|
|
{
|
|
+ struct stm32_pinctrl *pctl = dev_get_drvdata(bank->gpio_chip.parent);
|
|
unsigned long flags;
|
|
u32 val;
|
|
+ int err = 0;
|
|
|
|
clk_enable(bank->clk);
|
|
spin_lock_irqsave(&bank->lock, flags);
|
|
|
|
+ if (pctl->hwlock)
|
|
+ err = hwspin_lock_timeout(pctl->hwlock, HWSPINLOCK_TIMEOUT);
|
|
+
|
|
+ if (err) {
|
|
+ dev_err(pctl->dev, "Can't get hwspinlock\n");
|
|
+ goto unlock;
|
|
+ }
|
|
+
|
|
val = readl_relaxed(bank->base + STM32_GPIO_SPEEDR);
|
|
val &= ~GENMASK(offset * 2 + 1, offset * 2);
|
|
val |= speed << (offset * 2);
|
|
@@ -758,6 +800,10 @@ static void stm32_pconf_set_speed(struct stm32_gpio_bank *bank,
|
|
|
|
stm32_gpio_backup_speed(bank, offset, speed);
|
|
|
|
+ if (pctl->hwlock)
|
|
+ hwspin_unlock(pctl->hwlock);
|
|
+
|
|
+unlock:
|
|
spin_unlock_irqrestore(&bank->lock, flags);
|
|
clk_disable(bank->clk);
|
|
}
|
|
@@ -783,12 +829,22 @@ static u32 stm32_pconf_get_speed(struct stm32_gpio_bank *bank,
|
|
static void stm32_pconf_set_bias(struct stm32_gpio_bank *bank,
|
|
unsigned offset, u32 bias)
|
|
{
|
|
+ struct stm32_pinctrl *pctl = dev_get_drvdata(bank->gpio_chip.parent);
|
|
unsigned long flags;
|
|
u32 val;
|
|
+ int err = 0;
|
|
|
|
clk_enable(bank->clk);
|
|
spin_lock_irqsave(&bank->lock, flags);
|
|
|
|
+ if (pctl->hwlock)
|
|
+ err = hwspin_lock_timeout(pctl->hwlock, HWSPINLOCK_TIMEOUT);
|
|
+
|
|
+ if (err) {
|
|
+ dev_err(pctl->dev, "Can't get hwspinlock\n");
|
|
+ goto unlock;
|
|
+ }
|
|
+
|
|
val = readl_relaxed(bank->base + STM32_GPIO_PUPDR);
|
|
val &= ~GENMASK(offset * 2 + 1, offset * 2);
|
|
val |= bias << (offset * 2);
|
|
@@ -796,6 +852,10 @@ static void stm32_pconf_set_bias(struct stm32_gpio_bank *bank,
|
|
|
|
stm32_gpio_backup_bias(bank, offset, bias);
|
|
|
|
+ if (pctl->hwlock)
|
|
+ hwspin_unlock(pctl->hwlock);
|
|
+
|
|
+unlock:
|
|
spin_unlock_irqrestore(&bank->lock, flags);
|
|
clk_disable(bank->clk);
|
|
}
|
|
@@ -1206,7 +1266,7 @@ int stm32_pctl_probe(struct platform_device *pdev)
|
|
struct device *dev = &pdev->dev;
|
|
struct stm32_pinctrl *pctl;
|
|
struct pinctrl_pin_desc *pins;
|
|
- int i, ret, banks = 0;
|
|
+ int i, ret, hwlock_id, banks = 0;
|
|
|
|
if (!np)
|
|
return -EINVAL;
|
|
@@ -1226,6 +1286,15 @@ int stm32_pctl_probe(struct platform_device *pdev)
|
|
|
|
platform_set_drvdata(pdev, pctl);
|
|
|
|
+ /* hwspinlock is optional */
|
|
+ hwlock_id = of_hwspin_lock_get_id(pdev->dev.of_node, 0);
|
|
+ if (hwlock_id < 0) {
|
|
+ if (hwlock_id == -EPROBE_DEFER)
|
|
+ return hwlock_id;
|
|
+ } else {
|
|
+ pctl->hwlock = hwspin_lock_request_specific(hwlock_id);
|
|
+ }
|
|
+
|
|
pctl->dev = dev;
|
|
pctl->match_data = match->data;
|
|
|
|
diff --git a/include/linux/pinctrl/pinctrl.h b/include/linux/pinctrl/pinctrl.h
|
|
index 8f5dbb8..bbeaa29 100644
|
|
--- a/include/linux/pinctrl/pinctrl.h
|
|
+++ b/include/linux/pinctrl/pinctrl.h
|
|
@@ -148,6 +148,8 @@ extern int pinctrl_register_and_init(struct pinctrl_desc *pctldesc,
|
|
struct pinctrl_dev **pctldev);
|
|
extern int pinctrl_enable(struct pinctrl_dev *pctldev);
|
|
|
|
+extern int pinctrl_claim_hogs(struct pinctrl_dev *pctldev);
|
|
+
|
|
/* Please use pinctrl_register_and_init() and pinctrl_enable() instead */
|
|
extern struct pinctrl_dev *pinctrl_register(struct pinctrl_desc *pctldesc,
|
|
struct device *dev, void *driver_data);
|
|
--
|
|
2.7.4
|
|
|