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

125 lines
2.5 KiB
C
Raw Normal View History

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
* @brief Bitfield auxiliary functions
2015-06-09 12:10:20 +02:00
*
* @author Kaspar Schleiser <kaspar@schleiser.de>
* @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"
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;
}
int bf_get_unset(uint8_t field[], size_t size)
2015-06-09 12:10:20 +02:00
{
int result = -1;
unsigned nbytes = (size + 7) / 8;
2015-06-09 12:10:20 +02:00
unsigned state = irq_disable();
2015-06-09 12:10:20 +02:00
/* skip full bytes */
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;
}
}
irq_restore(state);
return result;
2015-06-09 12:10:20 +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);
}
}
void bf_clear_all(uint8_t field[], size_t size)
{
unsigned bytes = size >> 3;
unsigned bits = size & 0x7;
memset(field, 0, bytes);
if (bits) {
field[bytes] &= ((1U << (8 - bits)) - 1);
2022-11-02 13:55:13 +01:00
}
}
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;
}