2018-12-06 18:53:18 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2018 Freie Universität Berlin
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @defgroup net_eui48 IEEE EUI-48 identifier
|
|
|
|
* @ingroup net
|
|
|
|
* @brief Definition and IPv6 IID conversion for IEEE EUI-48 identifiers
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Definition and IPv6 IID conversion for IEEE EUI-48 identifiers
|
|
|
|
*
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef NET_EUI48_H
|
|
|
|
#define NET_EUI48_H
|
|
|
|
|
|
|
|
#include <stdint.h>
|
|
|
|
|
|
|
|
#include "net/eui64.h"
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Data type to represent an EUI-48
|
|
|
|
*/
|
2019-01-18 17:01:57 +01:00
|
|
|
typedef struct {
|
2018-12-06 18:53:18 +01:00
|
|
|
uint8_t uint8[6]; /**< split into 6 8-bit words. */
|
|
|
|
} eui48_t;
|
|
|
|
|
2020-09-30 11:16:47 +02:00
|
|
|
/**
|
|
|
|
* @name EUI-48 bit flags contained in the first octet
|
|
|
|
*
|
|
|
|
* @see IEEE 802-2001 section 9.2
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Locally administered address.
|
|
|
|
*/
|
|
|
|
#define EUI48_LOCAL_FLAG 0x02
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Group type address.
|
|
|
|
*/
|
|
|
|
#define EUI48_GROUP_FLAG 0x01
|
|
|
|
/** @} */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Set the locally administrated bit in the EUI-48 address.
|
|
|
|
*
|
|
|
|
* @see IEEE 802-2001 section 9.2
|
|
|
|
*
|
|
|
|
* @param addr ethernet address
|
|
|
|
*/
|
|
|
|
static inline void eui48_set_local(eui48_t *addr)
|
|
|
|
{
|
|
|
|
addr->uint8[0] |= EUI48_LOCAL_FLAG;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Clear the group address bit to signal the address as individual
|
|
|
|
* address
|
|
|
|
*
|
|
|
|
* @see IEEE 802-2001 section 9.2
|
|
|
|
*
|
|
|
|
* @param addr ethernet address
|
|
|
|
*/
|
|
|
|
static inline void eui48_clear_group(eui48_t *addr)
|
|
|
|
{
|
|
|
|
addr->uint8[0] &= ~EUI48_GROUP_FLAG;
|
|
|
|
}
|
|
|
|
|
2019-01-18 15:22:19 +01:00
|
|
|
/**
|
|
|
|
* @brief Generates an EUI-64 from a 48-bit device address
|
|
|
|
*
|
|
|
|
* @see [RFC 2464, section 4](https://tools.ietf.org/html/rfc2464#section-4)
|
|
|
|
*
|
|
|
|
* @param[out] eui64 the resulting EUI-64.
|
|
|
|
* @param[in] addr a 48-bit device address
|
|
|
|
*/
|
|
|
|
static inline void eui48_to_eui64(eui64_t *eui64, const eui48_t *addr)
|
|
|
|
{
|
|
|
|
eui64->uint8[0] = addr->uint8[0];
|
|
|
|
eui64->uint8[1] = addr->uint8[1];
|
|
|
|
eui64->uint8[2] = addr->uint8[2];
|
|
|
|
eui64->uint8[3] = 0xff;
|
|
|
|
eui64->uint8[4] = 0xfe;
|
|
|
|
eui64->uint8[5] = addr->uint8[3];
|
|
|
|
eui64->uint8[6] = addr->uint8[4];
|
|
|
|
eui64->uint8[7] = addr->uint8[5];
|
|
|
|
}
|
|
|
|
|
2018-12-06 18:53:18 +01:00
|
|
|
/**
|
|
|
|
* @brief Generates an IPv6 interface identifier from a 48-bit device address
|
|
|
|
*
|
2019-01-18 15:22:19 +01:00
|
|
|
* @note The IPv6 IID is derived from the EUI-64 by flipping the U/L bit.
|
2018-12-06 18:53:18 +01:00
|
|
|
* @see [RFC 2464, section 4](https://tools.ietf.org/html/rfc2464#section-4)
|
|
|
|
* @see [RFC 4291, section 2.5.1](https://tools.ietf.org/html/rfc4291#section-2.5.1)
|
|
|
|
*
|
|
|
|
* @param[out] iid the resulting EUI-64.
|
|
|
|
* @param[in] addr a 48-bit device address
|
|
|
|
*/
|
|
|
|
static inline void eui48_to_ipv6_iid(eui64_t *iid, const eui48_t *addr)
|
|
|
|
{
|
2019-01-18 15:25:17 +01:00
|
|
|
eui48_to_eui64(iid, addr);
|
|
|
|
iid->uint8[0] ^= 0x02;
|
2018-12-06 18:53:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Convert a 64-bit IPv6 IID into a EUI-48 device address
|
|
|
|
*
|
|
|
|
* @param[out] addr the resulting EUI-48
|
|
|
|
* @param[in] iid a 64-bit IPv6 interface identifier
|
|
|
|
*/
|
|
|
|
static inline void eui48_from_ipv6_iid(eui48_t *addr, const eui64_t *iid)
|
|
|
|
{
|
|
|
|
addr->uint8[0] = iid->uint8[0] ^ 0x02;
|
|
|
|
addr->uint8[1] = iid->uint8[1];
|
|
|
|
addr->uint8[2] = iid->uint8[2];
|
|
|
|
addr->uint8[3] = iid->uint8[5];
|
|
|
|
addr->uint8[4] = iid->uint8[6];
|
|
|
|
addr->uint8[5] = iid->uint8[7];
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif /* NET_EUI48_H */
|
|
|
|
/** @} */
|