2005-04-17 02:20:36 +04:00
/*
* linux / arch / arm / mach - footbridge / netwinder - leds . c
*
* Copyright ( C ) 1998 - 1999 Russell King
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*
* NetWinder LED control routines .
*
* The Netwinder uses the leds as follows :
* - Green - toggles state every 50 timer interrupts
* - Red - On if the system is not idle
*
* Changelog :
* 02 - 05 - 1999 RMK Various cleanups
*/
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/init.h>
# include <linux/spinlock.h>
2008-08-05 19:14:15 +04:00
# include <mach/hardware.h>
2005-04-17 02:20:36 +04:00
# include <asm/leds.h>
# include <asm/mach-types.h>
# define LED_STATE_ENABLED 1
# define LED_STATE_CLAIMED 2
static char led_state ;
static char hw_led_state ;
2009-07-03 17:44:46 +04:00
static DEFINE_RAW_SPINLOCK ( leds_lock ) ;
2005-04-17 02:20:36 +04:00
static void netwinder_leds_event ( led_event_t evt )
{
unsigned long flags ;
2009-07-03 17:44:46 +04:00
raw_spin_lock_irqsave ( & leds_lock , flags ) ;
2005-04-17 02:20:36 +04:00
switch ( evt ) {
case led_start :
led_state | = LED_STATE_ENABLED ;
hw_led_state = GPIO_GREEN_LED ;
break ;
case led_stop :
led_state & = ~ LED_STATE_ENABLED ;
break ;
case led_claim :
led_state | = LED_STATE_CLAIMED ;
hw_led_state = 0 ;
break ;
case led_release :
led_state & = ~ LED_STATE_CLAIMED ;
hw_led_state = 0 ;
break ;
# ifdef CONFIG_LEDS_TIMER
case led_timer :
if ( ! ( led_state & LED_STATE_CLAIMED ) )
hw_led_state ^ = GPIO_GREEN_LED ;
break ;
# endif
# ifdef CONFIG_LEDS_CPU
case led_idle_start :
if ( ! ( led_state & LED_STATE_CLAIMED ) )
hw_led_state & = ~ GPIO_RED_LED ;
break ;
case led_idle_end :
if ( ! ( led_state & LED_STATE_CLAIMED ) )
hw_led_state | = GPIO_RED_LED ;
break ;
# endif
case led_halted :
if ( ! ( led_state & LED_STATE_CLAIMED ) )
hw_led_state | = GPIO_RED_LED ;
break ;
case led_green_on :
if ( led_state & LED_STATE_CLAIMED )
hw_led_state | = GPIO_GREEN_LED ;
break ;
case led_green_off :
if ( led_state & LED_STATE_CLAIMED )
hw_led_state & = ~ GPIO_GREEN_LED ;
break ;
case led_amber_on :
if ( led_state & LED_STATE_CLAIMED )
hw_led_state | = GPIO_GREEN_LED | GPIO_RED_LED ;
break ;
case led_amber_off :
if ( led_state & LED_STATE_CLAIMED )
hw_led_state & = ~ ( GPIO_GREEN_LED | GPIO_RED_LED ) ;
break ;
case led_red_on :
if ( led_state & LED_STATE_CLAIMED )
hw_led_state | = GPIO_RED_LED ;
break ;
case led_red_off :
if ( led_state & LED_STATE_CLAIMED )
hw_led_state & = ~ GPIO_RED_LED ;
break ;
default :
break ;
}
2009-07-03 17:44:46 +04:00
raw_spin_unlock_irqrestore ( & leds_lock , flags ) ;
2005-04-17 02:20:36 +04:00
if ( led_state & LED_STATE_ENABLED ) {
2009-07-03 17:44:46 +04:00
raw_spin_lock_irqsave ( & nw_gpio_lock , flags ) ;
2008-12-06 11:25:16 +03:00
nw_gpio_modify_op ( GPIO_RED_LED | GPIO_GREEN_LED , hw_led_state ) ;
2009-07-03 17:44:46 +04:00
raw_spin_unlock_irqrestore ( & nw_gpio_lock , flags ) ;
2005-04-17 02:20:36 +04:00
}
}
static int __init leds_init ( void )
{
if ( machine_is_netwinder ( ) )
leds_event = netwinder_leds_event ;
leds_event ( led_start ) ;
return 0 ;
}
__initcall ( leds_init ) ;