Merge tag 'rust-6.4' of https://github.com/Rust-for-Linux/linux
[linux-2.6-microblaze.git] / drivers / gpio / gpio-siox.c
1 // SPDX-License-Identifier: GPL-2.0
2 /*
3  * Copyright (C) 2015-2018 Pengutronix, Uwe Kleine-König <kernel@pengutronix.de>
4  */
5
6 #include <linux/module.h>
7 #include <linux/siox.h>
8 #include <linux/gpio/driver.h>
9 #include <linux/of.h>
10
11 struct gpio_siox_ddata {
12         struct gpio_chip gchip;
13         struct mutex lock;
14         u8 setdata[1];
15         u8 getdata[3];
16
17         raw_spinlock_t irqlock;
18         u32 irq_enable;
19         u32 irq_status;
20         u32 irq_type[20];
21 };
22
23 /*
24  * Note that this callback only sets the value that is clocked out in the next
25  * cycle.
26  */
27 static int gpio_siox_set_data(struct siox_device *sdevice, u8 status, u8 buf[])
28 {
29         struct gpio_siox_ddata *ddata = dev_get_drvdata(&sdevice->dev);
30
31         mutex_lock(&ddata->lock);
32         buf[0] = ddata->setdata[0];
33         mutex_unlock(&ddata->lock);
34
35         return 0;
36 }
37
38 static int gpio_siox_get_data(struct siox_device *sdevice, const u8 buf[])
39 {
40         struct gpio_siox_ddata *ddata = dev_get_drvdata(&sdevice->dev);
41         size_t offset;
42         u32 trigger;
43
44         mutex_lock(&ddata->lock);
45
46         raw_spin_lock_irq(&ddata->irqlock);
47
48         for (offset = 0; offset < 12; ++offset) {
49                 unsigned int bitpos = 11 - offset;
50                 unsigned int gpiolevel = buf[bitpos / 8] & (1 << bitpos % 8);
51                 unsigned int prev_level =
52                         ddata->getdata[bitpos / 8] & (1 << (bitpos % 8));
53                 u32 irq_type = ddata->irq_type[offset];
54
55                 if (gpiolevel) {
56                         if ((irq_type & IRQ_TYPE_LEVEL_HIGH) ||
57                             ((irq_type & IRQ_TYPE_EDGE_RISING) && !prev_level))
58                                 ddata->irq_status |= 1 << offset;
59                 } else {
60                         if ((irq_type & IRQ_TYPE_LEVEL_LOW) ||
61                             ((irq_type & IRQ_TYPE_EDGE_FALLING) && prev_level))
62                                 ddata->irq_status |= 1 << offset;
63                 }
64         }
65
66         trigger = ddata->irq_status & ddata->irq_enable;
67
68         raw_spin_unlock_irq(&ddata->irqlock);
69
70         ddata->getdata[0] = buf[0];
71         ddata->getdata[1] = buf[1];
72         ddata->getdata[2] = buf[2];
73
74         mutex_unlock(&ddata->lock);
75
76         for (offset = 0; offset < 12; ++offset) {
77                 if (trigger & (1 << offset)) {
78                         struct irq_domain *irqdomain = ddata->gchip.irq.domain;
79                         unsigned int irq = irq_find_mapping(irqdomain, offset);
80
81                         /*
82                          * Conceptually handle_nested_irq should call the flow
83                          * handler of the irq chip. But it doesn't, so we have
84                          * to clean the irq_status here.
85                          */
86                         raw_spin_lock_irq(&ddata->irqlock);
87                         ddata->irq_status &= ~(1 << offset);
88                         raw_spin_unlock_irq(&ddata->irqlock);
89
90                         handle_nested_irq(irq);
91                 }
92         }
93
94         return 0;
95 }
96
97 static void gpio_siox_irq_ack(struct irq_data *d)
98 {
99         struct gpio_chip *gc = irq_data_get_irq_chip_data(d);
100         struct gpio_siox_ddata *ddata = gpiochip_get_data(gc);
101
102         raw_spin_lock(&ddata->irqlock);
103         ddata->irq_status &= ~(1 << d->hwirq);
104         raw_spin_unlock(&ddata->irqlock);
105 }
106
107 static void gpio_siox_irq_mask(struct irq_data *d)
108 {
109         struct gpio_chip *gc = irq_data_get_irq_chip_data(d);
110         struct gpio_siox_ddata *ddata = gpiochip_get_data(gc);
111
112         raw_spin_lock(&ddata->irqlock);
113         ddata->irq_enable &= ~(1 << d->hwirq);
114         raw_spin_unlock(&ddata->irqlock);
115         gpiochip_disable_irq(gc, irqd_to_hwirq(d));
116 }
117
118 static void gpio_siox_irq_unmask(struct irq_data *d)
119 {
120         struct gpio_chip *gc = irq_data_get_irq_chip_data(d);
121         struct gpio_siox_ddata *ddata = gpiochip_get_data(gc);
122
123         gpiochip_enable_irq(gc, irqd_to_hwirq(d));
124         raw_spin_lock(&ddata->irqlock);
125         ddata->irq_enable |= 1 << d->hwirq;
126         raw_spin_unlock(&ddata->irqlock);
127 }
128
129 static int gpio_siox_irq_set_type(struct irq_data *d, u32 type)
130 {
131         struct gpio_chip *gc = irq_data_get_irq_chip_data(d);
132         struct gpio_siox_ddata *ddata = gpiochip_get_data(gc);
133
134         raw_spin_lock(&ddata->irqlock);
135         ddata->irq_type[d->hwirq] = type;
136         raw_spin_unlock(&ddata->irqlock);
137
138         return 0;
139 }
140
141 static int gpio_siox_get(struct gpio_chip *chip, unsigned int offset)
142 {
143         struct gpio_siox_ddata *ddata = gpiochip_get_data(chip);
144         int ret;
145
146         mutex_lock(&ddata->lock);
147
148         if (offset >= 12) {
149                 unsigned int bitpos = 19 - offset;
150
151                 ret = ddata->setdata[0] & (1 << bitpos);
152         } else {
153                 unsigned int bitpos = 11 - offset;
154
155                 ret = ddata->getdata[bitpos / 8] & (1 << (bitpos % 8));
156         }
157
158         mutex_unlock(&ddata->lock);
159
160         return ret;
161 }
162
163 static void gpio_siox_set(struct gpio_chip *chip,
164                           unsigned int offset, int value)
165 {
166         struct gpio_siox_ddata *ddata = gpiochip_get_data(chip);
167         u8 mask = 1 << (19 - offset);
168
169         mutex_lock(&ddata->lock);
170
171         if (value)
172                 ddata->setdata[0] |= mask;
173         else
174                 ddata->setdata[0] &= ~mask;
175
176         mutex_unlock(&ddata->lock);
177 }
178
179 static int gpio_siox_direction_input(struct gpio_chip *chip,
180                                      unsigned int offset)
181 {
182         if (offset >= 12)
183                 return -EINVAL;
184
185         return 0;
186 }
187
188 static int gpio_siox_direction_output(struct gpio_chip *chip,
189                                       unsigned int offset, int value)
190 {
191         if (offset < 12)
192                 return -EINVAL;
193
194         gpio_siox_set(chip, offset, value);
195         return 0;
196 }
197
198 static int gpio_siox_get_direction(struct gpio_chip *chip, unsigned int offset)
199 {
200         if (offset < 12)
201                 return GPIO_LINE_DIRECTION_IN;
202         else
203                 return GPIO_LINE_DIRECTION_OUT;
204 }
205
206 static const struct irq_chip gpio_siox_irq_chip = {
207         .name = "siox-gpio",
208         .irq_ack = gpio_siox_irq_ack,
209         .irq_mask = gpio_siox_irq_mask,
210         .irq_unmask = gpio_siox_irq_unmask,
211         .irq_set_type = gpio_siox_irq_set_type,
212         .flags = IRQCHIP_IMMUTABLE,
213         GPIOCHIP_IRQ_RESOURCE_HELPERS,
214 };
215
216 static int gpio_siox_probe(struct siox_device *sdevice)
217 {
218         struct gpio_siox_ddata *ddata;
219         struct gpio_irq_chip *girq;
220         struct device *dev = &sdevice->dev;
221         struct gpio_chip *gc;
222         int ret;
223
224         ddata = devm_kzalloc(dev, sizeof(*ddata), GFP_KERNEL);
225         if (!ddata)
226                 return -ENOMEM;
227
228         dev_set_drvdata(dev, ddata);
229
230         mutex_init(&ddata->lock);
231         raw_spin_lock_init(&ddata->irqlock);
232
233         gc = &ddata->gchip;
234         gc->base = -1;
235         gc->can_sleep = 1;
236         gc->parent = dev;
237         gc->owner = THIS_MODULE;
238         gc->get = gpio_siox_get;
239         gc->set = gpio_siox_set;
240         gc->direction_input = gpio_siox_direction_input;
241         gc->direction_output = gpio_siox_direction_output;
242         gc->get_direction = gpio_siox_get_direction;
243         gc->ngpio = 20;
244
245         girq = &gc->irq;
246         gpio_irq_chip_set_chip(girq, &gpio_siox_irq_chip);
247         girq->default_type = IRQ_TYPE_NONE;
248         girq->handler = handle_level_irq;
249         girq->threaded = true;
250
251         ret = devm_gpiochip_add_data(dev, gc, ddata);
252         if (ret)
253                 dev_err(dev, "Failed to register gpio chip (%d)\n", ret);
254
255         return ret;
256 }
257
258 static struct siox_driver gpio_siox_driver = {
259         .probe = gpio_siox_probe,
260         .set_data = gpio_siox_set_data,
261         .get_data = gpio_siox_get_data,
262         .driver = {
263                 .name = "gpio-siox",
264         },
265 };
266 module_siox_driver(gpio_siox_driver);
267
268 MODULE_AUTHOR("Uwe Kleine-Koenig <u.kleine-koenig@pengutronix.de>");
269 MODULE_DESCRIPTION("SIOX gpio driver");
270 MODULE_LICENSE("GPL v2");