summaryrefslogtreecommitdiffstats
path: root/sys/isa
diff options
context:
space:
mode:
authorwollman <wollman@FreeBSD.org>1994-08-15 03:15:20 +0000
committerwollman <wollman@FreeBSD.org>1994-08-15 03:15:20 +0000
commitd1747d99343973bfb8d64254147b2e1b0ac5ab63 (patch)
treed219bc60ff8562e306a361c7b99cc4c08812feb6 /sys/isa
parentbe8746b39613d0e068e09206d4660427a1735576 (diff)
downloadFreeBSD-src-d1747d99343973bfb8d64254147b2e1b0ac5ab63.zip
FreeBSD-src-d1747d99343973bfb8d64254147b2e1b0ac5ab63.tar.gz
Enable use of the RTC chip for the statistical clock. While this does
not provide the full accuracy of a randomized statistical clock, it does provide greater accuracy than the previous method, while not significantly increasing overhead. It also provides profiling support at 1024 Hz. You must re-compile config before making a new kernel, or you will end up with unresolved symbols. Reviewed uy: Bruce evans said it worked for him.
Diffstat (limited to 'sys/isa')
-rw-r--r--sys/isa/atrtc.c60
-rw-r--r--sys/isa/rtc.h20
2 files changed, 75 insertions, 5 deletions
diff --git a/sys/isa/atrtc.c b/sys/isa/atrtc.c
index cf0a7f6..35f2e42 100644
--- a/sys/isa/atrtc.c
+++ b/sys/isa/atrtc.c
@@ -34,7 +34,7 @@
* SUCH DAMAGE.
*
* from: @(#)clock.c 7.2 (Berkeley) 5/12/91
- * $Id: clock.c,v 1.12 1994/08/11 00:28:24 wollman Exp $
+ * $Id: clock.c,v 1.13 1994/08/13 03:49:56 wollman Exp $
*/
/*
@@ -60,6 +60,7 @@
#define TIMER_DIV(x) ((TIMER_FREQ+(x)/2)/(x))
void hardclock();
+void statclock();
static int beeping;
int timer0_divisor = TIMER_DIV(100); /* XXX should be hz */
u_int timer0_prescale;
@@ -102,6 +103,44 @@ clkintr(frame)
hardclock(&frame);
}
+static u_char rtc_statusa = RTCSA_DIVIDER | RTCSA_NOPROF;
+
+/*
+ * This routine receives statistical clock interrupts from the RTC.
+ * As explained above, these occur at 128 interrupts per second.
+ * When profiling, we receive interrupts at a rate of 1024 Hz.
+ *
+ * This does not actually add as much overhead as it sounds, because
+ * when the statistical clock is active, the hardclock driver no longer
+ * needs to keep (inaccurate) statistics on its own. This decouples
+ * statistics gathering from scheduling interrupts.
+ *
+ * The RTC chip requires that we read status register C (RTC_INTR)
+ * to acknowledge an interrupt, before it will generate the next one.
+ */
+void
+rtcintr(struct clockframe frame)
+{
+ u_char stat;
+ stat = rtcin(RTC_INTR);
+ if(stat & RTCIR_PERIOD) {
+ statclock(&frame);
+ }
+}
+
+#ifdef DEBUG
+void
+printrtc(void)
+{
+ outb(IO_RTC, RTC_STATUSA);
+ printf("RTC status A = %x", inb(IO_RTC+1));
+ outb(IO_RTC, RTC_STATUSB);
+ printf(", B = %x", inb(IO_RTC+1));
+ outb(IO_RTC, RTC_INTR);
+ printf(", C = %x\n", inb(IO_RTC+1));
+}
+#endif
+
#if 0
void
timerintr(struct clockframe frame)
@@ -338,9 +377,9 @@ startrtclock()
/* initialize brain-dead battery powered clock */
outb (IO_RTC, RTC_STATUSA);
- outb (IO_RTC+1, 0x26);
+ outb (IO_RTC+1, rtc_statusa);
outb (IO_RTC, RTC_STATUSB);
- outb (IO_RTC+1, 2);
+ outb (IO_RTC+1, RTCSB_24HR);
outb (IO_RTC, RTC_DIAG);
if (s = inb (IO_RTC+1))
@@ -460,13 +499,17 @@ test_inittodr(time_t base)
*/
#define V(s) __CONCAT(V, s)
extern void V(clk)();
-
+extern void V(rtc)();
void
enablertclock()
{
setidt(ICU_OFFSET+0, &V(clk), SDT_SYS386IGT, SEL_KPL);
INTREN(IRQ0);
+ setidt(ICU_OFFSET+8, &V(rtc), SDT_SYS386IGT, SEL_KPL);
+ INTREN(IRQ8);
+ outb(IO_RTC, RTC_STATUSB);
+ outb(IO_RTC+1, RTCSB_PINTR | RTCSB_24HR);
}
@@ -482,10 +525,19 @@ spinwait(int millisecs)
void
cpu_initclocks()
{
+ stathz = RTC_NOPROFRATE;
+ profhz = RTC_PROFRATE;
enablertclock();
}
void
setstatclockrate(int newhz)
{
+ if(newhz == RTC_PROFRATE) {
+ rtc_statusa = RTCSA_DIVIDER | RTCSA_PROF;
+ } else {
+ rtc_statusa = RTCSA_DIVIDER | RTCSA_NOPROF;
+ }
+ outb(IO_RTC, RTC_STATUSA);
+ outb(IO_RTC+1, rtc_statusa);
}
diff --git a/sys/isa/rtc.h b/sys/isa/rtc.h
index ba008b6..9c0f501 100644
--- a/sys/isa/rtc.h
+++ b/sys/isa/rtc.h
@@ -34,7 +34,7 @@
* SUCH DAMAGE.
*
* from: @(#)rtc.h 7.1 (Berkeley) 5/12/91
- * $Id: rtc.h,v 1.3 1993/11/07 17:44:34 wollman Exp $
+ * $Id: rtc.h,v 1.4 1993/12/18 01:12:47 ache Exp $
*/
#ifndef _I386_ISA_RTC_H_
@@ -54,10 +54,28 @@
#define RTC_DAY 0x07 /* day of month */
#define RTC_MONTH 0x08 /* month of year */
#define RTC_YEAR 0x09 /* month of year */
+
#define RTC_STATUSA 0x0a /* status register A */
#define RTCSA_TUP 0x80 /* time update, don't look now */
+#define RTCSA_DIVIDER 0x20 /* divider correct for 32768 Hz */
+#define RTCSA_8192 0x03
+#define RTCSA_4096 0x04
+#define RTCSA_2048 0x05
+#define RTCSA_1024 0x06 /* default for profiling */
+#define RTCSA_PROF RTCSA_1024
+#define RTC_PROFRATE 1024
+#define RTCSA_512 0x07
+#define RTCSA_256 0x08
+#define RTCSA_128 0x09
+#define RTCSA_NOPROF RTCSA_128
+#define RTC_NOPROFRATE 128
+#define RTCSA_64 0x0a
+#define RTCSA_32 0x0b
#define RTC_STATUSB 0x0b /* status register B */
+#define RTCSB_HALT 0x80 /* stop clock updates */
+#define RTCSB_PINTR 0x40 /* periodic clock interrupt */
+#define RTCSB_24HR 0x02 /* 24-hour mode */
#define RTC_INTR 0x0c /* status register C (R) interrupt source */
#define RTCIR_UPDATE 0x10 /* update intr */
OpenPOWER on IntegriCloud