add support for target 3c24xx (more known as Openmoko GTA02 "Freerunner") and merge...
[openwrt/svn-archive/archive.git] / target / linux / s3c24xx / patches / 0032-gta02-leds.patch.patch
1 From fc22d87d11df9053f1a1b41b7b450c3af07b5059 Mon Sep 17 00:00:00 2001
2 From: mokopatches <mokopatches@openmoko.org>
3 Date: Wed, 16 Jul 2008 14:46:56 +0100
4 Subject: [PATCH] gta02-leds.patch
5
6 ---
7 drivers/leds/Kconfig | 6 +
8 drivers/leds/Makefile | 1 +
9 drivers/leds/leds-neo1973-gta02.c | 226 +++++++++++++++++++++++++++++++++++++
10 3 files changed, 233 insertions(+), 0 deletions(-)
11 create mode 100644 drivers/leds/leds-neo1973-gta02.c
12
13 diff --git a/drivers/leds/Kconfig b/drivers/leds/Kconfig
14 index 8c7d949..b6b1211 100644
15 --- a/drivers/leds/Kconfig
16 +++ b/drivers/leds/Kconfig
17 @@ -153,6 +153,12 @@ config LEDS_NEO1973_VIBRATOR
18 help
19 This option enables support for the vibrator on the FIC Neo1973.
20
21 +config LEDS_NEO1973_GTA02
22 + tristate "LED Support for the FIC Neo1973 (GTA02)"
23 + depends on LEDS_CLASS && MACH_NEO1973_GTA02
24 + help
25 + This option enables support for the LEDs on the FIC Neo1973.
26 +
27 comment "LED Triggers"
28
29 config LEDS_TRIGGERS
30 diff --git a/drivers/leds/Makefile b/drivers/leds/Makefile
31 index 148fe51..3a9df6a 100644
32 --- a/drivers/leds/Makefile
33 +++ b/drivers/leds/Makefile
34 @@ -22,6 +22,7 @@ obj-$(CONFIG_LEDS_CLEVO_MAIL) += leds-clevo-mail.o
35 obj-$(CONFIG_LEDS_HP6XX) += leds-hp6xx.o
36 obj-$(CONFIG_LEDS_FSG) += leds-fsg.o
37 obj-$(CONFIG_LEDS_NEO1973_VIBRATOR) += leds-neo1973-vibrator.o
38 +obj-$(CONFIG_LEDS_NEO1973_GTA02) += leds-neo1973-gta02.o
39
40 # LED Triggers
41 obj-$(CONFIG_LEDS_TRIGGER_TIMER) += ledtrig-timer.o
42 diff --git a/drivers/leds/leds-neo1973-gta02.c b/drivers/leds/leds-neo1973-gta02.c
43 new file mode 100644
44 index 0000000..bf1d540
45 --- /dev/null
46 +++ b/drivers/leds/leds-neo1973-gta02.c
47 @@ -0,0 +1,226 @@
48 +/*
49 + * LED driver for the FIC Neo1973 GTA02 GSM phone
50 + *
51 + * (C) 2006-2007 by OpenMoko, Inc.
52 + * Author: Harald Welte <laforge@openmoko.org>
53 + * All rights reserved.
54 + *
55 + * This program is free software; you can redistribute it and/or modify
56 + * it under the terms of the GNU General Public License version 2 as
57 + * published by the Free Software Foundation.
58 + *
59 + */
60 +
61 +#include <linux/kernel.h>
62 +#include <linux/init.h>
63 +#include <linux/platform_device.h>
64 +#include <linux/leds.h>
65 +#include <asm/hardware.h>
66 +#include <asm/mach-types.h>
67 +#include <asm/arch/pwm.h>
68 +#include <asm/arch/gta02.h>
69 +#include <asm/plat-s3c/regs-timer.h>
70 +
71 +#define MAX_LEDS 3
72 +#define COUNTER 256
73 +
74 +struct gta02_led_priv
75 +{
76 + struct mutex mutex;
77 + struct led_classdev cdev;
78 + struct s3c2410_pwm pwm;
79 + unsigned int gpio;
80 + unsigned int has_pwm;
81 +};
82 +
83 +struct gta02_led_bundle
84 +{
85 + int num_leds;
86 + struct gta02_led_priv led[MAX_LEDS];
87 +};
88 +
89 +static inline struct gta02_led_priv *to_priv(struct led_classdev *led_cdev)
90 +{
91 + return container_of(led_cdev, struct gta02_led_priv, cdev);
92 +}
93 +
94 +static inline struct gta02_led_bundle *to_bundle(struct led_classdev *led_cdev)
95 +{
96 + return dev_get_drvdata(led_cdev->dev);
97 +}
98 +
99 +static void gta02led_set(struct led_classdev *led_cdev,
100 + enum led_brightness value)
101 +{
102 + struct gta02_led_priv *lp = to_priv(led_cdev);
103 +
104 + /*
105 + * value == 255 -> 99% duty cycle (full power)
106 + * value == 128 -> 50% duty cycle (medium power)
107 + * value == 0 -> 0% duty cycle (zero power)
108 + */
109 + mutex_lock(&lp->mutex);
110 + if (lp->has_pwm) {
111 + s3c2410_pwm_duty_cycle(value, &lp->pwm);
112 + } else {
113 + if (value)
114 + s3c2410_gpio_setpin(lp->gpio, 1);
115 + else
116 + s3c2410_gpio_setpin(lp->gpio, 0);
117 + }
118 + mutex_unlock(&lp->mutex);
119 +}
120 +
121 +#ifdef CONFIG_PM
122 +static int gta02led_suspend(struct platform_device *pdev, pm_message_t state)
123 +{
124 + struct gta02_led_bundle *bundle = platform_get_drvdata(pdev);
125 + int i;
126 +
127 + for (i = 0; i < bundle->num_leds; i++)
128 + led_classdev_suspend(&bundle->led[i].cdev);
129 +
130 + return 0;
131 +}
132 +
133 +static int gta02led_resume(struct platform_device *pdev)
134 +{
135 + struct gta02_led_bundle *bundle = platform_get_drvdata(pdev);
136 + int i;
137 +
138 + for (i = 0; i < bundle->num_leds; i++)
139 + led_classdev_resume(&bundle->led[i].cdev);
140 +
141 + return 0;
142 +}
143 +#endif
144 +
145 +static int __init gta02led_probe(struct platform_device *pdev)
146 +{
147 + int i, rc;
148 + struct gta02_led_bundle *bundle;
149 +
150 + if (!machine_is_neo1973_gta02())
151 + return -EIO;
152 +
153 + bundle = kzalloc(sizeof(struct gta02_led_bundle), GFP_KERNEL);
154 + if (!bundle)
155 + return -ENOMEM;
156 + platform_set_drvdata(pdev, bundle);
157 +
158 + for (i = 0; i < pdev->num_resources; i++) {
159 + struct gta02_led_priv *lp;
160 + struct resource *r;
161 +
162 + if (i >= MAX_LEDS)
163 + break;
164 +
165 + r = platform_get_resource(pdev, 0, i);
166 + if (!r || !r->start || !r->name)
167 + continue;
168 +
169 + lp = &bundle->led[i];
170 +
171 + lp->gpio = r->start;
172 + lp->cdev.name = r->name;
173 + lp->cdev.brightness_set = gta02led_set;
174 +
175 + switch (lp->gpio) {
176 + case S3C2410_GPB0:
177 + lp->has_pwm = 1;
178 + lp->pwm.timerid = PWM0;
179 + s3c2410_gpio_cfgpin(lp->gpio, S3C2410_GPB0_TOUT0);
180 + break;
181 + case S3C2410_GPB1:
182 + lp->has_pwm = 1;
183 + lp->pwm.timerid = PWM1;
184 + s3c2410_gpio_cfgpin(lp->gpio, S3C2410_GPB1_TOUT1);
185 + break;
186 + case S3C2410_GPB2:
187 + lp->has_pwm = 1;
188 + lp->pwm.timerid = PWM2;
189 + s3c2410_gpio_cfgpin(lp->gpio, S3C2410_GPB2_TOUT2);
190 + break;
191 + case S3C2410_GPB3:
192 + lp->has_pwm = 1;
193 + lp->pwm.timerid = PWM3;
194 + s3c2410_gpio_cfgpin(lp->gpio, S3C2410_GPB3_TOUT3);
195 + break;
196 + default:
197 + break;
198 + }
199 +
200 + lp->pwm.prescaler = 0;
201 + lp->pwm.divider = S3C2410_TCFG1_MUX3_DIV8;
202 + lp->pwm.counter = COUNTER;
203 + lp->pwm.comparer = COUNTER;
204 + s3c2410_pwm_enable(&lp->pwm);
205 + s3c2410_pwm_start(&lp->pwm);
206 +
207 + switch (lp->gpio) {
208 + case S3C2410_GPB0:
209 + case S3C2410_GPB1:
210 + case S3C2410_GPB2:
211 + lp->has_pwm = 0;
212 + s3c2410_gpio_cfgpin(lp->gpio, S3C2410_GPIO_OUTPUT);
213 + s3c2410_gpio_setpin(lp->gpio, 0);
214 + break;
215 + default:
216 + break;
217 + }
218 +
219 + mutex_init(&lp->mutex);
220 + rc = led_classdev_register(&pdev->dev, &lp->cdev);
221 + }
222 +
223 + return 0;
224 +}
225 +
226 +static int gta02led_remove(struct platform_device *pdev)
227 +{
228 + struct gta02_led_bundle *bundle = platform_get_drvdata(pdev);
229 + int i;
230 +
231 + for (i = 0; i < bundle->num_leds; i++) {
232 + struct gta02_led_priv *lp = &bundle->led[i];
233 + if (lp->has_pwm)
234 + s3c2410_pwm_disable(&lp->pwm);
235 +
236 + led_classdev_unregister(&lp->cdev);
237 + mutex_destroy(&lp->mutex);
238 + }
239 +
240 + platform_set_drvdata(pdev, NULL);
241 + kfree(bundle);
242 +
243 + return 0;
244 +}
245 +
246 +static struct platform_driver gta02led_driver = {
247 + .probe = gta02led_probe,
248 + .remove = gta02led_remove,
249 +#ifdef CONFIG_PM
250 + .suspend = gta02led_suspend,
251 + .resume = gta02led_resume,
252 +#endif
253 + .driver = {
254 + .name = "gta02-led",
255 + },
256 +};
257 +
258 +static int __init gta02led_init(void)
259 +{
260 + return platform_driver_register(&gta02led_driver);
261 +}
262 +
263 +static void __exit gta02led_exit(void)
264 +{
265 + platform_driver_unregister(&gta02led_driver);
266 +}
267 +
268 +module_init(gta02led_init);
269 +module_exit(gta02led_exit);
270 +
271 +MODULE_AUTHOR("Harald Welte <laforge@openmoko.org>");
272 +MODULE_DESCRIPTION("FIC Neo1973 GTA02 LED driver");
273 +MODULE_LICENSE("GPL");
274 --
275 1.5.6.3
276