forked from len0rd/rockbox
		
	later. We still need to hunt down snippets used that are not. 1324 modified files... http://www.rockbox.org/mail/archive/rockbox-dev-archive-2008-06/0060.shtml git-svn-id: svn://svn.rockbox.org/rockbox/trunk@17847 a1c6a512-1295-4272-9138-f99709370657
		
			
				
	
	
		
			221 lines
		
	
	
	
		
			5.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			221 lines
		
	
	
	
		
			5.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /***************************************************************************
 | |
|  *             __________               __   ___.
 | |
|  *   Open      \______   \ ____   ____ |  | _\_ |__   _______  ___
 | |
|  *   Source     |       _//  _ \_/ ___\|  |/ /| __ \ /  _ \  \/  /
 | |
|  *   Jukebox    |    |   (  <_> )  \___|    < | \_\ (  <_> > <  <
 | |
|  *   Firmware   |____|_  /\____/ \___  >__|_ \|___  /\____/__/\_ \
 | |
|  *                     \/            \/     \/    \/            \/
 | |
|  * $Id$
 | |
|  *
 | |
|  * Copyright (C) 2002 by Linus Nielsen Feltzing, Uwe Freese, Laurent Baum, 
 | |
|  *                       Przemyslaw Holubowski
 | |
|  *
 | |
|  * 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 "i2c.h"
 | |
| #include "rtc.h"
 | |
| #include "kernel.h"
 | |
| #include "system.h"
 | |
| #include "i2c-pp.h"
 | |
| #include <stdbool.h>
 | |
| 
 | |
| /* RTC registers */
 | |
| #define RTC_CTRL1          0x00
 | |
| #define RTC_CTRL2          0x01
 | |
| #define RTC_ALARM_MINUTES  0x09
 | |
| #define RTC_ALARM_HOURS    0x0A
 | |
| #define RTC_ALARM_DAY      0x0B
 | |
| #define RTC_ALARM_WDAY     0x0C
 | |
| #define RTC_TIMER_CTRL     0x0E
 | |
| #define RTC_TIMER          0x0F
 | |
| 
 | |
| /* Control 2 register flags */
 | |
| #define RTC_TIE    0x01
 | |
| #define RTC_AIE    0x02
 | |
| #define RTC_TF     0x04
 | |
| #define RTC_AF     0x08
 | |
| #define RTC_TI_TP  0x10
 | |
| 
 | |
| /* Alarm registers flags */
 | |
| #define RTC_AE     0x80
 | |
| 
 | |
| /* Timer register flags */
 | |
| #define RTC_TE     0x80
 | |
| 
 | |
| bool rtc_lock_alarm_clear=true;
 | |
| 
 | |
| void rtc_init(void)
 | |
| {
 | |
|     unsigned char tmp;
 | |
|     int rv;
 | |
|     
 | |
|     /* initialize Control 1 register */
 | |
|     tmp = 0;
 | |
|     pp_i2c_send(0x51, RTC_CTRL1,tmp);
 | |
|     
 | |
|     /* read value of the Control 2 register - we'll need it to preserve alarm and timer interrupt assertion flags */
 | |
|     rv = i2c_readbytes(0x51,RTC_CTRL2,1,&tmp);
 | |
|     /* preserve alarm and timer interrupt flags */
 | |
|     tmp &= (RTC_TF | RTC_AF | RTC_TIE | RTC_AIE);
 | |
|     pp_i2c_send(0x51, RTC_CTRL2,tmp);
 | |
| }
 | |
| 
 | |
| int rtc_read_datetime(unsigned char* buf)
 | |
| {
 | |
|     unsigned char tmp;
 | |
|     int read;
 | |
|     
 | |
|     /*RTC_E8564's slave address is 0x51*/
 | |
|     read = i2c_readbytes(0x51,0x02,7,buf);
 | |
| 
 | |
|     /* swap wday and mday to be compatible with
 | |
|      * get_time() from firmware/common/timefuncs.c */
 | |
|     tmp=buf[3];
 | |
|     buf[3]=buf[4];
 | |
|     buf[4]=tmp;
 | |
|     
 | |
|     return read;
 | |
| }
 | |
| 
 | |
| int rtc_write_datetime(unsigned char* buf)
 | |
| {
 | |
|     int i;
 | |
|     unsigned char tmp;
 | |
|     
 | |
|     /* swap wday and mday to be compatible with
 | |
|      * set_time() in firmware/common/timefuncs.c */
 | |
|     tmp=buf[3];
 | |
|     buf[3]=buf[4];
 | |
|     buf[4]=tmp;
 | |
| 
 | |
|     for (i=0;i<7;i++){
 | |
|         pp_i2c_send(0x51, 0x02+i,buf[i]);
 | |
|     }
 | |
|     return 1;
 | |
| }
 | |
| 
 | |
| void rtc_set_alarm(int h, int m)
 | |
| {
 | |
|     unsigned char buf[4]={0};
 | |
|     int rv=0;
 | |
|     int i=0;
 | |
|     
 | |
|     /* clear alarm interrupt */
 | |
|     rv = i2c_readbytes(0x51,RTC_CTRL2,1,buf);
 | |
|     buf[0] &= RTC_AF;
 | |
|     pp_i2c_send(0x51, RTC_CTRL2,buf[0]);
 | |
|     
 | |
|     /* prepare new alarm */
 | |
|     if( m >= 0 )
 | |
|         buf[0] = (((m/10) << 4) | m%10);
 | |
|     else
 | |
|         /* ignore minutes comparison query */
 | |
|         buf[0] = RTC_AE;
 | |
|     
 | |
|     if( h >= 0 )
 | |
|         buf[1] = (((h/10) << 4) | h%10);
 | |
|     else
 | |
|         /* ignore hours comparison query */
 | |
|         buf[1] = RTC_AE;
 | |
|     
 | |
|     /* ignore day and wday */
 | |
|     buf[2] = RTC_AE;
 | |
|     buf[3] = RTC_AE;
 | |
|     
 | |
|     /* write new alarm */
 | |
|     for(;i<4;i++)
 | |
|         pp_i2c_send(0x51, RTC_ALARM_MINUTES+i,buf[i]);
 | |
|     
 | |
|     /* note: alarm is not enabled at the point */
 | |
| }
 | |
| 
 | |
| void rtc_get_alarm(int *h, int *m)
 | |
| {
 | |
|     unsigned char buf[4]={0};
 | |
|     
 | |
|     /* get alarm preset */
 | |
|     i2c_readbytes(0x51, RTC_ALARM_MINUTES,4,buf);
 | |
| 
 | |
|     *m = ((buf[0] >> 4) & 0x7)*10 + (buf[0] & 0x0f);
 | |
|     *h = ((buf[1] >> 4) & 0x3)*10 + (buf[1] & 0x0f);
 | |
| 
 | |
| }
 | |
| 
 | |
| bool rtc_enable_alarm(bool enable)
 | |
| {
 | |
|     unsigned char tmp=0;
 | |
|     int rv=0;
 | |
| 
 | |
|     if(enable)
 | |
|     {
 | |
|         /* enable alarm interrupt */
 | |
|         rv = i2c_readbytes(0x51,RTC_CTRL2,1,&tmp);
 | |
|         tmp |= RTC_AIE;
 | |
|         tmp &= ~RTC_AF;
 | |
|         pp_i2c_send(0x51, RTC_CTRL2,tmp);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         /* disable alarm interrupt */       
 | |
|         if(rtc_lock_alarm_clear)
 | |
|             /* lock disabling alarm before it was checked whether or not the unit was started by RTC alarm */
 | |
|             return false;        
 | |
|         rv = i2c_readbytes(0x51,RTC_CTRL2,1,&tmp);
 | |
|         tmp &= ~(RTC_AIE | RTC_AF);
 | |
|         pp_i2c_send(0x51, RTC_CTRL2,tmp);
 | |
|     }
 | |
|     
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| bool rtc_check_alarm_started(bool release_alarm)
 | |
| {
 | |
|     static bool alarm_state, run_before = false;
 | |
|     unsigned char tmp=0;
 | |
|     bool started;
 | |
|     int rv=0;
 | |
| 
 | |
|     if (run_before)
 | |
|     {
 | |
|         started = alarm_state;
 | |
|         alarm_state &= ~release_alarm;         
 | |
|     } 
 | |
|     else 
 | |
|     { 
 | |
|         /* read Control 2 register which contains alarm flag */
 | |
|         rv = i2c_readbytes(0x51,RTC_CTRL2,1,&tmp);
 | |
| 
 | |
|         alarm_state = started = ( (tmp & RTC_AF) && (tmp & RTC_AIE) );
 | |
|         
 | |
|         if(release_alarm && started)
 | |
|         {
 | |
|             rv = i2c_readbytes(0x51,RTC_CTRL2,1,&tmp);
 | |
|             /* clear alarm interrupt enable and alarm flag */
 | |
|             tmp &= ~(RTC_AF | RTC_AIE);
 | |
|             pp_i2c_send(0x51, RTC_CTRL2,tmp);
 | |
|         }
 | |
|         run_before = true;
 | |
|         rtc_lock_alarm_clear = false;
 | |
|     }
 | |
|     
 | |
|     return started;
 | |
| }
 | |
| 
 | |
| bool rtc_check_alarm_flag(void)
 | |
| {
 | |
|     unsigned char tmp=0;
 | |
|     int rv=0;
 | |
|     
 | |
|     /* read Control 2 register which contains alarm flag */
 | |
|     rv = i2c_readbytes(0x51,RTC_CTRL2,1,&tmp);
 | |
| 
 | |
|     return (tmp & RTC_AF);
 | |
| }
 |