2006-03-27 13:16:41 +04:00
/*
* 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>
2007-05-08 11:33:27 +04:00
# include "rtc-core.h"
2006-03-27 13:16:41 +04:00
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 ) ;
2006-10-01 10:28:17 +04:00
const struct rtc_class_ops * ops = rtc - > ops ;
2006-03-27 13:16:41 +04:00
/* We keep the lock as long as the device is in use
* and return immediately if busy
*/
if ( ! ( mutex_trylock ( & rtc - > char_lock ) ) )
return - EBUSY ;
2007-05-08 11:33:30 +04:00
file - > private_data = rtc ;
2006-03-27 13:16:41 +04:00
2007-05-08 11:33:40 +04:00
err = ops - > open ? ops - > open ( rtc - > dev . parent ) : 0 ;
2006-03-27 13:16:41 +04:00
if ( err = = 0 ) {
spin_lock_irq ( & rtc - > irq_lock ) ;
rtc - > irq_data = 0 ;
spin_unlock_irq ( & rtc - > irq_lock ) ;
return 0 ;
}
/* something has gone wrong, release the lock */
mutex_unlock ( & rtc - > char_lock ) ;
return err ;
}
2006-06-25 16:48:17 +04:00
# 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
*/
2006-11-22 17:57:56 +03:00
static void rtc_uie_task ( struct work_struct * work )
2006-06-25 16:48:17 +04:00
{
2006-11-22 17:57:56 +03:00
struct rtc_device * rtc =
container_of ( work , struct rtc_device , uie_task ) ;
2006-06-25 16:48:17 +04:00
struct rtc_time tm ;
int num = 0 ;
int err ;
2007-05-08 11:33:30 +04:00
err = rtc_read_time ( rtc , & tm ) ;
2006-11-25 22:09:28 +03:00
local_irq_disable ( ) ;
spin_lock ( & rtc - > irq_lock ) ;
2006-06-25 16:48:17 +04:00
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 ;
}
2006-11-25 22:09:28 +03:00
spin_unlock ( & rtc - > irq_lock ) ;
2006-06-25 16:48:17 +04:00
if ( num )
2007-05-08 11:33:30 +04:00
rtc_update_irq ( rtc , num , RTC_UF | RTC_IRQF ) ;
2006-11-25 22:09:28 +03:00
local_irq_enable ( ) ;
2006-06-25 16:48:17 +04:00
}
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 void clear_uie ( struct rtc_device * rtc )
{
spin_lock_irq ( & rtc - > irq_lock ) ;
if ( rtc - > 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 - > irq_active = 0 ;
}
spin_unlock_irq ( & rtc - > irq_lock ) ;
}
static int set_uie ( struct rtc_device * rtc )
{
struct rtc_time tm ;
int err ;
2007-05-08 11:33:30 +04:00
err = rtc_read_time ( rtc , & tm ) ;
2006-06-25 16:48:17 +04:00
if ( err )
return err ;
spin_lock_irq ( & rtc - > irq_lock ) ;
if ( ! rtc - > irq_active ) {
rtc - > 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 ;
}
# endif /* CONFIG_RTC_INTF_DEV_UIE_EMUL */
2006-03-27 13:16:41 +04:00
static ssize_t
rtc_dev_read ( struct file * file , char __user * buf , size_t count , loff_t * ppos )
{
struct rtc_device * rtc = to_rtc_device ( file - > private_data ) ;
DECLARE_WAITQUEUE ( wait , current ) ;
unsigned long data ;
ssize_t ret ;
2006-05-01 23:16:16 +04:00
if ( count ! = sizeof ( unsigned int ) & & count < sizeof ( unsigned long ) )
2006-03-27 13:16:41 +04:00
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 )
2007-05-08 11:33:40 +04:00
data = rtc - > ops - > read_callback ( rtc - > dev . parent ,
2006-05-01 23:16:16 +04:00
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 ) ;
2006-03-27 13:16:41 +04:00
}
return ret ;
}
static unsigned int rtc_dev_poll ( struct file * file , poll_table * wait )
{
struct rtc_device * rtc = to_rtc_device ( file - > private_data ) ;
unsigned long data ;
poll_wait ( file , & rtc - > irq_queue , wait ) ;
data = rtc - > irq_data ;
return ( data ! = 0 ) ? ( POLLIN | POLLRDNORM ) : 0 ;
}
static int rtc_dev_ioctl ( struct inode * inode , struct file * file ,
unsigned int cmd , unsigned long arg )
{
int err = 0 ;
2007-05-08 11:33:30 +04:00
struct rtc_device * rtc = file - > private_data ;
2006-10-01 10:28:17 +04:00
const struct rtc_class_ops * ops = rtc - > ops ;
2006-03-27 13:16:41 +04:00
struct rtc_time tm ;
struct rtc_wkalrm alarm ;
void __user * uarg = ( void __user * ) arg ;
2006-11-25 22:09:27 +03:00
/* check that the calling task has appropriate permissions
2006-06-25 16:48:20 +04:00
* 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 ) )
return - EACCES ;
break ;
case RTC_IRQP_SET :
if ( arg > rtc - > max_user_freq & & ! capable ( CAP_SYS_RESOURCE ) )
return - EACCES ;
break ;
case RTC_PIE_ON :
if ( ! capable ( CAP_SYS_RESOURCE ) )
return - EACCES ;
break ;
}
2006-03-27 13:16:41 +04:00
/* avoid conflicting IRQ users */
if ( cmd = = RTC_PIE_ON | | cmd = = RTC_PIE_OFF | | cmd = = RTC_IRQP_SET ) {
2006-11-25 22:09:28 +03:00
spin_lock_irq ( & rtc - > irq_task_lock ) ;
2006-03-27 13:16:41 +04:00
if ( rtc - > irq_task )
err = - EBUSY ;
2006-11-25 22:09:28 +03:00
spin_unlock_irq ( & rtc - > irq_task_lock ) ;
2006-03-27 13:16:41 +04:00
if ( err < 0 )
return err ;
}
/* try the driver's ioctl interface */
if ( ops - > ioctl ) {
2007-05-08 11:33:40 +04:00
err = ops - > ioctl ( rtc - > dev . parent , cmd , arg ) ;
2006-05-21 02:00:29 +04:00
if ( err ! = - ENOIOCTLCMD )
2006-03-27 13:16:41 +04:00
return err ;
}
/* if the driver does not provide the ioctl interface
* or if that particular ioctl was not implemented
2006-05-21 02:00:29 +04:00
* ( - ENOIOCTLCMD ) , we will try to emulate here .
2006-03-27 13:16:41 +04:00
*/
switch ( cmd ) {
case RTC_ALM_READ :
2007-05-08 11:33:30 +04:00
err = rtc_read_alarm ( rtc , & alarm ) ;
2006-03-27 13:16:41 +04:00
if ( err < 0 )
return err ;
if ( copy_to_user ( uarg , & alarm . time , sizeof ( tm ) ) )
return - EFAULT ;
break ;
case RTC_ALM_SET :
if ( copy_from_user ( & alarm . time , uarg , sizeof ( tm ) ) )
return - EFAULT ;
alarm . enabled = 0 ;
alarm . pending = 0 ;
alarm . time . tm_mday = - 1 ;
alarm . time . tm_mon = - 1 ;
alarm . time . tm_year = - 1 ;
alarm . time . tm_wday = - 1 ;
alarm . time . tm_yday = - 1 ;
alarm . time . tm_isdst = - 1 ;
2007-05-08 11:33:30 +04:00
err = rtc_set_alarm ( rtc , & alarm ) ;
2006-03-27 13:16:41 +04:00
break ;
case RTC_RD_TIME :
2007-05-08 11:33:30 +04:00
err = rtc_read_time ( rtc , & tm ) ;
2006-03-27 13:16:41 +04:00
if ( err < 0 )
return err ;
if ( copy_to_user ( uarg , & tm , sizeof ( tm ) ) )
return - EFAULT ;
break ;
case RTC_SET_TIME :
if ( copy_from_user ( & tm , uarg , sizeof ( tm ) ) )
return - EFAULT ;
2007-05-08 11:33:30 +04:00
err = rtc_set_time ( rtc , & tm ) ;
2006-03-27 13:16:41 +04:00
break ;
2006-11-25 22:09:27 +03:00
case RTC_IRQP_READ :
if ( ops - > irq_set_freq )
2007-02-09 19:38:05 +03:00
err = put_user ( rtc - > irq_freq , ( unsigned long __user * ) uarg ) ;
2006-11-25 22:09:27 +03:00
break ;
case RTC_IRQP_SET :
if ( ops - > irq_set_freq )
2007-05-08 11:33:30 +04:00
err = rtc_irq_set_freq ( rtc , rtc - > irq_task , arg ) ;
2006-11-25 22:09:27 +03:00
break ;
2006-03-27 13:16:41 +04:00
#if 0
case RTC_EPOCH_SET :
# ifndef rtc_epoch
/*
* There were no RTC clocks before 1900.
*/
if ( arg < 1900 ) {
err = - EINVAL ;
break ;
}
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 :
if ( copy_from_user ( & alarm , uarg , sizeof ( alarm ) ) )
return - EFAULT ;
2007-05-08 11:33:30 +04:00
err = rtc_set_alarm ( rtc , & alarm ) ;
2006-03-27 13:16:41 +04:00
break ;
case RTC_WKALM_RD :
2007-05-08 11:33:30 +04:00
err = rtc_read_alarm ( rtc , & alarm ) ;
2006-03-27 13:16:41 +04:00
if ( err < 0 )
return err ;
if ( copy_to_user ( uarg , & alarm , sizeof ( alarm ) ) )
return - EFAULT ;
break ;
2006-06-25 16:48:17 +04:00
# ifdef CONFIG_RTC_INTF_DEV_UIE_EMUL
case RTC_UIE_OFF :
clear_uie ( rtc ) ;
return 0 ;
case RTC_UIE_ON :
return set_uie ( rtc ) ;
# endif
2006-03-27 13:16:41 +04:00
default :
2006-05-21 02:00:29 +04:00
err = - ENOTTY ;
2006-03-27 13:16:41 +04:00
break ;
}
return err ;
}
static int rtc_dev_release ( struct inode * inode , struct file * file )
{
struct rtc_device * rtc = to_rtc_device ( file - > private_data ) ;
2006-06-25 16:48:17 +04:00
# ifdef CONFIG_RTC_INTF_DEV_UIE_EMUL
clear_uie ( rtc ) ;
# endif
2006-03-27 13:16:41 +04:00
if ( rtc - > ops - > release )
2007-05-08 11:33:40 +04:00
rtc - > ops - > release ( rtc - > dev . parent ) ;
2006-03-27 13:16:41 +04:00
mutex_unlock ( & rtc - > char_lock ) ;
return 0 ;
}
static int rtc_dev_fasync ( int fd , struct file * file , int on )
{
struct rtc_device * rtc = to_rtc_device ( file - > private_data ) ;
return fasync_helper ( fd , file , on , & rtc - > async_queue ) ;
}
2007-02-12 11:55:34 +03:00
static const struct file_operations rtc_dev_fops = {
2006-03-27 13:16:41 +04:00
. owner = THIS_MODULE ,
. llseek = no_llseek ,
. read = rtc_dev_read ,
. poll = rtc_dev_poll ,
. ioctl = rtc_dev_ioctl ,
. open = rtc_dev_open ,
. release = rtc_dev_release ,
. fasync = rtc_dev_fasync ,
} ;
/* insertion/removal hooks */
2007-05-08 11:33:46 +04:00
void rtc_dev_prepare ( struct rtc_device * rtc )
2006-03-27 13:16:41 +04:00
{
2007-05-08 11:33:27 +04:00
if ( ! rtc_devt )
return ;
2006-03-27 13:16:41 +04:00
if ( rtc - > id > = RTC_DEV_MAX ) {
2007-05-08 11:33:27 +04:00
pr_debug ( " %s: too many RTC devices \n " , rtc - > name ) ;
return ;
2006-03-27 13:16:41 +04:00
}
2007-05-08 11:33:40 +04:00
rtc - > dev . devt = MKDEV ( MAJOR ( rtc_devt ) , rtc - > id ) ;
2007-05-08 11:33:27 +04:00
2006-03-27 13:16:41 +04:00
mutex_init ( & rtc - > char_lock ) ;
spin_lock_init ( & rtc - > irq_lock ) ;
init_waitqueue_head ( & rtc - > irq_queue ) ;
2006-06-25 16:48:17 +04:00
# ifdef CONFIG_RTC_INTF_DEV_UIE_EMUL
2006-11-22 17:57:56 +03:00
INIT_WORK ( & rtc - > uie_task , rtc_uie_task ) ;
2006-06-25 16:48:17 +04:00
setup_timer ( & rtc - > uie_timer , rtc_uie_timer , ( unsigned long ) rtc ) ;
# endif
2006-03-27 13:16:41 +04:00
cdev_init ( & rtc - > char_dev , & rtc_dev_fops ) ;
rtc - > char_dev . owner = rtc - > owner ;
2007-05-08 11:33:46 +04:00
}
2006-03-27 13:16:41 +04:00
2007-05-08 11:33:46 +04:00
void rtc_dev_add_device ( struct rtc_device * rtc )
{
2007-05-08 11:33:40 +04:00
if ( cdev_add ( & rtc - > char_dev , rtc - > dev . devt , 1 ) )
2007-05-08 11:33:27 +04:00
printk ( KERN_WARNING " %s: failed to add char device %d:%d \n " ,
rtc - > name , MAJOR ( rtc_devt ) , rtc - > id ) ;
else
pr_debug ( " %s: dev (%d:%d) \n " , rtc - > name ,
2006-03-27 13:16:41 +04:00
MAJOR ( rtc_devt ) , rtc - > id ) ;
}
2007-05-08 11:33:27 +04:00
void rtc_dev_del_device ( struct rtc_device * rtc )
2006-03-27 13:16:41 +04:00
{
2007-05-08 11:33:40 +04:00
if ( rtc - > dev . devt )
2006-03-27 13:16:41 +04:00
cdev_del ( & rtc - > char_dev ) ;
}
2007-05-08 11:33:27 +04:00
void __init rtc_dev_init ( void )
2006-03-27 13:16:41 +04:00
{
int err ;
err = alloc_chrdev_region ( & rtc_devt , 0 , RTC_DEV_MAX , " rtc " ) ;
2007-05-08 11:33:27 +04:00
if ( err < 0 )
2006-03-27 13:16:41 +04:00
printk ( KERN_ERR " %s: failed to allocate char dev region \n " ,
__FILE__ ) ;
}
2007-05-08 11:33:27 +04:00
void __exit rtc_dev_exit ( void )
2006-03-27 13:16:41 +04:00
{
2007-05-08 11:33:27 +04:00
if ( rtc_devt )
unregister_chrdev_region ( rtc_devt , RTC_DEV_MAX ) ;
2006-03-27 13:16:41 +04:00
}