diff options
author | bde <bde@FreeBSD.org> | 1999-12-25 15:30:31 +0000 |
---|---|---|
committer | bde <bde@FreeBSD.org> | 1999-12-25 15:30:31 +0000 |
commit | b68c18d69fcaf2204eff83522d53c17853d42783 (patch) | |
tree | c50a249e175a05c275ef1af5a16272686e50f6b8 /sys/isa/atrtc.c | |
parent | afa829033b35c800bacce4c1ba1311d236578bd7 (diff) | |
download | FreeBSD-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/isa/atrtc.c')
-rw-r--r-- | sys/isa/atrtc.c | 20 |
1 files changed, 16 insertions, 4 deletions
diff --git a/sys/isa/atrtc.c b/sys/isa/atrtc.c index e421fb1..11b1426 100644 --- a/sys/isa/atrtc.c +++ b/sys/isa/atrtc.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: |