1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 09:52:45 +01:00
RIOT/sys/include/net/eui48.h
Benjamin Valentin 307ea6224a net/eui48: add eui64_to_eui48()
Provide helper function to mangle a EUI-64 into an EUI-48.
This can be useful to get predictable, locally unique addreses.
2020-09-30 11:18:36 +02:00

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 */
/** @} */