| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 1 | /* | 
|  | 2 | * LED control using Renesas TPU | 
|  | 3 | * | 
|  | 4 | *  Copyright (C) 2011 Magnus Damm | 
|  | 5 | * | 
|  | 6 | * This program is free software; you can redistribute it and/or modify | 
|  | 7 | * it under the terms of the GNU General Public License as published by | 
|  | 8 | * the Free Software Foundation; either version 2 of the License | 
|  | 9 | * | 
|  | 10 | * This program is distributed in the hope that it will be useful, | 
|  | 11 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | 12 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
|  | 13 | * GNU General Public License for more details. | 
|  | 14 | * | 
|  | 15 | * You should have received a copy of the GNU General Public License | 
|  | 16 | * along with this program; if not, write to the Free Software | 
|  | 17 | * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA | 
|  | 18 | */ | 
|  | 19 |  | 
|  | 20 | #include <linux/module.h> | 
|  | 21 | #include <linux/init.h> | 
|  | 22 | #include <linux/platform_device.h> | 
|  | 23 | #include <linux/spinlock.h> | 
|  | 24 | #include <linux/printk.h> | 
|  | 25 | #include <linux/ioport.h> | 
|  | 26 | #include <linux/io.h> | 
|  | 27 | #include <linux/clk.h> | 
|  | 28 | #include <linux/leds.h> | 
| Magnus Damm | 2b67c95 | 2011-10-31 17:12:09 -0700 | [diff] [blame] | 29 | #include <linux/platform_data/leds-renesas-tpu.h> | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 30 | #include <linux/gpio.h> | 
|  | 31 | #include <linux/err.h> | 
|  | 32 | #include <linux/slab.h> | 
|  | 33 | #include <linux/pm_runtime.h> | 
| Magnus Damm | 02c3294 | 2011-10-31 17:12:06 -0700 | [diff] [blame] | 34 | #include <linux/workqueue.h> | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 35 |  | 
|  | 36 | enum r_tpu_pin { R_TPU_PIN_UNUSED, R_TPU_PIN_GPIO, R_TPU_PIN_GPIO_FN }; | 
|  | 37 | enum r_tpu_timer { R_TPU_TIMER_UNUSED, R_TPU_TIMER_ON }; | 
|  | 38 |  | 
|  | 39 | struct r_tpu_priv { | 
|  | 40 | struct led_classdev ldev; | 
|  | 41 | void __iomem *mapbase; | 
|  | 42 | struct clk *clk; | 
|  | 43 | struct platform_device *pdev; | 
|  | 44 | enum r_tpu_pin pin_state; | 
|  | 45 | enum r_tpu_timer timer_state; | 
|  | 46 | unsigned long min_rate; | 
|  | 47 | unsigned int refresh_rate; | 
| Magnus Damm | 02c3294 | 2011-10-31 17:12:06 -0700 | [diff] [blame] | 48 | struct work_struct work; | 
|  | 49 | enum led_brightness new_brightness; | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 50 | }; | 
|  | 51 |  | 
|  | 52 | static DEFINE_SPINLOCK(r_tpu_lock); | 
|  | 53 |  | 
|  | 54 | #define TSTR -1 /* Timer start register (shared register) */ | 
|  | 55 | #define TCR  0 /* Timer control register (+0x00) */ | 
|  | 56 | #define TMDR 1 /* Timer mode register (+0x04) */ | 
|  | 57 | #define TIOR 2 /* Timer I/O control register (+0x08) */ | 
|  | 58 | #define TIER 3 /* Timer interrupt enable register (+0x0c) */ | 
|  | 59 | #define TSR  4 /* Timer status register (+0x10) */ | 
|  | 60 | #define TCNT 5 /* Timer counter (+0x14) */ | 
|  | 61 | #define TGRA 6 /* Timer general register A (+0x18) */ | 
|  | 62 | #define TGRB 7 /* Timer general register B (+0x1c) */ | 
|  | 63 | #define TGRC 8 /* Timer general register C (+0x20) */ | 
|  | 64 | #define TGRD 9 /* Timer general register D (+0x24) */ | 
|  | 65 |  | 
|  | 66 | static inline unsigned short r_tpu_read(struct r_tpu_priv *p, int reg_nr) | 
|  | 67 | { | 
|  | 68 | struct led_renesas_tpu_config *cfg = p->pdev->dev.platform_data; | 
|  | 69 | void __iomem *base = p->mapbase; | 
|  | 70 | unsigned long offs = reg_nr << 2; | 
|  | 71 |  | 
|  | 72 | if (reg_nr == TSTR) | 
|  | 73 | return ioread16(base - cfg->channel_offset); | 
|  | 74 |  | 
|  | 75 | return ioread16(base + offs); | 
|  | 76 | } | 
|  | 77 |  | 
|  | 78 | static inline void r_tpu_write(struct r_tpu_priv *p, int reg_nr, | 
|  | 79 | unsigned short value) | 
|  | 80 | { | 
|  | 81 | struct led_renesas_tpu_config *cfg = p->pdev->dev.platform_data; | 
|  | 82 | void __iomem *base = p->mapbase; | 
|  | 83 | unsigned long offs = reg_nr << 2; | 
|  | 84 |  | 
|  | 85 | if (reg_nr == TSTR) { | 
|  | 86 | iowrite16(value, base - cfg->channel_offset); | 
|  | 87 | return; | 
|  | 88 | } | 
|  | 89 |  | 
|  | 90 | iowrite16(value, base + offs); | 
|  | 91 | } | 
|  | 92 |  | 
|  | 93 | static void r_tpu_start_stop_ch(struct r_tpu_priv *p, int start) | 
|  | 94 | { | 
|  | 95 | struct led_renesas_tpu_config *cfg = p->pdev->dev.platform_data; | 
|  | 96 | unsigned long flags, value; | 
|  | 97 |  | 
|  | 98 | /* start stop register shared by multiple timer channels */ | 
|  | 99 | spin_lock_irqsave(&r_tpu_lock, flags); | 
|  | 100 | value = r_tpu_read(p, TSTR); | 
|  | 101 |  | 
|  | 102 | if (start) | 
|  | 103 | value |= 1 << cfg->timer_bit; | 
|  | 104 | else | 
|  | 105 | value &= ~(1 << cfg->timer_bit); | 
|  | 106 |  | 
|  | 107 | r_tpu_write(p, TSTR, value); | 
|  | 108 | spin_unlock_irqrestore(&r_tpu_lock, flags); | 
|  | 109 | } | 
|  | 110 |  | 
|  | 111 | static int r_tpu_enable(struct r_tpu_priv *p, enum led_brightness brightness) | 
|  | 112 | { | 
|  | 113 | struct led_renesas_tpu_config *cfg = p->pdev->dev.platform_data; | 
|  | 114 | int prescaler[] = { 1, 4, 16, 64 }; | 
|  | 115 | int k, ret; | 
|  | 116 | unsigned long rate, tmp; | 
|  | 117 |  | 
|  | 118 | if (p->timer_state == R_TPU_TIMER_ON) | 
|  | 119 | return 0; | 
|  | 120 |  | 
|  | 121 | /* wake up device and enable clock */ | 
|  | 122 | pm_runtime_get_sync(&p->pdev->dev); | 
|  | 123 | ret = clk_enable(p->clk); | 
|  | 124 | if (ret) { | 
|  | 125 | dev_err(&p->pdev->dev, "cannot enable clock\n"); | 
|  | 126 | return ret; | 
|  | 127 | } | 
|  | 128 |  | 
|  | 129 | /* make sure channel is disabled */ | 
|  | 130 | r_tpu_start_stop_ch(p, 0); | 
|  | 131 |  | 
|  | 132 | /* get clock rate after enabling it */ | 
|  | 133 | rate = clk_get_rate(p->clk); | 
|  | 134 |  | 
|  | 135 | /* pick the lowest acceptable rate */ | 
| Axel Lin | 4d79833 | 2013-01-10 20:03:38 -0800 | [diff] [blame] | 136 | for (k = ARRAY_SIZE(prescaler) - 1; k >= 0; k--) | 
|  | 137 | if ((rate / prescaler[k]) >= p->min_rate) | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 138 | break; | 
|  | 139 |  | 
| Axel Lin | 4d79833 | 2013-01-10 20:03:38 -0800 | [diff] [blame] | 140 | if (k < 0) { | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 141 | dev_err(&p->pdev->dev, "clock rate mismatch\n"); | 
|  | 142 | goto err0; | 
|  | 143 | } | 
|  | 144 | dev_dbg(&p->pdev->dev, "rate = %lu, prescaler %u\n", | 
| Axel Lin | 4d79833 | 2013-01-10 20:03:38 -0800 | [diff] [blame] | 145 | rate, prescaler[k]); | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 146 |  | 
|  | 147 | /* clear TCNT on TGRB match, count on rising edge, set prescaler */ | 
| Axel Lin | 4d79833 | 2013-01-10 20:03:38 -0800 | [diff] [blame] | 148 | r_tpu_write(p, TCR, 0x0040 | k); | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 149 |  | 
|  | 150 | /* output 0 until TGRA, output 1 until TGRB */ | 
|  | 151 | r_tpu_write(p, TIOR, 0x0002); | 
|  | 152 |  | 
| Axel Lin | 4d79833 | 2013-01-10 20:03:38 -0800 | [diff] [blame] | 153 | rate /= prescaler[k] * p->refresh_rate; | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 154 | r_tpu_write(p, TGRB, rate); | 
|  | 155 | dev_dbg(&p->pdev->dev, "TRGB = 0x%04lx\n", rate); | 
|  | 156 |  | 
|  | 157 | tmp = (cfg->max_brightness - brightness) * rate; | 
|  | 158 | r_tpu_write(p, TGRA, tmp / cfg->max_brightness); | 
|  | 159 | dev_dbg(&p->pdev->dev, "TRGA = 0x%04lx\n", tmp / cfg->max_brightness); | 
|  | 160 |  | 
|  | 161 | /* PWM mode */ | 
|  | 162 | r_tpu_write(p, TMDR, 0x0002); | 
|  | 163 |  | 
|  | 164 | /* enable channel */ | 
|  | 165 | r_tpu_start_stop_ch(p, 1); | 
|  | 166 |  | 
|  | 167 | p->timer_state = R_TPU_TIMER_ON; | 
|  | 168 | return 0; | 
|  | 169 | err0: | 
|  | 170 | clk_disable(p->clk); | 
|  | 171 | pm_runtime_put_sync(&p->pdev->dev); | 
|  | 172 | return -ENOTSUPP; | 
|  | 173 | } | 
|  | 174 |  | 
|  | 175 | static void r_tpu_disable(struct r_tpu_priv *p) | 
|  | 176 | { | 
|  | 177 | if (p->timer_state == R_TPU_TIMER_UNUSED) | 
|  | 178 | return; | 
|  | 179 |  | 
|  | 180 | /* disable channel */ | 
|  | 181 | r_tpu_start_stop_ch(p, 0); | 
|  | 182 |  | 
|  | 183 | /* stop clock and mark device as idle */ | 
|  | 184 | clk_disable(p->clk); | 
|  | 185 | pm_runtime_put_sync(&p->pdev->dev); | 
|  | 186 |  | 
|  | 187 | p->timer_state = R_TPU_TIMER_UNUSED; | 
|  | 188 | } | 
|  | 189 |  | 
|  | 190 | static void r_tpu_set_pin(struct r_tpu_priv *p, enum r_tpu_pin new_state, | 
|  | 191 | enum led_brightness brightness) | 
|  | 192 | { | 
|  | 193 | struct led_renesas_tpu_config *cfg = p->pdev->dev.platform_data; | 
|  | 194 |  | 
|  | 195 | if (p->pin_state == new_state) { | 
|  | 196 | if (p->pin_state == R_TPU_PIN_GPIO) | 
|  | 197 | gpio_set_value(cfg->pin_gpio, brightness); | 
|  | 198 | return; | 
|  | 199 | } | 
|  | 200 |  | 
|  | 201 | if (p->pin_state == R_TPU_PIN_GPIO) | 
|  | 202 | gpio_free(cfg->pin_gpio); | 
|  | 203 |  | 
|  | 204 | if (p->pin_state == R_TPU_PIN_GPIO_FN) | 
|  | 205 | gpio_free(cfg->pin_gpio_fn); | 
|  | 206 |  | 
| Jingoo Han | b0053aa | 2012-10-23 05:18:47 -0700 | [diff] [blame] | 207 | if (new_state == R_TPU_PIN_GPIO) | 
|  | 208 | gpio_request_one(cfg->pin_gpio, GPIOF_DIR_OUT | !!brightness, | 
|  | 209 | cfg->name); | 
|  | 210 |  | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 211 | if (new_state == R_TPU_PIN_GPIO_FN) | 
|  | 212 | gpio_request(cfg->pin_gpio_fn, cfg->name); | 
|  | 213 |  | 
|  | 214 | p->pin_state = new_state; | 
|  | 215 | } | 
|  | 216 |  | 
| Magnus Damm | 02c3294 | 2011-10-31 17:12:06 -0700 | [diff] [blame] | 217 | static void r_tpu_work(struct work_struct *work) | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 218 | { | 
| Magnus Damm | 02c3294 | 2011-10-31 17:12:06 -0700 | [diff] [blame] | 219 | struct r_tpu_priv *p = container_of(work, struct r_tpu_priv, work); | 
|  | 220 | enum led_brightness brightness = p->new_brightness; | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 221 |  | 
|  | 222 | r_tpu_disable(p); | 
|  | 223 |  | 
|  | 224 | /* off and maximum are handled as GPIO pins, in between PWM */ | 
| Magnus Damm | 02c3294 | 2011-10-31 17:12:06 -0700 | [diff] [blame] | 225 | if ((brightness == 0) || (brightness == p->ldev.max_brightness)) | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 226 | r_tpu_set_pin(p, R_TPU_PIN_GPIO, brightness); | 
|  | 227 | else { | 
|  | 228 | r_tpu_set_pin(p, R_TPU_PIN_GPIO_FN, 0); | 
|  | 229 | r_tpu_enable(p, brightness); | 
|  | 230 | } | 
|  | 231 | } | 
|  | 232 |  | 
| Magnus Damm | 02c3294 | 2011-10-31 17:12:06 -0700 | [diff] [blame] | 233 | static void r_tpu_set_brightness(struct led_classdev *ldev, | 
|  | 234 | enum led_brightness brightness) | 
|  | 235 | { | 
|  | 236 | struct r_tpu_priv *p = container_of(ldev, struct r_tpu_priv, ldev); | 
|  | 237 | p->new_brightness = brightness; | 
|  | 238 | schedule_work(&p->work); | 
|  | 239 | } | 
|  | 240 |  | 
| Bill Pemberton | 98ea1ea | 2012-11-19 13:23:02 -0500 | [diff] [blame] | 241 | static int r_tpu_probe(struct platform_device *pdev) | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 242 | { | 
|  | 243 | struct led_renesas_tpu_config *cfg = pdev->dev.platform_data; | 
|  | 244 | struct r_tpu_priv *p; | 
|  | 245 | struct resource *res; | 
| Bryan Wu | bfe4c04 | 2012-07-04 11:48:57 +0800 | [diff] [blame] | 246 | int ret; | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 247 |  | 
|  | 248 | if (!cfg) { | 
|  | 249 | dev_err(&pdev->dev, "missing platform data\n"); | 
| Arnd Bergmann | 4a5a418 | 2012-08-09 05:27:58 +0800 | [diff] [blame] | 250 | return -ENODEV; | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 251 | } | 
|  | 252 |  | 
| Bryan Wu | bfe4c04 | 2012-07-04 11:48:57 +0800 | [diff] [blame] | 253 | p = devm_kzalloc(&pdev->dev, sizeof(*p), GFP_KERNEL); | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 254 | if (p == NULL) { | 
|  | 255 | dev_err(&pdev->dev, "failed to allocate driver data\n"); | 
| Bryan Wu | bfe4c04 | 2012-07-04 11:48:57 +0800 | [diff] [blame] | 256 | return -ENOMEM; | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 257 | } | 
|  | 258 |  | 
|  | 259 | res = platform_get_resource(pdev, IORESOURCE_MEM, 0); | 
|  | 260 | if (!res) { | 
|  | 261 | dev_err(&pdev->dev, "failed to get I/O memory\n"); | 
| Bryan Wu | bfe4c04 | 2012-07-04 11:48:57 +0800 | [diff] [blame] | 262 | return -ENXIO; | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 263 | } | 
|  | 264 |  | 
|  | 265 | /* map memory, let mapbase point to our channel */ | 
| Jingoo Han | f9e007f | 2012-10-23 05:22:11 -0700 | [diff] [blame] | 266 | p->mapbase = devm_ioremap_nocache(&pdev->dev, res->start, | 
|  | 267 | resource_size(res)); | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 268 | if (p->mapbase == NULL) { | 
|  | 269 | dev_err(&pdev->dev, "failed to remap I/O memory\n"); | 
| Bryan Wu | bfe4c04 | 2012-07-04 11:48:57 +0800 | [diff] [blame] | 270 | return -ENXIO; | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 271 | } | 
|  | 272 |  | 
|  | 273 | /* get hold of clock */ | 
| Jingoo Han | f9e007f | 2012-10-23 05:22:11 -0700 | [diff] [blame] | 274 | p->clk = devm_clk_get(&pdev->dev, NULL); | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 275 | if (IS_ERR(p->clk)) { | 
|  | 276 | dev_err(&pdev->dev, "cannot get clock\n"); | 
| Jingoo Han | f9e007f | 2012-10-23 05:22:11 -0700 | [diff] [blame] | 277 | return PTR_ERR(p->clk); | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 278 | } | 
|  | 279 |  | 
|  | 280 | p->pdev = pdev; | 
|  | 281 | p->pin_state = R_TPU_PIN_UNUSED; | 
|  | 282 | p->timer_state = R_TPU_TIMER_UNUSED; | 
|  | 283 | p->refresh_rate = cfg->refresh_rate ? cfg->refresh_rate : 100; | 
|  | 284 | r_tpu_set_pin(p, R_TPU_PIN_GPIO, LED_OFF); | 
|  | 285 | platform_set_drvdata(pdev, p); | 
|  | 286 |  | 
| Magnus Damm | 02c3294 | 2011-10-31 17:12:06 -0700 | [diff] [blame] | 287 | INIT_WORK(&p->work, r_tpu_work); | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 288 |  | 
|  | 289 | p->ldev.name = cfg->name; | 
|  | 290 | p->ldev.brightness = LED_OFF; | 
|  | 291 | p->ldev.max_brightness = cfg->max_brightness; | 
|  | 292 | p->ldev.brightness_set = r_tpu_set_brightness; | 
|  | 293 | p->ldev.flags |= LED_CORE_SUSPENDRESUME; | 
|  | 294 | ret = led_classdev_register(&pdev->dev, &p->ldev); | 
|  | 295 | if (ret < 0) | 
| Jingoo Han | f9e007f | 2012-10-23 05:22:11 -0700 | [diff] [blame] | 296 | goto err0; | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 297 |  | 
|  | 298 | /* max_brightness may be updated by the LED core code */ | 
|  | 299 | p->min_rate = p->ldev.max_brightness * p->refresh_rate; | 
|  | 300 |  | 
|  | 301 | pm_runtime_enable(&pdev->dev); | 
|  | 302 | return 0; | 
|  | 303 |  | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 304 | err0: | 
| Jingoo Han | f9e007f | 2012-10-23 05:22:11 -0700 | [diff] [blame] | 305 | r_tpu_set_pin(p, R_TPU_PIN_UNUSED, LED_OFF); | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 306 | return ret; | 
|  | 307 | } | 
|  | 308 |  | 
| Bill Pemberton | 678e8a6 | 2012-11-19 13:26:00 -0500 | [diff] [blame] | 309 | static int r_tpu_remove(struct platform_device *pdev) | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 310 | { | 
|  | 311 | struct r_tpu_priv *p = platform_get_drvdata(pdev); | 
|  | 312 |  | 
|  | 313 | r_tpu_set_brightness(&p->ldev, LED_OFF); | 
|  | 314 | led_classdev_unregister(&p->ldev); | 
| Magnus Damm | 02c3294 | 2011-10-31 17:12:06 -0700 | [diff] [blame] | 315 | cancel_work_sync(&p->work); | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 316 | r_tpu_disable(p); | 
|  | 317 | r_tpu_set_pin(p, R_TPU_PIN_UNUSED, LED_OFF); | 
|  | 318 |  | 
|  | 319 | pm_runtime_disable(&pdev->dev); | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 320 |  | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 321 | return 0; | 
|  | 322 | } | 
|  | 323 |  | 
|  | 324 | static struct platform_driver r_tpu_device_driver = { | 
|  | 325 | .probe		= r_tpu_probe, | 
| Bill Pemberton | df07cf8 | 2012-11-19 13:20:20 -0500 | [diff] [blame] | 326 | .remove		= r_tpu_remove, | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 327 | .driver		= { | 
|  | 328 | .name	= "leds-renesas-tpu", | 
|  | 329 | } | 
|  | 330 | }; | 
|  | 331 |  | 
| Axel Lin | 892a884 | 2012-01-10 15:09:24 -0800 | [diff] [blame] | 332 | module_platform_driver(r_tpu_device_driver); | 
| Magnus Damm | f59b6f9 | 2011-10-31 17:11:55 -0700 | [diff] [blame] | 333 |  | 
|  | 334 | MODULE_AUTHOR("Magnus Damm"); | 
|  | 335 | MODULE_DESCRIPTION("Renesas TPU LED Driver"); | 
|  | 336 | MODULE_LICENSE("GPL v2"); |