summaryrefslogtreecommitdiffstats
path: root/sys/i386/isa/clock.c
diff options
context:
space:
mode:
authorbde <bde@FreeBSD.org>1999-12-25 15:30:31 +0000
committerbde <bde@FreeBSD.org>1999-12-25 15:30:31 +0000
commitb68c18d69fcaf2204eff83522d53c17853d42783 (patch)
treec50a249e175a05c275ef1af5a16272686e50f6b8 /sys/i386/isa/clock.c
parentafa829033b35c800bacce4c1ba1311d236578bd7 (diff)
downloadFreeBSD-src-b68c18d69fcaf2204eff83522d53c17853d42783.zip
FreeBSD-src-b68c18d69fcaf2204eff83522d53c17853d42783.tar.gz
Fixed races accessing the RTC. The races apparently caused
apm_default_resume() to sometimes set a very wrong time. (1) Accesses to the RTC index and data registers were not atomic enough. Interrupts were not masked. This was only good enough until an interrupt handler (rtcintr()) started accessing the RTC in FreeBSD-2.0. (2) Access to the block of time registers in inittodr() was not atomic enough. inittodr() has 244us to read the time registers. Interrupts were not masked. This was only good enough until something (apm) started calling inittodr() after boot time in FreeBSD-2.0. The fix for (2) also makes the timecounter update more atomic, although this is currently unimportant due to the low resolution of the RTC. Problem reported by: mckay
Diffstat (limited to 'sys/i386/isa/clock.c')
-rw-r--r--sys/i386/isa/clock.c20
1 files changed, 16 insertions, 4 deletions
diff --git a/sys/i386/isa/clock.c b/sys/i386/isa/clock.c
index e421fb1..11b1426 100644
--- a/sys/i386/isa/clock.c
+++ b/sys/i386/isa/clock.c
@@ -550,23 +550,30 @@ int
rtcin(reg)
int reg;
{
+ int s;
u_char val;
+ s = splhigh();
outb(IO_RTC, reg);
inb(0x84);
val = inb(IO_RTC + 1);
inb(0x84);
+ splx(s);
return (val);
}
static __inline void
writertc(u_char reg, u_char val)
{
+ int s;
+
+ s = splhigh();
inb(0x84);
outb(IO_RTC, reg);
inb(0x84);
outb(IO_RTC + 1, val);
inb(0x84); /* XXX work around wrong order in rtcin() */
+ splx(s);
}
static __inline int
@@ -847,7 +854,11 @@ inittodr(time_t base)
/* wait for time update to complete */
/* If RTCSA_TUP is zero, we have at least 244us before next update */
- while (rtcin(RTC_STATUSA) & RTCSA_TUP);
+ s = splhigh();
+ while (rtcin(RTC_STATUSA) & RTCSA_TUP) {
+ splx(s);
+ s = splhigh();
+ }
days = 0;
#ifdef USE_RTC_CENTURY
@@ -857,8 +868,10 @@ inittodr(time_t base)
if (year < 1970)
year += 100;
#endif
- if (year < 1970)
+ if (year < 1970) {
+ splx(s);
goto wrong_time;
+ }
month = readrtc(RTC_MONTH);
for (m = 1; m < month; m++)
days += daysinmonth[m-1];
@@ -880,12 +893,11 @@ inittodr(time_t base)
y = time_second - sec;
if (y <= -2 || y >= 2) {
/* badly off, adjust it */
- s = splclock();
ts.tv_sec = sec;
ts.tv_nsec = 0;
set_timecounter(&ts);
- splx(s);
}
+ splx(s);
return;
wrong_time:
OpenPOWER on IntegriCloud