2011-04-22 14:03:08 +04:00
/*
* PTP 1588 clock support - character device implementation .
*
* Copyright ( C ) 2010 OMICRON electronics GmbH
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
# include <linux/module.h>
# include <linux/posix-clock.h>
# include <linux/poll.h>
# include <linux/sched.h>
2012-11-26 05:44:34 +04:00
# include <linux/slab.h>
2016-02-22 14:15:25 +03:00
# include <linux/timekeeping.h>
2011-04-22 14:03:08 +04:00
# include "ptp_private.h"
2014-03-21 01:21:52 +04:00
static int ptp_disable_pinfunc ( struct ptp_clock_info * ops ,
enum ptp_pin_function func , unsigned int chan )
{
struct ptp_clock_request rq ;
int err = 0 ;
memset ( & rq , 0 , sizeof ( rq ) ) ;
switch ( func ) {
case PTP_PF_NONE :
break ;
case PTP_PF_EXTTS :
rq . type = PTP_CLK_REQ_EXTTS ;
rq . extts . index = chan ;
err = ops - > enable ( ops , & rq , 0 ) ;
break ;
case PTP_PF_PEROUT :
rq . type = PTP_CLK_REQ_PEROUT ;
rq . perout . index = chan ;
err = ops - > enable ( ops , & rq , 0 ) ;
break ;
case PTP_PF_PHYSYNC :
break ;
default :
return - EINVAL ;
}
return err ;
}
int ptp_set_pinfunc ( struct ptp_clock * ptp , unsigned int pin ,
enum ptp_pin_function func , unsigned int chan )
{
struct ptp_clock_info * info = ptp - > info ;
struct ptp_pin_desc * pin1 = NULL , * pin2 = & info - > pin_config [ pin ] ;
unsigned int i ;
/* Check to see if any other pin previously had this function. */
for ( i = 0 ; i < info - > n_pins ; i + + ) {
if ( info - > pin_config [ i ] . func = = func & &
info - > pin_config [ i ] . chan = = chan ) {
pin1 = & info - > pin_config [ i ] ;
break ;
}
}
if ( pin1 & & i = = pin )
return 0 ;
/* Check the desired function and channel. */
switch ( func ) {
case PTP_PF_NONE :
break ;
case PTP_PF_EXTTS :
if ( chan > = info - > n_ext_ts )
return - EINVAL ;
break ;
case PTP_PF_PEROUT :
if ( chan > = info - > n_per_out )
return - EINVAL ;
break ;
case PTP_PF_PHYSYNC :
2014-06-27 14:05:33 +04:00
if ( chan ! = 0 )
return - EINVAL ;
2014-03-21 01:21:52 +04:00
default :
return - EINVAL ;
}
if ( info - > verify ( info , pin , func , chan ) ) {
pr_err ( " driver cannot use function %u on pin %u \n " , func , chan ) ;
return - EOPNOTSUPP ;
}
/* Disable whatever function was previously assigned. */
if ( pin1 ) {
ptp_disable_pinfunc ( info , func , chan ) ;
pin1 - > func = PTP_PF_NONE ;
pin1 - > chan = 0 ;
}
ptp_disable_pinfunc ( info , pin2 - > func , pin2 - > chan ) ;
pin2 - > func = func ;
pin2 - > chan = chan ;
return 0 ;
}
2011-04-22 14:03:08 +04:00
int ptp_open ( struct posix_clock * pc , fmode_t fmode )
{
return 0 ;
}
long ptp_ioctl ( struct posix_clock * pc , unsigned int cmd , unsigned long arg )
{
struct ptp_clock_caps caps ;
struct ptp_clock_request req ;
2012-11-26 05:44:35 +04:00
struct ptp_sys_offset * sysoff = NULL ;
2016-02-22 14:15:25 +03:00
struct ptp_sys_offset_precise precise_offset ;
2014-03-21 01:21:52 +04:00
struct ptp_pin_desc pd ;
2011-04-22 14:03:08 +04:00
struct ptp_clock * ptp = container_of ( pc , struct ptp_clock , clock ) ;
struct ptp_clock_info * ops = ptp - > info ;
2012-10-31 10:19:07 +04:00
struct ptp_clock_time * pct ;
2015-03-30 00:11:52 +03:00
struct timespec64 ts ;
2016-02-22 14:15:25 +03:00
struct system_device_crosststamp xtstamp ;
2011-04-22 14:03:08 +04:00
int enable , err = 0 ;
2014-03-21 01:21:52 +04:00
unsigned int i , pin_index ;
2011-04-22 14:03:08 +04:00
switch ( cmd ) {
case PTP_CLOCK_GETCAPS :
memset ( & caps , 0 , sizeof ( caps ) ) ;
caps . max_adj = ptp - > info - > max_adj ;
caps . n_alarm = ptp - > info - > n_alarm ;
caps . n_ext_ts = ptp - > info - > n_ext_ts ;
caps . n_per_out = ptp - > info - > n_per_out ;
caps . pps = ptp - > info - > pps ;
2014-03-21 01:21:52 +04:00
caps . n_pins = ptp - > info - > n_pins ;
2016-02-22 14:15:25 +03:00
caps . cross_timestamping = ptp - > info - > getcrosststamp ! = NULL ;
2011-05-29 23:53:12 +04:00
if ( copy_to_user ( ( void __user * ) arg , & caps , sizeof ( caps ) ) )
err = - EFAULT ;
2011-04-22 14:03:08 +04:00
break ;
case PTP_EXTTS_REQUEST :
if ( copy_from_user ( & req . extts , ( void __user * ) arg ,
sizeof ( req . extts ) ) ) {
err = - EFAULT ;
break ;
}
if ( req . extts . index > = ops - > n_ext_ts ) {
err = - EINVAL ;
break ;
}
req . type = PTP_CLK_REQ_EXTTS ;
enable = req . extts . flags & PTP_ENABLE_FEATURE ? 1 : 0 ;
err = ops - > enable ( ops , & req , enable ) ;
break ;
case PTP_PEROUT_REQUEST :
if ( copy_from_user ( & req . perout , ( void __user * ) arg ,
sizeof ( req . perout ) ) ) {
err = - EFAULT ;
break ;
}
if ( req . perout . index > = ops - > n_per_out ) {
err = - EINVAL ;
break ;
}
req . type = PTP_CLK_REQ_PEROUT ;
enable = req . perout . period . sec | | req . perout . period . nsec ;
err = ops - > enable ( ops , & req , enable ) ;
break ;
case PTP_ENABLE_PPS :
if ( ! capable ( CAP_SYS_TIME ) )
return - EPERM ;
req . type = PTP_CLK_REQ_PPS ;
enable = arg ? 1 : 0 ;
err = ops - > enable ( ops , & req , enable ) ;
break ;
2016-02-22 14:15:25 +03:00
case PTP_SYS_OFFSET_PRECISE :
if ( ! ptp - > info - > getcrosststamp ) {
err = - EOPNOTSUPP ;
break ;
}
err = ptp - > info - > getcrosststamp ( ptp - > info , & xtstamp ) ;
if ( err )
break ;
2016-10-11 16:02:47 +03:00
memset ( & precise_offset , 0 , sizeof ( precise_offset ) ) ;
2016-02-22 14:15:25 +03:00
ts = ktime_to_timespec64 ( xtstamp . device ) ;
precise_offset . device . sec = ts . tv_sec ;
precise_offset . device . nsec = ts . tv_nsec ;
ts = ktime_to_timespec64 ( xtstamp . sys_realtime ) ;
precise_offset . sys_realtime . sec = ts . tv_sec ;
precise_offset . sys_realtime . nsec = ts . tv_nsec ;
ts = ktime_to_timespec64 ( xtstamp . sys_monoraw ) ;
precise_offset . sys_monoraw . sec = ts . tv_sec ;
precise_offset . sys_monoraw . nsec = ts . tv_nsec ;
if ( copy_to_user ( ( void __user * ) arg , & precise_offset ,
sizeof ( precise_offset ) ) )
err = - EFAULT ;
break ;
2012-10-31 10:19:07 +04:00
case PTP_SYS_OFFSET :
2016-05-20 15:21:02 +03:00
sysoff = memdup_user ( ( void __user * ) arg , sizeof ( * sysoff ) ) ;
if ( IS_ERR ( sysoff ) ) {
err = PTR_ERR ( sysoff ) ;
2016-05-26 09:46:22 +03:00
sysoff = NULL ;
2012-10-31 10:19:07 +04:00
break ;
}
2012-11-26 05:44:35 +04:00
if ( sysoff - > n_samples > PTP_MAX_SAMPLES ) {
2012-10-31 10:19:07 +04:00
err = - EINVAL ;
break ;
}
2012-11-26 05:44:35 +04:00
pct = & sysoff - > ts [ 0 ] ;
for ( i = 0 ; i < sysoff - > n_samples ; i + + ) {
2015-03-30 00:11:52 +03:00
getnstimeofday64 ( & ts ) ;
2012-10-31 10:19:07 +04:00
pct - > sec = ts . tv_sec ;
pct - > nsec = ts . tv_nsec ;
pct + + ;
2015-03-30 00:12:13 +03:00
ptp - > info - > gettime64 ( ptp - > info , & ts ) ;
2012-10-31 10:19:07 +04:00
pct - > sec = ts . tv_sec ;
pct - > nsec = ts . tv_nsec ;
pct + + ;
}
2015-03-30 00:11:52 +03:00
getnstimeofday64 ( & ts ) ;
2012-10-31 10:19:07 +04:00
pct - > sec = ts . tv_sec ;
pct - > nsec = ts . tv_nsec ;
2012-11-26 05:44:35 +04:00
if ( copy_to_user ( ( void __user * ) arg , sysoff , sizeof ( * sysoff ) ) )
2012-10-31 10:19:07 +04:00
err = - EFAULT ;
break ;
2014-03-21 01:21:52 +04:00
case PTP_PIN_GETFUNC :
if ( copy_from_user ( & pd , ( void __user * ) arg , sizeof ( pd ) ) ) {
err = - EFAULT ;
break ;
}
pin_index = pd . index ;
if ( pin_index > = ops - > n_pins ) {
err = - EINVAL ;
break ;
}
if ( mutex_lock_interruptible ( & ptp - > pincfg_mux ) )
return - ERESTARTSYS ;
pd = ops - > pin_config [ pin_index ] ;
mutex_unlock ( & ptp - > pincfg_mux ) ;
if ( ! err & & copy_to_user ( ( void __user * ) arg , & pd , sizeof ( pd ) ) )
err = - EFAULT ;
break ;
case PTP_PIN_SETFUNC :
if ( copy_from_user ( & pd , ( void __user * ) arg , sizeof ( pd ) ) ) {
err = - EFAULT ;
break ;
}
pin_index = pd . index ;
if ( pin_index > = ops - > n_pins ) {
err = - EINVAL ;
break ;
}
if ( mutex_lock_interruptible ( & ptp - > pincfg_mux ) )
return - ERESTARTSYS ;
err = ptp_set_pinfunc ( ptp , pin_index , pd . func , pd . chan ) ;
mutex_unlock ( & ptp - > pincfg_mux ) ;
break ;
2011-04-22 14:03:08 +04:00
default :
err = - ENOTTY ;
break ;
}
2012-11-26 05:44:35 +04:00
kfree ( sysoff ) ;
2011-04-22 14:03:08 +04:00
return err ;
}
unsigned int ptp_poll ( struct posix_clock * pc , struct file * fp , poll_table * wait )
{
struct ptp_clock * ptp = container_of ( pc , struct ptp_clock , clock ) ;
poll_wait ( fp , & ptp - > tsev_wq , wait ) ;
return queue_cnt ( & ptp - > tsevq ) ? POLLIN : 0 ;
}
2012-11-26 05:44:34 +04:00
# define EXTTS_BUFSIZE (PTP_BUF_TIMESTAMPS * sizeof(struct ptp_extts_event))
2011-04-22 14:03:08 +04:00
ssize_t ptp_read ( struct posix_clock * pc ,
uint rdflags , char __user * buf , size_t cnt )
{
struct ptp_clock * ptp = container_of ( pc , struct ptp_clock , clock ) ;
struct timestamp_event_queue * queue = & ptp - > tsevq ;
2012-11-26 05:44:34 +04:00
struct ptp_extts_event * event ;
2011-04-22 14:03:08 +04:00
unsigned long flags ;
size_t qcnt , i ;
2012-11-26 05:44:34 +04:00
int result ;
2011-04-22 14:03:08 +04:00
if ( cnt % sizeof ( struct ptp_extts_event ) ! = 0 )
return - EINVAL ;
2012-11-26 05:44:34 +04:00
if ( cnt > EXTTS_BUFSIZE )
cnt = EXTTS_BUFSIZE ;
2011-04-22 14:03:08 +04:00
cnt = cnt / sizeof ( struct ptp_extts_event ) ;
if ( mutex_lock_interruptible ( & ptp - > tsevq_mux ) )
return - ERESTARTSYS ;
if ( wait_event_interruptible ( ptp - > tsev_wq ,
ptp - > defunct | | queue_cnt ( queue ) ) ) {
mutex_unlock ( & ptp - > tsevq_mux ) ;
return - ERESTARTSYS ;
}
2011-05-29 23:54:07 +04:00
if ( ptp - > defunct ) {
mutex_unlock ( & ptp - > tsevq_mux ) ;
2011-04-22 14:03:08 +04:00
return - ENODEV ;
2011-05-29 23:54:07 +04:00
}
2011-04-22 14:03:08 +04:00
2012-11-26 05:44:34 +04:00
event = kmalloc ( EXTTS_BUFSIZE , GFP_KERNEL ) ;
if ( ! event ) {
mutex_unlock ( & ptp - > tsevq_mux ) ;
return - ENOMEM ;
}
2011-04-22 14:03:08 +04:00
spin_lock_irqsave ( & queue - > lock , flags ) ;
qcnt = queue_cnt ( queue ) ;
if ( cnt > qcnt )
cnt = qcnt ;
for ( i = 0 ; i < cnt ; i + + ) {
event [ i ] = queue - > buf [ queue - > head ] ;
queue - > head = ( queue - > head + 1 ) % PTP_MAX_TIMESTAMPS ;
}
spin_unlock_irqrestore ( & queue - > lock , flags ) ;
cnt = cnt * sizeof ( struct ptp_extts_event ) ;
mutex_unlock ( & ptp - > tsevq_mux ) ;
2012-11-26 05:44:34 +04:00
result = cnt ;
2011-05-29 23:54:07 +04:00
if ( copy_to_user ( buf , event , cnt ) )
2012-11-26 05:44:34 +04:00
result = - EFAULT ;
2011-04-22 14:03:08 +04:00
2012-11-26 05:44:34 +04:00
kfree ( event ) ;
return result ;
2011-04-22 14:03:08 +04:00
}