Blame view

drivers/rtc/rtc-dev.c 12 KB
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
  /*
   * RTC subsystem, dev interface
   *
   * Copyright (C) 2005 Tower Technologies
   * Author: Alessandro Zummo <a.zummo@towertech.it>
   *
   * based on arch/arm/common/rtctime.c
   *
   * This program is free software; you can redistribute it and/or modify
   * it under the terms of the GNU General Public License version 2 as
   * published by the Free Software Foundation.
  */
  
  #include <linux/module.h>
  #include <linux/rtc.h>
d43c36dc6   Alexey Dobriyan   headers: remove s...
16
  #include <linux/sched.h>
5726fb201   David Brownell   rtc: remove /sys/...
17
  #include "rtc-core.h"
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
18

e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
19
20
21
22
23
24
25
26
27
  static dev_t rtc_devt;
  
  #define RTC_DEV_MAX 16 /* 16 RTCs should be enough for everyone... */
  
  static int rtc_dev_open(struct inode *inode, struct file *file)
  {
  	int err;
  	struct rtc_device *rtc = container_of(inode->i_cdev,
  					struct rtc_device, char_dev);
ff8371ac9   David Brownell   [PATCH] constify ...
28
  	const struct rtc_class_ops *ops = rtc->ops;
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
29

b1c3c8982   David Brownell   revert "rtc: cdev...
30
31
  	if (test_and_set_bit_lock(RTC_DEV_BUSY, &rtc->flags))
  		return -EBUSY;
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
32

ab6a2d70d   David Brownell   rtc: rtc interfac...
33
  	file->private_data = rtc;
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
34

cd9662094   David Brownell   rtc: remove rest ...
35
  	err = ops->open ? ops->open(rtc->dev.parent) : 0;
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
36
37
38
39
  	if (err == 0) {
  		spin_lock_irq(&rtc->irq_lock);
  		rtc->irq_data = 0;
  		spin_unlock_irq(&rtc->irq_lock);
b1c3c8982   David Brownell   revert "rtc: cdev...
40
  		return 0;
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
41
  	}
8853c202b   Jiri Kosina   RTC: convert mute...
42
  	/* something has gone wrong */
372a302e9   Jiri Kosina   RTC: assure prope...
43
  	clear_bit_unlock(RTC_DEV_BUSY, &rtc->flags);
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
44
45
  	return err;
  }
6e57b1d6a   John Stultz   RTC: Revert UIE e...
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
  #ifdef CONFIG_RTC_INTF_DEV_UIE_EMUL
  /*
   * Routine to poll RTC seconds field for change as often as possible,
   * after first RTC_UIE use timer to reduce polling
   */
  static void rtc_uie_task(struct work_struct *work)
  {
  	struct rtc_device *rtc =
  		container_of(work, struct rtc_device, uie_task);
  	struct rtc_time tm;
  	int num = 0;
  	int err;
  
  	err = rtc_read_time(rtc, &tm);
  
  	spin_lock_irq(&rtc->irq_lock);
  	if (rtc->stop_uie_polling || err) {
  		rtc->uie_task_active = 0;
  	} else if (rtc->oldsecs != tm.tm_sec) {
  		num = (tm.tm_sec + 60 - rtc->oldsecs) % 60;
  		rtc->oldsecs = tm.tm_sec;
  		rtc->uie_timer.expires = jiffies + HZ - (HZ/10);
  		rtc->uie_timer_active = 1;
  		rtc->uie_task_active = 0;
  		add_timer(&rtc->uie_timer);
  	} else if (schedule_work(&rtc->uie_task) == 0) {
  		rtc->uie_task_active = 0;
  	}
  	spin_unlock_irq(&rtc->irq_lock);
  	if (num)
456d66ecd   John Stultz   RTC: Re-enable UI...
76
  		rtc_handle_legacy_irq(rtc, num, RTC_UF);
6e57b1d6a   John Stultz   RTC: Revert UIE e...
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
  }
  static void rtc_uie_timer(unsigned long data)
  {
  	struct rtc_device *rtc = (struct rtc_device *)data;
  	unsigned long flags;
  
  	spin_lock_irqsave(&rtc->irq_lock, flags);
  	rtc->uie_timer_active = 0;
  	rtc->uie_task_active = 1;
  	if ((schedule_work(&rtc->uie_task) == 0))
  		rtc->uie_task_active = 0;
  	spin_unlock_irqrestore(&rtc->irq_lock, flags);
  }
  
  static int clear_uie(struct rtc_device *rtc)
  {
  	spin_lock_irq(&rtc->irq_lock);
  	if (rtc->uie_irq_active) {
  		rtc->stop_uie_polling = 1;
  		if (rtc->uie_timer_active) {
  			spin_unlock_irq(&rtc->irq_lock);
  			del_timer_sync(&rtc->uie_timer);
  			spin_lock_irq(&rtc->irq_lock);
  			rtc->uie_timer_active = 0;
  		}
  		if (rtc->uie_task_active) {
  			spin_unlock_irq(&rtc->irq_lock);
  			flush_scheduled_work();
  			spin_lock_irq(&rtc->irq_lock);
  		}
  		rtc->uie_irq_active = 0;
  	}
  	spin_unlock_irq(&rtc->irq_lock);
  	return 0;
  }
  
  static int set_uie(struct rtc_device *rtc)
  {
  	struct rtc_time tm;
  	int err;
  
  	err = rtc_read_time(rtc, &tm);
  	if (err)
  		return err;
  	spin_lock_irq(&rtc->irq_lock);
  	if (!rtc->uie_irq_active) {
  		rtc->uie_irq_active = 1;
  		rtc->stop_uie_polling = 0;
  		rtc->oldsecs = tm.tm_sec;
  		rtc->uie_task_active = 1;
  		if (schedule_work(&rtc->uie_task) == 0)
  			rtc->uie_task_active = 0;
  	}
  	rtc->irq_data = 0;
  	spin_unlock_irq(&rtc->irq_lock);
  	return 0;
  }
  
  int rtc_dev_update_irq_enable_emul(struct rtc_device *rtc, unsigned int enabled)
  {
  	if (enabled)
  		return set_uie(rtc);
  	else
  		return clear_uie(rtc);
  }
  EXPORT_SYMBOL(rtc_dev_update_irq_enable_emul);
  
  #endif /* CONFIG_RTC_INTF_DEV_UIE_EMUL */
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
145
146
147
148
  
  static ssize_t
  rtc_dev_read(struct file *file, char __user *buf, size_t count, loff_t *ppos)
  {
88efe1373   Mark Zhan   rtc-dev: no need ...
149
  	struct rtc_device *rtc = file->private_data;
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
150
151
152
153
  
  	DECLARE_WAITQUEUE(wait, current);
  	unsigned long data;
  	ssize_t ret;
3418ff761   Atsushi Nemoto   [PATCH] RTC: rtc-...
154
  	if (count != sizeof(unsigned int) && count < sizeof(unsigned long))
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
  		return -EINVAL;
  
  	add_wait_queue(&rtc->irq_queue, &wait);
  	do {
  		__set_current_state(TASK_INTERRUPTIBLE);
  
  		spin_lock_irq(&rtc->irq_lock);
  		data = rtc->irq_data;
  		rtc->irq_data = 0;
  		spin_unlock_irq(&rtc->irq_lock);
  
  		if (data != 0) {
  			ret = 0;
  			break;
  		}
  		if (file->f_flags & O_NONBLOCK) {
  			ret = -EAGAIN;
  			break;
  		}
  		if (signal_pending(current)) {
  			ret = -ERESTARTSYS;
  			break;
  		}
  		schedule();
  	} while (1);
  	set_current_state(TASK_RUNNING);
  	remove_wait_queue(&rtc->irq_queue, &wait);
  
  	if (ret == 0) {
  		/* Check for any data updates */
  		if (rtc->ops->read_callback)
cd9662094   David Brownell   rtc: remove rest ...
186
  			data = rtc->ops->read_callback(rtc->dev.parent,
3418ff761   Atsushi Nemoto   [PATCH] RTC: rtc-...
187
188
189
190
191
192
193
194
195
  						       data);
  
  		if (sizeof(int) != sizeof(long) &&
  		    count == sizeof(unsigned int))
  			ret = put_user(data, (unsigned int __user *)buf) ?:
  				sizeof(unsigned int);
  		else
  			ret = put_user(data, (unsigned long __user *)buf) ?:
  				sizeof(unsigned long);
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
196
197
198
199
200
201
  	}
  	return ret;
  }
  
  static unsigned int rtc_dev_poll(struct file *file, poll_table *wait)
  {
88efe1373   Mark Zhan   rtc-dev: no need ...
202
  	struct rtc_device *rtc = file->private_data;
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
203
204
205
206
207
208
209
210
  	unsigned long data;
  
  	poll_wait(file, &rtc->irq_queue, wait);
  
  	data = rtc->irq_data;
  
  	return (data != 0) ? (POLLIN | POLLRDNORM) : 0;
  }
5ad31a575   David Brownell   rtc: remove BKL f...
211
  static long rtc_dev_ioctl(struct file *file,
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
212
213
214
  		unsigned int cmd, unsigned long arg)
  {
  	int err = 0;
ab6a2d70d   David Brownell   rtc: rtc interfac...
215
  	struct rtc_device *rtc = file->private_data;
ff8371ac9   David Brownell   [PATCH] constify ...
216
  	const struct rtc_class_ops *ops = rtc->ops;
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
217
218
219
  	struct rtc_time tm;
  	struct rtc_wkalrm alarm;
  	void __user *uarg = (void __user *) arg;
5ad31a575   David Brownell   rtc: remove BKL f...
220
221
  	err = mutex_lock_interruptible(&rtc->ops_lock);
  	if (err)
b68bb2632   David Brownell   rtc: don't return...
222
  		return err;
5ad31a575   David Brownell   rtc: remove BKL f...
223

2601a4647   David Brownell   [PATCH] rtc frame...
224
  	/* check that the calling task has appropriate permissions
110d693d5   Alessandro Zummo   [PATCH] rtc subsy...
225
226
227
228
229
230
231
  	 * for certain ioctls. doing this check here is useful
  	 * to avoid duplicate code in each driver.
  	 */
  	switch (cmd) {
  	case RTC_EPOCH_SET:
  	case RTC_SET_TIME:
  		if (!capable(CAP_SYS_TIME))
5ad31a575   David Brownell   rtc: remove BKL f...
232
  			err = -EACCES;
110d693d5   Alessandro Zummo   [PATCH] rtc subsy...
233
234
235
236
  		break;
  
  	case RTC_IRQP_SET:
  		if (arg > rtc->max_user_freq && !capable(CAP_SYS_RESOURCE))
5ad31a575   David Brownell   rtc: remove BKL f...
237
  			err = -EACCES;
110d693d5   Alessandro Zummo   [PATCH] rtc subsy...
238
239
240
  		break;
  
  	case RTC_PIE_ON:
9d013d3b1   Bryan Kadzban   rtc: allow valida...
241
242
  		if (rtc->irq_freq > rtc->max_user_freq &&
  				!capable(CAP_SYS_RESOURCE))
5ad31a575   David Brownell   rtc: remove BKL f...
243
  			err = -EACCES;
110d693d5   Alessandro Zummo   [PATCH] rtc subsy...
244
245
  		break;
  	}
5ad31a575   David Brownell   rtc: remove BKL f...
246
247
  	if (err)
  		goto done;
ac54cd2bd   John Stultz   RTC: Fix rtc driv...
248
  	/*
8a0bdfd7a   David Brownell   rtc-cmos alarm ac...
249
250
251
252
253
254
255
256
  	 * Drivers *SHOULD NOT* provide ioctl implementations
  	 * for these requests.  Instead, provide methods to
  	 * support the following code, so that the RTC's main
  	 * features are accessible without using ioctls.
  	 *
  	 * RTC and alarm times will be in UTC, by preference,
  	 * but dual-booting with MS-Windows implies RTCs must
  	 * use the local wall clock time.
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
257
258
259
260
  	 */
  
  	switch (cmd) {
  	case RTC_ALM_READ:
5ad31a575   David Brownell   rtc: remove BKL f...
261
  		mutex_unlock(&rtc->ops_lock);
ab6a2d70d   David Brownell   rtc: rtc interfac...
262
  		err = rtc_read_alarm(rtc, &alarm);
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
263
264
265
266
  		if (err < 0)
  			return err;
  
  		if (copy_to_user(uarg, &alarm.time, sizeof(tm)))
5ad31a575   David Brownell   rtc: remove BKL f...
267
268
  			err = -EFAULT;
  		return err;
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
269
270
  
  	case RTC_ALM_SET:
5ad31a575   David Brownell   rtc: remove BKL f...
271
  		mutex_unlock(&rtc->ops_lock);
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
272
273
274
275
276
  		if (copy_from_user(&alarm.time, uarg, sizeof(tm)))
  			return -EFAULT;
  
  		alarm.enabled = 0;
  		alarm.pending = 0;
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
277
278
279
  		alarm.time.tm_wday = -1;
  		alarm.time.tm_yday = -1;
  		alarm.time.tm_isdst = -1;
f8245c268   David Brownell   rtc: remove "RTC_...
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
  
  		/* RTC_ALM_SET alarms may be up to 24 hours in the future.
  		 * Rather than expecting every RTC to implement "don't care"
  		 * for day/month/year fields, just force the alarm to have
  		 * the right values for those fields.
  		 *
  		 * RTC_WKALM_SET should be used instead.  Not only does it
  		 * eliminate the need for a separate RTC_AIE_ON call, it
  		 * doesn't have the "alarm 23:59:59 in the future" race.
  		 *
  		 * NOTE:  some legacy code may have used invalid fields as
  		 * wildcards, exposing hardware "periodic alarm" capabilities.
  		 * Not supported here.
  		 */
  		{
  			unsigned long now, then;
  
  			err = rtc_read_time(rtc, &tm);
  			if (err < 0)
  				return err;
  			rtc_tm_to_time(&tm, &now);
  
  			alarm.time.tm_mday = tm.tm_mday;
  			alarm.time.tm_mon = tm.tm_mon;
  			alarm.time.tm_year = tm.tm_year;
  			err  = rtc_valid_tm(&alarm.time);
  			if (err < 0)
  				return err;
  			rtc_tm_to_time(&alarm.time, &then);
  
  			/* alarm may need to wrap into tomorrow */
  			if (then < now) {
  				rtc_time_to_tm(now + 24 * 60 * 60, &tm);
  				alarm.time.tm_mday = tm.tm_mday;
  				alarm.time.tm_mon = tm.tm_mon;
  				alarm.time.tm_year = tm.tm_year;
  			}
  		}
5ad31a575   David Brownell   rtc: remove BKL f...
318
  		return rtc_set_alarm(rtc, &alarm);
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
319
320
  
  	case RTC_RD_TIME:
5ad31a575   David Brownell   rtc: remove BKL f...
321
  		mutex_unlock(&rtc->ops_lock);
ab6a2d70d   David Brownell   rtc: rtc interfac...
322
  		err = rtc_read_time(rtc, &tm);
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
323
324
325
326
  		if (err < 0)
  			return err;
  
  		if (copy_to_user(uarg, &tm, sizeof(tm)))
5ad31a575   David Brownell   rtc: remove BKL f...
327
328
  			err = -EFAULT;
  		return err;
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
329
330
  
  	case RTC_SET_TIME:
5ad31a575   David Brownell   rtc: remove BKL f...
331
  		mutex_unlock(&rtc->ops_lock);
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
332
333
  		if (copy_from_user(&tm, uarg, sizeof(tm)))
  			return -EFAULT;
5ad31a575   David Brownell   rtc: remove BKL f...
334
  		return rtc_set_time(rtc, &tm);
2601a4647   David Brownell   [PATCH] rtc frame...
335

d691eb901   Alessandro Zummo   RTC: periodic irq...
336
337
338
339
340
341
  	case RTC_PIE_ON:
  		err = rtc_irq_set_state(rtc, NULL, 1);
  		break;
  
  	case RTC_PIE_OFF:
  		err = rtc_irq_set_state(rtc, NULL, 0);
2601a4647   David Brownell   [PATCH] rtc frame...
342
  		break;
099e65762   Alessandro Zummo   rtc: add alarm/up...
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
  	case RTC_AIE_ON:
  		mutex_unlock(&rtc->ops_lock);
  		return rtc_alarm_irq_enable(rtc, 1);
  
  	case RTC_AIE_OFF:
  		mutex_unlock(&rtc->ops_lock);
  		return rtc_alarm_irq_enable(rtc, 0);
  
  	case RTC_UIE_ON:
  		mutex_unlock(&rtc->ops_lock);
  		return rtc_update_irq_enable(rtc, 1);
  
  	case RTC_UIE_OFF:
  		mutex_unlock(&rtc->ops_lock);
  		return rtc_update_irq_enable(rtc, 0);
2601a4647   David Brownell   [PATCH] rtc frame...
358
  	case RTC_IRQP_SET:
d691eb901   Alessandro Zummo   RTC: periodic irq...
359
360
361
362
363
  		err = rtc_irq_set_freq(rtc, NULL, arg);
  		break;
  
  	case RTC_IRQP_READ:
  		err = put_user(rtc->irq_freq, (unsigned long __user *)uarg);
2601a4647   David Brownell   [PATCH] rtc frame...
364
  		break;
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
365
366
367
368
369
370
371
372
373
374
  #if 0
  	case RTC_EPOCH_SET:
  #ifndef rtc_epoch
  		/*
  		 * There were no RTC clocks before 1900.
  		 */
  		if (arg < 1900) {
  			err = -EINVAL;
  			break;
  		}
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
375
376
377
378
379
380
381
382
383
384
  		rtc_epoch = arg;
  		err = 0;
  #endif
  		break;
  
  	case RTC_EPOCH_READ:
  		err = put_user(rtc_epoch, (unsigned long __user *)uarg);
  		break;
  #endif
  	case RTC_WKALM_SET:
5ad31a575   David Brownell   rtc: remove BKL f...
385
  		mutex_unlock(&rtc->ops_lock);
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
386
387
  		if (copy_from_user(&alarm, uarg, sizeof(alarm)))
  			return -EFAULT;
5ad31a575   David Brownell   rtc: remove BKL f...
388
  		return rtc_set_alarm(rtc, &alarm);
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
389
390
  
  	case RTC_WKALM_RD:
5ad31a575   David Brownell   rtc: remove BKL f...
391
  		mutex_unlock(&rtc->ops_lock);
ab6a2d70d   David Brownell   rtc: rtc interfac...
392
  		err = rtc_read_alarm(rtc, &alarm);
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
393
394
395
396
  		if (err < 0)
  			return err;
  
  		if (copy_to_user(uarg, &alarm, sizeof(alarm)))
5ad31a575   David Brownell   rtc: remove BKL f...
397
398
  			err = -EFAULT;
  		return err;
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
399
400
  
  	default:
ac54cd2bd   John Stultz   RTC: Fix rtc driv...
401
402
403
404
405
  		/* Finally try the driver's ioctl interface */
  		if (ops->ioctl) {
  			err = ops->ioctl(rtc->dev.parent, cmd, arg);
  			if (err == -ENOIOCTLCMD)
  				err = -ENOTTY;
e17fd4ba2   John Stultz   rtc: Fix ioctl er...
406
407
  		} else
  			err = -ENOTTY;
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
408
409
  		break;
  	}
5ad31a575   David Brownell   rtc: remove BKL f...
410
411
  done:
  	mutex_unlock(&rtc->ops_lock);
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
412
413
  	return err;
  }
2e4a75cdc   Marcin Slusarz   rtc: fix kernel p...
414
415
416
417
418
  static int rtc_dev_fasync(int fd, struct file *file, int on)
  {
  	struct rtc_device *rtc = file->private_data;
  	return fasync_helper(fd, file, on, &rtc->async_queue);
  }
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
419
420
  static int rtc_dev_release(struct inode *inode, struct file *file)
  {
88efe1373   Mark Zhan   rtc-dev: no need ...
421
  	struct rtc_device *rtc = file->private_data;
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
422

743e6a504   David Brownell   rtc: file close()...
423
424
425
426
427
428
429
430
  	/* We shut down the repeating IRQs that userspace enabled,
  	 * since nothing is listening to them.
  	 *  - Update (UIE) ... currently only managed through ioctls
  	 *  - Periodic (PIE) ... also used through rtc_*() interface calls
  	 *
  	 * Leave the alarm alone; it may be set to trigger a system wakeup
  	 * later, or be used by kernel code, and is a one-shot event anyway.
  	 */
099e65762   Alessandro Zummo   rtc: add alarm/up...
431
432
  
  	/* Keep ioctl until all drivers are converted */
743e6a504   David Brownell   rtc: file close()...
433
  	rtc_dev_ioctl(file, RTC_UIE_OFF, 0);
099e65762   Alessandro Zummo   rtc: add alarm/up...
434
  	rtc_update_irq_enable(rtc, 0);
5cdc98b8f   Tomas Janousek   rtc-dev: stop per...
435
  	rtc_irq_set_state(rtc, NULL, 0);
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
436
  	if (rtc->ops->release)
cd9662094   David Brownell   rtc: remove rest ...
437
  		rtc->ops->release(rtc->dev.parent);
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
438

372a302e9   Jiri Kosina   RTC: assure prope...
439
  	clear_bit_unlock(RTC_DEV_BUSY, &rtc->flags);
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
440
441
  	return 0;
  }
d54b1fdb1   Arjan van de Ven   [PATCH] mark stru...
442
  static const struct file_operations rtc_dev_fops = {
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
443
444
445
446
  	.owner		= THIS_MODULE,
  	.llseek		= no_llseek,
  	.read		= rtc_dev_read,
  	.poll		= rtc_dev_poll,
5ad31a575   David Brownell   rtc: remove BKL f...
447
  	.unlocked_ioctl	= rtc_dev_ioctl,
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
448
449
450
451
452
453
  	.open		= rtc_dev_open,
  	.release	= rtc_dev_release,
  	.fasync		= rtc_dev_fasync,
  };
  
  /* insertion/removal hooks */
cb3a58d2a   David Brownell   rtc: update to cl...
454
  void rtc_dev_prepare(struct rtc_device *rtc)
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
455
  {
5726fb201   David Brownell   rtc: remove /sys/...
456
457
  	if (!rtc_devt)
  		return;
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
458
459
  
  	if (rtc->id >= RTC_DEV_MAX) {
5726fb201   David Brownell   rtc: remove /sys/...
460
461
462
  		pr_debug("%s: too many RTC devices
  ", rtc->name);
  		return;
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
463
  	}
cd9662094   David Brownell   rtc: remove rest ...
464
  	rtc->dev.devt = MKDEV(MAJOR(rtc_devt), rtc->id);
5726fb201   David Brownell   rtc: remove /sys/...
465

6e57b1d6a   John Stultz   RTC: Revert UIE e...
466
467
468
469
  #ifdef CONFIG_RTC_INTF_DEV_UIE_EMUL
  	INIT_WORK(&rtc->uie_task, rtc_uie_task);
  	setup_timer(&rtc->uie_timer, rtc_uie_timer, (unsigned long)rtc);
  #endif
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
470
471
  	cdev_init(&rtc->char_dev, &rtc_dev_fops);
  	rtc->char_dev.owner = rtc->owner;
cb3a58d2a   David Brownell   rtc: update to cl...
472
  }
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
473

cb3a58d2a   David Brownell   rtc: update to cl...
474
475
  void rtc_dev_add_device(struct rtc_device *rtc)
  {
cd9662094   David Brownell   rtc: remove rest ...
476
  	if (cdev_add(&rtc->char_dev, rtc->dev.devt, 1))
5726fb201   David Brownell   rtc: remove /sys/...
477
478
479
480
481
482
  		printk(KERN_WARNING "%s: failed to add char device %d:%d
  ",
  			rtc->name, MAJOR(rtc_devt), rtc->id);
  	else
  		pr_debug("%s: dev (%d:%d)
  ", rtc->name,
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
483
  			MAJOR(rtc_devt), rtc->id);
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
484
  }
5726fb201   David Brownell   rtc: remove /sys/...
485
  void rtc_dev_del_device(struct rtc_device *rtc)
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
486
  {
cd9662094   David Brownell   rtc: remove rest ...
487
  	if (rtc->dev.devt)
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
488
  		cdev_del(&rtc->char_dev);
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
489
  }
5726fb201   David Brownell   rtc: remove /sys/...
490
  void __init rtc_dev_init(void)
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
491
492
  {
  	int err;
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
493
  	err = alloc_chrdev_region(&rtc_devt, 0, RTC_DEV_MAX, "rtc");
5726fb201   David Brownell   rtc: remove /sys/...
494
  	if (err < 0)
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
495
496
497
  		printk(KERN_ERR "%s: failed to allocate char dev region
  ",
  			__FILE__);
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
498
  }
5726fb201   David Brownell   rtc: remove /sys/...
499
  void __exit rtc_dev_exit(void)
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
500
  {
5726fb201   David Brownell   rtc: remove /sys/...
501
502
  	if (rtc_devt)
  		unregister_chrdev_region(rtc_devt, RTC_DEV_MAX);
e824290e5   Alessandro Zummo   [PATCH] RTC subsy...
503
  }