2015-06-09 12:10:20 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015 Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
2019-09-14 15:47:10 +02:00
|
|
|
* @brief Bitfield auxiliary functions
|
2015-06-09 12:10:20 +02:00
|
|
|
*
|
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
2021-10-24 13:21:09 +02:00
|
|
|
* @author Benjamin Valentin <benjamin.valentin@ml-pa.com>
|
2015-06-09 12:10:20 +02:00
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdint.h>
|
2022-11-02 13:55:13 +01:00
|
|
|
#include <string.h>
|
2015-06-09 12:10:20 +02:00
|
|
|
#include "bitfield.h"
|
2022-11-10 23:40:03 +01:00
|
|
|
#include "bitarithm.h"
|
2015-06-09 12:10:20 +02:00
|
|
|
#include "irq.h"
|
|
|
|
|
2021-10-24 13:21:09 +02:00
|
|
|
static inline unsigned _skip_bytes(const uint8_t field[], unsigned nbytes, uint8_t byte)
|
|
|
|
{
|
|
|
|
unsigned i = 0;
|
|
|
|
for (unsigned j = 0; (j < nbytes) && (field[j] == byte); j++) {
|
|
|
|
i += 8;
|
|
|
|
}
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
2022-03-01 00:33:51 +01:00
|
|
|
int bf_get_unset(uint8_t field[], size_t size)
|
2015-06-09 12:10:20 +02:00
|
|
|
{
|
|
|
|
int result = -1;
|
2021-10-24 13:21:09 +02:00
|
|
|
unsigned nbytes = (size + 7) / 8;
|
2015-06-09 12:10:20 +02:00
|
|
|
|
2016-03-19 09:25:47 +01:00
|
|
|
unsigned state = irq_disable();
|
2015-06-09 12:10:20 +02:00
|
|
|
|
|
|
|
/* skip full bytes */
|
2021-10-24 13:21:09 +02:00
|
|
|
unsigned i = _skip_bytes(field, nbytes, 0xff);
|
2015-06-09 12:10:20 +02:00
|
|
|
|
|
|
|
for (; i < size; i++) {
|
|
|
|
if (!bf_isset(field, i)) {
|
|
|
|
bf_set(field, i);
|
|
|
|
result = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-19 09:25:47 +01:00
|
|
|
irq_restore(state);
|
2022-03-01 00:33:51 +01:00
|
|
|
return result;
|
2015-06-09 12:10:20 +02:00
|
|
|
}
|
2021-10-24 13:21:09 +02:00
|
|
|
|
|
|
|
int bf_find_first_set(const uint8_t field[], size_t size)
|
|
|
|
{
|
|
|
|
unsigned nbytes = (size + 7) / 8;
|
|
|
|
unsigned i = _skip_bytes(field, nbytes, 0);
|
|
|
|
|
|
|
|
for (; i < size; i++) {
|
|
|
|
if (bf_isset(field, i)) {
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int bf_find_first_unset(const uint8_t field[], size_t size)
|
|
|
|
{
|
|
|
|
unsigned nbytes = (size + 7) / 8;
|
|
|
|
unsigned i = _skip_bytes(field, nbytes, 0xff);
|
|
|
|
|
|
|
|
for (; i < size; i++) {
|
|
|
|
if (!bf_isset(field, i)) {
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
2022-11-02 13:55:13 +01:00
|
|
|
|
|
|
|
void bf_set_all(uint8_t field[], size_t size)
|
|
|
|
{
|
|
|
|
unsigned bytes = size >> 3;
|
|
|
|
unsigned bits = size & 0x7;
|
|
|
|
|
|
|
|
memset(field, 0xff, bytes);
|
|
|
|
if (bits) {
|
|
|
|
field[bytes] = ~((1U << (8 - bits)) - 1);
|
|
|
|
}
|
|
|
|
}
|
2022-11-10 23:40:03 +01:00
|
|
|
|
|
|
|
unsigned bf_popcnt(const uint8_t field[], size_t size)
|
|
|
|
{
|
|
|
|
unsigned bytes = size >> 3;
|
|
|
|
unsigned bits = size & 0x7;
|
|
|
|
unsigned mask = ~((1U << (8 - bits)) - 1);
|
|
|
|
|
|
|
|
unsigned count = 0;
|
|
|
|
for (unsigned i = 0; i < bytes; ++i) {
|
|
|
|
count += bitarithm_bits_set(field[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bits) {
|
|
|
|
count += bitarithm_bits_set(field[bytes] & mask);
|
|
|
|
}
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|