2015-12-29 15:34:48 +01:00
|
|
|
/*
|
|
|
|
* BIRD Library -- ID Map
|
|
|
|
*
|
|
|
|
* (c) 2013--2015 Ondrej Zajicek <santiago@crfreenet.org>
|
|
|
|
* (c) 2013--2015 CZ.NIC z.s.p.o.
|
|
|
|
*
|
|
|
|
* Can be freely distributed and used under the terms of the GNU GPL.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include "nest/bird.h"
|
|
|
|
#include "lib/idm.h"
|
|
|
|
#include "lib/resource.h"
|
|
|
|
#include "lib/string.h"
|
|
|
|
|
|
|
|
|
|
|
|
void
|
2019-04-09 13:38:39 +02:00
|
|
|
idm_init(struct idm *m, pool *p, u64 size, u64 max)
|
2015-12-29 15:34:48 +01:00
|
|
|
{
|
|
|
|
m->pos = 0;
|
|
|
|
m->used = 1;
|
|
|
|
m->size = size;
|
2019-04-09 13:38:39 +02:00
|
|
|
m->max = max;
|
2015-12-29 15:34:48 +01:00
|
|
|
m->data = mb_allocz(p, m->size * sizeof(u32));
|
|
|
|
|
|
|
|
/* ID 0 is reserved */
|
|
|
|
m->data[0] = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int u32_cto(uint x) { return ffs(~x) - 1; }
|
|
|
|
|
2019-04-09 13:38:39 +02:00
|
|
|
u64
|
2015-12-29 15:34:48 +01:00
|
|
|
idm_alloc(struct idm *m)
|
|
|
|
{
|
2019-04-09 13:38:39 +02:00
|
|
|
u64 i, j;
|
2015-12-29 15:34:48 +01:00
|
|
|
|
|
|
|
for (i = m->pos; i < m->size; i++)
|
|
|
|
if (m->data[i] != 0xffffffff)
|
|
|
|
goto found;
|
|
|
|
|
2019-04-09 13:38:39 +02:00
|
|
|
/* If we are at least 7/8 full, expand (if we are allowed to) */
|
2019-04-09 22:51:40 +02:00
|
|
|
if ((m->used < m->max) && (m->used > m->size * 28))
|
2015-12-29 15:34:48 +01:00
|
|
|
{
|
|
|
|
m->size *= 2;
|
|
|
|
m->data = mb_realloc(m->data, m->size * sizeof(u32));
|
|
|
|
memset(m->data + i, 0, (m->size - i) * sizeof(u32));
|
|
|
|
goto found;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < m->pos; i++)
|
|
|
|
if (m->data[i] != 0xffffffff)
|
|
|
|
goto found;
|
|
|
|
|
2019-04-09 13:38:39 +02:00
|
|
|
return 0;
|
2015-12-29 15:34:48 +01:00
|
|
|
|
2016-11-08 19:27:58 +01:00
|
|
|
found:
|
2015-12-29 15:34:48 +01:00
|
|
|
m->pos = i;
|
|
|
|
j = u32_cto(m->data[i]);
|
|
|
|
|
2019-04-09 13:38:39 +02:00
|
|
|
u64 id = 32 * i + j;
|
|
|
|
|
|
|
|
ASSERT(id < m->max);
|
|
|
|
|
2015-12-29 15:34:48 +01:00
|
|
|
m->data[i] |= (1 << j);
|
|
|
|
m->used++;
|
2019-04-09 13:38:39 +02:00
|
|
|
return id;
|
2015-12-29 15:34:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2019-04-09 13:38:39 +02:00
|
|
|
idm_free(struct idm *m, u64 id)
|
2015-12-29 15:34:48 +01:00
|
|
|
{
|
2019-04-09 13:38:39 +02:00
|
|
|
u64 i = id / 32;
|
|
|
|
u64 j = id % 32;
|
2015-12-29 15:34:48 +01:00
|
|
|
|
|
|
|
ASSERT((i < m->size) && (m->data[i] & (1 << j)));
|
|
|
|
m->data[i] &= ~(1 << j);
|
|
|
|
m->used--;
|
|
|
|
}
|