2019-05-20 10:19:02 +03:00
// SPDX-License-Identifier: GPL-2.0-or-later
2009-06-18 03:28:37 +04:00
/*
* PPS core file
*
* Copyright ( C ) 2005 - 2009 Rodolfo Giometti < giometti @ linux . it >
*/
2011-01-13 04:00:52 +03:00
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2009-06-18 03:28:37 +04:00
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/init.h>
# include <linux/sched.h>
# include <linux/uaccess.h>
# include <linux/idr.h>
2011-01-13 04:00:53 +03:00
# include <linux/mutex.h>
2009-06-18 03:28:37 +04:00
# include <linux/cdev.h>
# include <linux/poll.h>
# include <linux/pps_kernel.h>
2011-01-13 04:00:53 +03:00
# include <linux/slab.h>
2009-06-18 03:28:37 +04:00
2011-01-13 04:00:58 +03:00
# include "kc.h"
2009-06-18 03:28:37 +04:00
/*
* Local variables
*/
static dev_t pps_devt ;
static struct class * pps_class ;
2011-01-13 04:00:53 +03:00
static DEFINE_MUTEX ( pps_idr_lock ) ;
2011-01-13 04:00:53 +03:00
static DEFINE_IDR ( pps_idr ) ;
2009-06-18 03:28:37 +04:00
/*
* Char device methods
*/
2017-07-03 13:39:46 +03:00
static __poll_t pps_cdev_poll ( struct file * file , poll_table * wait )
2009-06-18 03:28:37 +04:00
{
struct pps_device * pps = file - > private_data ;
poll_wait ( file , & pps - > queue , wait ) ;
2018-02-12 01:34:03 +03:00
return EPOLLIN | EPOLLRDNORM ;
2009-06-18 03:28:37 +04:00
}
static int pps_cdev_fasync ( int fd , struct file * file , int on )
{
struct pps_device * pps = file - > private_data ;
return fasync_helper ( fd , file , on , & pps - > async_queue ) ;
}
2017-03-11 02:19:45 +03:00
static int pps_cdev_pps_fetch ( struct pps_device * pps , struct pps_fdata * fdata )
{
unsigned int ev = pps - > last_ev ;
int err = 0 ;
/* Manage the timeout */
if ( fdata - > timeout . flags & PPS_TIME_INVALID )
err = wait_event_interruptible ( pps - > queue ,
ev ! = pps - > last_ev ) ;
else {
unsigned long ticks ;
dev_dbg ( pps - > dev , " timeout %lld.%09d \n " ,
( long long ) fdata - > timeout . sec ,
fdata - > timeout . nsec ) ;
ticks = fdata - > timeout . sec * HZ ;
ticks + = fdata - > timeout . nsec / ( NSEC_PER_SEC / HZ ) ;
if ( ticks ! = 0 ) {
err = wait_event_interruptible_timeout (
pps - > queue ,
ev ! = pps - > last_ev ,
ticks ) ;
if ( err = = 0 )
return - ETIMEDOUT ;
}
}
/* Check for pending signals */
if ( err = = - ERESTARTSYS ) {
dev_dbg ( pps - > dev , " pending signal caught \n " ) ;
return - EINTR ;
}
return 0 ;
}
2009-06-18 03:28:37 +04:00
static long pps_cdev_ioctl ( struct file * file ,
unsigned int cmd , unsigned long arg )
{
struct pps_device * pps = file - > private_data ;
struct pps_kparams params ;
void __user * uarg = ( void __user * ) arg ;
int __user * iuarg = ( int __user * ) arg ;
int err ;
switch ( cmd ) {
case PPS_GETPARAMS :
2011-01-13 04:00:52 +03:00
dev_dbg ( pps - > dev , " PPS_GETPARAMS \n " ) ;
2009-06-18 03:28:37 +04:00
2009-11-12 01:26:52 +03:00
spin_lock_irq ( & pps - > lock ) ;
/* Get the current parameters */
params = pps - > params ;
spin_unlock_irq ( & pps - > lock ) ;
err = copy_to_user ( uarg , & params , sizeof ( struct pps_kparams ) ) ;
2009-06-18 03:28:37 +04:00
if ( err )
return - EFAULT ;
break ;
case PPS_SETPARAMS :
2011-01-13 04:00:52 +03:00
dev_dbg ( pps - > dev , " PPS_SETPARAMS \n " ) ;
2009-06-18 03:28:37 +04:00
/* Check the capabilities */
if ( ! capable ( CAP_SYS_TIME ) )
return - EPERM ;
err = copy_from_user ( & params , uarg , sizeof ( struct pps_kparams ) ) ;
if ( err )
return - EFAULT ;
if ( ! ( params . mode & ( PPS_CAPTUREASSERT | PPS_CAPTURECLEAR ) ) ) {
2011-01-13 04:00:52 +03:00
dev_dbg ( pps - > dev , " capture mode unspecified (%x) \n " ,
2009-06-18 03:28:37 +04:00
params . mode ) ;
return - EINVAL ;
}
/* Check for supported capabilities */
if ( ( params . mode & ~ pps - > info . mode ) ! = 0 ) {
2011-01-13 04:00:52 +03:00
dev_dbg ( pps - > dev , " unsupported capabilities (%x) \n " ,
2009-06-18 03:28:37 +04:00
params . mode ) ;
return - EINVAL ;
}
spin_lock_irq ( & pps - > lock ) ;
/* Save the new parameters */
pps - > params = params ;
/* Restore the read only parameters */
if ( ( params . mode & ( PPS_TSFMT_TSPEC | PPS_TSFMT_NTPFP ) ) = = 0 ) {
/* section 3.3 of RFC 2783 interpreted */
2011-01-13 04:00:52 +03:00
dev_dbg ( pps - > dev , " time format unspecified (%x) \n " ,
2009-06-18 03:28:37 +04:00
params . mode ) ;
pps - > params . mode | = PPS_TSFMT_TSPEC ;
}
if ( pps - > info . mode & PPS_CANWAIT )
pps - > params . mode | = PPS_CANWAIT ;
pps - > params . api_version = PPS_API_VERS ;
2019-07-17 02:30:09 +03:00
/*
* Clear unused fields of pps_kparams to avoid leaking
* uninitialized data of the PPS_SETPARAMS caller via
* PPS_GETPARAMS
*/
pps - > params . assert_off_tu . flags = 0 ;
pps - > params . clear_off_tu . flags = 0 ;
2009-06-18 03:28:37 +04:00
spin_unlock_irq ( & pps - > lock ) ;
break ;
case PPS_GETCAP :
2011-01-13 04:00:52 +03:00
dev_dbg ( pps - > dev , " PPS_GETCAP \n " ) ;
2009-06-18 03:28:37 +04:00
err = put_user ( pps - > info . mode , iuarg ) ;
if ( err )
return - EFAULT ;
break ;
2011-01-13 04:00:49 +03:00
case PPS_FETCH : {
struct pps_fdata fdata ;
2011-01-13 04:00:52 +03:00
dev_dbg ( pps - > dev , " PPS_FETCH \n " ) ;
2009-06-18 03:28:37 +04:00
err = copy_from_user ( & fdata , uarg , sizeof ( struct pps_fdata ) ) ;
if ( err )
return - EFAULT ;
2017-03-11 02:19:45 +03:00
err = pps_cdev_pps_fetch ( pps , & fdata ) ;
if ( err )
return err ;
2009-06-18 03:28:37 +04:00
/* Return the fetched timestamp */
spin_lock_irq ( & pps - > lock ) ;
fdata . info . assert_sequence = pps - > assert_sequence ;
fdata . info . clear_sequence = pps - > clear_sequence ;
fdata . info . assert_tu = pps - > assert_tu ;
fdata . info . clear_tu = pps - > clear_tu ;
fdata . info . current_mode = pps - > current_mode ;
spin_unlock_irq ( & pps - > lock ) ;
err = copy_to_user ( uarg , & fdata , sizeof ( struct pps_fdata ) ) ;
if ( err )
return - EFAULT ;
break ;
2011-01-13 04:00:49 +03:00
}
2011-01-13 04:00:58 +03:00
case PPS_KC_BIND : {
struct pps_bind_args bind_args ;
dev_dbg ( pps - > dev , " PPS_KC_BIND \n " ) ;
/* Check the capabilities */
if ( ! capable ( CAP_SYS_TIME ) )
return - EPERM ;
if ( copy_from_user ( & bind_args , uarg ,
sizeof ( struct pps_bind_args ) ) )
return - EFAULT ;
/* Check for supported capabilities */
if ( ( bind_args . edge & ~ pps - > info . mode ) ! = 0 ) {
dev_err ( pps - > dev , " unsupported capabilities (%x) \n " ,
bind_args . edge ) ;
return - EINVAL ;
}
/* Validate parameters roughly */
if ( bind_args . tsformat ! = PPS_TSFMT_TSPEC | |
( bind_args . edge & ~ PPS_CAPTUREBOTH ) ! = 0 | |
bind_args . consumer ! = PPS_KC_HARDPPS ) {
dev_err ( pps - > dev , " invalid kernel consumer bind "
" parameters (%x) \n " , bind_args . edge ) ;
return - EINVAL ;
}
err = pps_kc_bind ( pps , & bind_args ) ;
if ( err < 0 )
return err ;
break ;
}
2009-06-18 03:28:37 +04:00
default :
return - ENOTTY ;
}
return 0 ;
}
2017-03-11 02:19:44 +03:00
# ifdef CONFIG_COMPAT
static long pps_cdev_compat_ioctl ( struct file * file ,
unsigned int cmd , unsigned long arg )
{
2017-03-11 02:19:45 +03:00
struct pps_device * pps = file - > private_data ;
void __user * uarg = ( void __user * ) arg ;
2017-03-11 02:19:44 +03:00
cmd = _IOC ( _IOC_DIR ( cmd ) , _IOC_TYPE ( cmd ) , _IOC_NR ( cmd ) , sizeof ( void * ) ) ;
2017-03-11 02:19:45 +03:00
if ( cmd = = PPS_FETCH ) {
struct pps_fdata_compat compat ;
struct pps_fdata fdata ;
int err ;
dev_dbg ( pps - > dev , " PPS_FETCH \n " ) ;
err = copy_from_user ( & compat , uarg , sizeof ( struct pps_fdata_compat ) ) ;
if ( err )
return - EFAULT ;
memcpy ( & fdata . timeout , & compat . timeout ,
sizeof ( struct pps_ktime_compat ) ) ;
err = pps_cdev_pps_fetch ( pps , & fdata ) ;
if ( err )
return err ;
/* Return the fetched timestamp */
spin_lock_irq ( & pps - > lock ) ;
compat . info . assert_sequence = pps - > assert_sequence ;
compat . info . clear_sequence = pps - > clear_sequence ;
compat . info . current_mode = pps - > current_mode ;
memcpy ( & compat . info . assert_tu , & pps - > assert_tu ,
sizeof ( struct pps_ktime_compat ) ) ;
memcpy ( & compat . info . clear_tu , & pps - > clear_tu ,
sizeof ( struct pps_ktime_compat ) ) ;
spin_unlock_irq ( & pps - > lock ) ;
return copy_to_user ( uarg , & compat ,
sizeof ( struct pps_fdata_compat ) ) ? - EFAULT : 0 ;
}
2017-03-11 02:19:44 +03:00
return pps_cdev_ioctl ( file , cmd , arg ) ;
}
# else
# define pps_cdev_compat_ioctl NULL
# endif
2009-06-18 03:28:37 +04:00
static int pps_cdev_open ( struct inode * inode , struct file * file )
{
struct pps_device * pps = container_of ( inode - > i_cdev ,
struct pps_device , cdev ) ;
file - > private_data = pps ;
2013-02-12 11:27:20 +04:00
kobject_get ( & pps - > dev - > kobj ) ;
2009-06-18 03:28:37 +04:00
return 0 ;
}
static int pps_cdev_release ( struct inode * inode , struct file * file )
{
2013-02-12 11:27:20 +04:00
struct pps_device * pps = container_of ( inode - > i_cdev ,
struct pps_device , cdev ) ;
kobject_put ( & pps - > dev - > kobj ) ;
2009-06-18 03:28:37 +04:00
return 0 ;
}
/*
* Char device stuff
*/
static const struct file_operations pps_cdev_fops = {
. owner = THIS_MODULE ,
. llseek = no_llseek ,
. poll = pps_cdev_poll ,
. fasync = pps_cdev_fasync ,
2017-03-11 02:19:44 +03:00
. compat_ioctl = pps_cdev_compat_ioctl ,
2009-06-18 03:28:37 +04:00
. unlocked_ioctl = pps_cdev_ioctl ,
. open = pps_cdev_open ,
. release = pps_cdev_release ,
} ;
2011-01-13 04:00:53 +03:00
static void pps_device_destruct ( struct device * dev )
{
struct pps_device * pps = dev_get_drvdata ( dev ) ;
2013-02-12 11:27:20 +04:00
cdev_del ( & pps - > cdev ) ;
/* Now we can release the ID for re-use */
pr_debug ( " deallocating pps%d \n " , pps - > id ) ;
2011-01-13 04:00:53 +03:00
mutex_lock ( & pps_idr_lock ) ;
2011-01-13 04:00:53 +03:00
idr_remove ( & pps_idr , pps - > id ) ;
2011-01-13 04:00:53 +03:00
mutex_unlock ( & pps_idr_lock ) ;
2011-01-13 04:00:53 +03:00
kfree ( dev ) ;
kfree ( pps ) ;
}
2009-06-18 03:28:37 +04:00
int pps_register_cdev ( struct pps_device * pps )
{
int err ;
2011-01-13 04:00:51 +03:00
dev_t devt ;
2011-01-13 04:00:53 +03:00
mutex_lock ( & pps_idr_lock ) ;
2013-02-28 05:04:38 +04:00
/*
* Get new ID for the new PPS source . After idr_alloc ( ) calling
* the new source will be freely available into the kernel .
2011-01-13 04:00:53 +03:00
*/
2013-02-28 05:04:38 +04:00
err = idr_alloc ( & pps_idr , pps , 0 , PPS_MAX_SOURCES , GFP_KERNEL ) ;
if ( err < 0 ) {
if ( err = = - ENOSPC ) {
pr_err ( " %s: too many PPS sources in the system \n " ,
pps - > info . name ) ;
err = - EBUSY ;
}
goto out_unlock ;
2011-01-13 04:00:53 +03:00
}
2013-02-28 05:04:38 +04:00
pps - > id = err ;
mutex_unlock ( & pps_idr_lock ) ;
2011-01-13 04:00:53 +03:00
2011-01-13 04:00:51 +03:00
devt = MKDEV ( MAJOR ( pps_devt ) , pps - > id ) ;
2009-06-18 03:28:37 +04:00
cdev_init ( & pps - > cdev , & pps_cdev_fops ) ;
pps - > cdev . owner = pps - > info . owner ;
2011-01-13 04:00:51 +03:00
err = cdev_add ( & pps - > cdev , devt , 1 ) ;
2009-06-18 03:28:37 +04:00
if ( err ) {
2011-01-13 04:00:52 +03:00
pr_err ( " %s: failed to add char device %d:%d \n " ,
2009-06-18 03:28:37 +04:00
pps - > info . name , MAJOR ( pps_devt ) , pps - > id ) ;
2011-01-13 04:00:53 +03:00
goto free_idr ;
2009-06-18 03:28:37 +04:00
}
2011-01-13 04:00:51 +03:00
pps - > dev = device_create ( pps_class , pps - > info . dev , devt , pps ,
2009-06-18 03:28:37 +04:00
" pps%d " , pps - > id ) ;
2012-07-31 01:42:51 +04:00
if ( IS_ERR ( pps - > dev ) ) {
err = PTR_ERR ( pps - > dev ) ;
2009-06-18 03:28:37 +04:00
goto del_cdev ;
2012-07-31 01:42:51 +04:00
}
2009-06-18 03:28:37 +04:00
2013-02-12 11:27:20 +04:00
/* Override the release function with our own */
2011-01-13 04:00:53 +03:00
pps - > dev - > release = pps_device_destruct ;
2009-06-18 03:28:37 +04:00
pr_debug ( " source %s got cdev (%d:%d) \n " , pps - > info . name ,
MAJOR ( pps_devt ) , pps - > id ) ;
return 0 ;
del_cdev :
cdev_del ( & pps - > cdev ) ;
2011-01-13 04:00:53 +03:00
free_idr :
2011-01-13 04:00:53 +03:00
mutex_lock ( & pps_idr_lock ) ;
2011-01-13 04:00:53 +03:00
idr_remove ( & pps_idr , pps - > id ) ;
2013-02-28 05:04:38 +04:00
out_unlock :
2011-01-13 04:00:53 +03:00
mutex_unlock ( & pps_idr_lock ) ;
2009-06-18 03:28:37 +04:00
return err ;
}
void pps_unregister_cdev ( struct pps_device * pps )
{
2013-02-12 11:27:20 +04:00
pr_debug ( " unregistering pps%d \n " , pps - > id ) ;
2013-02-10 13:08:32 +04:00
pps - > lookup_cookie = NULL ;
2011-01-13 04:00:51 +03:00
device_destroy ( pps_class , pps - > dev - > devt ) ;
2009-06-18 03:28:37 +04:00
}
2013-02-10 13:08:32 +04:00
/*
* Look up a pps device by magic cookie .
* The cookie is usually a pointer to some enclosing device , but this
* code doesn ' t care ; you should never be dereferencing it .
*
* This is a bit of a kludge that is currently used only by the PPS
* serial line discipline . It may need to be tweaked when a second user
* is found .
*
* There is no function interface for setting the lookup_cookie field .
* It ' s initialized to NULL when the pps device is created , and if a
* client wants to use it , just fill it in afterward .
*
* The cookie is automatically set to NULL in pps_unregister_source ( )
* so that it will not be used again , even if the pps device cannot
* be removed from the idr due to pending references holding the minor
* number in use .
*/
struct pps_device * pps_lookup_dev ( void const * cookie )
{
struct pps_device * pps ;
unsigned id ;
rcu_read_lock ( ) ;
idr_for_each_entry ( & pps_idr , pps , id )
if ( cookie = = pps - > lookup_cookie )
break ;
rcu_read_unlock ( ) ;
return pps ;
}
EXPORT_SYMBOL ( pps_lookup_dev ) ;
2009-06-18 03:28:37 +04:00
/*
* Module stuff
*/
static void __exit pps_exit ( void )
{
class_destroy ( pps_class ) ;
unregister_chrdev_region ( pps_devt , PPS_MAX_SOURCES ) ;
}
static int __init pps_init ( void )
{
int err ;
2023-03-13 21:18:35 +03:00
pps_class = class_create ( " pps " ) ;
2012-03-06 02:59:15 +04:00
if ( IS_ERR ( pps_class ) ) {
2011-01-13 04:00:52 +03:00
pr_err ( " failed to allocate class \n " ) ;
2012-03-06 02:59:15 +04:00
return PTR_ERR ( pps_class ) ;
2009-06-18 03:28:37 +04:00
}
2013-07-25 02:05:19 +04:00
pps_class - > dev_groups = pps_groups ;
2009-06-18 03:28:37 +04:00
err = alloc_chrdev_region ( & pps_devt , 0 , PPS_MAX_SOURCES , " pps " ) ;
if ( err < 0 ) {
2011-01-13 04:00:52 +03:00
pr_err ( " failed to allocate char device region \n " ) ;
2009-06-18 03:28:37 +04:00
goto remove_class ;
}
pr_info ( " LinuxPPS API ver. %d registered \n " , PPS_API_VERS ) ;
pr_info ( " Software ver. %s - Copyright 2005-2007 Rodolfo Giometti "
" <giometti@linux.it> \n " , PPS_VERSION ) ;
return 0 ;
remove_class :
class_destroy ( pps_class ) ;
return err ;
}
subsys_initcall ( pps_init ) ;
module_exit ( pps_exit ) ;
MODULE_AUTHOR ( " Rodolfo Giometti <giometti@linux.it> " ) ;
MODULE_DESCRIPTION ( " LinuxPPS support (RFC 2783) - ver. " PPS_VERSION ) ;
MODULE_LICENSE ( " GPL " ) ;