// SPDX-License-Identifier: GPL-2.0 /* Test program for SIOC{G,S}HWTSTAMP * Copyright 2013 Solarflare Communications * Author: Ben Hutchings */ #include <errno.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <sys/socket.h> #include <sys/ioctl.h> #include <linux/if.h> #include <linux/net_tstamp.h> #include <linux/sockios.h> #include "kselftest.h" static int lookup_value(const char **names, int size, const char *name) { int value; for (value = 0; value < size; value++) if (names[value] && strcasecmp(names[value], name) == 0) return value; return -1; } static const char * lookup_name(const char **names, int size, int value) { return (value >= 0 && value < size) ? names[value] : NULL; } static void list_names(FILE *f, const char **names, int size) { int value; for (value = 0; value < size; value++) if (names[value]) fprintf(f, " %s\n", names[value]); } static const char *tx_types[] = { #define TX_TYPE(name) [HWTSTAMP_TX_ ## name] = #name TX_TYPE(OFF), TX_TYPE(ON), TX_TYPE(ONESTEP_SYNC) #undef TX_TYPE }; #define N_TX_TYPES ((int)(ARRAY_SIZE(tx_types))) static const char *rx_filters[] = { #define RX_FILTER(name) [HWTSTAMP_FILTER_ ## name] = #name RX_FILTER(NONE), RX_FILTER(ALL), RX_FILTER(SOME), RX_FILTER(PTP_V1_L4_EVENT), RX_FILTER(PTP_V1_L4_SYNC), RX_FILTER(PTP_V1_L4_DELAY_REQ), RX_FILTER(PTP_V2_L4_EVENT), RX_FILTER(PTP_V2_L4_SYNC), RX_FILTER(PTP_V2_L4_DELAY_REQ), RX_FILTER(PTP_V2_L2_EVENT), RX_FILTER(PTP_V2_L2_SYNC), RX_FILTER(PTP_V2_L2_DELAY_REQ), RX_FILTER(PTP_V2_EVENT), RX_FILTER(PTP_V2_SYNC), RX_FILTER(PTP_V2_DELAY_REQ), #undef RX_FILTER }; #define N_RX_FILTERS ((int)(ARRAY_SIZE(rx_filters))) static void usage(void) { fputs("Usage: hwtstamp_config if_name [tx_type rx_filter]\n" "tx_type is any of (case-insensitive):\n", stderr); list_names(stderr, tx_types, N_TX_TYPES); fputs("rx_filter is any of (case-insensitive):\n", stderr); list_names(stderr, rx_filters, N_RX_FILTERS); } int main(int argc, char **argv) { struct ifreq ifr; struct hwtstamp_config config; const char *name; int sock; if ((argc != 2 && argc != 4) || (strlen(argv[1]) >= IFNAMSIZ)) { usage(); return 2; } if (argc == 4) { config.flags = 0; config.tx_type = lookup_value(tx_types, N_TX_TYPES, argv[2]); config.rx_filter = lookup_value(rx_filters, N_RX_FILTERS, argv[3]); if (config.tx_type < 0 || config.rx_filter < 0) { usage(); return 2; } } sock = socket(AF_INET, SOCK_DGRAM, 0); if (sock < 0) { perror("socket"); return 1; } strcpy(ifr.ifr_name, argv[1]); ifr.ifr_data = (caddr_t)&config; if (ioctl(sock, (argc == 2) ? SIOCGHWTSTAMP : SIOCSHWTSTAMP, &ifr)) { perror("ioctl"); return 1; } printf("flags = %#x\n", config.flags); name = lookup_name(tx_types, N_TX_TYPES, config.tx_type); if (name) printf("tx_type = %s\n", name); else printf("tx_type = %d\n", config.tx_type); name = lookup_name(rx_filters, N_RX_FILTERS, config.rx_filter); if (name) printf("rx_filter = %s\n", name); else printf("rx_filter = %d\n", config.rx_filter); return 0; }