|
|
e6f414 |
commit c0e49c708814ec783726fe92202371847703c5ed
|
|
|
e6f414 |
Author: Miroslav Lichvar <mlichvar@redhat.com>
|
|
|
e6f414 |
Date: Mon Nov 12 17:27:58 2018 +0100
|
|
|
e6f414 |
|
|
|
e6f414 |
sysoff: Initialize data for ioctl(PTP_SYS_OFFSET).
|
|
|
e6f414 |
|
|
|
e6f414 |
This fixes valgrind errors.
|
|
|
e6f414 |
|
|
|
e6f414 |
Signed-off-by: Miroslav Lichvar <mlichvar@redhat.com>
|
|
|
e6f414 |
|
|
|
e6f414 |
diff --git a/sysoff.c b/sysoff.c
|
|
|
e6f414 |
index f7b6240..407a01c 100644
|
|
|
e6f414 |
--- a/sysoff.c
|
|
|
e6f414 |
+++ b/sysoff.c
|
|
|
e6f414 |
@@ -18,6 +18,7 @@
|
|
|
e6f414 |
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
e6f414 |
*/
|
|
|
e6f414 |
#include <stdio.h>
|
|
|
e6f414 |
+#include <string.h>
|
|
|
e6f414 |
#include <sys/ioctl.h>
|
|
|
e6f414 |
#include <linux/ptp_clock.h>
|
|
|
e6f414 |
|
|
|
e6f414 |
@@ -76,6 +77,7 @@ int sysoff_measure(int fd, int n_samples,
|
|
|
e6f414 |
int64_t *result, uint64_t *ts, int64_t *delay)
|
|
|
e6f414 |
{
|
|
|
e6f414 |
struct ptp_sys_offset pso;
|
|
|
e6f414 |
+ memset(&pso, 0, sizeof(pso));
|
|
|
e6f414 |
pso.n_samples = n_samples;
|
|
|
e6f414 |
if (ioctl(fd, PTP_SYS_OFFSET, &pso)) {
|
|
|
e6f414 |
perror("ioctl PTP_SYS_OFFSET");
|
|
|
e6f414 |
|
|
|
e6f414 |
commit 93baf34adb81046a5e1c3b9a3e685029f2046993
|
|
|
e6f414 |
Author: Miroslav Lichvar <mlichvar@redhat.com>
|
|
|
e6f414 |
Date: Mon Nov 12 17:27:59 2018 +0100
|
|
|
e6f414 |
|
|
|
e6f414 |
sysoff: Extend API for different sysoff methods.
|
|
|
e6f414 |
|
|
|
e6f414 |
The kernel supports different PTP_SYS_OFFSET* ioctls. Use the sysoff
|
|
|
e6f414 |
enum to allow selecting between them in sysoff_measure().
|
|
|
e6f414 |
|
|
|
e6f414 |
Signed-off-by: Miroslav Lichvar <mlichvar@redhat.com>
|
|
|
e6f414 |
|
|
|
e6f414 |
diff --git a/phc2sys.c b/phc2sys.c
|
|
|
e6f414 |
index 15f8d75..2cd477a 100644
|
|
|
e6f414 |
--- a/phc2sys.c
|
|
|
e6f414 |
+++ b/phc2sys.c
|
|
|
e6f414 |
@@ -74,7 +74,7 @@ struct clock {
|
|
|
e6f414 |
LIST_ENTRY(clock) dst_list;
|
|
|
e6f414 |
clockid_t clkid;
|
|
|
e6f414 |
int phc_index;
|
|
|
e6f414 |
- int sysoff_supported;
|
|
|
e6f414 |
+ int sysoff_method;
|
|
|
e6f414 |
int is_utc;
|
|
|
e6f414 |
int dest_only;
|
|
|
e6f414 |
int state;
|
|
|
e6f414 |
@@ -255,9 +255,8 @@ static struct clock *clock_add(struct node *node, char *device)
|
|
|
e6f414 |
c->servo = servo_add(node, c);
|
|
|
e6f414 |
|
|
|
e6f414 |
if (clkid != CLOCK_INVALID && clkid != CLOCK_REALTIME)
|
|
|
e6f414 |
- c->sysoff_supported = (SYSOFF_SUPPORTED ==
|
|
|
e6f414 |
- sysoff_probe(CLOCKID_TO_FD(clkid),
|
|
|
e6f414 |
- node->phc_readings));
|
|
|
e6f414 |
+ c->sysoff_method = sysoff_probe(CLOCKID_TO_FD(clkid),
|
|
|
e6f414 |
+ node->phc_readings);
|
|
|
e6f414 |
|
|
|
e6f414 |
LIST_INSERT_HEAD(&node->clocks, c, list);
|
|
|
e6f414 |
return c;
|
|
|
e6f414 |
@@ -784,11 +783,12 @@ static int do_loop(struct node *node, int subscriptions)
|
|
|
e6f414 |
continue;
|
|
|
e6f414 |
|
|
|
e6f414 |
if (clock->clkid == CLOCK_REALTIME &&
|
|
|
e6f414 |
- node->master->sysoff_supported) {
|
|
|
e6f414 |
+ node->master->sysoff_method >= 0) {
|
|
|
e6f414 |
/* use sysoff */
|
|
|
e6f414 |
if (sysoff_measure(CLOCKID_TO_FD(node->master->clkid),
|
|
|
e6f414 |
+ node->master->sysoff_method,
|
|
|
e6f414 |
node->phc_readings,
|
|
|
e6f414 |
- &offset, &ts, &delay))
|
|
|
e6f414 |
+ &offset, &ts, &delay) < 0)
|
|
|
e6f414 |
return -1;
|
|
|
e6f414 |
} else {
|
|
|
e6f414 |
/* use phc */
|
|
|
e6f414 |
diff --git a/phc_ctl.c b/phc_ctl.c
|
|
|
e6f414 |
index 4a78a19..b9a9cf4 100644
|
|
|
e6f414 |
--- a/phc_ctl.c
|
|
|
e6f414 |
+++ b/phc_ctl.c
|
|
|
e6f414 |
@@ -367,10 +367,12 @@ static int do_cmp(clockid_t clkid, int cmdc, char *cmdv[])
|
|
|
e6f414 |
struct timespec ts, rta, rtb;
|
|
|
e6f414 |
int64_t sys_offset, delay = 0, offset;
|
|
|
e6f414 |
uint64_t sys_ts;
|
|
|
e6f414 |
+ int method;
|
|
|
e6f414 |
|
|
|
e6f414 |
- if (SYSOFF_SUPPORTED ==
|
|
|
e6f414 |
- sysoff_measure(CLOCKID_TO_FD(clkid),
|
|
|
e6f414 |
- 9, &sys_offset, &sys_ts, &delay)) {
|
|
|
e6f414 |
+ method = sysoff_probe(CLOCKID_TO_FD(clkid), 9);
|
|
|
e6f414 |
+
|
|
|
e6f414 |
+ if (method >= 0 && sysoff_measure(CLOCKID_TO_FD(clkid), method, 9,
|
|
|
e6f414 |
+ &sys_offset, &sys_ts, &delay) >= 0) {
|
|
|
e6f414 |
pr_notice( "offset from CLOCK_REALTIME is %"PRId64"ns\n",
|
|
|
e6f414 |
sys_offset);
|
|
|
e6f414 |
return 0;
|
|
|
e6f414 |
diff --git a/sysoff.c b/sysoff.c
|
|
|
e6f414 |
index 407a01c..f709a9b 100644
|
|
|
e6f414 |
--- a/sysoff.c
|
|
|
e6f414 |
+++ b/sysoff.c
|
|
|
e6f414 |
@@ -73,8 +73,8 @@ static int64_t sysoff_estimate(struct ptp_clock_time *pct, int n_samples,
|
|
|
e6f414 |
return samples[0].offset;
|
|
|
e6f414 |
}
|
|
|
e6f414 |
|
|
|
e6f414 |
-int sysoff_measure(int fd, int n_samples,
|
|
|
e6f414 |
- int64_t *result, uint64_t *ts, int64_t *delay)
|
|
|
e6f414 |
+static int sysoff_basic(int fd, int n_samples,
|
|
|
e6f414 |
+ int64_t *result, uint64_t *ts, int64_t *delay)
|
|
|
e6f414 |
{
|
|
|
e6f414 |
struct ptp_sys_offset pso;
|
|
|
e6f414 |
memset(&pso, 0, sizeof(pso));
|
|
|
e6f414 |
@@ -84,13 +84,24 @@ int sysoff_measure(int fd, int n_samples,
|
|
|
e6f414 |
return SYSOFF_RUN_TIME_MISSING;
|
|
|
e6f414 |
}
|
|
|
e6f414 |
*result = sysoff_estimate(pso.ts, n_samples, ts, delay);
|
|
|
e6f414 |
- return SYSOFF_SUPPORTED;
|
|
|
e6f414 |
+ return SYSOFF_BASIC;
|
|
|
e6f414 |
+}
|
|
|
e6f414 |
+
|
|
|
e6f414 |
+int sysoff_measure(int fd, int method, int n_samples,
|
|
|
e6f414 |
+ int64_t *result, uint64_t *ts, int64_t *delay)
|
|
|
e6f414 |
+{
|
|
|
e6f414 |
+ switch (method) {
|
|
|
e6f414 |
+ case SYSOFF_BASIC:
|
|
|
e6f414 |
+ return sysoff_basic(fd, n_samples, result, ts, delay);
|
|
|
e6f414 |
+ }
|
|
|
e6f414 |
+ return SYSOFF_COMPILE_TIME_MISSING;
|
|
|
e6f414 |
}
|
|
|
e6f414 |
|
|
|
e6f414 |
int sysoff_probe(int fd, int n_samples)
|
|
|
e6f414 |
{
|
|
|
e6f414 |
int64_t junk, delay;
|
|
|
e6f414 |
uint64_t ts;
|
|
|
e6f414 |
+ int i;
|
|
|
e6f414 |
|
|
|
e6f414 |
if (n_samples > PTP_MAX_SAMPLES) {
|
|
|
e6f414 |
fprintf(stderr, "warning: %d exceeds kernel max readings %d\n",
|
|
|
e6f414 |
@@ -99,7 +110,13 @@ int sysoff_probe(int fd, int n_samples)
|
|
|
e6f414 |
return SYSOFF_RUN_TIME_MISSING;
|
|
|
e6f414 |
}
|
|
|
e6f414 |
|
|
|
e6f414 |
- return sysoff_measure(fd, n_samples, &junk, &ts, &delay);
|
|
|
e6f414 |
+ for (i = 0; i < SYSOFF_LAST; i++) {
|
|
|
e6f414 |
+ if (sysoff_measure(fd, i, n_samples, &junk, &ts, &delay) < 0)
|
|
|
e6f414 |
+ continue;
|
|
|
e6f414 |
+ return i;
|
|
|
e6f414 |
+ }
|
|
|
e6f414 |
+
|
|
|
e6f414 |
+ return SYSOFF_RUN_TIME_MISSING;
|
|
|
e6f414 |
}
|
|
|
e6f414 |
|
|
|
e6f414 |
#else /* !PTP_SYS_OFFSET */
|
|
|
e6f414 |
diff --git a/sysoff.h b/sysoff.h
|
|
|
e6f414 |
index cb70265..02ecdfa 100644
|
|
|
e6f414 |
--- a/sysoff.h
|
|
|
e6f414 |
+++ b/sysoff.h
|
|
|
e6f414 |
@@ -21,13 +21,14 @@
|
|
|
e6f414 |
#include <stdint.h>
|
|
|
e6f414 |
|
|
|
e6f414 |
enum {
|
|
|
e6f414 |
- SYSOFF_SUPPORTED,
|
|
|
e6f414 |
- SYSOFF_COMPILE_TIME_MISSING,
|
|
|
e6f414 |
- SYSOFF_RUN_TIME_MISSING,
|
|
|
e6f414 |
+ SYSOFF_COMPILE_TIME_MISSING = -2,
|
|
|
e6f414 |
+ SYSOFF_RUN_TIME_MISSING = -1,
|
|
|
e6f414 |
+ SYSOFF_BASIC,
|
|
|
e6f414 |
+ SYSOFF_LAST,
|
|
|
e6f414 |
};
|
|
|
e6f414 |
|
|
|
e6f414 |
/**
|
|
|
e6f414 |
- * Check to see if the PTP_SYS_OFFSET ioctl is supported.
|
|
|
e6f414 |
+ * Check to see if a PTP_SYS_OFFSET ioctl is supported.
|
|
|
e6f414 |
* @param fd An open file descriptor to a PHC device.
|
|
|
e6f414 |
* @return One of the SYSOFF_ enumeration values.
|
|
|
e6f414 |
*/
|
|
|
e6f414 |
@@ -36,11 +37,12 @@ int sysoff_probe(int fd, int n_samples);
|
|
|
e6f414 |
/**
|
|
|
e6f414 |
* Measure the offset between a PHC and the system time.
|
|
|
e6f414 |
* @param fd An open file descriptor to a PHC device.
|
|
|
e6f414 |
+ * @param method A non-negative SYSOFF_ value returned by sysoff_probe().
|
|
|
e6f414 |
* @param n_samples The number of consecutive readings to make.
|
|
|
e6f414 |
* @param result The estimated offset in nanoseconds.
|
|
|
e6f414 |
* @param ts The system time corresponding to the 'result'.
|
|
|
e6f414 |
* @param delay The delay in reading of the clock in nanoseconds.
|
|
|
e6f414 |
* @return One of the SYSOFF_ enumeration values.
|
|
|
e6f414 |
*/
|
|
|
e6f414 |
-int sysoff_measure(int fd, int n_samples,
|
|
|
e6f414 |
+int sysoff_measure(int fd, int method, int n_samples,
|
|
|
e6f414 |
int64_t *result, uint64_t *ts, int64_t *delay);
|
|
|
e6f414 |
|
|
|
e6f414 |
commit 192b8e315c4585489d7aa7f59683035998805e40
|
|
|
e6f414 |
Author: Miroslav Lichvar <mlichvar@redhat.com>
|
|
|
e6f414 |
Date: Mon Nov 12 17:28:00 2018 +0100
|
|
|
e6f414 |
|
|
|
e6f414 |
sysoff: Add support for PTP_SYS_OFFSET_PRECISE ioctl.
|
|
|
e6f414 |
|
|
|
e6f414 |
This ioctl uses cross timestamping for a more accurate measurement of
|
|
|
e6f414 |
the offset. It is supported on some onboard Intel NICs using the e1000e
|
|
|
e6f414 |
driver and a virtual PHC with the ptp_kvm driver.
|
|
|
e6f414 |
|
|
|
e6f414 |
Signed-off-by: Miroslav Lichvar <mlichvar@redhat.com>
|
|
|
e6f414 |
|
|
|
e6f414 |
diff --git a/sysoff.c b/sysoff.c
|
|
|
e6f414 |
index f709a9b..9f65d95 100644
|
|
|
e6f414 |
--- a/sysoff.c
|
|
|
e6f414 |
+++ b/sysoff.c
|
|
|
e6f414 |
@@ -22,6 +22,7 @@
|
|
|
e6f414 |
#include <sys/ioctl.h>
|
|
|
e6f414 |
#include <linux/ptp_clock.h>
|
|
|
e6f414 |
|
|
|
e6f414 |
+#include "print.h"
|
|
|
e6f414 |
#include "sysoff.h"
|
|
|
e6f414 |
|
|
|
e6f414 |
#define NS_PER_SEC 1000000000LL
|
|
|
e6f414 |
@@ -39,6 +40,23 @@ static struct {
|
|
|
e6f414 |
uint64_t timestamp;
|
|
|
e6f414 |
} samples[PTP_MAX_SAMPLES];
|
|
|
e6f414 |
|
|
|
e6f414 |
+static int sysoff_precise(int fd, int64_t *result, uint64_t *ts)
|
|
|
e6f414 |
+{
|
|
|
e6f414 |
+#ifdef PTP_SYS_OFFSET_PRECISE
|
|
|
e6f414 |
+ struct ptp_sys_offset_precise pso;
|
|
|
e6f414 |
+ memset(&pso, 0, sizeof(pso));
|
|
|
e6f414 |
+ if (ioctl(fd, PTP_SYS_OFFSET_PRECISE, &pso)) {
|
|
|
e6f414 |
+ pr_debug("ioctl PTP_SYS_OFFSET_PRECISE: %m");
|
|
|
e6f414 |
+ return SYSOFF_RUN_TIME_MISSING;
|
|
|
e6f414 |
+ }
|
|
|
e6f414 |
+ *result = pctns(&pso.sys_realtime) - pctns(&pso.device);
|
|
|
e6f414 |
+ *ts = pctns(&pso.sys_realtime);
|
|
|
e6f414 |
+ return SYSOFF_PRECISE;
|
|
|
e6f414 |
+#else
|
|
|
e6f414 |
+ return SYSOFF_COMPILE_TIME_MISSING;
|
|
|
e6f414 |
+#endif
|
|
|
e6f414 |
+}
|
|
|
e6f414 |
+
|
|
|
e6f414 |
static void insertion_sort(int length, int64_t interval, int64_t offset, uint64_t ts)
|
|
|
e6f414 |
{
|
|
|
e6f414 |
int i = length - 1;
|
|
|
e6f414 |
@@ -91,6 +109,9 @@ int sysoff_measure(int fd, int method, int n_samples,
|
|
|
e6f414 |
int64_t *result, uint64_t *ts, int64_t *delay)
|
|
|
e6f414 |
{
|
|
|
e6f414 |
switch (method) {
|
|
|
e6f414 |
+ case SYSOFF_PRECISE:
|
|
|
e6f414 |
+ *delay = 0;
|
|
|
e6f414 |
+ return sysoff_precise(fd, result, ts);
|
|
|
e6f414 |
case SYSOFF_BASIC:
|
|
|
e6f414 |
return sysoff_basic(fd, n_samples, result, ts, delay);
|
|
|
e6f414 |
}
|
|
|
e6f414 |
diff --git a/sysoff.h b/sysoff.h
|
|
|
e6f414 |
index 02ecdfa..37f7353 100644
|
|
|
e6f414 |
--- a/sysoff.h
|
|
|
e6f414 |
+++ b/sysoff.h
|
|
|
e6f414 |
@@ -23,6 +23,7 @@
|
|
|
e6f414 |
enum {
|
|
|
e6f414 |
SYSOFF_COMPILE_TIME_MISSING = -2,
|
|
|
e6f414 |
SYSOFF_RUN_TIME_MISSING = -1,
|
|
|
e6f414 |
+ SYSOFF_PRECISE,
|
|
|
e6f414 |
SYSOFF_BASIC,
|
|
|
e6f414 |
SYSOFF_LAST,
|
|
|
e6f414 |
};
|
|
|
e6f414 |
|
|
|
e6f414 |
commit 68a9011c9d7d859920da339ba59c14dc1d617a45
|
|
|
e6f414 |
Author: Miroslav Lichvar <mlichvar@redhat.com>
|
|
|
e6f414 |
Date: Mon Nov 12 17:28:01 2018 +0100
|
|
|
e6f414 |
|
|
|
e6f414 |
sysoff: Add support for PTP_SYS_OFFSET_EXTENDED ioctl.
|
|
|
e6f414 |
|
|
|
e6f414 |
This is a more accurate variant of the the PTP_SYS_OFFSET ioctl, which
|
|
|
e6f414 |
will probably be supported in future kernel versions.
|
|
|
e6f414 |
|
|
|
e6f414 |
Signed-off-by: Miroslav Lichvar <mlichvar@redhat.com>
|
|
|
e6f414 |
|
|
|
e6f414 |
diff --git a/sysoff.c b/sysoff.c
|
|
|
e6f414 |
index 9f65d95..b993ee9 100644
|
|
|
e6f414 |
--- a/sysoff.c
|
|
|
e6f414 |
+++ b/sysoff.c
|
|
|
e6f414 |
@@ -71,17 +71,23 @@ static void insertion_sort(int length, int64_t interval, int64_t offset, uint64_
|
|
|
e6f414 |
samples[i+1].timestamp = ts;
|
|
|
e6f414 |
}
|
|
|
e6f414 |
|
|
|
e6f414 |
-static int64_t sysoff_estimate(struct ptp_clock_time *pct, int n_samples,
|
|
|
e6f414 |
- uint64_t *ts, int64_t *delay)
|
|
|
e6f414 |
+static int64_t sysoff_estimate(struct ptp_clock_time *pct, int extended,
|
|
|
e6f414 |
+ int n_samples, uint64_t *ts, int64_t *delay)
|
|
|
e6f414 |
{
|
|
|
e6f414 |
int64_t t1, t2, tp;
|
|
|
e6f414 |
int64_t interval, offset;
|
|
|
e6f414 |
int i;
|
|
|
e6f414 |
|
|
|
e6f414 |
for (i = 0; i < n_samples; i++) {
|
|
|
e6f414 |
- t1 = pctns(&pct[2*i]);
|
|
|
e6f414 |
- tp = pctns(&pct[2*i+1]);
|
|
|
e6f414 |
- t2 = pctns(&pct[2*i+2]);
|
|
|
e6f414 |
+ if (extended) {
|
|
|
e6f414 |
+ t1 = pctns(&pct[3*i]);
|
|
|
e6f414 |
+ tp = pctns(&pct[3*i+1]);
|
|
|
e6f414 |
+ t2 = pctns(&pct[3*i+2]);
|
|
|
e6f414 |
+ } else {
|
|
|
e6f414 |
+ t1 = pctns(&pct[2*i]);
|
|
|
e6f414 |
+ tp = pctns(&pct[2*i+1]);
|
|
|
e6f414 |
+ t2 = pctns(&pct[2*i+2]);
|
|
|
e6f414 |
+ }
|
|
|
e6f414 |
interval = t2 - t1;
|
|
|
e6f414 |
offset = (t2 + t1) / 2 - tp;
|
|
|
e6f414 |
insertion_sort(i, interval, offset, (t2 + t1) / 2);
|
|
|
e6f414 |
@@ -91,6 +97,24 @@ static int64_t sysoff_estimate(struct ptp_clock_time *pct, int n_samples,
|
|
|
e6f414 |
return samples[0].offset;
|
|
|
e6f414 |
}
|
|
|
e6f414 |
|
|
|
e6f414 |
+static int sysoff_extended(int fd, int n_samples,
|
|
|
e6f414 |
+ int64_t *result, uint64_t *ts, int64_t *delay)
|
|
|
e6f414 |
+{
|
|
|
e6f414 |
+#ifdef PTP_SYS_OFFSET_EXTENDED
|
|
|
e6f414 |
+ struct ptp_sys_offset_extended pso;
|
|
|
e6f414 |
+ memset(&pso, 0, sizeof(pso));
|
|
|
e6f414 |
+ pso.n_samples = n_samples;
|
|
|
e6f414 |
+ if (ioctl(fd, PTP_SYS_OFFSET_EXTENDED, &pso)) {
|
|
|
e6f414 |
+ pr_debug("ioctl PTP_SYS_OFFSET_EXTENDED: %m");
|
|
|
e6f414 |
+ return SYSOFF_RUN_TIME_MISSING;
|
|
|
e6f414 |
+ }
|
|
|
e6f414 |
+ *result = sysoff_estimate(&pso.ts[0][0], 1, n_samples, ts, delay);
|
|
|
e6f414 |
+ return SYSOFF_EXTENDED;
|
|
|
e6f414 |
+#else
|
|
|
e6f414 |
+ return SYSOFF_COMPILE_TIME_MISSING;
|
|
|
e6f414 |
+#endif
|
|
|
e6f414 |
+}
|
|
|
e6f414 |
+
|
|
|
e6f414 |
static int sysoff_basic(int fd, int n_samples,
|
|
|
e6f414 |
int64_t *result, uint64_t *ts, int64_t *delay)
|
|
|
e6f414 |
{
|
|
|
e6f414 |
@@ -101,7 +125,7 @@ static int sysoff_basic(int fd, int n_samples,
|
|
|
e6f414 |
perror("ioctl PTP_SYS_OFFSET");
|
|
|
e6f414 |
return SYSOFF_RUN_TIME_MISSING;
|
|
|
e6f414 |
}
|
|
|
e6f414 |
- *result = sysoff_estimate(pso.ts, n_samples, ts, delay);
|
|
|
e6f414 |
+ *result = sysoff_estimate(pso.ts, 0, n_samples, ts, delay);
|
|
|
e6f414 |
return SYSOFF_BASIC;
|
|
|
e6f414 |
}
|
|
|
e6f414 |
|
|
|
e6f414 |
@@ -112,6 +136,8 @@ int sysoff_measure(int fd, int method, int n_samples,
|
|
|
e6f414 |
case SYSOFF_PRECISE:
|
|
|
e6f414 |
*delay = 0;
|
|
|
e6f414 |
return sysoff_precise(fd, result, ts);
|
|
|
e6f414 |
+ case SYSOFF_EXTENDED:
|
|
|
e6f414 |
+ return sysoff_extended(fd, n_samples, result, ts, delay);
|
|
|
e6f414 |
case SYSOFF_BASIC:
|
|
|
e6f414 |
return sysoff_basic(fd, n_samples, result, ts, delay);
|
|
|
e6f414 |
}
|
|
|
e6f414 |
diff --git a/sysoff.h b/sysoff.h
|
|
|
e6f414 |
index 37f7353..79d2290 100644
|
|
|
e6f414 |
--- a/sysoff.h
|
|
|
e6f414 |
+++ b/sysoff.h
|
|
|
e6f414 |
@@ -24,6 +24,7 @@ enum {
|
|
|
e6f414 |
SYSOFF_COMPILE_TIME_MISSING = -2,
|
|
|
e6f414 |
SYSOFF_RUN_TIME_MISSING = -1,
|
|
|
e6f414 |
SYSOFF_PRECISE,
|
|
|
e6f414 |
+ SYSOFF_EXTENDED,
|
|
|
e6f414 |
SYSOFF_BASIC,
|
|
|
e6f414 |
SYSOFF_LAST,
|
|
|
e6f414 |
};
|
|
|
e6f414 |
|
|
|
e6f414 |
commit 8142da41b61fb5b9ee4ad8f5ab56adb0447cd37b
|
|
|
e6f414 |
Author: Miroslav Lichvar <mlichvar@redhat.com>
|
|
|
e6f414 |
Date: Mon Nov 12 17:28:02 2018 +0100
|
|
|
e6f414 |
|
|
|
e6f414 |
phc2sys: Use reversed sysoff when synchronizing to system clock.
|
|
|
e6f414 |
|
|
|
e6f414 |
If synchronizing a PHC to the system clock, use one of the
|
|
|
e6f414 |
PTP_SYS_OFFSET ioctls (if supported) to measure the offset between the
|
|
|
e6f414 |
two clocks. Negate the offset and switch the timestamp before passing
|
|
|
e6f414 |
them to the servo.
|
|
|
e6f414 |
|
|
|
e6f414 |
This makes the synchronization between PHC and system clock symmetric.
|
|
|
e6f414 |
|
|
|
e6f414 |
Signed-off-by: Miroslav Lichvar <mlichvar@redhat.com>
|
|
|
e6f414 |
|
|
|
e6f414 |
diff --git a/phc2sys.c b/phc2sys.c
|
|
|
e6f414 |
index 2cd477a..b8f1ea0 100644
|
|
|
e6f414 |
--- a/phc2sys.c
|
|
|
e6f414 |
+++ b/phc2sys.c
|
|
|
e6f414 |
@@ -790,6 +790,16 @@ static int do_loop(struct node *node, int subscriptions)
|
|
|
e6f414 |
node->phc_readings,
|
|
|
e6f414 |
&offset, &ts, &delay) < 0)
|
|
|
e6f414 |
return -1;
|
|
|
e6f414 |
+ } else if (node->master->clkid == CLOCK_REALTIME &&
|
|
|
e6f414 |
+ clock->sysoff_method >= 0) {
|
|
|
e6f414 |
+ /* use reversed sysoff */
|
|
|
e6f414 |
+ if (sysoff_measure(CLOCKID_TO_FD(clock->clkid),
|
|
|
e6f414 |
+ clock->sysoff_method,
|
|
|
e6f414 |
+ node->phc_readings,
|
|
|
e6f414 |
+ &offset, &ts, &delay) < 0)
|
|
|
e6f414 |
+ return -1;
|
|
|
e6f414 |
+ ts += offset;
|
|
|
e6f414 |
+ offset = -offset;
|
|
|
e6f414 |
} else {
|
|
|
e6f414 |
/* use phc */
|
|
|
e6f414 |
if (!read_phc(node->master->clkid, clock->clkid,
|
|
|
e6f414 |
commit e0580929f451e685d92cd10d80b76f39e9b09a97
|
|
|
e6f414 |
Author: Richard Cochran <richardcochran@gmail.com>
|
|
|
e6f414 |
Date: Tue Dec 24 11:09:34 2019 -0800
|
|
|
e6f414 |
|
|
|
e6f414 |
phc2sys: Fix frequency estimation when synchronizing a PHC to the system clock.
|
|
|
e6f414 |
|
|
|
e6f414 |
When synchronizing a PHC to the Linux system clock (CLOCK_REALTIME),
|
|
|
e6f414 |
the phc2sys uses the sysoff method, reversing the master and slave
|
|
|
e6f414 |
roles.
|
|
|
e6f414 |
|
|
|
e6f414 |
The offset between a master clock and a slave clock is given by
|
|
|
e6f414 |
|
|
|
e6f414 |
offset = slave_ts - master_ts,
|
|
|
e6f414 |
|
|
|
e6f414 |
and the call to sysoff_measure() provides the 'offset' and 'slave_ts'
|
|
|
e6f414 |
values. The needed local time stamp on the 'master' is given by
|
|
|
e6f414 |
|
|
|
e6f414 |
master_ts = slave_ts - offset,
|
|
|
e6f414 |
|
|
|
e6f414 |
but the code calcuates
|
|
|
e6f414 |
|
|
|
e6f414 |
master_ts = slave_ts + offset.
|
|
|
e6f414 |
|
|
|
e6f414 |
When passed to the servo, the local time stamp is used to estimate the
|
|
|
e6f414 |
frequency offset between the two clocks before starting the main
|
|
|
e6f414 |
synchronization loop. The effect of the bug may be seen with a simple
|
|
|
e6f414 |
test. Here is a sample output with the existing code.
|
|
|
e6f414 |
|
|
|
e6f414 |
$ sudo testptp -d /dev/ptp1 -f 62400000
|
|
|
e6f414 |
frequency adjustment okay
|
|
|
e6f414 |
$ sudo ./phc2sys -m -q -c eth6 -s CLOCK_REALTIME -O0
|
|
|
e6f414 |
phc2sys[90221.239]: eth6 sys offset 191001318 s0 freq -62400000 delay 5547
|
|
|
e6f414 |
phc2sys[90222.239]: eth6 sys offset 253380897 s1 freq +8265884 delay 5507
|
|
|
e6f414 |
phc2sys[90223.239]: eth6 sys offset -8301685 s2 freq -35801 delay 5487
|
|
|
e6f414 |
phc2sys[90224.239]: eth6 sys offset -8297136 s2 freq -2521757 delay 5531
|
|
|
e6f414 |
phc2sys[90225.239]: eth6 sys offset -5806117 s2 freq -2519879 delay 5542
|
|
|
e6f414 |
phc2sys[90226.239]: eth6 sys offset -3317009 s2 freq -1772606 delay 5495
|
|
|
e6f414 |
phc2sys[90227.240]: eth6 sys offset -1575231 s2 freq -1025931 delay 5505
|
|
|
e6f414 |
phc2sys[90228.240]: eth6 sys offset -580249 s2 freq -503518 delay 5524
|
|
|
e6f414 |
phc2sys[90229.240]: eth6 sys offset -107770 s2 freq -205114 delay 5519
|
|
|
e6f414 |
phc2sys[90230.240]: eth6 sys offset 66298 s2 freq -63377 delay 5490
|
|
|
e6f414 |
phc2sys[90230.881]: eth6 sys offset 86942 s2 freq -22844 delay 5495
|
|
|
e6f414 |
|
|
|
e6f414 |
And this is the output with the bug fix in place.
|
|
|
e6f414 |
|
|
|
e6f414 |
$ sudo testptp -d /dev/ptp1 -f 62400000
|
|
|
e6f414 |
frequency adjustment okay
|
|
|
e6f414 |
$ sudo ./phc2sys -m -q -c eth6 -s CLOCK_REALTIME -O0
|
|
|
e6f414 |
phc2sys[90365.624]: eth6 sys offset 311912675 s0 freq -62400000 delay 5490
|
|
|
e6f414 |
phc2sys[90366.624]: eth6 sys offset 374292766 s1 freq -31098 delay 5642
|
|
|
e6f414 |
phc2sys[90367.624]: eth6 sys offset -3825 s2 freq -34923 delay 5617
|
|
|
e6f414 |
phc2sys[90368.625]: eth6 sys offset 6 s2 freq -32240 delay 5564
|
|
|
e6f414 |
phc2sys[90369.625]: eth6 sys offset 1241 s2 freq -31003 delay 5605
|
|
|
e6f414 |
phc2sys[90370.625]: eth6 sys offset 1131 s2 freq -30741 delay 5600
|
|
|
e6f414 |
phc2sys[90371.625]: eth6 sys offset 801 s2 freq -30732 delay 5621
|
|
|
e6f414 |
phc2sys[90372.625]: eth6 sys offset 458 s2 freq -30834 delay 5640
|
|
|
e6f414 |
phc2sys[90373.626]: eth6 sys offset 186 s2 freq -30969 delay 5598
|
|
|
e6f414 |
phc2sys[90374.626]: eth6 sys offset 134 s2 freq -30965 delay 5599
|
|
|
e6f414 |
phc2sys[90375.626]: eth6 sys offset 43 s2 freq -31016 delay 5595
|
|
|
e6f414 |
phc2sys[90375.681]: eth6 sys offset -32 s2 freq -31078 delay 5541
|
|
|
e6f414 |
|
|
|
e6f414 |
This patch fixes the issue by correcting the calculation of the local
|
|
|
e6f414 |
time stamp value.
|
|
|
e6f414 |
|
|
|
e6f414 |
Fixes: 8142da41b61f ("phc2sys: Use reversed sysoff when synchronizing to system clock.")
|
|
|
e6f414 |
Signed-off-by: Richard Cochran <richardcochran@gmail.com>
|
|
|
e6f414 |
Reported-by: Cliff Spradlin <cspradlin@google.com>
|
|
|
e6f414 |
Tested-by: Vladimir Oltean <olteanv@gmail.com>
|
|
|
e6f414 |
|
|
|
e6f414 |
diff --git a/phc2sys.c b/phc2sys.c
|
|
|
e6f414 |
index 28c657a..c0b7b3d 100644
|
|
|
e6f414 |
--- a/phc2sys.c
|
|
|
e6f414 |
+++ b/phc2sys.c
|
|
|
e6f414 |
@@ -770,8 +770,8 @@ static int do_loop(struct node *node, int subscriptions)
|
|
|
e6f414 |
node->phc_readings,
|
|
|
e6f414 |
&offset, &ts, &delay) < 0)
|
|
|
e6f414 |
return -1;
|
|
|
e6f414 |
- ts += offset;
|
|
|
e6f414 |
offset = -offset;
|
|
|
e6f414 |
+ ts += offset;
|
|
|
e6f414 |
} else {
|
|
|
e6f414 |
/* use phc */
|
|
|
e6f414 |
if (!read_phc(node->master->clkid, clock->clkid,
|