summaryrefslogtreecommitdiff
path: root/sys/arch/i386/isa/clock.c
diff options
context:
space:
mode:
authorTheo de Raadt <deraadt@cvs.openbsd.org>1995-10-18 08:53:40 +0000
committerTheo de Raadt <deraadt@cvs.openbsd.org>1995-10-18 08:53:40 +0000
commitd6583bb2a13f329cf0332ef2570eb8bb8fc0e39c (patch)
treeece253b876159b39c620e62b6c9b1174642e070e /sys/arch/i386/isa/clock.c
initial import of NetBSD tree
Diffstat (limited to 'sys/arch/i386/isa/clock.c')
-rw-r--r--sys/arch/i386/isa/clock.c519
1 files changed, 519 insertions, 0 deletions
diff --git a/sys/arch/i386/isa/clock.c b/sys/arch/i386/isa/clock.c
new file mode 100644
index 00000000000..ff2f768d41c
--- /dev/null
+++ b/sys/arch/i386/isa/clock.c
@@ -0,0 +1,519 @@
+/* $NetBSD: clock.c,v 1.34 1995/08/13 04:06:29 mycroft Exp $ */
+
+/*-
+ * Copyright (c) 1993, 1994 Charles Hannum.
+ * Copyright (c) 1990 The Regents of the University of California.
+ * All rights reserved.
+ *
+ * This code is derived from software contributed to Berkeley by
+ * William Jolitz and Don Ahn.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. All advertising materials mentioning features or use of this software
+ * must display the following acknowledgement:
+ * This product includes software developed by the University of
+ * California, Berkeley and its contributors.
+ * 4. Neither the name of the University nor the names of its contributors
+ * may be used to endorse or promote products derived from this software
+ * without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ *
+ * @(#)clock.c 7.2 (Berkeley) 5/12/91
+ */
+/*
+ * Mach Operating System
+ * Copyright (c) 1991,1990,1989 Carnegie Mellon University
+ * All Rights Reserved.
+ *
+ * Permission to use, copy, modify and distribute this software and its
+ * documentation is hereby granted, provided that both the copyright
+ * notice and this permission notice appear in all copies of the
+ * software, derivative works or modified versions, and any portions
+ * thereof, and that both notices appear in supporting documentation.
+ *
+ * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS"
+ * CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
+ * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
+ *
+ * Carnegie Mellon requests users of this software to return to
+ *
+ * Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU
+ * School of Computer Science
+ * Carnegie Mellon University
+ * Pittsburgh PA 15213-3890
+ *
+ * any improvements or extensions that they make and grant Carnegie Mellon
+ * the rights to redistribute these changes.
+ */
+/*
+ Copyright 1988, 1989 by Intel Corporation, Santa Clara, California.
+
+ All Rights Reserved
+
+Permission to use, copy, modify, and distribute this software and
+its documentation for any purpose and without fee is hereby
+granted, provided that the above copyright notice appears in all
+copies and that both the copyright notice and this permission notice
+appear in supporting documentation, and that the name of Intel
+not be used in advertising or publicity pertaining to distribution
+of the software without specific, written prior permission.
+
+INTEL DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE
+INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS,
+IN NO EVENT SHALL INTEL BE LIABLE FOR ANY SPECIAL, INDIRECT, OR
+CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
+LOSS OF USE, DATA OR PROFITS, WHETHER IN ACTION OF CONTRACT,
+NEGLIGENCE, OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION
+WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+*/
+
+/*
+ * Primitive clock interrupt routines.
+ */
+#include <sys/param.h>
+#include <sys/systm.h>
+#include <sys/time.h>
+#include <sys/kernel.h>
+#include <sys/device.h>
+
+#include <machine/cpu.h>
+#include <machine/pio.h>
+#include <machine/cpufunc.h>
+
+#include <dev/isa/isareg.h>
+#include <dev/isa/isavar.h>
+#include <dev/ic/mc146818reg.h>
+#include <i386/isa/nvram.h>
+#include <i386/isa/timerreg.h>
+#include <i386/isa/spkrreg.h>
+
+void spinwait __P((int));
+
+#define SECMIN ((unsigned)60) /* seconds per minute */
+#define SECHOUR ((unsigned)(60*SECMIN)) /* seconds per hour */
+#define SECDAY ((unsigned)(24*SECHOUR)) /* seconds per day */
+#define SECYR ((unsigned)(365*SECDAY)) /* seconds per common year */
+
+__inline u_int
+mc146818_read(sc, reg)
+ void *sc; /* XXX use it? */
+ u_int reg;
+{
+
+ outb(IO_RTC, reg);
+ return (inb(IO_RTC+1));
+}
+
+__inline void
+mc146818_write(sc, reg, datum)
+ void *sc; /* XXX use it? */
+ u_int reg, datum;
+{
+
+ outb(IO_RTC, reg);
+ outb(IO_RTC+1, datum);
+}
+
+void
+startrtclock()
+{
+ int s;
+
+ findcpuspeed(); /* use the clock (while it's free)
+ to find the cpu speed */
+ /* initialize 8253 clock */
+ outb(TIMER_MODE, TIMER_SEL0|TIMER_RATEGEN|TIMER_16BIT);
+
+ /* Correct rounding will buy us a better precision in timekeeping */
+ outb(IO_TIMER1, TIMER_DIV(hz) % 256);
+ outb(IO_TIMER1, TIMER_DIV(hz) / 256);
+
+ /* Check diagnostic status */
+ if (s = mc146818_read(NULL, NVRAM_DIAG)) /* XXX softc */
+ printf("RTC BIOS diagnostic error %b\n", s, NVRAM_DIAG_BITS);
+}
+
+int
+clockintr(arg)
+ void *arg;
+{
+ struct clockframe *frame = arg; /* not strictly necessary */
+
+ hardclock(frame);
+ return -1;
+}
+
+int
+gettick()
+{
+ u_char lo, hi;
+
+ /* Don't want someone screwing with the counter while we're here. */
+ disable_intr();
+ /* Select counter 0 and latch it. */
+ outb(TIMER_MODE, TIMER_SEL0 | TIMER_LATCH);
+ lo = inb(TIMER_CNTR0);
+ hi = inb(TIMER_CNTR0);
+ enable_intr();
+ return ((hi << 8) | lo);
+}
+
+/*
+ * Wait "n" microseconds.
+ * Relies on timer 1 counting down from (TIMER_FREQ / hz) at TIMER_FREQ Hz.
+ * Note: timer had better have been programmed before this is first used!
+ * (Note that we use `rate generator' mode, which counts at 1:1; `square
+ * wave' mode counts at 2:1).
+ */
+void
+delay(n)
+ int n;
+{
+ int limit, tick, otick;
+
+ /*
+ * Read the counter first, so that the rest of the setup overhead is
+ * counted.
+ */
+ otick = gettick();
+
+#ifdef __GNUC__
+ /*
+ * Calculate ((n * TIMER_FREQ) / 1e6) using explicit assembler code so
+ * we can take advantage of the intermediate 64-bit quantity to prevent
+ * loss of significance.
+ */
+ n -= 5;
+ if (n < 0)
+ return;
+ {register int m;
+ __asm __volatile("mul %3"
+ : "=a" (n), "=d" (m)
+ : "0" (n), "r" (TIMER_FREQ));
+ __asm __volatile("div %3"
+ : "=a" (n)
+ : "0" (n), "d" (m), "r" (1000000)
+ : "%edx");}
+#else
+ /*
+ * Calculate ((n * TIMER_FREQ) / 1e6) without using floating point and
+ * without any avoidable overflows.
+ */
+ n -= 20;
+ {
+ int sec = n / 1000000,
+ usec = n % 1000000;
+ n = sec * TIMER_FREQ +
+ usec * (TIMER_FREQ / 1000000) +
+ usec * ((TIMER_FREQ % 1000000) / 1000) / 1000 +
+ usec * (TIMER_FREQ % 1000) / 1000000;
+ }
+#endif
+
+ limit = TIMER_FREQ / hz;
+
+ while (n > 0) {
+ tick = gettick();
+ if (tick > otick)
+ n -= limit - (tick - otick);
+ else
+ n -= otick - tick;
+ otick = tick;
+ }
+}
+
+static int beeping;
+
+void
+sysbeepstop(arg)
+ void *arg;
+{
+
+ /* disable counter 2 */
+ disable_intr();
+ outb(PITAUX_PORT, inb(PITAUX_PORT) & ~PIT_SPKR);
+ enable_intr();
+ beeping = 0;
+}
+
+void
+sysbeep(pitch, period)
+ int pitch, period;
+{
+ static int last_pitch;
+
+ if (beeping)
+ untimeout(sysbeepstop, 0);
+ if (pitch == 0 || period == 0) {
+ sysbeepstop(0);
+ last_pitch = 0;
+ return;
+ }
+ if (!beeping || last_pitch != pitch) {
+ disable_intr();
+ outb(TIMER_MODE, TIMER_SEL2 | TIMER_16BIT | TIMER_SQWAVE);
+ outb(TIMER_CNTR2, TIMER_DIV(pitch) % 256);
+ outb(TIMER_CNTR2, TIMER_DIV(pitch) / 256);
+ outb(PITAUX_PORT, inb(PITAUX_PORT) | PIT_SPKR); /* enable counter 2 */
+ enable_intr();
+ }
+ last_pitch = pitch;
+ beeping = 1;
+ timeout(sysbeepstop, 0, period);
+}
+
+unsigned int delaycount; /* calibrated loop variable (1 millisecond) */
+
+#define FIRST_GUESS 0x2000
+findcpuspeed()
+{
+ int i;
+ int remainder;
+
+ /* Put counter in count down mode */
+ outb(TIMER_MODE, TIMER_SEL0 | TIMER_16BIT | TIMER_RATEGEN);
+ outb(TIMER_CNTR0, 0xff);
+ outb(TIMER_CNTR0, 0xff);
+ for (i = FIRST_GUESS; i; i--)
+ ;
+ /* Read the value left in the counter */
+ remainder = gettick();
+ /*
+ * Formula for delaycount is:
+ * (loopcount * timer clock speed) / (counter ticks * 1000)
+ */
+ delaycount = (FIRST_GUESS * TIMER_DIV(1000)) / (0xffff-remainder);
+}
+
+void
+cpu_initclocks()
+{
+
+ /*
+ * XXX If you're doing strange things with multiple clocks, you might
+ * want to keep track of clock handlers.
+ */
+ (void)isa_intr_establish(0, ISA_IST_PULSE, ISA_IPL_CLOCK,
+ clockintr, 0);
+}
+
+void
+rtcinit()
+{
+ static int first_rtcopen_ever = 1;
+
+ if (!first_rtcopen_ever)
+ return;
+ first_rtcopen_ever = 0;
+
+ mc146818_write(NULL, MC_REGA, /* XXX softc */
+ MC_BASE_32_KHz | MC_RATE_1024_Hz);
+ mc146818_write(NULL, MC_REGB, MC_REGB_24HR); /* XXX softc */
+}
+
+int
+rtcget(regs)
+ mc_todregs *regs;
+{
+
+ rtcinit();
+ if (mc146818_read(NULL, MC_REGD) & MC_REGD_VRT == 0) /* XXX softc */
+ return (-1);
+ MC146818_GETTOD(NULL, regs); /* XXX softc */
+ return (0);
+}
+
+void
+rtcput(regs)
+ mc_todregs *regs;
+{
+
+ rtcinit();
+ MC146818_PUTTOD(NULL, regs); /* XXX softc */
+}
+
+static int month[12] = {31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};
+
+static int
+yeartoday(year)
+ int year;
+{
+
+ return ((year % 4) ? 365 : 366);
+}
+
+int
+hexdectodec(n)
+ char n;
+{
+
+ return (((n >> 4) & 0x0f) * 10 + (n & 0x0f));
+}
+
+char
+dectohexdec(n)
+ int n;
+{
+
+ return ((char)(((n / 10) << 4) & 0xf0) | ((n % 10) & 0x0f));
+}
+
+static int timeset;
+
+/*
+ * Initialize the time of day register, based on the time base which is, e.g.
+ * from a filesystem.
+ */
+void
+inittodr(base)
+ time_t base;
+{
+ mc_todregs rtclk;
+ time_t n;
+ int sec, min, hr, dom, mon, yr;
+ int i, days = 0;
+ int s;
+
+ /*
+ * We mostly ignore the suggested time and go for the RTC clock time
+ * stored in the CMOS RAM. If the time can't be obtained from the
+ * CMOS, or if the time obtained from the CMOS is 5 or more years
+ * less than the suggested time, we used the suggested time. (In
+ * the latter case, it's likely that the CMOS battery has died.)
+ */
+
+ if (base < 15*SECYR) { /* if before 1985, something's odd... */
+ printf("WARNING: preposterous time in file system\n");
+ /* read the system clock anyway */
+ base = 17*SECYR + 186*SECDAY + SECDAY/2;
+ }
+
+ s = splclock();
+ if (rtcget(&rtclk)) {
+ splx(s);
+ printf("WARNING: invalid time in clock chip\n");
+ goto fstime;
+ }
+ splx(s);
+
+ sec = hexdectodec(rtclk[MC_SEC]);
+ min = hexdectodec(rtclk[MC_MIN]);
+ hr = hexdectodec(rtclk[MC_HOUR]);
+ dom = hexdectodec(rtclk[MC_DOM]);
+ mon = hexdectodec(rtclk[MC_MONTH]);
+ yr = hexdectodec(rtclk[MC_YEAR]);
+ yr = (yr < 70) ? yr+100 : yr;
+
+ n = sec + 60 * min + 3600 * hr;
+ n += (dom - 1) * 3600 * 24;
+
+ if (yeartoday(yr) == 366)
+ month[1] = 29;
+ for (i = mon - 2; i >= 0; i--)
+ days += month[i];
+ month[1] = 28;
+ for (i = 70; i < yr; i++)
+ days += yeartoday(i);
+ n += days * 3600 * 24;
+
+ n += tz.tz_minuteswest * 60;
+ if (tz.tz_dsttime)
+ n -= 3600;
+
+ if (base < n - 5*SECYR)
+ printf("WARNING: file system time much less than clock time\n");
+ else if (base > n + 5*SECYR) {
+ printf("WARNING: clock time much less than file system time\n");
+ printf("WARNING: using file system time\n");
+ goto fstime;
+ }
+
+ timeset = 1;
+ time.tv_sec = n;
+ time.tv_usec = 0;
+ return;
+
+fstime:
+ timeset = 1;
+ time.tv_sec = base;
+ time.tv_usec = 0;
+ printf("WARNING: CHECK AND RESET THE DATE!\n");
+}
+
+/*
+ * Reset the clock.
+ */
+void
+resettodr()
+{
+ mc_todregs rtclk;
+ time_t n;
+ int diff, i, j;
+ int s;
+
+ /*
+ * We might have been called by boot() due to a crash early
+ * on. Don't reset the clock chip in this case.
+ */
+ if (!timeset)
+ return;
+
+ s = splclock();
+ if (rtcget(&rtclk))
+ bzero(&rtclk, sizeof(rtclk));
+ splx(s);
+
+ diff = tz.tz_minuteswest * 60;
+ if (tz.tz_dsttime)
+ diff -= 3600;
+ n = (time.tv_sec - diff) % (3600 * 24); /* hrs+mins+secs */
+ rtclk[MC_SEC] = dectohexdec(n % 60);
+ n /= 60;
+ rtclk[MC_MIN] = dectohexdec(n % 60);
+ rtclk[MC_HOUR] = dectohexdec(n / 60);
+
+ n = (time.tv_sec - diff) / (3600 * 24); /* days */
+ rtclk[MC_DOW] = (n + 4) % 7; /* 1/1/70 is Thursday */
+
+ for (j = 1970, i = yeartoday(j); n >= i; j++, i = yeartoday(j))
+ n -= i;
+
+ rtclk[MC_YEAR] = dectohexdec(j - 1900);
+
+ if (i == 366)
+ month[1] = 29;
+ for (i = 0; n >= month[i]; i++)
+ n -= month[i];
+ month[1] = 28;
+ rtclk[MC_MONTH] = dectohexdec(++i);
+
+ rtclk[MC_DOM] = dectohexdec(++n);
+
+ s = splclock();
+ rtcput(&rtclk);
+ splx(s);
+}
+
+void
+setstatclockrate(arg)
+ int arg;
+{
+}