mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
307ea6224a
Provide helper function to mangle a EUI-64 into an EUI-48. This can be useful to get predictable, locally unique addreses.
165 lines
4.0 KiB
C
165 lines
4.0 KiB
C
/*
|
|
* 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
|
|
*/
|
|
typedef struct {
|
|
uint8_t uint8[6]; /**< split into 6 8-bit words. */
|
|
} eui48_t;
|
|
|
|
/**
|
|
* @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;
|
|
}
|
|
|
|
/**
|
|
* @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];
|
|
}
|
|
|
|
/**
|
|
* @brief Generates an EUI-48 from a 64-bit device address
|
|
*
|
|
* @warning The resulting EUI-48 is not guaranteed to be unique
|
|
* and, hence, marked as only locally unique.
|
|
*
|
|
* @param[out] eui48 the resulting EUI-48.
|
|
* @param[in] addr a 64-bit device address
|
|
*/
|
|
static inline void eui64_to_eui48(eui48_t *eui48, const eui64_t *addr)
|
|
{
|
|
/* Preserve vendor id */
|
|
eui48->uint8[0] = addr->uint8[0];
|
|
eui48->uint8[1] = addr->uint8[1];
|
|
eui48->uint8[2] = addr->uint8[2];
|
|
|
|
/* Use most volatile bits */
|
|
eui48->uint8[3] = addr->uint8[5];
|
|
eui48->uint8[4] = addr->uint8[6];
|
|
eui48->uint8[5] = addr->uint8[7];
|
|
|
|
/* EUI is only locally unique */
|
|
eui48_set_local(eui48);
|
|
}
|
|
|
|
/**
|
|
* @brief Generates an IPv6 interface identifier from a 48-bit device address
|
|
*
|
|
* @note The IPv6 IID is derived from the EUI-64 by flipping the U/L bit.
|
|
* @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)
|
|
{
|
|
eui48_to_eui64(iid, addr);
|
|
iid->uint8[0] ^= 0x02;
|
|
}
|
|
|
|
/**
|
|
* @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 */
|
|
/** @} */
|