aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/rtc
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/rtc')
-rw-r--r--drivers/rtc/rtc-sa1100.c66
1 files changed, 11 insertions, 55 deletions
diff --git a/drivers/rtc/rtc-sa1100.c b/drivers/rtc/rtc-sa1100.c
index fef52c801a13..91d58bdc7f88 100644
--- a/drivers/rtc/rtc-sa1100.c
+++ b/drivers/rtc/rtc-sa1100.c
@@ -44,54 +44,6 @@
44static const unsigned long RTC_FREQ = 1024; 44static const unsigned long RTC_FREQ = 1024;
45static DEFINE_SPINLOCK(sa1100_rtc_lock); 45static DEFINE_SPINLOCK(sa1100_rtc_lock);
46 46
47/*
48 * Calculate the next alarm time given the requested alarm time mask
49 * and the current time.
50 */
51static void rtc_next_alarm_time(struct rtc_time *next, struct rtc_time *now,
52 struct rtc_time *alrm)
53{
54 unsigned long next_time;
55 unsigned long now_time;
56
57 next->tm_year = now->tm_year;
58 next->tm_mon = now->tm_mon;
59 next->tm_mday = now->tm_mday;
60 next->tm_hour = alrm->tm_hour;
61 next->tm_min = alrm->tm_min;
62 next->tm_sec = alrm->tm_sec;
63
64 rtc_tm_to_time(now, &now_time);
65 rtc_tm_to_time(next, &next_time);
66
67 if (next_time < now_time) {
68 /* Advance one day */
69 next_time += 60 * 60 * 24;
70 rtc_time_to_tm(next_time, next);
71 }
72}
73
74static int rtc_update_alarm(struct rtc_time *alrm)
75{
76 struct rtc_time alarm_tm, now_tm;
77 unsigned long now, time;
78 int ret;
79
80 do {
81 now = RCNR;
82 rtc_time_to_tm(now, &now_tm);
83 rtc_next_alarm_time(&alarm_tm, &now_tm, alrm);
84 ret = rtc_tm_to_time(&alarm_tm, &time);
85 if (ret != 0)
86 break;
87
88 RTSR = RTSR & (RTSR_HZE|RTSR_ALE|RTSR_AL);
89 RTAR = time;
90 } while (now != RCNR);
91
92 return ret;
93}
94
95static irqreturn_t sa1100_rtc_interrupt(int irq, void *dev_id) 47static irqreturn_t sa1100_rtc_interrupt(int irq, void *dev_id)
96{ 48{
97 struct platform_device *pdev = to_platform_device(dev_id); 49 struct platform_device *pdev = to_platform_device(dev_id);
@@ -219,16 +171,20 @@ static int sa1100_rtc_read_alarm(struct device *dev, struct rtc_wkalrm *alrm)
219 171
220static int sa1100_rtc_set_alarm(struct device *dev, struct rtc_wkalrm *alrm) 172static int sa1100_rtc_set_alarm(struct device *dev, struct rtc_wkalrm *alrm)
221{ 173{
174 unsigned long time;
222 int ret; 175 int ret;
223 176
224 spin_lock_irq(&sa1100_rtc_lock); 177 spin_lock_irq(&sa1100_rtc_lock);
225 ret = rtc_update_alarm(&alrm->time); 178 ret = rtc_tm_to_time(&alrm->time, &time);
226 if (ret == 0) { 179 if (ret != 0)
227 if (alrm->enabled) 180 goto out;
228 RTSR |= RTSR_ALE; 181 RTSR = RTSR & (RTSR_HZE|RTSR_ALE|RTSR_AL);
229 else 182 RTAR = time;
230 RTSR &= ~RTSR_ALE; 183 if (alrm->enabled)
231 } 184 RTSR |= RTSR_ALE;
185 else
186 RTSR &= ~RTSR_ALE;
187out:
232 spin_unlock_irq(&sa1100_rtc_lock); 188 spin_unlock_irq(&sa1100_rtc_lock);
233 189
234 return ret; 190 return ret;