2011-11-13 00:41:20 +08:00
|
|
|
/**
|
2012-03-21 20:57:05 +08:00
|
|
|
* @file ptp4l.c
|
|
|
|
* @brief PTP Boundary Clock main program
|
2011-11-13 00:41:20 +08:00
|
|
|
* @note Copyright (C) 2011 Richard Cochran <richardcochran@gmail.com>
|
|
|
|
*
|
|
|
|
* 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.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#include "clock.h"
|
2011-12-31 16:18:25 +08:00
|
|
|
#include "config.h"
|
2011-11-13 00:41:20 +08:00
|
|
|
#include "print.h"
|
2012-03-17 21:53:10 +08:00
|
|
|
#include "sk.h"
|
2011-11-13 00:41:20 +08:00
|
|
|
#include "transport.h"
|
2012-08-05 20:11:18 +08:00
|
|
|
#include "util.h"
|
2011-11-13 00:41:20 +08:00
|
|
|
|
2012-07-28 04:34:34 +08:00
|
|
|
int assume_two_step;
|
2012-05-03 17:17:34 +08:00
|
|
|
double configured_pi_kp, configured_pi_ki; /*see pi.c*/
|
2012-07-27 17:30:22 +08:00
|
|
|
extern unsigned char ptp_dst_mac[]; /*see raw.c*/
|
|
|
|
extern unsigned char p2p_dst_mac[]; /*see raw.c*/
|
2012-03-21 03:16:58 +08:00
|
|
|
|
2011-11-13 00:41:20 +08:00
|
|
|
static int running = 1;
|
2012-08-21 01:56:29 +08:00
|
|
|
|
|
|
|
static struct config cfg_settings = {
|
|
|
|
.dds = {
|
|
|
|
.slaveOnly = FALSE,
|
|
|
|
.priority1 = 128,
|
|
|
|
.clockQuality.clockClass = 248,
|
|
|
|
.clockQuality.clockAccuracy = 0xfe,
|
|
|
|
.clockQuality.offsetScaledLogVariance = 0xffff,
|
|
|
|
.priority2 = 128,
|
|
|
|
},
|
|
|
|
|
|
|
|
.pod = {
|
|
|
|
.logAnnounceInterval = 1,
|
|
|
|
.logSyncInterval = 0,
|
|
|
|
.logMinDelayReqInterval = 0,
|
|
|
|
.logMinPdelayReqInterval = 0,
|
|
|
|
.announceReceiptTimeout = 3,
|
|
|
|
.transportSpecific = 0,
|
|
|
|
.path_trace_enabled = 0,
|
|
|
|
.follow_up_info = 0,
|
|
|
|
},
|
|
|
|
|
2012-08-21 01:56:35 +08:00
|
|
|
.timestamping = TS_HARDWARE,
|
|
|
|
.dm = DM_E2E,
|
|
|
|
.transport = TRANS_UDP_IPV4,
|
|
|
|
|
2012-08-21 01:56:29 +08:00
|
|
|
.assume_two_step = &assume_two_step,
|
|
|
|
.tx_timestamp_retries = &sk_tx_retries,
|
|
|
|
.rx_timestamp_l2only = &sk_prefer_layer2,
|
|
|
|
.pi_proportional_const = &configured_pi_kp,
|
|
|
|
.pi_integral_const = &configured_pi_ki,
|
|
|
|
.ptp_dst_mac = ptp_dst_mac,
|
|
|
|
.p2p_dst_mac = p2p_dst_mac,
|
|
|
|
};
|
2011-11-13 00:41:20 +08:00
|
|
|
|
|
|
|
static void usage(char *progname)
|
|
|
|
{
|
|
|
|
fprintf(stderr,
|
|
|
|
"\nusage: %s [options]\n\n"
|
2012-04-05 18:04:11 +08:00
|
|
|
" Delay Mechanism (per interface)\n\n"
|
|
|
|
" -A Auto, starting with E2E\n"
|
|
|
|
" -E E2E, delay request-response (default)\n"
|
|
|
|
" -P P2P, peer delay mechanism\n\n"
|
|
|
|
" Network Transport (per interface)\n\n"
|
2011-11-13 00:41:20 +08:00
|
|
|
" -2 IEEE 802.3\n"
|
|
|
|
" -4 UDP IPV4 (default)\n"
|
|
|
|
" -6 UDP IPV6\n\n"
|
|
|
|
" Time Stamping\n\n"
|
2012-08-21 01:56:03 +08:00
|
|
|
" -H HARDWARE (default)\n"
|
|
|
|
" -S SOFTWARE\n"
|
|
|
|
" -L LEGACY HW\n\n"
|
2011-11-13 00:41:20 +08:00
|
|
|
" Other Options\n\n"
|
2011-12-31 16:18:25 +08:00
|
|
|
" -f [file] read configuration from 'file'\n"
|
2011-11-13 00:41:20 +08:00
|
|
|
" -i [dev] interface device to use, for example 'eth0'\n"
|
|
|
|
" (may be specified multiple times)\n"
|
2012-05-10 02:46:16 +08:00
|
|
|
" -p [dev] PTP hardware clock device to use, default auto\n"
|
2012-01-08 18:46:48 +08:00
|
|
|
" (ignored for SOFTWARE/LEGACY HW time stamping)\n"
|
2012-08-21 01:56:03 +08:00
|
|
|
" -s slave only mode (overrides configuration file)\n"
|
|
|
|
" -l [num] set the logging level to 'num'\n"
|
2012-08-21 01:56:08 +08:00
|
|
|
" -q do not print messages to the syslog\n"
|
|
|
|
" -v print messages to stdout\n"
|
2012-08-21 01:56:03 +08:00
|
|
|
" -h prints this message and exits\n"
|
2012-01-08 18:46:48 +08:00
|
|
|
"\n",
|
2012-05-10 02:46:16 +08:00
|
|
|
progname);
|
2011-11-13 00:41:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
2012-05-10 02:46:16 +08:00
|
|
|
char *config = NULL, *req_phc = NULL, *progname;
|
2012-08-21 01:56:24 +08:00
|
|
|
int c, slaveonly = 0;
|
|
|
|
struct interface *iface = cfg_settings.iface;
|
|
|
|
int *nports = &cfg_settings.nports;
|
2012-08-21 01:56:35 +08:00
|
|
|
enum delay_mechanism *dm = &cfg_settings.dm;
|
|
|
|
enum transport_type *transport = &cfg_settings.transport;
|
|
|
|
enum timestamp_type *timestamping = &cfg_settings.timestamping;
|
2011-11-13 00:41:20 +08:00
|
|
|
struct clock *clock;
|
2012-08-21 01:56:29 +08:00
|
|
|
struct defaultDS *ds = &cfg_settings.dds;
|
|
|
|
struct port_defaults *pod = &cfg_settings.pod;
|
2012-05-10 02:46:16 +08:00
|
|
|
int phc_index = -1;
|
2011-11-13 00:41:20 +08:00
|
|
|
|
|
|
|
/* Process the command line arguments. */
|
|
|
|
progname = strrchr(argv[0], '/');
|
|
|
|
progname = progname ? 1+progname : argv[0];
|
2012-08-21 01:56:03 +08:00
|
|
|
while (EOF != (c = getopt(argc, argv, "AEP246HSLf:i:p:sl:qvh"))) {
|
2011-11-13 00:41:20 +08:00
|
|
|
switch (c) {
|
2012-08-21 01:56:03 +08:00
|
|
|
case 'A':
|
2012-08-21 01:56:35 +08:00
|
|
|
*dm = DM_AUTO;
|
2012-08-21 01:56:03 +08:00
|
|
|
break;
|
|
|
|
case 'E':
|
2012-08-21 01:56:35 +08:00
|
|
|
*dm = DM_E2E;
|
2012-08-21 01:56:03 +08:00
|
|
|
break;
|
|
|
|
case 'P':
|
2012-08-21 01:56:35 +08:00
|
|
|
*dm = DM_P2P;
|
2012-08-21 01:56:03 +08:00
|
|
|
break;
|
2011-11-13 00:41:20 +08:00
|
|
|
case '2':
|
2012-08-21 01:56:35 +08:00
|
|
|
*transport = TRANS_IEEE_802_3;
|
2011-11-13 00:41:20 +08:00
|
|
|
break;
|
|
|
|
case '4':
|
2012-08-21 01:56:35 +08:00
|
|
|
*transport = TRANS_UDP_IPV4;
|
2011-11-13 00:41:20 +08:00
|
|
|
break;
|
|
|
|
case '6':
|
2012-08-21 01:56:35 +08:00
|
|
|
*transport = TRANS_UDP_IPV6;
|
2011-11-13 00:41:20 +08:00
|
|
|
break;
|
2012-08-21 01:56:03 +08:00
|
|
|
case 'H':
|
2012-08-21 01:56:35 +08:00
|
|
|
*timestamping = TS_HARDWARE;
|
2012-04-05 18:04:11 +08:00
|
|
|
break;
|
2012-08-21 01:56:03 +08:00
|
|
|
case 'S':
|
2012-08-21 01:56:35 +08:00
|
|
|
*timestamping = TS_SOFTWARE;
|
2012-08-21 01:56:03 +08:00
|
|
|
break;
|
|
|
|
case 'L':
|
2012-08-21 01:56:35 +08:00
|
|
|
*timestamping = TS_LEGACY_HW;
|
2012-04-05 18:04:11 +08:00
|
|
|
break;
|
2011-12-31 16:18:25 +08:00
|
|
|
case 'f':
|
|
|
|
config = optarg;
|
|
|
|
break;
|
2011-11-13 00:41:20 +08:00
|
|
|
case 'i':
|
2012-08-21 01:56:24 +08:00
|
|
|
if (*nports < MAX_PORTS) {
|
|
|
|
iface[*nports].name = optarg;
|
2012-08-21 01:56:35 +08:00
|
|
|
iface[*nports].dm = *dm;
|
|
|
|
iface[*nports].transport = *transport;
|
2012-08-21 01:56:24 +08:00
|
|
|
(*nports)++;
|
2011-11-13 00:41:20 +08:00
|
|
|
} else {
|
|
|
|
fprintf(stderr, "too many interfaces\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
break;
|
2012-08-21 01:56:03 +08:00
|
|
|
case 'p':
|
|
|
|
req_phc = optarg;
|
2011-12-30 18:27:02 +08:00
|
|
|
break;
|
2012-08-21 01:56:03 +08:00
|
|
|
case 's':
|
2011-12-28 02:18:52 +08:00
|
|
|
slaveonly = 1;
|
|
|
|
break;
|
2012-08-21 01:56:03 +08:00
|
|
|
case 'l':
|
|
|
|
print_set_level(atoi(optarg));
|
2011-11-13 00:41:20 +08:00
|
|
|
break;
|
2012-01-08 18:46:48 +08:00
|
|
|
case 'q':
|
2012-08-21 01:56:08 +08:00
|
|
|
print_set_syslog(1);
|
2012-01-08 18:46:48 +08:00
|
|
|
break;
|
|
|
|
case 'v':
|
2012-08-21 01:56:08 +08:00
|
|
|
print_set_verbose(1);
|
2012-01-08 18:46:48 +08:00
|
|
|
break;
|
2011-11-13 00:41:20 +08:00
|
|
|
case 'h':
|
|
|
|
usage(progname);
|
|
|
|
return 0;
|
|
|
|
case '?':
|
|
|
|
usage(progname);
|
|
|
|
return -1;
|
|
|
|
default:
|
|
|
|
usage(progname);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-08-21 01:56:24 +08:00
|
|
|
if (!*nports) {
|
2012-03-27 17:50:54 +08:00
|
|
|
fprintf(stderr, "no interface specified\n");
|
2011-11-13 00:41:20 +08:00
|
|
|
usage(progname);
|
|
|
|
return -1;
|
|
|
|
}
|
2012-05-10 02:46:16 +08:00
|
|
|
|
|
|
|
/* determine PHC Clock index */
|
2012-08-21 01:56:35 +08:00
|
|
|
if (*timestamping == TS_SOFTWARE || *timestamping == TS_LEGACY_HW) {
|
2012-05-10 02:46:16 +08:00
|
|
|
phc_index = -1;
|
|
|
|
} else if (req_phc) {
|
|
|
|
if (1 != sscanf(req_phc, "/dev/ptp%d", &phc_index)) {
|
|
|
|
fprintf(stderr, "bad ptp device string\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
} else if (sk_interface_phc(iface[0].name, &phc_index)) {
|
|
|
|
fprintf(stderr, "get_ts_info not supported\n"
|
|
|
|
"please specify ptp device\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (phc_index >= 0) {
|
2012-07-08 16:44:05 +08:00
|
|
|
pr_info("selected /dev/ptp%d as PTP clock", phc_index);
|
2011-11-13 00:41:20 +08:00
|
|
|
}
|
|
|
|
|
2012-08-21 01:56:29 +08:00
|
|
|
if (generate_clock_identity(&ds->clockIdentity, iface[0].name)) {
|
2011-11-13 00:41:20 +08:00
|
|
|
fprintf(stderr, "failed to generate a clock identity\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2012-03-21 03:09:40 +08:00
|
|
|
if (config && config_read(config, &cfg_settings)) {
|
2011-12-31 16:18:25 +08:00
|
|
|
fprintf(stderr, "failed to read configuration file\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (slaveonly) {
|
2012-08-21 01:56:29 +08:00
|
|
|
ds->slaveOnly = TRUE;
|
|
|
|
ds->clockQuality.clockClass = 255;
|
2011-12-31 16:18:25 +08:00
|
|
|
}
|
|
|
|
|
2012-08-21 01:56:35 +08:00
|
|
|
clock = clock_create(phc_index, iface, *nports, *timestamping, ds, pod);
|
2011-11-13 00:41:20 +08:00
|
|
|
if (!clock) {
|
|
|
|
fprintf(stderr, "failed to create a clock\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (running) {
|
|
|
|
if (clock_poll(clock))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|