Merge branch 'siocghwtstamp' of git://git.kernel.org/pub/scm/linux/kernel/git/bwh/sfc-next
Ben Hutchings says: ==================== SIOCGHWTSTAMP ioctl 1. Add the SIOCGHWTSTAMP ioctl and update the timestamping documentation. 2. Implement SIOCGHWTSTAMP in most drivers that support SIOCSHWTSTAMP. 3. Add a test program to exercise SIOC{G,S}HWTSTAMP. ==================== Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
@@ -85,7 +85,7 @@ Filled in if SOF_TIMESTAMPING_SYS_HARDWARE is set. Requires support
|
||||
by the network device and will be empty without that support.
|
||||
|
||||
|
||||
SIOCSHWTSTAMP:
|
||||
SIOCSHWTSTAMP, SIOCGHWTSTAMP:
|
||||
|
||||
Hardware time stamping must also be initialized for each device driver
|
||||
that is expected to do hardware time stamping. The parameter is defined in
|
||||
@@ -115,6 +115,10 @@ Only a processes with admin rights may change the configuration. User
|
||||
space is responsible to ensure that multiple processes don't interfere
|
||||
with each other and that the settings are reset.
|
||||
|
||||
Any process can read the actual configuration by passing this
|
||||
structure to ioctl(SIOCGHWTSTAMP) in the same way. However, this has
|
||||
not been implemented in all drivers.
|
||||
|
||||
/* possible values for hwtstamp_config->tx_type */
|
||||
enum {
|
||||
/*
|
||||
@@ -157,7 +161,8 @@ DEVICE IMPLEMENTATION
|
||||
|
||||
A driver which supports hardware time stamping must support the
|
||||
SIOCSHWTSTAMP ioctl and update the supplied struct hwtstamp_config with
|
||||
the actual values as described in the section on SIOCSHWTSTAMP.
|
||||
the actual values as described in the section on SIOCSHWTSTAMP. It
|
||||
should also support SIOCGHWTSTAMP.
|
||||
|
||||
Time stamps for received packets must be stored in the skb. To get a pointer
|
||||
to the shared time stamp structure of the skb call skb_hwtstamps(). Then
|
||||
|
@@ -1 +1,2 @@
|
||||
timestamping
|
||||
hwtstamp_config
|
||||
|
@@ -2,12 +2,13 @@
|
||||
obj- := dummy.o
|
||||
|
||||
# List of programs to build
|
||||
hostprogs-y := timestamping
|
||||
hostprogs-y := timestamping hwtstamp_config
|
||||
|
||||
# Tell kbuild to always build the programs
|
||||
always := $(hostprogs-y)
|
||||
|
||||
HOSTCFLAGS_timestamping.o += -I$(objtree)/usr/include
|
||||
HOSTCFLAGS_hwtstamp_config.o += -I$(objtree)/usr/include
|
||||
|
||||
clean:
|
||||
rm -f timestamping
|
||||
rm -f timestamping hwtstamp_config
|
||||
|
134
Documentation/networking/timestamping/hwtstamp_config.c
Normal file
134
Documentation/networking/timestamping/hwtstamp_config.c
Normal file
@@ -0,0 +1,134 @@
|
||||
/* 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>
|
||||
|
||||
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)(sizeof(tx_types) / sizeof(tx_types[0])))
|
||||
|
||||
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)(sizeof(rx_filters) / sizeof(rx_filters[0])))
|
||||
|
||||
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;
|
||||
}
|
Reference in New Issue
Block a user