Annotation of prex/dev/i386/pc/rtc.c, Revision 1.1.1.1
1.1 nbrk 1: /*-
2: * Copyright (c) 2005, Kohsuke Ohtani
3: * All rights reserved.
4: *
5: * Redistribution and use in source and binary forms, with or without
6: * modification, are permitted provided that the following conditions
7: * are met:
8: * 1. Redistributions of source code must retain the above copyright
9: * notice, this list of conditions and the following disclaimer.
10: * 2. Redistributions in binary form must reproduce the above copyright
11: * notice, this list of conditions and the following disclaimer in the
12: * documentation and/or other materials provided with the distribution.
13: * 3. Neither the name of the author nor the names of any co-contributors
14: * may be used to endorse or promote products derived from this software
15: * without specific prior written permission.
16: *
17: * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
18: * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19: * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20: * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
21: * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22: * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23: * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24: * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25: * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26: * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27: * SUCH DAMAGE.
28: */
29:
30: /*
31: * rtc.c - Real time clock driver
32: */
33:
34: #include <sys/time.h>
35: #include <sys/ioctl.h>
36:
37: #include <driver.h>
38: #include <cpufunc.h>
39:
40: /* Cmos */
41: #define CMOS_INDEX 0x70
42: #define CMOS_DATA 0x71
43:
44: /* CMOS address */
45: #define CMOS_SEC 0x00
46: #define CMOS_MIN 0x02
47: #define CMOS_HOUR 0x04
48: #define CMOS_DAY 0x07
49: #define CMOS_MON 0x08
50: #define CMOS_YEAR 0x09
51: #define CMOS_STS_A 0x0a
52: #define CMOS_UIP 0x80
53: #define CMOS_STS_B 0x0b
54: #define CMOS_BCD 0x04
55:
56:
57: #define DAYSPERYEAR (31+28+31+30+31+30+31+31+30+31+30+31)
58:
59: static int rtc_read(device_t dev, char *buf, size_t *nbyte, int blkno);
60: static int rtc_ioctl(device_t dev, u_long cmd, void *arg);
61: static int rtc_init(void);
62:
63: /*
64: * Driver structure
65: */
66: struct driver rtc_drv = {
67: /* name */ "Realtime Clock",
68: /* order */ 4,
69: /* init */ rtc_init,
70: };
71:
72: /*
73: * Device I/O table
74: */
75: static struct devio rtc_io = {
76: /* open */ NULL,
77: /* close */ NULL,
78: /* read */ rtc_read,
79: /* write */ NULL,
80: /* ioctl */ rtc_ioctl,
81: /* event */ NULL,
82: };
83:
84: static const int daysinmonth[] =
85: { 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };
86:
87: static device_t rtc_dev; /* Device object */
88: static u_long boot_sec; /* Time (sec) at system boot */
89: static u_long boot_ticks; /* Time (ticks) at system boot */
90:
91:
92: static u_int
93: cmos_read(int index)
94: {
95: u_int value;
96:
97: irq_lock();
98: outb(index, CMOS_INDEX);
99: value = inb(CMOS_DATA);
100: irq_unlock();
101: return value;
102: }
103:
104: /*
105: static void
106: cmos_write(int index, int value)
107: {
108: irq_lock();
109: outb(index, CMOS_INDEX);
110: outb(value, CMOS_DATA);
111: irq_unlock();
112: }
113: */
114:
115: static u_int
116: bcd2bin(u_int bcd)
117: {
118:
119: return (bcd & 0x0f) + ((bcd >> 4) & 0xf) * 10;
120: }
121:
122: /*
123: static int bin2bcd(int bin)
124: {
125: return ((bin / 10) << 4) + (bin % 10);
126: }
127: */
128:
129: static int
130: is_leap(u_int year)
131: {
132:
133: if ((year % 4 == 0) && (year % 100 != 0))
134: return 1;
135: if (year % 400 == 0)
136: return 1;
137: return 0;
138: }
139:
140: /*
141: * Return current seconds (seconds since Epoch 1970/1/1 0:0:0)
142: */
143: static u_long
144: cmos_gettime(void)
145: {
146: u_int sec, min, hour, day, mon, year;
147: u_int i;
148: u_int days;
149:
150: /* Wait until data ready */
151: for (i = 0; i < 1000000; i++)
152: if (!(cmos_read(CMOS_STS_A) & CMOS_UIP))
153: break;
154:
155: sec = cmos_read(CMOS_SEC);
156: min = cmos_read(CMOS_MIN);
157: hour = cmos_read(CMOS_HOUR);
158: day = cmos_read(CMOS_DAY);
159: mon = cmos_read(CMOS_MON);
160: year = cmos_read(CMOS_YEAR);
161:
162: if (!(cmos_read(CMOS_STS_B) & CMOS_BCD)) {
163: sec = bcd2bin(sec);
164: min = bcd2bin(min);
165: hour = bcd2bin(hour);
166: day = bcd2bin(day);
167: mon = bcd2bin(mon);
168: year = bcd2bin(year);
169: }
170: if (year < 80)
171: year += 2000;
172: else
173: year += 1900;
174: #ifdef DEBUG
175: printf("rtc: system time was %d/%d/%d %d:%d:%d\n",
176: year, mon, day, hour, min, sec);
177: #endif
178:
179: days = 0;
180: for (i = 1970; i < year; i++)
181: days += DAYSPERYEAR + is_leap(i);
182: for (i = 1; i < mon; i++)
183: days += daysinmonth[i - 1];
184: if ((mon > 2) && is_leap(year))
185: days++;
186: days += day - 1;
187:
188: sec = (((days * 24 + hour) * 60) + min) * 60 + sec;
189: return sec;
190: }
191:
192: static int
193: rtc_read(device_t dev, char *buf, size_t *nbyte, int blkno)
194: {
195: u_long time;
196:
197: if (*nbyte < sizeof(u_long))
198: return 0;
199:
200: time = cmos_gettime();
201: if (umem_copyout(&time, buf, sizeof(u_long)))
202: return EFAULT;
203: *nbyte = sizeof(u_long);
204: return 0;
205: }
206:
207: static int
208: rtc_ioctl(device_t dev, u_long cmd, void *arg)
209: {
210: struct timeval tv;
211: int err = 0;
212: u_long msec;
213:
214: switch (cmd) {
215: case RTCIOC_GET_TIME:
216: /*
217: * Calculate current time (sec/usec) from
218: * boot time and current tick count.
219: */
220: msec = tick_to_msec(timer_count() - boot_ticks);
221: tv.tv_sec = (long)(boot_sec + (msec / 1000));
222: tv.tv_usec = (long)((msec * 1000) % 1000000);
223: if (umem_copyout(&tv, arg, sizeof(tv)))
224: return EFAULT;
225: break;
226: case RTCIOC_SET_TIME:
227: err = EINVAL;
228: break;
229: default:
230: return EINVAL;
231: }
232: return err;
233: }
234:
235: /*
236: * Initialize
237: */
238: static int
239: rtc_init(void)
240: {
241:
242: /* Create device object */
243: rtc_dev = device_create(&rtc_io, "rtc", DF_CHR);
244: ASSERT(rtc_dev);
245: boot_sec = cmos_gettime();
246: boot_ticks = timer_count();
247: return 0;
248: }
CVSweb