1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/sys/luid/luid.c

151 lines
3.1 KiB
C
Raw Normal View History

2017-01-10 14:44:49 +01:00
/*
* Copyright (C) 2017 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.
*/
/**
* @ingroup sys_luid
2017-01-10 14:44:49 +01:00
* @{
*
* @file
* @brief LUID module implementation
2017-01-10 14:44:49 +01:00
*
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
*
* @}
*/
#include <stdint.h>
#include <string.h>
#include "assert.h"
#include "periph/cpuid.h"
#include "luid.h"
2017-01-10 14:44:49 +01:00
#ifndef MAX
#define MAX(a, b) (((a) > (b)) ? (a) : (b))
#endif
void __attribute__((weak)) luid_base(void *buf, size_t len)
{
memset(buf, LUID_BACKUP_SEED, len);
#if CPUID_LEN
uint8_t *out = buf;
uint8_t cid[CPUID_LEN];
uint8_t sum = 0;
cpuid_get(cid);
/* CPU ID is XORed with LUID_BACKUP_SEED.
* Then FisherYates shuffle is performed to mix
* the bytes, as often large continuous chunks of
* CPU ID are equal on machines with the same CPU.
*/
for (size_t i = 0; i < MAX(len, CPUID_LEN); i++) {
uint8_t tmp, j, k;
j = i % len;
k = i % CPUID_LEN;
/* xor with CPU ID */
out[j] ^= cid[k];
/* get 'random' position */
sum += out[j];
k = sum % (j + 1);
/* shuffle bytes */
tmp = out[j];
out[j] = out[k];
out[k] = tmp;
}
#endif
}
static uint8_t lastused;
2017-01-10 14:44:49 +01:00
void luid_get(void *buf, size_t len)
2017-01-10 14:44:49 +01:00
{
luid_base(buf, len);
2017-01-10 14:44:49 +01:00
((uint8_t *)buf)[0] ^= lastused++;
}
void luid_get_lb(void *buf, size_t len)
{
luid_base(buf, len);
((uint8_t *)buf)[len - 1] ^= lastused++;
}
void luid_custom(void *buf, size_t len, int gen)
2017-01-10 14:44:49 +01:00
{
luid_base(buf, len);
2017-01-10 14:44:49 +01:00
2017-02-02 16:43:12 +01:00
for (size_t i = 0; i < sizeof(gen); i++) {
2017-01-10 14:44:49 +01:00
((uint8_t *)buf)[i % len] ^= ((gen >> (i * 8)) & 0xff);
}
}
void luid_get_short(network_uint16_t *addr)
{
luid_base(addr, sizeof(*addr));
addr->u8[1] ^= lastused++;
/* https://tools.ietf.org/html/rfc4944#section-12 requires the first bit to
* 0 for unicast addresses */
addr->u8[0] &= 0x7F;
}
void luid_get_eui48(eui48_t *addr)
{
luid_base(addr, sizeof(*addr));
addr->uint8[5] ^= lastused++;
eui48_set_local(addr);
eui48_clear_group(addr);
}
void luid_netdev_get_eui48(const netdev_t *netdev, eui48_t *addr)
{
luid_base(addr, sizeof(*addr));
#ifdef MODULE_NETDEV_REGISTER
addr->uint8[4] ^= netdev->type;
addr->uint8[5] ^= netdev->index;
#else
/* we should only get here with gnrc_netif_single */
(void)netdev;
#endif
eui48_set_local(addr);
eui48_clear_group(addr);
}
void luid_get_eui64(eui64_t *addr)
{
luid_base(addr, sizeof(*addr));
addr->uint8[7] ^= lastused++;
eui64_set_local(addr);
eui64_clear_group(addr);
}
void luid_netdev_get_eui64(const netdev_t *netdev, eui64_t *addr)
{
luid_base(addr, sizeof(*addr));
#ifdef MODULE_NETDEV_REGISTER
addr->uint8[6] ^= netdev->type;
addr->uint8[7] ^= netdev->index;
#else
/* we should only get here with gnrc_netif_single */
(void)netdev;
#endif
eui64_set_local(addr);
eui64_clear_group(addr);
}