xref: /openbmc/linux/drivers/gpio/gpio-cs5535.c (revision 5ef12cb4a3a78ffb331c03a795a15eea4ae35155)
1 /*
2  * AMD CS5535/CS5536 GPIO driver
3  * Copyright (C) 2006  Advanced Micro Devices, Inc.
4  * Copyright (C) 2007-2009  Andres Salomon <dilinger@collabora.co.uk>
5  *
6  * This program is free software; you can redistribute it and/or
7  * modify it under the terms of version 2 of the GNU General Public License
8  * as published by the Free Software Foundation.
9  */
10 
11 #include <linux/kernel.h>
12 #include <linux/spinlock.h>
13 #include <linux/module.h>
14 #include <linux/platform_device.h>
15 #include <linux/gpio/driver.h>
16 #include <linux/io.h>
17 #include <linux/cs5535.h>
18 #include <asm/msr.h>
19 
20 #define DRV_NAME "cs5535-gpio"
21 
22 /*
23  * Some GPIO pins
24  *  31-29,23 : reserved (always mask out)
25  *  28       : Power Button
26  *  26       : PME#
27  *  22-16    : LPC
28  *  14,15    : SMBus
29  *  9,8      : UART1
30  *  7        : PCI INTB
31  *  3,4      : UART2/DDC
32  *  2        : IDE_IRQ0
33  *  1        : AC_BEEP
34  *  0        : PCI INTA
35  *
36  * If a mask was not specified, allow all except
37  * reserved and Power Button
38  */
39 #define GPIO_DEFAULT_MASK 0x0F7FFFFF
40 
41 static ulong mask = GPIO_DEFAULT_MASK;
42 module_param_named(mask, mask, ulong, 0444);
43 MODULE_PARM_DESC(mask, "GPIO channel mask.");
44 
45 /*
46  * FIXME: convert this singleton driver to use the state container
47  * design pattern, see Documentation/driver-model/design-patterns.txt
48  */
49 static struct cs5535_gpio_chip {
50 	struct gpio_chip chip;
51 	resource_size_t base;
52 
53 	struct platform_device *pdev;
54 	spinlock_t lock;
55 } cs5535_gpio_chip;
56 
57 /*
58  * The CS5535/CS5536 GPIOs support a number of extra features not defined
59  * by the gpio_chip API, so these are exported.  For a full list of the
60  * registers, see include/linux/cs5535.h.
61  */
62 
63 static void errata_outl(struct cs5535_gpio_chip *chip, u32 val,
64 		unsigned int reg)
65 {
66 	unsigned long addr = chip->base + 0x80 + reg;
67 
68 	/*
69 	 * According to the CS5536 errata (#36), after suspend
70 	 * a write to the high bank GPIO register will clear all
71 	 * non-selected bits; the recommended workaround is a
72 	 * read-modify-write operation.
73 	 *
74 	 * Don't apply this errata to the edge status GPIOs, as writing
75 	 * to their lower bits will clear them.
76 	 */
77 	if (reg != GPIO_POSITIVE_EDGE_STS && reg != GPIO_NEGATIVE_EDGE_STS) {
78 		if (val & 0xffff)
79 			val |= (inl(addr) & 0xffff); /* ignore the high bits */
80 		else
81 			val |= (inl(addr) ^ (val >> 16));
82 	}
83 	outl(val, addr);
84 }
85 
86 static void __cs5535_gpio_set(struct cs5535_gpio_chip *chip, unsigned offset,
87 		unsigned int reg)
88 {
89 	if (offset < 16)
90 		/* low bank register */
91 		outl(1 << offset, chip->base + reg);
92 	else
93 		/* high bank register */
94 		errata_outl(chip, 1 << (offset - 16), reg);
95 }
96 
97 void cs5535_gpio_set(unsigned offset, unsigned int reg)
98 {
99 	struct cs5535_gpio_chip *chip = &cs5535_gpio_chip;
100 	unsigned long flags;
101 
102 	spin_lock_irqsave(&chip->lock, flags);
103 	__cs5535_gpio_set(chip, offset, reg);
104 	spin_unlock_irqrestore(&chip->lock, flags);
105 }
106 EXPORT_SYMBOL_GPL(cs5535_gpio_set);
107 
108 static void __cs5535_gpio_clear(struct cs5535_gpio_chip *chip, unsigned offset,
109 		unsigned int reg)
110 {
111 	if (offset < 16)
112 		/* low bank register */
113 		outl(1 << (offset + 16), chip->base + reg);
114 	else
115 		/* high bank register */
116 		errata_outl(chip, 1 << offset, reg);
117 }
118 
119 void cs5535_gpio_clear(unsigned offset, unsigned int reg)
120 {
121 	struct cs5535_gpio_chip *chip = &cs5535_gpio_chip;
122 	unsigned long flags;
123 
124 	spin_lock_irqsave(&chip->lock, flags);
125 	__cs5535_gpio_clear(chip, offset, reg);
126 	spin_unlock_irqrestore(&chip->lock, flags);
127 }
128 EXPORT_SYMBOL_GPL(cs5535_gpio_clear);
129 
130 int cs5535_gpio_isset(unsigned offset, unsigned int reg)
131 {
132 	struct cs5535_gpio_chip *chip = &cs5535_gpio_chip;
133 	unsigned long flags;
134 	long val;
135 
136 	spin_lock_irqsave(&chip->lock, flags);
137 	if (offset < 16)
138 		/* low bank register */
139 		val = inl(chip->base + reg);
140 	else {
141 		/* high bank register */
142 		val = inl(chip->base + 0x80 + reg);
143 		offset -= 16;
144 	}
145 	spin_unlock_irqrestore(&chip->lock, flags);
146 
147 	return (val & (1 << offset)) ? 1 : 0;
148 }
149 EXPORT_SYMBOL_GPL(cs5535_gpio_isset);
150 
151 int cs5535_gpio_set_irq(unsigned group, unsigned irq)
152 {
153 	uint32_t lo, hi;
154 
155 	if (group > 7 || irq > 15)
156 		return -EINVAL;
157 
158 	rdmsr(MSR_PIC_ZSEL_HIGH, lo, hi);
159 
160 	lo &= ~(0xF << (group * 4));
161 	lo |= (irq & 0xF) << (group * 4);
162 
163 	wrmsr(MSR_PIC_ZSEL_HIGH, lo, hi);
164 	return 0;
165 }
166 EXPORT_SYMBOL_GPL(cs5535_gpio_set_irq);
167 
168 void cs5535_gpio_setup_event(unsigned offset, int pair, int pme)
169 {
170 	struct cs5535_gpio_chip *chip = &cs5535_gpio_chip;
171 	uint32_t shift = (offset % 8) * 4;
172 	unsigned long flags;
173 	uint32_t val;
174 
175 	if (offset >= 24)
176 		offset = GPIO_MAP_W;
177 	else if (offset >= 16)
178 		offset = GPIO_MAP_Z;
179 	else if (offset >= 8)
180 		offset = GPIO_MAP_Y;
181 	else
182 		offset = GPIO_MAP_X;
183 
184 	spin_lock_irqsave(&chip->lock, flags);
185 	val = inl(chip->base + offset);
186 
187 	/* Clear whatever was there before */
188 	val &= ~(0xF << shift);
189 
190 	/* Set the new value */
191 	val |= ((pair & 7) << shift);
192 
193 	/* Set the PME bit if this is a PME event */
194 	if (pme)
195 		val |= (1 << (shift + 3));
196 
197 	outl(val, chip->base + offset);
198 	spin_unlock_irqrestore(&chip->lock, flags);
199 }
200 EXPORT_SYMBOL_GPL(cs5535_gpio_setup_event);
201 
202 /*
203  * Generic gpio_chip API support.
204  */
205 
206 static int chip_gpio_request(struct gpio_chip *c, unsigned offset)
207 {
208 	struct cs5535_gpio_chip *chip = gpiochip_get_data(c);
209 	unsigned long flags;
210 
211 	spin_lock_irqsave(&chip->lock, flags);
212 
213 	/* check if this pin is available */
214 	if ((mask & (1 << offset)) == 0) {
215 		dev_info(&chip->pdev->dev,
216 			"pin %u is not available (check mask)\n", offset);
217 		spin_unlock_irqrestore(&chip->lock, flags);
218 		return -EINVAL;
219 	}
220 
221 	/* disable output aux 1 & 2 on this pin */
222 	__cs5535_gpio_clear(chip, offset, GPIO_OUTPUT_AUX1);
223 	__cs5535_gpio_clear(chip, offset, GPIO_OUTPUT_AUX2);
224 
225 	/* disable input aux 1 on this pin */
226 	__cs5535_gpio_clear(chip, offset, GPIO_INPUT_AUX1);
227 
228 	spin_unlock_irqrestore(&chip->lock, flags);
229 
230 	return 0;
231 }
232 
233 static int chip_gpio_get(struct gpio_chip *chip, unsigned offset)
234 {
235 	return cs5535_gpio_isset(offset, GPIO_READ_BACK);
236 }
237 
238 static void chip_gpio_set(struct gpio_chip *chip, unsigned offset, int val)
239 {
240 	if (val)
241 		cs5535_gpio_set(offset, GPIO_OUTPUT_VAL);
242 	else
243 		cs5535_gpio_clear(offset, GPIO_OUTPUT_VAL);
244 }
245 
246 static int chip_direction_input(struct gpio_chip *c, unsigned offset)
247 {
248 	struct cs5535_gpio_chip *chip = gpiochip_get_data(c);
249 	unsigned long flags;
250 
251 	spin_lock_irqsave(&chip->lock, flags);
252 	__cs5535_gpio_set(chip, offset, GPIO_INPUT_ENABLE);
253 	__cs5535_gpio_clear(chip, offset, GPIO_OUTPUT_ENABLE);
254 	spin_unlock_irqrestore(&chip->lock, flags);
255 
256 	return 0;
257 }
258 
259 static int chip_direction_output(struct gpio_chip *c, unsigned offset, int val)
260 {
261 	struct cs5535_gpio_chip *chip = gpiochip_get_data(c);
262 	unsigned long flags;
263 
264 	spin_lock_irqsave(&chip->lock, flags);
265 
266 	__cs5535_gpio_set(chip, offset, GPIO_INPUT_ENABLE);
267 	__cs5535_gpio_set(chip, offset, GPIO_OUTPUT_ENABLE);
268 	if (val)
269 		__cs5535_gpio_set(chip, offset, GPIO_OUTPUT_VAL);
270 	else
271 		__cs5535_gpio_clear(chip, offset, GPIO_OUTPUT_VAL);
272 
273 	spin_unlock_irqrestore(&chip->lock, flags);
274 
275 	return 0;
276 }
277 
278 static const char * const cs5535_gpio_names[] = {
279 	"GPIO0", "GPIO1", "GPIO2", "GPIO3",
280 	"GPIO4", "GPIO5", "GPIO6", "GPIO7",
281 	"GPIO8", "GPIO9", "GPIO10", "GPIO11",
282 	"GPIO12", "GPIO13", "GPIO14", "GPIO15",
283 	"GPIO16", "GPIO17", "GPIO18", "GPIO19",
284 	"GPIO20", "GPIO21", "GPIO22", NULL,
285 	"GPIO24", "GPIO25", "GPIO26", "GPIO27",
286 	"GPIO28", NULL, NULL, NULL,
287 };
288 
289 static struct cs5535_gpio_chip cs5535_gpio_chip = {
290 	.chip = {
291 		.owner = THIS_MODULE,
292 		.label = DRV_NAME,
293 
294 		.base = 0,
295 		.ngpio = 32,
296 		.names = cs5535_gpio_names,
297 		.request = chip_gpio_request,
298 
299 		.get = chip_gpio_get,
300 		.set = chip_gpio_set,
301 
302 		.direction_input = chip_direction_input,
303 		.direction_output = chip_direction_output,
304 	},
305 };
306 
307 static int cs5535_gpio_probe(struct platform_device *pdev)
308 {
309 	struct resource *res;
310 	int err = -EIO;
311 	ulong mask_orig = mask;
312 
313 	/* There are two ways to get the GPIO base address; one is by
314 	 * fetching it from MSR_LBAR_GPIO, the other is by reading the
315 	 * PCI BAR info.  The latter method is easier (especially across
316 	 * different architectures), so we'll stick with that for now.  If
317 	 * it turns out to be unreliable in the face of crappy BIOSes, we
318 	 * can always go back to using MSRs.. */
319 
320 	res = platform_get_resource(pdev, IORESOURCE_IO, 0);
321 	if (!res) {
322 		dev_err(&pdev->dev, "can't fetch device resource info\n");
323 		return err;
324 	}
325 
326 	if (!devm_request_region(&pdev->dev, res->start, resource_size(res),
327 				 pdev->name)) {
328 		dev_err(&pdev->dev, "can't request region\n");
329 		return err;
330 	}
331 
332 	/* set up the driver-specific struct */
333 	cs5535_gpio_chip.base = res->start;
334 	cs5535_gpio_chip.pdev = pdev;
335 	spin_lock_init(&cs5535_gpio_chip.lock);
336 
337 	dev_info(&pdev->dev, "reserved resource region %pR\n", res);
338 
339 	/* mask out reserved pins */
340 	mask &= 0x1F7FFFFF;
341 
342 	/* do not allow pin 28, Power Button, as there's special handling
343 	 * in the PMC needed. (note 12, p. 48) */
344 	mask &= ~(1 << 28);
345 
346 	if (mask_orig != mask)
347 		dev_info(&pdev->dev, "mask changed from 0x%08lX to 0x%08lX\n",
348 				mask_orig, mask);
349 
350 	/* finally, register with the generic GPIO API */
351 	err = devm_gpiochip_add_data(&pdev->dev, &cs5535_gpio_chip.chip,
352 				     &cs5535_gpio_chip);
353 	if (err)
354 		return err;
355 
356 	return 0;
357 }
358 
359 static struct platform_driver cs5535_gpio_driver = {
360 	.driver = {
361 		.name = DRV_NAME,
362 	},
363 	.probe = cs5535_gpio_probe,
364 };
365 
366 module_platform_driver(cs5535_gpio_driver);
367 
368 MODULE_AUTHOR("Andres Salomon <dilinger@queued.net>");
369 MODULE_DESCRIPTION("AMD CS5535/CS5536 GPIO driver");
370 MODULE_LICENSE("GPL");
371 MODULE_ALIAS("platform:" DRV_NAME);
372