blob: 936ed426b84f4d9f7841e217cf14b37114fbfcc5 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
Tony Lindgren7c38cf02005-09-08 23:07:38 +01002 * linux/arch/arm/mach-omap1/leds-osk.c
Linus Torvalds1da177e2005-04-16 15:20:36 -07003 *
David Brownelld94577d2008-02-28 22:09:15 -08004 * LED driver for OSK with optional Mistral QVGA board
Linus Torvalds1da177e2005-04-16 15:20:36 -07005 */
Russell King2f8163b2011-07-26 10:53:52 +01006#include <linux/gpio.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -07007#include <linux/init.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -07008
Russell Kinga09e64f2008-08-05 16:14:15 +01009#include <mach/hardware.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070010#include <asm/leds.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070011
Linus Torvalds1da177e2005-04-16 15:20:36 -070012#include "leds.h"
13
14
15#define LED_STATE_ENABLED (1 << 0)
16#define LED_STATE_CLAIMED (1 << 1)
17static u8 led_state;
18
Linus Torvalds1da177e2005-04-16 15:20:36 -070019#define TIMER_LED (1 << 3) /* Mistral board */
20#define IDLE_LED (1 << 4) /* Mistral board */
21static u8 hw_led_state;
22
23
Tony Lindgren7c38cf02005-09-08 23:07:38 +010024#ifdef CONFIG_OMAP_OSK_MISTRAL
Linus Torvalds1da177e2005-04-16 15:20:36 -070025
26/* For now, all system indicators require the Mistral board, since that
27 * LED can be manipulated without a task context. This LED is either red,
28 * or green, but not both; it can't give the full "disco led" effect.
29 */
30
31#define GPIO_LED_RED 3
32#define GPIO_LED_GREEN OMAP_MPUIO(4)
33
34static void mistral_setled(void)
35{
36 int red = 0;
37 int green = 0;
38
39 if (hw_led_state & TIMER_LED)
40 red = 1;
41 else if (hw_led_state & IDLE_LED)
42 green = 1;
David Cohen6e2d4102007-12-13 22:27:15 -040043 /* else both sides are disabled */
Linus Torvalds1da177e2005-04-16 15:20:36 -070044
David Brownell0b84b5c2008-12-10 17:35:25 -080045 gpio_set_value(GPIO_LED_GREEN, green);
46 gpio_set_value(GPIO_LED_RED, red);
Linus Torvalds1da177e2005-04-16 15:20:36 -070047}
48
49#endif
50
51void osk_leds_event(led_event_t evt)
52{
53 unsigned long flags;
54 u16 leds;
55
56 local_irq_save(flags);
57
58 if (!(led_state & LED_STATE_ENABLED) && evt != led_start)
59 goto done;
60
61 leds = hw_led_state;
62 switch (evt) {
63 case led_start:
64 led_state |= LED_STATE_ENABLED;
65 hw_led_state = 0;
66 leds = ~0;
67 break;
68
69 case led_halted:
70 case led_stop:
71 led_state &= ~LED_STATE_ENABLED;
72 hw_led_state = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -070073 break;
74
75 case led_claim:
76 led_state |= LED_STATE_CLAIMED;
77 hw_led_state = 0;
78 leds = ~0;
79 break;
80
81 case led_release:
82 led_state &= ~LED_STATE_CLAIMED;
83 hw_led_state = 0;
84 break;
85
Tony Lindgren7c38cf02005-09-08 23:07:38 +010086#ifdef CONFIG_OMAP_OSK_MISTRAL
Linus Torvalds1da177e2005-04-16 15:20:36 -070087
Linus Torvalds1da177e2005-04-16 15:20:36 -070088 case led_timer:
89 hw_led_state ^= TIMER_LED;
90 mistral_setled();
91 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -070092
David Brownelleebdf7d72007-04-02 12:48:10 -070093 case led_idle_start: /* idle == off */
94 hw_led_state &= ~IDLE_LED;
Linus Torvalds1da177e2005-04-16 15:20:36 -070095 mistral_setled();
96 break;
97
98 case led_idle_end:
David Brownelleebdf7d72007-04-02 12:48:10 -070099 hw_led_state |= IDLE_LED;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700100 mistral_setled();
101 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700102
Tony Lindgren7c38cf02005-09-08 23:07:38 +0100103#endif /* CONFIG_OMAP_OSK_MISTRAL */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700104
Linus Torvalds1da177e2005-04-16 15:20:36 -0700105 default:
106 break;
107 }
108
109 leds ^= hw_led_state;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700110
111done:
112 local_irq_restore(flags);
113}