forked from len0rd/rockbox
		
	git-svn-id: svn://svn.rockbox.org/rockbox/trunk@23730 a1c6a512-1295-4272-9138-f99709370657
		
			
				
	
	
		
			1000 lines
		
	
	
	
		
			26 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1000 lines
		
	
	
	
		
			26 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /***************************************************************************
 | |
|  *             __________               __   ___.
 | |
|  *   Open      \______   \ ____   ____ |  | _\_ |__   _______  ___
 | |
|  *   Source     |       _//  _ \_/ ___\|  |/ /| __ \ /  _ \  \/  /
 | |
|  *   Jukebox    |    |   (  <_> )  \___|    < | \_\ (  <_> > <  <
 | |
|  *   Firmware   |____|_  /\____/ \___  >__|_ \|___  /\____/__/\_ \
 | |
|  *                     \/            \/     \/    \/            \/
 | |
|  * $Id$
 | |
|  *
 | |
|  * Copyright (C) 2002 by Linus Nielsen Feltzing
 | |
|  * Additional work by Martin Ritter (2007) and Thomas Martitz (2008)
 | |
|  *                  for backlight thread fading
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or
 | |
|  * modify it under the terms of the GNU General Public License
 | |
|  * as published by the Free Software Foundation; either version 2
 | |
|  * of the License, or (at your option) any later version.
 | |
|  *
 | |
|  * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
 | |
|  * KIND, either express or implied.
 | |
|  *
 | |
|  ****************************************************************************/
 | |
| #include "config.h"
 | |
| #include <stdlib.h>
 | |
| #include "cpu.h"
 | |
| #include "kernel.h"
 | |
| #include "thread.h"
 | |
| #include "i2c.h"
 | |
| #include "debug.h"
 | |
| #include "rtc.h"
 | |
| #include "usb.h"
 | |
| #include "power.h"
 | |
| #include "system.h"
 | |
| #include "button.h"
 | |
| #include "timer.h"
 | |
| #include "backlight.h"
 | |
| #include "lcd.h"
 | |
| #include "screendump.h"
 | |
| 
 | |
| #ifdef HAVE_REMOTE_LCD
 | |
| #include "lcd-remote.h"
 | |
| #endif
 | |
| #ifndef SIMULATOR
 | |
| #include "backlight-target.h"
 | |
| #else
 | |
| #include "backlight-sim.h"
 | |
| #endif
 | |
| 
 | |
| #if (CONFIG_BACKLIGHT_FADING == BACKLIGHT_FADING_SW_SETTING) \
 | |
|     || (CONFIG_BACKLIGHT_FADING == BACKLIGHT_FADING_SW_HW_REG)
 | |
| #include "backlight-sw-fading.h"
 | |
| #endif
 | |
| 
 | |
| #if defined(HAVE_BACKLIGHT) && defined(BACKLIGHT_FULL_INIT)
 | |
| 
 | |
| #define BACKLIGHT_FADE_IN_THREAD \
 | |
|     (CONFIG_BACKLIGHT_FADING &  (BACKLIGHT_FADING_SW_SETTING \
 | |
|                                 |BACKLIGHT_FADING_SW_HW_REG \
 | |
|                                 |BACKLIGHT_FADING_PWM) )
 | |
| 
 | |
| #define BACKLIGHT_THREAD_TIMEOUT HZ
 | |
| 
 | |
| enum {
 | |
|     BACKLIGHT_ON,
 | |
|     BACKLIGHT_OFF,
 | |
|     BACKLIGHT_TMO_CHANGED,
 | |
| #ifdef HAVE_BACKLIGHT_BRIGHTNESS
 | |
|     BACKLIGHT_BRIGHTNESS_CHANGED,
 | |
| #endif
 | |
| #ifdef HAVE_REMOTE_LCD
 | |
|     REMOTE_BACKLIGHT_ON,
 | |
|     REMOTE_BACKLIGHT_OFF,
 | |
|     REMOTE_BACKLIGHT_TMO_CHANGED,
 | |
| #endif
 | |
| #if defined(_BACKLIGHT_FADE_BOOST) || defined(_BACKLIGHT_FADE_ENABLE)
 | |
|     BACKLIGHT_FADE_FINISH,
 | |
| #endif
 | |
| #ifdef HAVE_LCD_SLEEP
 | |
|     LCD_SLEEP,
 | |
| #endif
 | |
| #ifdef HAVE_BUTTON_LIGHT
 | |
|     BUTTON_LIGHT_ON,
 | |
|     BUTTON_LIGHT_OFF,
 | |
|     BUTTON_LIGHT_TMO_CHANGED,
 | |
| #ifdef HAVE_BUTTONLIGHT_BRIGHTNESS
 | |
|     BUTTON_LIGHT_BRIGHTNESS_CHANGED,
 | |
| #endif
 | |
| #endif /* HAVE_BUTTON_LIGHT */
 | |
| #ifdef BACKLIGHT_DRIVER_CLOSE
 | |
|     BACKLIGHT_QUIT,
 | |
| #endif
 | |
| };
 | |
| 
 | |
| static void backlight_thread(void);
 | |
| static long backlight_stack[DEFAULT_STACK_SIZE/sizeof(long)];
 | |
| static const char backlight_thread_name[] = "backlight";
 | |
| static struct event_queue backlight_queue;
 | |
| #ifdef BACKLIGHT_DRIVER_CLOSE
 | |
| static unsigned int backlight_thread_id = 0;
 | |
| #endif
 | |
| 
 | |
| #ifdef HAVE_BACKLIGHT_BRIGHTNESS
 | |
| int backlight_brightness = DEFAULT_BRIGHTNESS_SETTING;
 | |
| #endif
 | |
| static int backlight_timer SHAREDBSS_ATTR;
 | |
| static int backlight_timeout_normal = 5*HZ;
 | |
| #if CONFIG_CHARGING
 | |
| static int backlight_timeout_plugged = 5*HZ;
 | |
| #endif
 | |
| #ifdef HAS_BUTTON_HOLD
 | |
| static int backlight_on_button_hold = 0;
 | |
| #endif
 | |
| static void backlight_timeout_handler(void);
 | |
| 
 | |
| #ifdef HAVE_BUTTON_LIGHT
 | |
| static int buttonlight_timer;
 | |
| static int buttonlight_timeout = 5*HZ;
 | |
| 
 | |
| /* Update state of buttonlight according to timeout setting */
 | |
| static void buttonlight_update_state(void)
 | |
| {
 | |
|     buttonlight_timer = buttonlight_timeout;
 | |
| 
 | |
|     /* Buttonlight == OFF in the setting? */
 | |
|     if (buttonlight_timer < 0)
 | |
|     {
 | |
|         buttonlight_timer = 0; /* Disable the timeout */
 | |
|         _buttonlight_off();
 | |
|     }
 | |
|     else
 | |
|         _buttonlight_on();
 | |
| }
 | |
| 
 | |
| /* external interface */
 | |
| void buttonlight_on(void)
 | |
| {
 | |
|     queue_remove_from_head(&backlight_queue, BUTTON_LIGHT_ON);
 | |
|     queue_post(&backlight_queue, BUTTON_LIGHT_ON, 0);
 | |
| }
 | |
| 
 | |
| void buttonlight_off(void)
 | |
| {
 | |
|     queue_post(&backlight_queue, BUTTON_LIGHT_OFF, 0);
 | |
| }
 | |
| 
 | |
| void buttonlight_set_timeout(int value)
 | |
| {
 | |
|     buttonlight_timeout = HZ * value;
 | |
|     queue_post(&backlight_queue, BUTTON_LIGHT_TMO_CHANGED, 0);
 | |
| }
 | |
| 
 | |
| int buttonlight_get_current_timeout(void)
 | |
| {
 | |
|     return buttonlight_timeout;
 | |
| }
 | |
| 
 | |
| #endif /* HAVE_BUTTON_LIGHT */
 | |
| 
 | |
| #ifdef HAVE_REMOTE_LCD
 | |
| static int remote_backlight_timer;
 | |
| static int remote_backlight_timeout_normal = 5*HZ;
 | |
| #if CONFIG_CHARGING
 | |
| static int remote_backlight_timeout_plugged = 5*HZ;
 | |
| #endif
 | |
| #ifdef HAS_REMOTE_BUTTON_HOLD
 | |
| static int remote_backlight_on_button_hold = 0;
 | |
| #endif
 | |
| #endif /* HAVE_REMOTE_LCD */
 | |
| 
 | |
| #ifdef HAVE_LCD_SLEEP
 | |
| #ifdef HAVE_LCD_SLEEP_SETTING
 | |
| const signed char lcd_sleep_timeout_value[10] =
 | |
| {
 | |
|     -1, 0, 5, 10, 15, 20, 30, 45, 60, 90
 | |
| };
 | |
| static int lcd_sleep_timeout = 10*HZ;
 | |
| #else
 | |
| /* Target defines needed value */
 | |
| #define lcd_sleep_timeout LCD_SLEEP_TIMEOUT
 | |
| #endif
 | |
| 
 | |
| static int lcd_sleep_timer SHAREDDATA_ATTR = 0;
 | |
| 
 | |
| static void backlight_lcd_sleep_countdown(bool start)
 | |
| {
 | |
|     if (!start)
 | |
|     {
 | |
|         /* Cancel the LCD sleep countdown */
 | |
|         lcd_sleep_timer = 0;
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* Start LCD sleep countdown */
 | |
|     if (lcd_sleep_timeout < 0)
 | |
|     {
 | |
|         lcd_sleep_timer = 0; /* Setting == Always */
 | |
|         /* Ensure lcd_sleep() is called from backlight_thread() */
 | |
| #if (CONFIG_BACKLIGHT_FADING == BACKLIGHT_FADING_PWM)
 | |
|         queue_post(&backlight_queue, LCD_SLEEP, 0);
 | |
| #else
 | |
|         lcd_sleep();
 | |
| #endif
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         lcd_sleep_timer = lcd_sleep_timeout;
 | |
|     }
 | |
| }
 | |
| #endif /* HAVE_LCD_SLEEP */
 | |
| 
 | |
| #if (CONFIG_BACKLIGHT_FADING == BACKLIGHT_FADING_SW_SETTING) \
 | |
|     || (CONFIG_BACKLIGHT_FADING == BACKLIGHT_FADING_SW_HW_REG)
 | |
| static int backlight_fading_type = (FADING_UP|FADING_DOWN);
 | |
| static int backlight_fading_state = NOT_FADING;
 | |
| #endif
 | |
| 
 | |
| 
 | |
| /* backlight fading */
 | |
| #if (CONFIG_BACKLIGHT_FADING == BACKLIGHT_FADING_PWM)
 | |
| #define BL_PWM_INTERVAL 5  /* Cycle interval in ms */
 | |
| #define BL_PWM_BITS     8
 | |
| #define BL_PWM_COUNT    (1<<BL_PWM_BITS)
 | |
| 
 | |
| /* s15.16 fixed point variables */
 | |
| static int32_t bl_fade_in_step  = ((BL_PWM_INTERVAL*BL_PWM_COUNT)<<16)/300;
 | |
| static int32_t bl_fade_out_step = ((BL_PWM_INTERVAL*BL_PWM_COUNT)<<16)/2000;
 | |
| static int32_t bl_dim_fraction  = 0;     
 | |
| 
 | |
| static int bl_dim_target  = 0;
 | |
| static int bl_dim_current = 0;
 | |
| static enum {DIM_STATE_START, DIM_STATE_MAIN} bl_dim_state = DIM_STATE_START;
 | |
| static bool bl_timer_active = false;
 | |
| 
 | |
| static void backlight_isr(void)
 | |
| {
 | |
|     int timer_period = (TIMER_FREQ*BL_PWM_INTERVAL/1000);
 | |
|     bool idle = false;
 | |
| 
 | |
|     switch (bl_dim_state)
 | |
|     {
 | |
|       /* New cycle */
 | |
|       case DIM_STATE_START:
 | |
|         bl_dim_current = bl_dim_fraction >> 16;
 | |
| 
 | |
|         if (bl_dim_current > 0 && bl_dim_current < BL_PWM_COUNT)
 | |
|         {
 | |
|             _backlight_on_isr();
 | |
|             timer_period = (timer_period * bl_dim_current) >> BL_PWM_BITS;
 | |
|             bl_dim_state = DIM_STATE_MAIN;
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             if (bl_dim_current)
 | |
|                 _backlight_on_isr();
 | |
|             else
 | |
|                 _backlight_off_isr();
 | |
|             if (bl_dim_current == bl_dim_target)
 | |
|                 idle = true;
 | |
|         }
 | |
|         if (bl_dim_current < bl_dim_target)
 | |
|         {
 | |
|             bl_dim_fraction = MIN(bl_dim_fraction + bl_fade_in_step,
 | |
|                                   (BL_PWM_COUNT<<16));
 | |
|         }
 | |
|         else if (bl_dim_current > bl_dim_target)
 | |
|         {
 | |
|             bl_dim_fraction = MAX(bl_dim_fraction - bl_fade_out_step, 0);
 | |
|         }
 | |
|         break;
 | |
| 
 | |
|       /* Dim main screen */
 | |
|       case DIM_STATE_MAIN:
 | |
|         _backlight_off_isr();
 | |
|         timer_period = (timer_period * (BL_PWM_COUNT - bl_dim_current))
 | |
|                      >> BL_PWM_BITS;
 | |
|         bl_dim_state = DIM_STATE_START;
 | |
|         break ;
 | |
|     }
 | |
|     if (idle)
 | |
|     {
 | |
| #if defined(_BACKLIGHT_FADE_BOOST) || defined(_BACKLIGHT_FADE_ENABLE)
 | |
|         queue_post(&backlight_queue, BACKLIGHT_FADE_FINISH, 0);
 | |
| #endif
 | |
|         timer_unregister();
 | |
|         bl_timer_active = false;
 | |
| 
 | |
| #ifdef HAVE_LCD_SLEEP
 | |
|         if (bl_dim_current == 0)
 | |
|             backlight_lcd_sleep_countdown(true);
 | |
| #endif
 | |
|     }
 | |
|     else
 | |
|         timer_set_period(timer_period);
 | |
| }
 | |
| 
 | |
| static void backlight_switch(void)
 | |
| {
 | |
|     if (bl_dim_target > (BL_PWM_COUNT/2))
 | |
|     {
 | |
|         _backlight_on_normal();
 | |
|         bl_dim_fraction = (BL_PWM_COUNT<<16);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         _backlight_off_normal();
 | |
|         bl_dim_fraction = 0;
 | |
| 
 | |
| #ifdef HAVE_LCD_SLEEP
 | |
|         backlight_lcd_sleep_countdown(true);
 | |
| #endif
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void backlight_release_timer(void)
 | |
| {
 | |
| #ifdef _BACKLIGHT_FADE_BOOST
 | |
|     cpu_boost(false);
 | |
| #endif
 | |
|     timer_unregister();
 | |
|     bl_timer_active = false;
 | |
|     backlight_switch();
 | |
| }
 | |
| 
 | |
| static void backlight_dim(int value)
 | |
| {
 | |
|     /* protect from extraneous calls with the same target value */
 | |
|     if (value == bl_dim_target)
 | |
|         return;
 | |
| 
 | |
|     bl_dim_target = value;
 | |
| 
 | |
|     if (bl_timer_active)
 | |
|         return ;
 | |
| 
 | |
|     if (timer_register(0, backlight_release_timer, 2, backlight_isr
 | |
|                        IF_COP(, CPU)))
 | |
|     {
 | |
| #ifdef _BACKLIGHT_FADE_BOOST
 | |
|         /* Prevent cpu frequency changes while dimming. */
 | |
|         cpu_boost(true);
 | |
| #endif
 | |
|         bl_timer_active = true;
 | |
|     }
 | |
|     else
 | |
|         backlight_switch();
 | |
| }
 | |
| 
 | |
| static void backlight_setup_fade_up(void)
 | |
| {
 | |
|     if (bl_fade_in_step > 0)
 | |
|     {
 | |
| #ifdef _BACKLIGHT_FADE_ENABLE
 | |
|         _backlight_hw_enable(true);
 | |
| #endif
 | |
|         backlight_dim(BL_PWM_COUNT);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         bl_dim_target = BL_PWM_COUNT;
 | |
|         bl_dim_fraction = (BL_PWM_COUNT<<16);
 | |
|         _backlight_on_normal();
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void backlight_setup_fade_down(void)
 | |
| {
 | |
|     if (bl_fade_out_step > 0)
 | |
|     {
 | |
|         backlight_dim(0);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         bl_dim_target = bl_dim_fraction = 0;
 | |
|         _backlight_off_normal();
 | |
| #ifdef HAVE_LCD_SLEEP
 | |
|         backlight_lcd_sleep_countdown(true);
 | |
| #endif
 | |
|     }
 | |
| }
 | |
| 
 | |
| void backlight_set_fade_in(int value)
 | |
| {
 | |
|     if (value > 0)
 | |
|         bl_fade_in_step = ((BL_PWM_INTERVAL*BL_PWM_COUNT)<<16) / value;
 | |
|     else
 | |
|         bl_fade_in_step = 0;
 | |
| }
 | |
| 
 | |
| void backlight_set_fade_out(int value)
 | |
| {
 | |
|     if (value > 0)
 | |
|         bl_fade_out_step = ((BL_PWM_INTERVAL*BL_PWM_COUNT)<<16) / value;
 | |
|     else
 | |
|         bl_fade_out_step = 0;
 | |
| }
 | |
| 
 | |
| #elif  (CONFIG_BACKLIGHT_FADING == BACKLIGHT_FADING_SW_SETTING) \
 | |
|     || (CONFIG_BACKLIGHT_FADING == BACKLIGHT_FADING_SW_HW_REG)
 | |
| 
 | |
| void backlight_set_fade_out(bool value)
 | |
| {
 | |
|     if(value) /* on */
 | |
|         backlight_fading_type |= FADING_DOWN;
 | |
|     else
 | |
|         backlight_fading_type &= FADING_UP;
 | |
| }
 | |
| 
 | |
| void backlight_set_fade_in(bool value)
 | |
| {
 | |
|     if(value) /* on */
 | |
|         backlight_fading_type |= FADING_UP;
 | |
|     else
 | |
|         backlight_fading_type &= FADING_DOWN;
 | |
| }
 | |
| 
 | |
| static void backlight_setup_fade_up(void)
 | |
| {
 | |
|     if (backlight_fading_type & FADING_UP)
 | |
|     {
 | |
|         if (backlight_fading_state == NOT_FADING)
 | |
|         {
 | |
|             /* make sure the backlight is at lowest level */
 | |
|             _backlight_on();
 | |
|         }
 | |
|         backlight_fading_state = FADING_UP;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         backlight_fading_state = NOT_FADING;
 | |
|         _backlight_fade_update_state(backlight_brightness);
 | |
|         _backlight_on();
 | |
|         _backlight_set_brightness(backlight_brightness);
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void backlight_setup_fade_down(void)
 | |
| {
 | |
|     if (backlight_fading_type & FADING_DOWN)
 | |
|     {
 | |
|         backlight_fading_state = FADING_DOWN;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         backlight_fading_state = NOT_FADING;
 | |
|         _backlight_fade_update_state(MIN_BRIGHTNESS_SETTING-1);
 | |
|         _backlight_off();
 | |
| #if (CONFIG_BACKLIGHT_FADING == BACKLIGHT_FADING_SW_HW_REG)
 | |
|         /* write the lowest brightness level to the hardware so that
 | |
|          * fading up is glitch free */
 | |
|         _backlight_set_brightness(MIN_BRIGHTNESS_SETTING);
 | |
| #endif
 | |
| #ifdef HAVE_LCD_SLEEP
 | |
|         backlight_lcd_sleep_countdown(true);
 | |
| #endif
 | |
|     }
 | |
| }
 | |
| #endif /* CONFIG_BACKLIGHT_FADING */
 | |
| 
 | |
| static inline void do_backlight_off(void)
 | |
| {
 | |
|     backlight_timer = 0;
 | |
| #if BACKLIGHT_FADE_IN_THREAD
 | |
|     backlight_setup_fade_down();
 | |
| #else
 | |
|     _backlight_off();
 | |
|     /* targets that have fading need to start the countdown when done with
 | |
|      * fading */
 | |
| #ifdef HAVE_LCD_SLEEP
 | |
|     backlight_lcd_sleep_countdown(true);
 | |
| #endif
 | |
| #endif
 | |
| }
 | |
| 
 | |
| /* Update state of backlight according to timeout setting */
 | |
| static void backlight_update_state(void)
 | |
| {
 | |
| 
 | |
|     int timeout = backlight_get_current_timeout();
 | |
| 
 | |
|     /* Backlight == OFF in the setting? */
 | |
|     if (UNLIKELY(timeout < 0))
 | |
|     {
 | |
|         do_backlight_off();
 | |
| #if (CONFIG_BACKLIGHT_FADING == BACKLIGHT_FADING_SW_SETTING) \
 | |
|     || (CONFIG_BACKLIGHT_FADING == BACKLIGHT_FADING_SW_HW_REG)
 | |
|         /* necessary step to issue fading down when the setting is selected */
 | |
|         if (queue_empty(&backlight_queue))
 | |
|             queue_post(&backlight_queue, SYS_TIMEOUT, 0);
 | |
| #endif
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         backlight_timer = timeout;
 | |
| 
 | |
| #ifdef HAVE_LCD_SLEEP
 | |
|         backlight_lcd_sleep_countdown(false); /* wake up lcd */
 | |
| #endif
 | |
| 
 | |
| #if BACKLIGHT_FADE_IN_THREAD
 | |
|         backlight_setup_fade_up();
 | |
| #else
 | |
|         _backlight_on();
 | |
| #endif
 | |
|     }
 | |
| }
 | |
| 
 | |
| #ifdef HAVE_REMOTE_LCD
 | |
| /* Update state of remote backlight according to timeout setting */
 | |
| static void remote_backlight_update_state(void)
 | |
| {
 | |
|     int timeout = remote_backlight_get_current_timeout();
 | |
|     /* Backlight == OFF in the setting? */
 | |
|     if (timeout < 0)
 | |
|     {
 | |
|         remote_backlight_timer = 0; /* Disable the timeout */
 | |
|         _remote_backlight_off();
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         remote_backlight_timer = timeout;
 | |
|         _remote_backlight_on();
 | |
|     }
 | |
| }
 | |
| #endif /* HAVE_REMOTE_LCD */
 | |
| 
 | |
| void backlight_thread(void)
 | |
| {
 | |
|     struct queue_event ev;
 | |
|     bool locked = false;
 | |
| 
 | |
|     while(1)
 | |
|     {
 | |
| #if  (CONFIG_BACKLIGHT_FADING == BACKLIGHT_FADING_SW_SETTING) \
 | |
|     || (CONFIG_BACKLIGHT_FADING == BACKLIGHT_FADING_SW_HW_REG)
 | |
|         if (backlight_fading_state != NOT_FADING)
 | |
|             queue_wait_w_tmo(&backlight_queue, &ev, FADE_DELAY);
 | |
|         else
 | |
| #endif
 | |
|             queue_wait_w_tmo(&backlight_queue, &ev, BACKLIGHT_THREAD_TIMEOUT);
 | |
|         switch(ev.id)
 | |
|         {   /* These events must always be processed */
 | |
| #ifdef _BACKLIGHT_FADE_BOOST
 | |
|             case BACKLIGHT_FADE_FINISH:
 | |
|                 cpu_boost(false);
 | |
|                 break;
 | |
| #endif
 | |
| #ifdef _BACKLIGHT_FADE_ENABLE
 | |
|             case BACKLIGHT_FADE_FINISH:
 | |
|                 _backlight_hw_enable((bl_dim_current|bl_dim_target) != 0);
 | |
|                 break;
 | |
| #endif
 | |
| 
 | |
| #ifndef SIMULATOR
 | |
|             /* Here for now or else the aggressive init messes up scrolling */
 | |
| #ifdef HAVE_REMOTE_LCD
 | |
|             case SYS_REMOTE_PLUGGED:
 | |
|                 lcd_remote_on();
 | |
|                 lcd_remote_update();
 | |
|                 break;
 | |
| 
 | |
|             case SYS_REMOTE_UNPLUGGED:
 | |
|                 lcd_remote_off();
 | |
|                 break;
 | |
| #elif defined HAVE_REMOTE_LCD_AS_MAIN
 | |
|             case SYS_REMOTE_PLUGGED:
 | |
|                 lcd_on();
 | |
|                 lcd_update();
 | |
|                 break;
 | |
| 
 | |
|             case SYS_REMOTE_UNPLUGGED:
 | |
|                 lcd_off();
 | |
|                 break;
 | |
| #endif /* HAVE_REMOTE_LCD/ HAVE_REMOTE_LCD_AS_MAIN */
 | |
| #endif /* !SIMULATOR */
 | |
|             case SYS_USB_CONNECTED:
 | |
|                 usb_acknowledge(SYS_USB_CONNECTED_ACK);
 | |
|                 break;
 | |
| 
 | |
|             case SYS_USB_DISCONNECTED:
 | |
|                 usb_acknowledge(SYS_USB_DISCONNECTED_ACK);
 | |
|                 break;
 | |
| 
 | |
| #ifdef BACKLIGHT_DRIVER_CLOSE
 | |
|             /* Get out of here */
 | |
|             case BACKLIGHT_QUIT:
 | |
|                 return;
 | |
| #endif
 | |
|         }
 | |
|         if (locked)
 | |
|             continue;
 | |
| 
 | |
|         switch(ev.id)
 | |
|         {   /* These events are only processed if backlight isn't locked */
 | |
| #ifdef HAVE_REMOTE_LCD
 | |
|             case REMOTE_BACKLIGHT_TMO_CHANGED:
 | |
|             case REMOTE_BACKLIGHT_ON:
 | |
|                 remote_backlight_update_state();
 | |
|                 break;
 | |
| 
 | |
|             case REMOTE_BACKLIGHT_OFF:
 | |
|                 remote_backlight_timer = 0; /* Disable the timeout */
 | |
|                 _remote_backlight_off();
 | |
|                 break;
 | |
| #endif /* HAVE_REMOTE_LCD */
 | |
| 
 | |
|             case BACKLIGHT_TMO_CHANGED:
 | |
|             case BACKLIGHT_ON:
 | |
|                 backlight_update_state();
 | |
|                 break;
 | |
| 
 | |
|             case BACKLIGHT_OFF:
 | |
|                 do_backlight_off();
 | |
|                 break;
 | |
| #ifdef HAVE_BACKLIGHT_BRIGHTNESS
 | |
|             case BACKLIGHT_BRIGHTNESS_CHANGED:
 | |
|                 backlight_brightness = (int)ev.data;
 | |
|                 _backlight_set_brightness((int)ev.data);
 | |
| #if  (CONFIG_BACKLIGHT_FADING == BACKLIGHT_FADING_SW_SETTING) \
 | |
|     || (CONFIG_BACKLIGHT_FADING == BACKLIGHT_FADING_SW_HW_REG)
 | |
|                 /* receive backlight brightness */
 | |
|                 _backlight_fade_update_state((int)ev.data);
 | |
| #endif
 | |
|                 break;
 | |
| #endif /* HAVE_BACKLIGHT_BRIGHTNESS */
 | |
| #ifdef HAVE_LCD_SLEEP
 | |
|             case LCD_SLEEP:
 | |
|                 lcd_sleep();
 | |
|                 break;
 | |
| #endif
 | |
| #ifdef HAVE_BUTTON_LIGHT
 | |
|             case BUTTON_LIGHT_TMO_CHANGED:
 | |
|             case BUTTON_LIGHT_ON:
 | |
|                 buttonlight_update_state();
 | |
|                 break;
 | |
| 
 | |
|             case BUTTON_LIGHT_OFF:
 | |
|                 buttonlight_timer = 0;
 | |
|                 _buttonlight_off();
 | |
|                 break;
 | |
| #ifdef HAVE_BUTTONLIGHT_BRIGHTNESS
 | |
|             case BUTTON_LIGHT_BRIGHTNESS_CHANGED:                
 | |
|                 _buttonlight_set_brightness((int)ev.data);
 | |
|                 break;
 | |
| #endif /* HAVE_BUTTONLIGHT_BRIGHTNESS */
 | |
| #endif /* HAVE_BUTTON_LIGHT */
 | |
| 
 | |
|             case SYS_POWEROFF:  /* Lock backlight on poweroff so it doesn't */
 | |
|                 locked = true;      /* go off before power is actually cut. */
 | |
|                 /* fall through */
 | |
| #if CONFIG_CHARGING
 | |
|             case SYS_CHARGER_CONNECTED:
 | |
|             case SYS_CHARGER_DISCONNECTED:
 | |
| #endif
 | |
|                 backlight_update_state();
 | |
| #ifdef HAVE_REMOTE_LCD
 | |
|                 remote_backlight_update_state();
 | |
| #endif
 | |
|                 break;
 | |
|             case SYS_TIMEOUT:
 | |
| #if  (CONFIG_BACKLIGHT_FADING == BACKLIGHT_FADING_SW_SETTING) \
 | |
|     || (CONFIG_BACKLIGHT_FADING == BACKLIGHT_FADING_SW_HW_REG)
 | |
|                 if (backlight_fading_state != NOT_FADING)
 | |
|                 {
 | |
|                     if ((_backlight_fade_step(backlight_fading_state)))
 | |
|                     {   /* finished fading */
 | |
| #ifdef HAVE_LCD_SLEEP
 | |
|                         if (backlight_fading_state == FADING_DOWN)
 | |
|                         {   /* start sleep countdown */
 | |
|                              backlight_lcd_sleep_countdown(true);
 | |
|                         }
 | |
| #endif
 | |
|                         backlight_fading_state = NOT_FADING;
 | |
|                     }
 | |
|                 }
 | |
|                 else
 | |
| #endif /* CONFIG_BACKLIGHT_FADING */
 | |
|                     backlight_timeout_handler();
 | |
|                 break;
 | |
|         }
 | |
|     } /* end while */
 | |
| }
 | |
| 
 | |
| static void backlight_timeout_handler(void)
 | |
| {
 | |
|     if(backlight_timer > 0)
 | |
|     {
 | |
|         backlight_timer -= BACKLIGHT_THREAD_TIMEOUT;
 | |
|         if(backlight_timer <= 0)
 | |
|         {
 | |
|             do_backlight_off();
 | |
|         }
 | |
|     }
 | |
| #ifdef HAVE_LCD_SLEEP
 | |
|     else if(lcd_sleep_timer > 0)
 | |
|     {
 | |
|         lcd_sleep_timer -= BACKLIGHT_THREAD_TIMEOUT;
 | |
|         if(lcd_sleep_timer <= 0)
 | |
|         {
 | |
|             lcd_sleep();
 | |
|         }
 | |
|     }
 | |
| #endif /* HAVE_LCD_SLEEP */
 | |
| #ifdef HAVE_REMOTE_LCD
 | |
|     if(remote_backlight_timer > 0)
 | |
|     {
 | |
|         remote_backlight_timer -= BACKLIGHT_THREAD_TIMEOUT;
 | |
|         if(remote_backlight_timer <= 0)
 | |
|         {
 | |
|             _remote_backlight_off();
 | |
|         }
 | |
|     }
 | |
| #endif /* HAVE_REMOVE_LCD */
 | |
| #ifdef HAVE_BUTTON_LIGHT
 | |
|     if (buttonlight_timer > 0)
 | |
|     {
 | |
|         buttonlight_timer -= BACKLIGHT_THREAD_TIMEOUT;
 | |
|         if (buttonlight_timer <= 0)
 | |
|         {
 | |
|             _buttonlight_off();
 | |
|         }
 | |
|     }
 | |
| #endif /* HAVE_BUTTON_LIGHT */
 | |
| }
 | |
| 
 | |
| void backlight_init(void)
 | |
| {
 | |
|     queue_init(&backlight_queue, true);
 | |
| 
 | |
|     if (_backlight_init())
 | |
|     {
 | |
| #if (CONFIG_BACKLIGHT_FADING == BACKLIGHT_FADING_PWM)
 | |
|         /* If backlight is already on, don't fade in. */
 | |
|         bl_dim_target = BL_PWM_COUNT;
 | |
|         bl_dim_fraction = (BL_PWM_COUNT<<16);
 | |
| #endif
 | |
|     }
 | |
|     /* Leave all lights as set by the bootloader here. The settings load will
 | |
|      * call the appropriate backlight_set_*() functions, only changing light
 | |
|      * status if necessary. */
 | |
| #ifdef BACKLIGHT_DRIVER_CLOSE
 | |
|     backlight_thread_id =
 | |
| #endif
 | |
|     create_thread(backlight_thread, backlight_stack,
 | |
|                   sizeof(backlight_stack), 0, backlight_thread_name
 | |
|                   IF_PRIO(, PRIORITY_USER_INTERFACE)
 | |
|                   IF_COP(, CPU));
 | |
| }
 | |
| 
 | |
| #ifdef BACKLIGHT_DRIVER_CLOSE
 | |
| void backlight_close(void)
 | |
| {
 | |
|     unsigned int thread = backlight_thread_id;
 | |
| 
 | |
|     /* Wait for thread to exit */
 | |
|     if (thread == 0)
 | |
|         return;
 | |
| 
 | |
|     backlight_thread_id = 0;
 | |
| 
 | |
|     queue_post(&backlight_queue, BACKLIGHT_QUIT, 0);
 | |
|     thread_wait(thread);
 | |
| }
 | |
| #endif /* BACKLIGHT_DRIVER_CLOSE */
 | |
| 
 | |
| void backlight_on(void)
 | |
| {
 | |
|     queue_remove_from_head(&backlight_queue, BACKLIGHT_ON);
 | |
|     queue_post(&backlight_queue, BACKLIGHT_ON, 0);
 | |
| }
 | |
| 
 | |
| void backlight_off(void)
 | |
| {
 | |
|     queue_post(&backlight_queue, BACKLIGHT_OFF, 0);
 | |
| }
 | |
| 
 | |
| /* returns true when the backlight is on,
 | |
|  * and optionally when it's set to always off. */
 | |
| bool is_backlight_on(bool ignore_always_off)
 | |
| {
 | |
|     int timeout = backlight_get_current_timeout();
 | |
|     return (backlight_timer > 0)   /* countdown */
 | |
|         || (timeout == 0) /* always on */
 | |
|         || ((timeout < 0) && !ignore_always_off);
 | |
| }
 | |
| 
 | |
| /* return value in ticks; 0 means always on, <0 means always off */
 | |
| int backlight_get_current_timeout(void)
 | |
| {
 | |
| #ifdef HAS_BUTTON_HOLD
 | |
|     if ((backlight_on_button_hold != 0)
 | |
| #ifdef HAVE_REMOTE_LCD_AS_MAIN
 | |
|         && remote_button_hold()
 | |
| #else
 | |
|         && button_hold()
 | |
| #endif
 | |
|         )
 | |
|         return (backlight_on_button_hold == 2) ? 0 : -1;
 | |
|         /* always on or always off */
 | |
|     else
 | |
| #endif
 | |
| #if CONFIG_CHARGING
 | |
|         if (power_input_present())
 | |
|             return backlight_timeout_plugged;
 | |
|         else
 | |
| #endif
 | |
|             return backlight_timeout_normal;
 | |
| }
 | |
| 
 | |
| void backlight_set_timeout(int value)
 | |
| {
 | |
|     backlight_timeout_normal = HZ * value;
 | |
|     queue_post(&backlight_queue, BACKLIGHT_TMO_CHANGED, 0);
 | |
| }
 | |
| 
 | |
| #if CONFIG_CHARGING
 | |
| void backlight_set_timeout_plugged(int value)
 | |
| {
 | |
|     backlight_timeout_plugged = HZ * value;
 | |
|     queue_post(&backlight_queue, BACKLIGHT_TMO_CHANGED, 0);
 | |
| }
 | |
| #endif /* CONFIG_CHARGING */
 | |
| 
 | |
| #ifdef HAS_BUTTON_HOLD
 | |
| /* Hold button change event handler. */
 | |
| void backlight_hold_changed(bool hold_button)
 | |
| {
 | |
|     if (!hold_button || (backlight_on_button_hold > 0))
 | |
|         /* if unlocked or override in effect */
 | |
|         backlight_on();
 | |
| }
 | |
| 
 | |
| void backlight_set_on_button_hold(int index)
 | |
| {
 | |
|     if ((unsigned)index >= 3)
 | |
|         /* if given a weird value, use default */
 | |
|         index = 0;
 | |
| 
 | |
|     backlight_on_button_hold = index;
 | |
|     queue_post(&backlight_queue, BACKLIGHT_TMO_CHANGED, 0);
 | |
| }
 | |
| #endif /* HAS_BUTTON_HOLD */
 | |
| 
 | |
| #ifdef HAVE_LCD_SLEEP_SETTING
 | |
| void lcd_set_sleep_after_backlight_off(int index)
 | |
| {
 | |
|     if ((unsigned)index >= sizeof(lcd_sleep_timeout_value))
 | |
|         /* if given a weird value, use default */
 | |
|         index = 3;
 | |
| 
 | |
|     lcd_sleep_timeout = HZ * lcd_sleep_timeout_value[index];
 | |
| 
 | |
|     if (is_backlight_on(true))
 | |
|         /* Timer will be set when bl turns off or bl set to on. */
 | |
|         return;
 | |
| 
 | |
|     /* Backlight is Off */
 | |
|     if (lcd_sleep_timeout < 0)
 | |
|         lcd_sleep_timer = 1; /* Always - sleep next tick */
 | |
|     else
 | |
|         lcd_sleep_timer = lcd_sleep_timeout; /* Never, other */
 | |
| }
 | |
| #endif /* HAVE_LCD_SLEEP_SETTING */
 | |
| 
 | |
| #ifdef HAVE_REMOTE_LCD
 | |
| void remote_backlight_on(void)
 | |
| {
 | |
|     queue_post(&backlight_queue, REMOTE_BACKLIGHT_ON, 0);
 | |
| }
 | |
| 
 | |
| void remote_backlight_off(void)
 | |
| {
 | |
|     queue_post(&backlight_queue, REMOTE_BACKLIGHT_OFF, 0);
 | |
| }
 | |
| 
 | |
| void remote_backlight_set_timeout(int value)
 | |
| {
 | |
|     remote_backlight_timeout_normal = HZ * value;
 | |
|     queue_post(&backlight_queue, REMOTE_BACKLIGHT_TMO_CHANGED, 0);
 | |
| }
 | |
| 
 | |
| #if CONFIG_CHARGING
 | |
| void remote_backlight_set_timeout_plugged(int value)
 | |
| {
 | |
|     remote_backlight_timeout_plugged = HZ * value;
 | |
|     queue_post(&backlight_queue, REMOTE_BACKLIGHT_TMO_CHANGED, 0);
 | |
| }
 | |
| #endif /* CONFIG_CHARGING */
 | |
| 
 | |
| #ifdef HAS_REMOTE_BUTTON_HOLD
 | |
| /* Remote hold button change event handler. */
 | |
| void remote_backlight_hold_changed(bool rc_hold_button)
 | |
| {
 | |
|     if (!rc_hold_button || (remote_backlight_on_button_hold > 0))
 | |
|         /* if unlocked or override */
 | |
|         remote_backlight_on();
 | |
| }
 | |
| 
 | |
| void remote_backlight_set_on_button_hold(int index)
 | |
| {
 | |
|     if ((unsigned)index >= 3)
 | |
|         /* if given a weird value, use default */
 | |
|         index = 0;
 | |
| 
 | |
|     remote_backlight_on_button_hold = index;
 | |
|     queue_post(&backlight_queue, REMOTE_BACKLIGHT_TMO_CHANGED, 0);
 | |
| }
 | |
| #endif /* HAS_REMOTE_BUTTON_HOLD */
 | |
| 
 | |
| /* return value in ticks; 0 means always on, <0 means always off */
 | |
| int remote_backlight_get_current_timeout(void)
 | |
| {
 | |
| #ifdef HAS_REMOTE_BUTTON_HOLD
 | |
|     if (remote_button_hold() && (remote_backlight_on_button_hold != 0))
 | |
|         return (remote_backlight_on_button_hold == 2)
 | |
|                                  ? 0 : -1;  /* always on or always off */
 | |
|     else
 | |
| #endif
 | |
| #if CONFIG_CHARGING
 | |
|         if (power_input_present())
 | |
|             return remote_backlight_timeout_plugged;
 | |
|         else
 | |
| #endif
 | |
|             return remote_backlight_timeout_normal;
 | |
| }
 | |
| 
 | |
| /* returns true when the backlight is on, and
 | |
|  * optionally  when it's set to always off */
 | |
| bool is_remote_backlight_on(bool ignore_always_off)
 | |
| {
 | |
|     int timeout = remote_backlight_get_current_timeout();
 | |
|     return (remote_backlight_timer > 0)   /* countdown */
 | |
|         || (timeout == 0) /* always on */
 | |
|         || ((timeout < 0) && !ignore_always_off);
 | |
| }
 | |
| 
 | |
| #endif /* HAVE_REMOTE_LCD */
 | |
| 
 | |
| #ifdef HAVE_BACKLIGHT_BRIGHTNESS
 | |
| void backlight_set_brightness(int val)
 | |
| {
 | |
|     if (val < MIN_BRIGHTNESS_SETTING)
 | |
|         val = MIN_BRIGHTNESS_SETTING;
 | |
|     else if (val > MAX_BRIGHTNESS_SETTING)
 | |
|         val = MAX_BRIGHTNESS_SETTING;
 | |
| 
 | |
|     queue_post(&backlight_queue, BACKLIGHT_BRIGHTNESS_CHANGED, val);
 | |
| }
 | |
| #endif /* HAVE_BACKLIGHT_BRIGHTNESS */
 | |
| 
 | |
| #ifdef HAVE_BUTTONLIGHT_BRIGHTNESS
 | |
| void buttonlight_set_brightness(int val)
 | |
| {
 | |
|     if (val < MIN_BRIGHTNESS_SETTING)
 | |
|         val = MIN_BRIGHTNESS_SETTING;
 | |
|     else if (val > MAX_BRIGHTNESS_SETTING)
 | |
|         val = MAX_BRIGHTNESS_SETTING;
 | |
| 
 | |
|      queue_post(&backlight_queue, BUTTON_LIGHT_BRIGHTNESS_CHANGED, val);
 | |
| }
 | |
| #endif /* HAVE_BUTTONLIGHT_BRIGHTNESS */
 | |
| 
 | |
| #else /* !defined(HAVE_BACKLIGHT) || !defined(BACKLIGHT_FULL_INIT)
 | |
|     -- no backlight, empty dummy functions */
 | |
| 
 | |
| #if defined(HAVE_BACKLIGHT) && !defined(BACKLIGHT_FULL_INIT)
 | |
| void backlight_init(void)
 | |
| {
 | |
|     (void)_backlight_init();
 | |
|     _backlight_on();
 | |
| }
 | |
| #endif
 | |
| 
 | |
| void backlight_on(void) {}
 | |
| void backlight_off(void) {}
 | |
| void buttonlight_on(void) {}
 | |
| void backlight_set_timeout(int value) {(void)value;}
 | |
| 
 | |
| bool is_backlight_on(bool ignore_always_off)
 | |
| {
 | |
|     (void)ignore_always_off;
 | |
|     return true;
 | |
| }
 | |
| #ifdef HAVE_REMOTE_LCD
 | |
| void remote_backlight_on(void) {}
 | |
| void remote_backlight_off(void) {}
 | |
| void remote_backlight_set_timeout(int value) {(void)value;}
 | |
| 
 | |
| bool is_remote_backlight_on(bool ignore_always_off)
 | |
| {
 | |
|     (void)ignore_always_off;
 | |
|     return true;
 | |
| }
 | |
| #endif /* HAVE_REMOTE_LCD */
 | |
| #ifdef HAVE_BACKLIGHT_BRIGHTNESS
 | |
| void backlight_set_brightness(int val) { (void)val; }
 | |
| #endif
 | |
| #ifdef HAVE_BUTTONLIGHT_BRIGHTNESS
 | |
| void buttonlight_set_brightness(int val) { (void)val; }
 | |
| #endif
 | |
| #endif /* defined(HAVE_BACKLIGHT) && defined(BACKLIGHT_FULL_INIT) */
 |