ptp4l: Add IPoIB interface support for ptp4l

The current implementation of ptp4l always assumes 6 octets MAC
address, which is correct for Ethernet interfaces but not for IPoIB
interfaces (that have 20 octets MAC), therefore running ptp4l over
IPoIB interface does not function correctly.

In Infiniband, every interface has three identifiers:
GUID, GID, and LID.
The GUID is similar in concept to a MAC address. From RFC4392:
The EUI-64 portion of a GID is referred to as the Global Unique
Identifier (GUID) and is the only persistent identifier of a port.

Therefore, to support IPoIB interfaces, the GUID of the port should
be used instead of the MAC.
This patch checks the interface type before creating the clock identity,
for Infiniband ports, it retrieves the GUID of the port using sysfs
and use it to create the clock identity.

sysfs method was chosen since the GUID is the 6 lsb bytes of
the 20 byte device address, and SIOCGIFHWADDR ioctl call returns
the 14 msb bytes of the device address, so it is not possible to
get the GUID using SIOCGIFHWADDR ioctl call.

[ RC: fixed trivial coding style error, space after switch keyword. ]

Signed-off-by: Feras Daoud <ferasda@mellanox.com>
Reviewed-by: Alex Vesker <valex@mellanox.com>
master
Feras Daoud 2017-08-09 18:27:32 +03:00 committed by Richard Cochran
parent 7c3f9579f0
commit 7546434030
4 changed files with 96 additions and 12 deletions

View File

@ -24,6 +24,7 @@
#include <netpacket/packet.h>
#include <sys/socket.h>
#include <sys/un.h>
#include <net/if_arp.h>
struct address {
socklen_t len;

View File

@ -22,7 +22,13 @@
#include <stdint.h>
#define MAC_LEN 6
#define EUI48 6
#define EUI64 8
#define MAC_LEN EUI48
#define GUID_LEN EUI64
#define GUID_OFFSET 36
typedef uint8_t eth_addr[MAC_LEN];

65
sk.c
View File

@ -159,10 +159,55 @@ failed:
return -1;
}
static int sk_interface_guidaddr(const char *name, unsigned char *guid)
{
char file_name[64], buf[64], addr[8];
FILE *f;
char *err;
int res;
snprintf(file_name, sizeof buf, "/sys/class/net/%s/address", name);
f = fopen(file_name, "r");
if (!f) {
pr_err("failed to open %s: %m", buf);
return -1;
}
/* Set the file position to the beginning of the GUID */
res = fseek(f, GUID_OFFSET, SEEK_SET);
if (res) {
pr_err("fseek failed: %m");
goto error;
}
err = fgets(buf, sizeof buf, f);
if (err == NULL) {
pr_err("fseek failed: %m");
goto error;
}
res = sscanf(buf, "%hhx:%hhx:%hhx:%hhx:%hhx:%hhx:%hhx:%hhx",
&addr[0], &addr[1], &addr[2], &addr[3],
&addr[4], &addr[5], &addr[6], &addr[7]);
if (res != GUID_LEN) {
pr_err("sscanf failed: %m");
goto error;
}
memcpy(guid, addr, GUID_LEN);
fclose(f);
return 0;
error:
fclose(f);
return -1;
}
int sk_interface_macaddr(const char *name, struct address *mac)
{
struct ifreq ifreq;
int err, fd;
int err, fd, type;
memset(&ifreq, 0, sizeof(ifreq));
strncpy(ifreq.ifr_name, name, sizeof(ifreq.ifr_name) - 1);
@ -180,9 +225,23 @@ int sk_interface_macaddr(const char *name, struct address *mac)
return -1;
}
/* Get interface type */
type = ifreq.ifr_hwaddr.sa_family;
switch (type) {
case ARPHRD_INFINIBAND:
err = sk_interface_guidaddr(name, mac->sll.sll_addr);
if (err) {
pr_err("fail to get address using sysfs: %m");
return -1;
}
mac->sll.sll_halen = EUI64;
break;
default:
memcpy(mac->sll.sll_addr, &ifreq.ifr_hwaddr.sa_data, MAC_LEN);
mac->sll.sll_halen = EUI48;
}
mac->sll.sll_family = AF_PACKET;
mac->sll.sll_halen = MAC_LEN;
memcpy(mac->sll.sll_addr, &ifreq.ifr_hwaddr.sa_data, MAC_LEN);
mac->len = sizeof(mac->sll);
close(fd);
return 0;

34
util.c
View File

@ -135,14 +135,32 @@ int generate_clock_identity(struct ClockIdentity *ci, const char *name)
if (sk_interface_macaddr(name, &addr))
return -1;
ci->id[0] = addr.sll.sll_addr[0];
ci->id[1] = addr.sll.sll_addr[1];
ci->id[2] = addr.sll.sll_addr[2];
ci->id[3] = 0xFF;
ci->id[4] = 0xFE;
ci->id[5] = addr.sll.sll_addr[3];
ci->id[6] = addr.sll.sll_addr[4];
ci->id[7] = addr.sll.sll_addr[5];
switch (addr.sll.sll_halen) {
case EUI48:
ci->id[0] = addr.sll.sll_addr[0];
ci->id[1] = addr.sll.sll_addr[1];
ci->id[2] = addr.sll.sll_addr[2];
ci->id[3] = 0xFF;
ci->id[4] = 0xFE;
ci->id[5] = addr.sll.sll_addr[3];
ci->id[6] = addr.sll.sll_addr[4];
ci->id[7] = addr.sll.sll_addr[5];
break;
case EUI64:
ci->id[0] = addr.sll.sll_addr[0];
ci->id[1] = addr.sll.sll_addr[1];
ci->id[2] = addr.sll.sll_addr[2];
ci->id[3] = addr.sll.sll_addr[3];
ci->id[4] = addr.sll.sll_addr[4];
ci->id[5] = addr.sll.sll_addr[5];
ci->id[6] = addr.sll.sll_addr[6];
ci->id[7] = addr.sll.sll_addr[7];
break;
default:
return -1;
}
return 0;
}