diff options
author | Linus Torvalds <torvalds@linux-foundation.org> | 2009-06-14 16:42:43 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2009-06-14 16:42:43 -0400 |
commit | 2cf4d4514d5b43c1f3b64bd0ec8b9853bde8f1dc (patch) | |
tree | e35a625496acc6ac852846d40b8851186b9d1ac4 /arch/arm/plat-orion/gpio.c | |
parent | 44b7532b8b464f606053562400719c9c21276037 (diff) | |
parent | ce53895a5d24e0ee19fb92f56c17323fb4c9ab27 (diff) |
Merge branch 'for-linus' of master.kernel.org:/home/rmk/linux-2.6-arm
* 'for-linus' of master.kernel.org:/home/rmk/linux-2.6-arm: (417 commits)
MAINTAINERS: EB110ATX is not ebsa110
MAINTAINERS: update Eric Miao's email address and status
fb: add support of LCD display controller on pxa168/910 (base layer)
[ARM] 5552/1: ep93xx get_uart_rate(): use EP93XX_SYSCON_PWRCNT and EP93XX_SYSCON_PWRCN
[ARM] pxa/sharpsl_pm: zaurus needs generic pxa suspend/resume routines
[ARM] 5544/1: Trust PrimeCell resource sizes
[ARM] pxa/sharpsl_pm: cleanup of gpio-related code.
[ARM] pxa/sharpsl_pm: drop set_irq_type calls
[ARM] pxa/sharpsl_pm: merge pxa-specific code into generic one
[ARM] pxa/sharpsl_pm: merge the two sharpsl_pm.c since it's now pxa specific
[ARM] sa1100: remove unused collie_pm.c
[ARM] pxa: fix the conflicting non-static declarations of global_gpios[]
[ARM] 5550/1: Add default configure file for w90p910 platform
[ARM] 5549/1: Add clock api for w90p910 platform.
[ARM] 5548/1: Add gpio api for w90p910 platform
[ARM] 5551/1: Add multi-function pin api for w90p910 platform.
[ARM] Make ARM_VIC_NR depend on ARM_VIC
[ARM] 5546/1: ARM PL022 SSP/SPI driver v3
ARM: OMAP4: SMP: Update defconfig for OMAP4430
ARM: OMAP4: SMP: Enable SMP support for OMAP4430
...
Diffstat (limited to 'arch/arm/plat-orion/gpio.c')
-rw-r--r-- | arch/arm/plat-orion/gpio.c | 194 |
1 files changed, 75 insertions, 119 deletions
diff --git a/arch/arm/plat-orion/gpio.c b/arch/arm/plat-orion/gpio.c index 32eb9e33bebb..e814803d4741 100644 --- a/arch/arm/plat-orion/gpio.c +++ b/arch/arm/plat-orion/gpio.c | |||
@@ -15,10 +15,9 @@ | |||
15 | #include <linux/spinlock.h> | 15 | #include <linux/spinlock.h> |
16 | #include <linux/bitops.h> | 16 | #include <linux/bitops.h> |
17 | #include <linux/io.h> | 17 | #include <linux/io.h> |
18 | #include <asm/gpio.h> | 18 | #include <linux/gpio.h> |
19 | 19 | ||
20 | static DEFINE_SPINLOCK(gpio_lock); | 20 | static DEFINE_SPINLOCK(gpio_lock); |
21 | static const char *gpio_label[GPIO_MAX]; /* non null for allocated GPIOs */ | ||
22 | static unsigned long gpio_valid_input[BITS_TO_LONGS(GPIO_MAX)]; | 21 | static unsigned long gpio_valid_input[BITS_TO_LONGS(GPIO_MAX)]; |
23 | static unsigned long gpio_valid_output[BITS_TO_LONGS(GPIO_MAX)]; | 22 | static unsigned long gpio_valid_output[BITS_TO_LONGS(GPIO_MAX)]; |
24 | 23 | ||
@@ -46,82 +45,54 @@ static void __set_level(unsigned pin, int high) | |||
46 | writel(u, GPIO_OUT(pin)); | 45 | writel(u, GPIO_OUT(pin)); |
47 | } | 46 | } |
48 | 47 | ||
49 | 48 | static inline void __set_blinking(unsigned pin, int blink) | |
50 | /* | ||
51 | * GENERIC_GPIO primitives. | ||
52 | */ | ||
53 | int gpio_direction_input(unsigned pin) | ||
54 | { | 49 | { |
55 | unsigned long flags; | 50 | u32 u; |
56 | |||
57 | if (pin >= GPIO_MAX || !test_bit(pin, gpio_valid_input)) { | ||
58 | pr_debug("%s: invalid GPIO %d\n", __func__, pin); | ||
59 | return -EINVAL; | ||
60 | } | ||
61 | |||
62 | spin_lock_irqsave(&gpio_lock, flags); | ||
63 | |||
64 | /* | ||
65 | * Some callers might not have used gpio_request(), | ||
66 | * so flag this pin as requested now. | ||
67 | */ | ||
68 | if (gpio_label[pin] == NULL) | ||
69 | gpio_label[pin] = "?"; | ||
70 | 51 | ||
71 | /* | 52 | u = readl(GPIO_BLINK_EN(pin)); |
72 | * Configure GPIO direction. | 53 | if (blink) |
73 | */ | 54 | u |= 1 << (pin & 31); |
74 | __set_direction(pin, 1); | 55 | else |
56 | u &= ~(1 << (pin & 31)); | ||
57 | writel(u, GPIO_BLINK_EN(pin)); | ||
58 | } | ||
75 | 59 | ||
76 | spin_unlock_irqrestore(&gpio_lock, flags); | 60 | static inline int orion_gpio_is_valid(unsigned pin, int mode) |
61 | { | ||
62 | if (pin < GPIO_MAX) { | ||
63 | if ((mode & GPIO_INPUT_OK) && !test_bit(pin, gpio_valid_input)) | ||
64 | goto err_out; | ||
65 | if ((mode & GPIO_OUTPUT_OK) && !test_bit(pin, gpio_valid_output)) | ||
66 | goto err_out; | ||
67 | return true; | ||
68 | } | ||
77 | 69 | ||
78 | return 0; | 70 | err_out: |
71 | pr_debug("%s: invalid GPIO %d\n", __func__, pin); | ||
72 | return false; | ||
79 | } | 73 | } |
80 | EXPORT_SYMBOL(gpio_direction_input); | ||
81 | 74 | ||
82 | int gpio_direction_output(unsigned pin, int value) | 75 | /* |
76 | * GENERIC_GPIO primitives. | ||
77 | */ | ||
78 | static int orion_gpio_direction_input(struct gpio_chip *chip, unsigned pin) | ||
83 | { | 79 | { |
84 | unsigned long flags; | 80 | unsigned long flags; |
85 | u32 u; | ||
86 | 81 | ||
87 | if (pin >= GPIO_MAX || !test_bit(pin, gpio_valid_output)) { | 82 | if (!orion_gpio_is_valid(pin, GPIO_INPUT_OK)) |
88 | pr_debug("%s: invalid GPIO %d\n", __func__, pin); | ||
89 | return -EINVAL; | 83 | return -EINVAL; |
90 | } | ||
91 | 84 | ||
92 | spin_lock_irqsave(&gpio_lock, flags); | 85 | spin_lock_irqsave(&gpio_lock, flags); |
93 | 86 | ||
94 | /* | 87 | /* Configure GPIO direction. */ |
95 | * Some callers might not have used gpio_request(), | 88 | __set_direction(pin, 1); |
96 | * so flag this pin as requested now. | ||
97 | */ | ||
98 | if (gpio_label[pin] == NULL) | ||
99 | gpio_label[pin] = "?"; | ||
100 | |||
101 | /* | ||
102 | * Disable blinking. | ||
103 | */ | ||
104 | u = readl(GPIO_BLINK_EN(pin)); | ||
105 | u &= ~(1 << (pin & 31)); | ||
106 | writel(u, GPIO_BLINK_EN(pin)); | ||
107 | |||
108 | /* | ||
109 | * Configure GPIO output value. | ||
110 | */ | ||
111 | __set_level(pin, value); | ||
112 | |||
113 | /* | ||
114 | * Configure GPIO direction. | ||
115 | */ | ||
116 | __set_direction(pin, 0); | ||
117 | 89 | ||
118 | spin_unlock_irqrestore(&gpio_lock, flags); | 90 | spin_unlock_irqrestore(&gpio_lock, flags); |
119 | 91 | ||
120 | return 0; | 92 | return 0; |
121 | } | 93 | } |
122 | EXPORT_SYMBOL(gpio_direction_output); | ||
123 | 94 | ||
124 | int gpio_get_value(unsigned pin) | 95 | static int orion_gpio_get_value(struct gpio_chip *chip, unsigned pin) |
125 | { | 96 | { |
126 | int val; | 97 | int val; |
127 | 98 | ||
@@ -132,83 +103,75 @@ int gpio_get_value(unsigned pin) | |||
132 | 103 | ||
133 | return (val >> (pin & 31)) & 1; | 104 | return (val >> (pin & 31)) & 1; |
134 | } | 105 | } |
135 | EXPORT_SYMBOL(gpio_get_value); | ||
136 | 106 | ||
137 | void gpio_set_value(unsigned pin, int value) | 107 | static int orion_gpio_direction_output(struct gpio_chip *chip, unsigned pin, |
108 | int value) | ||
138 | { | 109 | { |
139 | unsigned long flags; | 110 | unsigned long flags; |
140 | u32 u; | 111 | |
112 | if (!orion_gpio_is_valid(pin, GPIO_OUTPUT_OK)) | ||
113 | return -EINVAL; | ||
141 | 114 | ||
142 | spin_lock_irqsave(&gpio_lock, flags); | 115 | spin_lock_irqsave(&gpio_lock, flags); |
143 | 116 | ||
144 | /* | 117 | /* Disable blinking. */ |
145 | * Disable blinking. | 118 | __set_blinking(pin, 0); |
146 | */ | ||
147 | u = readl(GPIO_BLINK_EN(pin)); | ||
148 | u &= ~(1 << (pin & 31)); | ||
149 | writel(u, GPIO_BLINK_EN(pin)); | ||
150 | 119 | ||
151 | /* | 120 | /* Configure GPIO output value. */ |
152 | * Configure GPIO output value. | ||
153 | */ | ||
154 | __set_level(pin, value); | 121 | __set_level(pin, value); |
155 | 122 | ||
123 | /* Configure GPIO direction. */ | ||
124 | __set_direction(pin, 0); | ||
125 | |||
156 | spin_unlock_irqrestore(&gpio_lock, flags); | 126 | spin_unlock_irqrestore(&gpio_lock, flags); |
127 | |||
128 | return 0; | ||
157 | } | 129 | } |
158 | EXPORT_SYMBOL(gpio_set_value); | ||
159 | 130 | ||
160 | int gpio_request(unsigned pin, const char *label) | 131 | static void orion_gpio_set_value(struct gpio_chip *chip, unsigned pin, |
132 | int value) | ||
161 | { | 133 | { |
162 | unsigned long flags; | 134 | unsigned long flags; |
163 | int ret; | ||
164 | |||
165 | if (pin >= GPIO_MAX || | ||
166 | !(test_bit(pin, gpio_valid_input) || | ||
167 | test_bit(pin, gpio_valid_output))) { | ||
168 | pr_debug("%s: invalid GPIO %d\n", __func__, pin); | ||
169 | return -EINVAL; | ||
170 | } | ||
171 | 135 | ||
172 | spin_lock_irqsave(&gpio_lock, flags); | 136 | spin_lock_irqsave(&gpio_lock, flags); |
173 | if (gpio_label[pin] == NULL) { | ||
174 | gpio_label[pin] = label ? label : "?"; | ||
175 | ret = 0; | ||
176 | } else { | ||
177 | pr_debug("%s: GPIO %d already used as %s\n", | ||
178 | __func__, pin, gpio_label[pin]); | ||
179 | ret = -EBUSY; | ||
180 | } | ||
181 | spin_unlock_irqrestore(&gpio_lock, flags); | ||
182 | 137 | ||
183 | return ret; | 138 | /* Configure GPIO output value. */ |
139 | __set_level(pin, value); | ||
140 | |||
141 | spin_unlock_irqrestore(&gpio_lock, flags); | ||
184 | } | 142 | } |
185 | EXPORT_SYMBOL(gpio_request); | ||
186 | 143 | ||
187 | void gpio_free(unsigned pin) | 144 | static int orion_gpio_request(struct gpio_chip *chip, unsigned pin) |
188 | { | 145 | { |
189 | if (pin >= GPIO_MAX || | 146 | if (orion_gpio_is_valid(pin, GPIO_INPUT_OK) || |
190 | !(test_bit(pin, gpio_valid_input) || | 147 | orion_gpio_is_valid(pin, GPIO_OUTPUT_OK)) |
191 | test_bit(pin, gpio_valid_output))) { | 148 | return 0; |
192 | pr_debug("%s: invalid GPIO %d\n", __func__, pin); | 149 | return -EINVAL; |
193 | return; | ||
194 | } | ||
195 | |||
196 | if (gpio_label[pin] == NULL) | ||
197 | pr_warning("%s: GPIO %d already freed\n", __func__, pin); | ||
198 | else | ||
199 | gpio_label[pin] = NULL; | ||
200 | } | 150 | } |
201 | EXPORT_SYMBOL(gpio_free); | ||
202 | 151 | ||
152 | static struct gpio_chip orion_gpiochip = { | ||
153 | .label = "orion_gpio", | ||
154 | .direction_input = orion_gpio_direction_input, | ||
155 | .get = orion_gpio_get_value, | ||
156 | .direction_output = orion_gpio_direction_output, | ||
157 | .set = orion_gpio_set_value, | ||
158 | .request = orion_gpio_request, | ||
159 | .base = 0, | ||
160 | .ngpio = GPIO_MAX, | ||
161 | .can_sleep = 0, | ||
162 | }; | ||
163 | |||
164 | void __init orion_gpio_init(void) | ||
165 | { | ||
166 | gpiochip_add(&orion_gpiochip); | ||
167 | } | ||
203 | 168 | ||
204 | /* | 169 | /* |
205 | * Orion-specific GPIO API extensions. | 170 | * Orion-specific GPIO API extensions. |
206 | */ | 171 | */ |
207 | void __init orion_gpio_set_unused(unsigned pin) | 172 | void __init orion_gpio_set_unused(unsigned pin) |
208 | { | 173 | { |
209 | /* | 174 | /* Configure as output, drive low. */ |
210 | * Configure as output, drive low. | ||
211 | */ | ||
212 | __set_level(pin, 0); | 175 | __set_level(pin, 0); |
213 | __set_direction(pin, 0); | 176 | __set_direction(pin, 0); |
214 | } | 177 | } |
@@ -230,21 +193,14 @@ void __init orion_gpio_set_valid(unsigned pin, int mode) | |||
230 | void orion_gpio_set_blink(unsigned pin, int blink) | 193 | void orion_gpio_set_blink(unsigned pin, int blink) |
231 | { | 194 | { |
232 | unsigned long flags; | 195 | unsigned long flags; |
233 | u32 u; | ||
234 | 196 | ||
235 | spin_lock_irqsave(&gpio_lock, flags); | 197 | spin_lock_irqsave(&gpio_lock, flags); |
236 | 198 | ||
237 | /* | 199 | /* Set output value to zero. */ |
238 | * Set output value to zero. | ||
239 | */ | ||
240 | __set_level(pin, 0); | 200 | __set_level(pin, 0); |
241 | 201 | ||
242 | u = readl(GPIO_BLINK_EN(pin)); | 202 | /* Set blinking. */ |
243 | if (blink) | 203 | __set_blinking(pin, blink); |
244 | u |= 1 << (pin & 31); | ||
245 | else | ||
246 | u &= ~(1 << (pin & 31)); | ||
247 | writel(u, GPIO_BLINK_EN(pin)); | ||
248 | 204 | ||
249 | spin_unlock_irqrestore(&gpio_lock, flags); | 205 | spin_unlock_irqrestore(&gpio_lock, flags); |
250 | } | 206 | } |
@@ -368,7 +324,7 @@ static int gpio_irq_set_type(u32 irq, u32 type) | |||
368 | } | 324 | } |
369 | 325 | ||
370 | struct irq_chip orion_gpio_irq_chip = { | 326 | struct irq_chip orion_gpio_irq_chip = { |
371 | .name = "orion_gpio", | 327 | .name = "orion_gpio_irq", |
372 | .ack = gpio_irq_ack, | 328 | .ack = gpio_irq_ack, |
373 | .mask = gpio_irq_mask, | 329 | .mask = gpio_irq_mask, |
374 | .unmask = gpio_irq_unmask, | 330 | .unmask = gpio_irq_unmask, |