ARM: dts: npcm8xx: add npcm845 function node
[platform/kernel/u-boot.git] / drivers / gpio / mvebu_gpio.c
1 // SPDX-License-Identifier: GPL-2.0+
2 /*
3  * Copyright (C) 2016 Stefan Roese <sr@denx.de>
4  */
5
6 #include <common.h>
7 #include <dm.h>
8 #include <dm/pinctrl.h>
9 #include <asm/gpio.h>
10 #include <asm/io.h>
11 #include <errno.h>
12 #include <linux/bitops.h>
13
14 #define MVEBU_GPIOS_PER_BANK    32
15
16 struct mvebu_gpio_regs {
17         u32 data_out;
18         u32 io_conf;
19         u32 blink_en;
20         u32 in_pol;
21         u32 data_in;
22 };
23
24 struct mvebu_gpio_priv {
25         struct mvebu_gpio_regs *regs;
26         char name[sizeof("mvebuX_")];
27 };
28
29 static int mvebu_gpio_direction_input(struct udevice *dev, unsigned int gpio)
30 {
31         struct mvebu_gpio_priv *priv = dev_get_priv(dev);
32         struct mvebu_gpio_regs *regs = priv->regs;
33
34         setbits_le32(&regs->io_conf, BIT(gpio));
35
36         return 0;
37 }
38
39 static int mvebu_gpio_direction_output(struct udevice *dev, unsigned gpio,
40                                        int value)
41 {
42         struct mvebu_gpio_priv *priv = dev_get_priv(dev);
43         struct mvebu_gpio_regs *regs = priv->regs;
44
45         if (value)
46                 setbits_le32(&regs->data_out, BIT(gpio));
47         else
48                 clrbits_le32(&regs->data_out, BIT(gpio));
49         clrbits_le32(&regs->io_conf, BIT(gpio));
50
51         return 0;
52 }
53
54 static int mvebu_gpio_get_function(struct udevice *dev, unsigned gpio)
55 {
56         struct mvebu_gpio_priv *priv = dev_get_priv(dev);
57         struct mvebu_gpio_regs *regs = priv->regs;
58         u32 val;
59
60         val = readl(&regs->io_conf) & BIT(gpio);
61         if (val)
62                 return GPIOF_INPUT;
63         else
64                 return GPIOF_OUTPUT;
65 }
66
67 static int mvebu_gpio_set_value(struct udevice *dev, unsigned gpio,
68                                 int value)
69 {
70         struct mvebu_gpio_priv *priv = dev_get_priv(dev);
71         struct mvebu_gpio_regs *regs = priv->regs;
72
73         if (value)
74                 setbits_le32(&regs->data_out, BIT(gpio));
75         else
76                 clrbits_le32(&regs->data_out, BIT(gpio));
77
78         return 0;
79 }
80
81 static int mvebu_gpio_get_value(struct udevice *dev, unsigned gpio)
82 {
83         struct mvebu_gpio_priv *priv = dev_get_priv(dev);
84         struct mvebu_gpio_regs *regs = priv->regs;
85
86         return !!(readl(&regs->data_in) & BIT(gpio));
87 }
88
89 static int mvebu_gpio_probe(struct udevice *dev)
90 {
91         struct gpio_dev_priv *uc_priv = dev_get_uclass_priv(dev);
92         struct mvebu_gpio_priv *priv = dev_get_priv(dev);
93
94         priv->regs = dev_read_addr_ptr(dev);
95         uc_priv->gpio_count = dev_read_u32_default(dev, "ngpios", MVEBU_GPIOS_PER_BANK);
96         sprintf(priv->name, "mvebu%d_", dev_seq(dev));
97         uc_priv->bank_name = priv->name;
98
99         return 0;
100 }
101
102 static const struct dm_gpio_ops mvebu_gpio_ops = {
103 #if CONFIG_IS_ENABLED(PINCTRL_ARMADA_38X)
104         .request                = pinctrl_gpio_request,
105         .rfree                  = pinctrl_gpio_free,
106 #endif
107         .direction_input        = mvebu_gpio_direction_input,
108         .direction_output       = mvebu_gpio_direction_output,
109         .get_function           = mvebu_gpio_get_function,
110         .get_value              = mvebu_gpio_get_value,
111         .set_value              = mvebu_gpio_set_value,
112 };
113
114 static const struct udevice_id mvebu_gpio_ids[] = {
115         { .compatible = "marvell,orion-gpio" },
116         { }
117 };
118
119 U_BOOT_DRIVER(gpio_mvebu) = {
120         .name                   = "gpio_mvebu",
121         .id                     = UCLASS_GPIO,
122         .of_match               = mvebu_gpio_ids,
123         .ops                    = &mvebu_gpio_ops,
124         .probe                  = mvebu_gpio_probe,
125         .priv_auto      = sizeof(struct mvebu_gpio_priv),
126 };