Commit bbbd9995 authored by David P. Reed's avatar David P. Reed Committed by Thomas Gleixner

x86: on x86_64, correct reading of PC RTC when update in progress in time_64.c

Correct potentially unstable PC RTC time register reading in time_64.c

Stop the use of an incorrect technique for reading the standard PC RTC
timer, which is documented to "disconnect" time registers from the bus
while updates are in progress.  The use of UIP flag while interrupts
are disabled to protect a 244 microsecond window is one of the
Motorola spec sheet's documented ways to read the RTC time registers
reliably.

tglx: removed locking changes from original patch, as they gain nothing
(read_persistent_clock is only called during boot, suspend, resume - so
no hot path affected) and conflict with the paravirt locking scheme
(see 32bit code), which we do not want to complicate for no benefit.
Signed-off-by: default avatarDavid P. Reed <dpreed@reed.com>
Signed-off-by: default avatarThomas Gleixner <tglx@linutronix.de>
parent c399da0d
...@@ -161,21 +161,27 @@ unsigned long read_persistent_clock(void) ...@@ -161,21 +161,27 @@ unsigned long read_persistent_clock(void)
unsigned century = 0; unsigned century = 0;
spin_lock_irqsave(&rtc_lock, flags); spin_lock_irqsave(&rtc_lock, flags);
/*
* if UIP is clear, then we have >= 244 microseconds before RTC
* registers will be updated. Spec sheet says that this is the
* reliable way to read RTC - registers invalid (off bus) during update
*/
while ((CMOS_READ(RTC_FREQ_SELECT) & RTC_UIP))
cpu_relax();
do {
sec = CMOS_READ(RTC_SECONDS); /* now read all RTC registers while stable with interrupts disabled */
min = CMOS_READ(RTC_MINUTES); sec = CMOS_READ(RTC_SECONDS);
hour = CMOS_READ(RTC_HOURS); min = CMOS_READ(RTC_MINUTES);
day = CMOS_READ(RTC_DAY_OF_MONTH); hour = CMOS_READ(RTC_HOURS);
mon = CMOS_READ(RTC_MONTH); day = CMOS_READ(RTC_DAY_OF_MONTH);
year = CMOS_READ(RTC_YEAR); mon = CMOS_READ(RTC_MONTH);
year = CMOS_READ(RTC_YEAR);
#ifdef CONFIG_ACPI #ifdef CONFIG_ACPI
if (acpi_gbl_FADT.header.revision >= FADT2_REVISION_ID && if (acpi_gbl_FADT.header.revision >= FADT2_REVISION_ID &&
acpi_gbl_FADT.century) acpi_gbl_FADT.century)
century = CMOS_READ(acpi_gbl_FADT.century); century = CMOS_READ(acpi_gbl_FADT.century);
#endif #endif
} while (sec != CMOS_READ(RTC_SECONDS));
spin_unlock_irqrestore(&rtc_lock, flags); spin_unlock_irqrestore(&rtc_lock, flags);
/* /*
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment