aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorRichard Cochran <richardcochran@gmail.com>2012-10-31 02:19:07 -0400
committerDavid S. Miller <davem@davemloft.net>2012-11-01 11:41:34 -0400
commit215b13dd288c2e1e4461c1530a801f5f83e8cd90 (patch)
treef4963077f13ab200a676636fe2d48993e73a2a8a
parenta24006ed12616bde1bbdb26868495906a212d8dc (diff)
ptp: add an ioctl to compare PHC time with system time
This patch adds an ioctl for PTP Hardware Clock (PHC) devices that allows user space to measure the time offset between the PHC and the system clock. Rather than hard coding any kind of estimation algorithm into the kernel, this patch takes the more flexible approach of just delivering an array of raw clock readings. In that way, the user space clock servo may be adapted to new and different hardware clocks. Signed-off-by: Richard Cochran <richardcochran@gmail.com> Acked-by: Jacob Keller <jacob.e.keller@intel.com> Signed-off-by: David S. Miller <davem@davemloft.net>
-rw-r--r--drivers/ptp/ptp_chardev.c32
-rw-r--r--include/uapi/linux/ptp_clock.h14
2 files changed, 46 insertions, 0 deletions
diff --git a/drivers/ptp/ptp_chardev.c b/drivers/ptp/ptp_chardev.c
index e7f301da2902..4f8ae8057a7e 100644
--- a/drivers/ptp/ptp_chardev.c
+++ b/drivers/ptp/ptp_chardev.c
@@ -33,9 +33,13 @@ long ptp_ioctl(struct posix_clock *pc, unsigned int cmd, unsigned long arg)
33{ 33{
34 struct ptp_clock_caps caps; 34 struct ptp_clock_caps caps;
35 struct ptp_clock_request req; 35 struct ptp_clock_request req;
36 struct ptp_sys_offset sysoff;
36 struct ptp_clock *ptp = container_of(pc, struct ptp_clock, clock); 37 struct ptp_clock *ptp = container_of(pc, struct ptp_clock, clock);
37 struct ptp_clock_info *ops = ptp->info; 38 struct ptp_clock_info *ops = ptp->info;
39 struct ptp_clock_time *pct;
40 struct timespec ts;
38 int enable, err = 0; 41 int enable, err = 0;
42 unsigned int i;
39 43
40 switch (cmd) { 44 switch (cmd) {
41 45
@@ -88,6 +92,34 @@ long ptp_ioctl(struct posix_clock *pc, unsigned int cmd, unsigned long arg)
88 err = ops->enable(ops, &req, enable); 92 err = ops->enable(ops, &req, enable);
89 break; 93 break;
90 94
95 case PTP_SYS_OFFSET:
96 if (copy_from_user(&sysoff, (void __user *)arg,
97 sizeof(sysoff))) {
98 err = -EFAULT;
99 break;
100 }
101 if (sysoff.n_samples > PTP_MAX_SAMPLES) {
102 err = -EINVAL;
103 break;
104 }
105 pct = &sysoff.ts[0];
106 for (i = 0; i < sysoff.n_samples; i++) {
107 getnstimeofday(&ts);
108 pct->sec = ts.tv_sec;
109 pct->nsec = ts.tv_nsec;
110 pct++;
111 ptp->info->gettime(ptp->info, &ts);
112 pct->sec = ts.tv_sec;
113 pct->nsec = ts.tv_nsec;
114 pct++;
115 }
116 getnstimeofday(&ts);
117 pct->sec = ts.tv_sec;
118 pct->nsec = ts.tv_nsec;
119 if (copy_to_user((void __user *)arg, &sysoff, sizeof(sysoff)))
120 err = -EFAULT;
121 break;
122
91 default: 123 default:
92 err = -ENOTTY; 124 err = -ENOTTY;
93 break; 125 break;
diff --git a/include/uapi/linux/ptp_clock.h b/include/uapi/linux/ptp_clock.h
index 94e981f810a2..b65c834f83e9 100644
--- a/include/uapi/linux/ptp_clock.h
+++ b/include/uapi/linux/ptp_clock.h
@@ -67,12 +67,26 @@ struct ptp_perout_request {
67 unsigned int rsv[4]; /* Reserved for future use. */ 67 unsigned int rsv[4]; /* Reserved for future use. */
68}; 68};
69 69
70#define PTP_MAX_SAMPLES 25 /* Maximum allowed offset measurement samples. */
71
72struct ptp_sys_offset {
73 unsigned int n_samples; /* Desired number of measurements. */
74 unsigned int rsv[3]; /* Reserved for future use. */
75 /*
76 * Array of interleaved system/phc time stamps. The kernel
77 * will provide 2*n_samples + 1 time stamps, with the last
78 * one as a system time stamp.
79 */
80 struct ptp_clock_time ts[2 * PTP_MAX_SAMPLES + 1];
81};
82
70#define PTP_CLK_MAGIC '=' 83#define PTP_CLK_MAGIC '='
71 84
72#define PTP_CLOCK_GETCAPS _IOR(PTP_CLK_MAGIC, 1, struct ptp_clock_caps) 85#define PTP_CLOCK_GETCAPS _IOR(PTP_CLK_MAGIC, 1, struct ptp_clock_caps)
73#define PTP_EXTTS_REQUEST _IOW(PTP_CLK_MAGIC, 2, struct ptp_extts_request) 86#define PTP_EXTTS_REQUEST _IOW(PTP_CLK_MAGIC, 2, struct ptp_extts_request)
74#define PTP_PEROUT_REQUEST _IOW(PTP_CLK_MAGIC, 3, struct ptp_perout_request) 87#define PTP_PEROUT_REQUEST _IOW(PTP_CLK_MAGIC, 3, struct ptp_perout_request)
75#define PTP_ENABLE_PPS _IOW(PTP_CLK_MAGIC, 4, int) 88#define PTP_ENABLE_PPS _IOW(PTP_CLK_MAGIC, 4, int)
89#define PTP_SYS_OFFSET _IOW(PTP_CLK_MAGIC, 5, struct ptp_sys_offset)
76 90
77struct ptp_extts_event { 91struct ptp_extts_event {
78 struct ptp_clock_time t; /* Time event occured. */ 92 struct ptp_clock_time t; /* Time event occured. */