aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/rtc/interface.c
diff options
context:
space:
mode:
authorAlessandro Zummo <alessandro.zummo@towertech.it>2007-10-16 04:28:15 -0400
committerLinus Torvalds <torvalds@woody.linux-foundation.org>2007-10-16 12:43:13 -0400
commitd691eb901e044065de10756ea78a5758d457c7fd (patch)
tree3b166f4d54bba5677a04b885d0da52c28db3a8a3 /drivers/rtc/interface.c
parent16a72c455a67bb23eed7292a31c6ba17729e78e6 (diff)
RTC: periodic irq fix
Add kernel/kernel and kernel/user locking for the periodic irq feature of the rtc class. PIE ioctls are also supported. Signed-off-by: Alessandro Zummo <a.zummo@towertech.it> Cc: David Brownell <david-b@pacbell.net> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'drivers/rtc/interface.c')
-rw-r--r--drivers/rtc/interface.c15
1 files changed, 12 insertions, 3 deletions
diff --git a/drivers/rtc/interface.c b/drivers/rtc/interface.c
index ad66c6ecf36..a43afa54f0d 100644
--- a/drivers/rtc/interface.c
+++ b/drivers/rtc/interface.c
@@ -210,6 +210,10 @@ int rtc_irq_register(struct rtc_device *rtc, struct rtc_task *task)
210 if (task == NULL || task->func == NULL) 210 if (task == NULL || task->func == NULL)
211 return -EINVAL; 211 return -EINVAL;
212 212
213 /* Cannot register while the char dev is in use */
214 if (!(mutex_trylock(&rtc->char_lock)))
215 return -EBUSY;
216
213 spin_lock_irq(&rtc->irq_task_lock); 217 spin_lock_irq(&rtc->irq_task_lock);
214 if (rtc->irq_task == NULL) { 218 if (rtc->irq_task == NULL) {
215 rtc->irq_task = task; 219 rtc->irq_task = task;
@@ -217,13 +221,14 @@ int rtc_irq_register(struct rtc_device *rtc, struct rtc_task *task)
217 } 221 }
218 spin_unlock_irq(&rtc->irq_task_lock); 222 spin_unlock_irq(&rtc->irq_task_lock);
219 223
224 mutex_unlock(&rtc->char_lock);
225
220 return retval; 226 return retval;
221} 227}
222EXPORT_SYMBOL_GPL(rtc_irq_register); 228EXPORT_SYMBOL_GPL(rtc_irq_register);
223 229
224void rtc_irq_unregister(struct rtc_device *rtc, struct rtc_task *task) 230void rtc_irq_unregister(struct rtc_device *rtc, struct rtc_task *task)
225{ 231{
226
227 spin_lock_irq(&rtc->irq_task_lock); 232 spin_lock_irq(&rtc->irq_task_lock);
228 if (rtc->irq_task == task) 233 if (rtc->irq_task == task)
229 rtc->irq_task = NULL; 234 rtc->irq_task = NULL;
@@ -240,8 +245,10 @@ int rtc_irq_set_state(struct rtc_device *rtc, struct rtc_task *task, int enabled
240 return -ENXIO; 245 return -ENXIO;
241 246
242 spin_lock_irqsave(&rtc->irq_task_lock, flags); 247 spin_lock_irqsave(&rtc->irq_task_lock, flags);
248 if (rtc->irq_task != NULL && task == NULL)
249 err = -EBUSY;
243 if (rtc->irq_task != task) 250 if (rtc->irq_task != task)
244 err = -ENXIO; 251 err = -EACCES;
245 spin_unlock_irqrestore(&rtc->irq_task_lock, flags); 252 spin_unlock_irqrestore(&rtc->irq_task_lock, flags);
246 253
247 if (err == 0) 254 if (err == 0)
@@ -260,8 +267,10 @@ int rtc_irq_set_freq(struct rtc_device *rtc, struct rtc_task *task, int freq)
260 return -ENXIO; 267 return -ENXIO;
261 268
262 spin_lock_irqsave(&rtc->irq_task_lock, flags); 269 spin_lock_irqsave(&rtc->irq_task_lock, flags);
270 if (rtc->irq_task != NULL && task == NULL)
271 err = -EBUSY;
263 if (rtc->irq_task != task) 272 if (rtc->irq_task != task)
264 err = -ENXIO; 273 err = -EACCES;
265 spin_unlock_irqrestore(&rtc->irq_task_lock, flags); 274 spin_unlock_irqrestore(&rtc->irq_task_lock, flags);
266 275
267 if (err == 0) { 276 if (err == 0) {