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.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/**
|
2017-02-24 08:57:11 +01:00
|
|
|
|
* @ingroup sys_luid
|
2017-01-10 14:44:49 +01:00
|
|
|
|
* @{
|
|
|
|
|
*
|
|
|
|
|
* @file
|
2017-02-24 08:57:11 +01:00
|
|
|
|
* @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"
|
|
|
|
|
|
2017-02-24 08:57:11 +01:00
|
|
|
|
#include "luid.h"
|
2017-01-10 14:44:49 +01:00
|
|
|
|
|
2019-11-04 21:07:23 +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
|
2020-09-09 14:21:19 +02:00
|
|
|
|
uint8_t *out = buf;
|
2019-11-04 21:07:23 +01:00
|
|
|
|
uint8_t cid[CPUID_LEN];
|
2020-09-09 14:21:19 +02:00
|
|
|
|
uint8_t sum = 0;
|
2019-11-04 21:07:23 +01:00
|
|
|
|
|
|
|
|
|
cpuid_get(cid);
|
2020-09-09 14:21:19 +02:00
|
|
|
|
|
|
|
|
|
/* CPU ID is XORed with LUID_BACKUP_SEED.
|
|
|
|
|
* Then Fisher–Yates shuffle is performed to mix
|
|
|
|
|
* the bytes, as often large continuous chunks of
|
|
|
|
|
* CPU ID are equal on machines with the same CPU.
|
|
|
|
|
*/
|
2019-11-04 21:07:23 +01:00
|
|
|
|
for (size_t i = 0; i < MAX(len, CPUID_LEN); i++) {
|
2020-09-09 14:21:19 +02:00
|
|
|
|
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;
|
2019-11-04 21:07:23 +01:00
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static uint8_t lastused;
|
2017-01-10 14:44:49 +01:00
|
|
|
|
|
2017-02-24 08:57:11 +01:00
|
|
|
|
void luid_get(void *buf, size_t len)
|
2017-01-10 14:44:49 +01:00
|
|
|
|
{
|
2017-02-24 08:57:11 +01:00
|
|
|
|
luid_base(buf, len);
|
2017-01-10 14:44:49 +01:00
|
|
|
|
|
|
|
|
|
((uint8_t *)buf)[0] ^= lastused++;
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-04 16:29:35 +02:00
|
|
|
|
void luid_get_lb(void *buf, size_t len)
|
|
|
|
|
{
|
|
|
|
|
luid_base(buf, len);
|
|
|
|
|
|
|
|
|
|
((uint8_t *)buf)[len - 1] ^= lastused++;
|
|
|
|
|
}
|
|
|
|
|
|
2017-02-24 08:57:11 +01:00
|
|
|
|
void luid_custom(void *buf, size_t len, int gen)
|
2017-01-10 14:44:49 +01:00
|
|
|
|
{
|
2017-02-24 08:57:11 +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);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-04 21:08:23 +01:00
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
|
2020-08-18 10:39:58 +02:00
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-04 21:08:23 +01:00
|
|
|
|
void luid_get_eui64(eui64_t *addr)
|
|
|
|
|
{
|
|
|
|
|
luid_base(addr, sizeof(*addr));
|
|
|
|
|
addr->uint8[7] ^= lastused++;
|
|
|
|
|
|
|
|
|
|
eui64_set_local(addr);
|
|
|
|
|
eui64_clear_group(addr);
|
|
|
|
|
}
|
2020-08-18 10:39:58 +02:00
|
|
|
|
|
|
|
|
|
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);
|
|
|
|
|
}
|