diff options
author | David Brownell <david-b@pacbell.net> | 2007-05-08 03:34:07 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@woody.linux-foundation.org> | 2007-05-08 14:15:19 -0400 |
commit | f8245c26886c912627ebc49f714e4491261224c4 (patch) | |
tree | 132a34d1e482f96a5d4bba5f744658aea9a5fcc9 /drivers/rtc/rtc-omap.c | |
parent | 416ce32e704d778c283f2f86cadd836cd5d3696c (diff) |
rtc: remove "RTC_ALM_SET mode" bugs
This fixes a common glitch in how RTC drivers handle two "set alarm" modes,
by getting rid of the surprising/hidden one that was rarely implemented
correctly (and which could expose nonportable hardware-specific behavior).
The glitch comes from the /dev/rtcX logic implementing the legacy
RTC_ALM_SET (limited to 24 hours, needing RTC_AIE_ON) ioctl on top of the
RTC driver call providing access to the newer RTC_WKALM_SET (without those
limitations) by initializing the day/month/year fields to be invalid ...
that second mode.
Now, since few RTC drivers check those fields, and most hardware misbehaves
when faced with invalid date fields, many RTC drivers will set bogus alarm
times on those RTC_ALM_SET code paths. (Several in-tree drivers have that
issue, and I also noticed it with code reviews on several new RTC drivers.)
This patch ensures that RTC drivers never see such invalid alarm fields, by
moving some logic out of rtc-omap into the RTC_ALM_SET code and adding an
explicit check (which will prevent the issue on other code paths).
Signed-off-by: David Brownell <dbrownell@users.sourceforge.net>
Cc: Scott Wood <scottwood@freescale.com>
Cc: Alessandro Zummo <a.zummo@towertech.it>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'drivers/rtc/rtc-omap.c')
-rw-r--r-- | drivers/rtc/rtc-omap.c | 28 |
1 files changed, 0 insertions, 28 deletions
diff --git a/drivers/rtc/rtc-omap.c b/drivers/rtc/rtc-omap.c index e6c7b0149f27..60a8a4bb8bd2 100644 --- a/drivers/rtc/rtc-omap.c +++ b/drivers/rtc/rtc-omap.c | |||
@@ -289,34 +289,6 @@ static int omap_rtc_set_alarm(struct device *dev, struct rtc_wkalrm *alm) | |||
289 | { | 289 | { |
290 | u8 reg; | 290 | u8 reg; |
291 | 291 | ||
292 | /* Much userspace code uses RTC_ALM_SET, thus "don't care" for | ||
293 | * day/month/year specifies alarms up to 24 hours in the future. | ||
294 | * So we need to handle that ... but let's ignore the "don't care" | ||
295 | * values for hours/minutes/seconds. | ||
296 | */ | ||
297 | if (alm->time.tm_mday <= 0 | ||
298 | && alm->time.tm_mon < 0 | ||
299 | && alm->time.tm_year < 0) { | ||
300 | struct rtc_time tm; | ||
301 | unsigned long now, then; | ||
302 | |||
303 | omap_rtc_read_time(dev, &tm); | ||
304 | rtc_tm_to_time(&tm, &now); | ||
305 | |||
306 | alm->time.tm_mday = tm.tm_mday; | ||
307 | alm->time.tm_mon = tm.tm_mon; | ||
308 | alm->time.tm_year = tm.tm_year; | ||
309 | rtc_tm_to_time(&alm->time, &then); | ||
310 | |||
311 | /* sometimes the alarm wraps into tomorrow */ | ||
312 | if (then < now) { | ||
313 | rtc_time_to_tm(now + 24 * 60 * 60, &tm); | ||
314 | alm->time.tm_mday = tm.tm_mday; | ||
315 | alm->time.tm_mon = tm.tm_mon; | ||
316 | alm->time.tm_year = tm.tm_year; | ||
317 | } | ||
318 | } | ||
319 | |||
320 | if (tm2bcd(&alm->time) < 0) | 292 | if (tm2bcd(&alm->time) < 0) |
321 | return -EINVAL; | 293 | return -EINVAL; |
322 | 294 | ||