blob: f7cf1064f9052de9b2e56fcc7de3d53544826efa [file] [log] [blame]
wdenkaffae2b2002-08-17 09:36:01 +00001/*
2 * (C) Copyright 2001
3 * Denis Peter MPL AG Switzerland. d.peter@mpl.ch
4 *
Wolfgang Denkd79de1d2013-07-08 09:37:19 +02005 * SPDX-License-Identifier: GPL-2.0+
wdenkaffae2b2002-08-17 09:36:01 +00006 */
7
8/*
9 * Date & Time support for the MC146818 (PIXX4) RTC
10 */
11
12/*#define DEBUG*/
13
14#include <common.h>
15#include <command.h>
16#include <rtc.h>
17
Paul Burtonc028f9b2013-11-08 11:18:55 +000018#if defined(__I386__) || defined(CONFIG_MALTA)
Graeme Russb87c30f2011-02-12 15:11:43 +110019#include <asm/io.h>
20#define in8(p) inb(p)
21#define out8(p, v) outb(v, p)
22#endif
23
Michal Simekc3e6c552008-07-14 19:45:37 +020024#if defined(CONFIG_CMD_DATE)
wdenkaffae2b2002-08-17 09:36:01 +000025
26static uchar rtc_read (uchar reg);
27static void rtc_write (uchar reg, uchar val);
wdenkaffae2b2002-08-17 09:36:01 +000028
Jean-Christophe PLAGNIOL-VILLARD03836942008-10-16 15:01:15 +020029#define RTC_PORT_MC146818 CONFIG_SYS_ISA_IO_BASE_ADDRESS + 0x70
Wolfgang Denka1be4762008-05-20 16:00:29 +020030#define RTC_SECONDS 0x00
31#define RTC_SECONDS_ALARM 0x01
32#define RTC_MINUTES 0x02
33#define RTC_MINUTES_ALARM 0x03
34#define RTC_HOURS 0x04
35#define RTC_HOURS_ALARM 0x05
36#define RTC_DAY_OF_WEEK 0x06
37#define RTC_DATE_OF_MONTH 0x07
38#define RTC_MONTH 0x08
39#define RTC_YEAR 0x09
40#define RTC_CONFIG_A 0x0A
41#define RTC_CONFIG_B 0x0B
42#define RTC_CONFIG_C 0x0C
43#define RTC_CONFIG_D 0x0D
wdenkaffae2b2002-08-17 09:36:01 +000044
45
46/* ------------------------------------------------------------------------- */
47
Yuri Tikhonov9bacd942008-03-20 17:56:04 +030048int rtc_get (struct rtc_time *tmp)
wdenkaffae2b2002-08-17 09:36:01 +000049{
50 uchar sec, min, hour, mday, wday, mon, year;
51 /* here check if rtc can be accessed */
52 while((rtc_read(RTC_CONFIG_A)&0x80)==0x80);
Wolfgang Denka1be4762008-05-20 16:00:29 +020053 sec = rtc_read (RTC_SECONDS);
54 min = rtc_read (RTC_MINUTES);
wdenkaffae2b2002-08-17 09:36:01 +000055 hour = rtc_read (RTC_HOURS);
56 mday = rtc_read (RTC_DATE_OF_MONTH);
57 wday = rtc_read (RTC_DAY_OF_WEEK);
Wolfgang Denka1be4762008-05-20 16:00:29 +020058 mon = rtc_read (RTC_MONTH);
wdenkaffae2b2002-08-17 09:36:01 +000059 year = rtc_read (RTC_YEAR);
60#ifdef RTC_DEBUG
61 printf ( "Get RTC year: %02x mon/cent: %02x mday: %02x wday: %02x "
62 "hr: %02x min: %02x sec: %02x\n",
wdenk452cfd62002-11-19 11:04:11 +000063 year, mon, mday, wday,
wdenkaffae2b2002-08-17 09:36:01 +000064 hour, min, sec );
65 printf ( "Alarms: month: %02x hour: %02x min: %02x sec: %02x\n",
66 rtc_read (RTC_CONFIG_D) & 0x3F,
67 rtc_read (RTC_HOURS_ALARM),
68 rtc_read (RTC_MINUTES_ALARM),
69 rtc_read (RTC_SECONDS_ALARM) );
70#endif
71 tmp->tm_sec = bcd2bin (sec & 0x7F);
72 tmp->tm_min = bcd2bin (min & 0x7F);
73 tmp->tm_hour = bcd2bin (hour & 0x3F);
74 tmp->tm_mday = bcd2bin (mday & 0x3F);
75 tmp->tm_mon = bcd2bin (mon & 0x1F);
76 tmp->tm_year = bcd2bin (year);
77 tmp->tm_wday = bcd2bin (wday & 0x07);
78 if(tmp->tm_year<70)
79 tmp->tm_year+=2000;
80 else
81 tmp->tm_year+=1900;
82 tmp->tm_yday = 0;
83 tmp->tm_isdst= 0;
84#ifdef RTC_DEBUG
85 printf ( "Get DATE: %4d-%02d-%02d (wday=%d) TIME: %2d:%02d:%02d\n",
86 tmp->tm_year, tmp->tm_mon, tmp->tm_mday, tmp->tm_wday,
87 tmp->tm_hour, tmp->tm_min, tmp->tm_sec);
88#endif
Yuri Tikhonov9bacd942008-03-20 17:56:04 +030089
90 return 0;
wdenkaffae2b2002-08-17 09:36:01 +000091}
92
Jean-Christophe PLAGNIOL-VILLARD97a2e102008-09-01 23:06:23 +020093int rtc_set (struct rtc_time *tmp)
wdenkaffae2b2002-08-17 09:36:01 +000094{
95#ifdef RTC_DEBUG
96 printf ( "Set DATE: %4d-%02d-%02d (wday=%d) TIME: %2d:%02d:%02d\n",
97 tmp->tm_year, tmp->tm_mon, tmp->tm_mday, tmp->tm_wday,
98 tmp->tm_hour, tmp->tm_min, tmp->tm_sec);
99#endif
100 rtc_write(RTC_CONFIG_B,0x82); /* disables the RTC to update the regs */
101
102 rtc_write (RTC_YEAR, bin2bcd(tmp->tm_year % 100));
103 rtc_write (RTC_MONTH, bin2bcd(tmp->tm_mon));
wdenkaffae2b2002-08-17 09:36:01 +0000104 rtc_write (RTC_DAY_OF_WEEK, bin2bcd(tmp->tm_wday));
wdenkaffae2b2002-08-17 09:36:01 +0000105 rtc_write (RTC_DATE_OF_MONTH, bin2bcd(tmp->tm_mday));
106 rtc_write (RTC_HOURS, bin2bcd(tmp->tm_hour));
107 rtc_write (RTC_MINUTES, bin2bcd(tmp->tm_min ));
108 rtc_write (RTC_SECONDS, bin2bcd(tmp->tm_sec ));
109 rtc_write(RTC_CONFIG_B,0x02); /* enables the RTC to update the regs */
110
Jean-Christophe PLAGNIOL-VILLARD97a2e102008-09-01 23:06:23 +0200111 return 0;
wdenkaffae2b2002-08-17 09:36:01 +0000112}
113
114void rtc_reset (void)
115{
116 rtc_write(RTC_CONFIG_B,0x82); /* disables the RTC to update the regs */
117 rtc_write(RTC_CONFIG_A,0x20); /* Normal OP */
118 rtc_write(RTC_CONFIG_B,0x00);
119 rtc_write(RTC_CONFIG_B,0x00);
120 rtc_write(RTC_CONFIG_B,0x02); /* enables the RTC to update the regs */
121}
122
123/* ------------------------------------------------------------------------- */
124
Jean-Christophe PLAGNIOL-VILLARD03836942008-10-16 15:01:15 +0200125#ifdef CONFIG_SYS_RTC_REG_BASE_ADDR
wdenkaffae2b2002-08-17 09:36:01 +0000126/*
127 * use direct memory access
128 */
129static uchar rtc_read (uchar reg)
130{
Jean-Christophe PLAGNIOL-VILLARD03836942008-10-16 15:01:15 +0200131 return(in8(CONFIG_SYS_RTC_REG_BASE_ADDR+reg));
wdenkaffae2b2002-08-17 09:36:01 +0000132}
133
134static void rtc_write (uchar reg, uchar val)
135{
Jean-Christophe PLAGNIOL-VILLARD03836942008-10-16 15:01:15 +0200136 out8(CONFIG_SYS_RTC_REG_BASE_ADDR+reg, val);
wdenkaffae2b2002-08-17 09:36:01 +0000137}
138#else
139static uchar rtc_read (uchar reg)
140{
141 out8(RTC_PORT_MC146818,reg);
142 return(in8(RTC_PORT_MC146818+1));
143}
144
145static void rtc_write (uchar reg, uchar val)
146{
147 out8(RTC_PORT_MC146818,reg);
148 out8(RTC_PORT_MC146818+1,val);
149}
150#endif
151
Jon Loeliger07efe2a2007-07-10 10:27:39 -0500152#endif