2014-04-08 21:12:29 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 René Kijewski <rene.kijewski@fu-berlin.de>
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup tests
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Measure the speed of the function in bitarithm.c
|
|
|
|
*
|
|
|
|
* @author René Kijewski <rene.kijewski@fu-berlin.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
2020-10-21 15:59:16 +02:00
|
|
|
#include <assert.h>
|
2019-11-15 09:22:18 +01:00
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdatomic.h>
|
2014-04-08 21:12:29 +02:00
|
|
|
#include <stdio.h>
|
2020-11-23 11:41:12 +01:00
|
|
|
#include <limits.h>
|
2014-04-08 21:12:29 +02:00
|
|
|
|
|
|
|
#include "bitarithm.h"
|
2015-09-03 18:47:43 +02:00
|
|
|
#include "xtimer.h"
|
2014-04-08 21:12:29 +02:00
|
|
|
|
2015-07-06 14:19:18 +02:00
|
|
|
#define TIMEOUT_S (5ul)
|
2017-01-17 16:44:05 +01:00
|
|
|
#define TIMEOUT (TIMEOUT_S * US_PER_SEC)
|
2014-04-08 21:12:29 +02:00
|
|
|
#define PER_ITERATION (4)
|
|
|
|
|
2020-07-19 22:39:06 +02:00
|
|
|
#if ARCH_32_BIT
|
|
|
|
#define TEST_AND_CLEAR_TEST_MASK_0 (0xA2101045UL)
|
|
|
|
#define TEST_AND_CLEAR_TEST_MASK_1 (0x22101044UL)
|
|
|
|
#else
|
|
|
|
#define TEST_AND_CLEAR_TEST_MASK_0 (0x9045UL)
|
|
|
|
#define TEST_AND_CLEAR_TEST_MASK_1 (0x1244UL)
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2019-11-15 09:22:18 +01:00
|
|
|
static atomic_bool done;
|
|
|
|
|
|
|
|
static void callback(void *unused)
|
2014-04-08 21:12:29 +02:00
|
|
|
{
|
2019-11-15 09:22:18 +01:00
|
|
|
(void)unused;
|
|
|
|
atomic_store(&done, true);
|
2014-04-08 21:12:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void run_test(const char *name, unsigned (*test)(unsigned))
|
|
|
|
{
|
|
|
|
unsigned i = 0;
|
|
|
|
unsigned long count = 0;
|
2019-11-15 09:22:18 +01:00
|
|
|
atomic_store(&done, false);
|
2014-04-08 21:12:29 +02:00
|
|
|
|
2019-11-15 09:22:18 +01:00
|
|
|
xtimer_t xtimer = { .callback = callback };
|
2015-09-03 18:47:43 +02:00
|
|
|
xtimer_set(&xtimer, TIMEOUT);
|
|
|
|
|
2014-04-08 21:12:29 +02:00
|
|
|
do {
|
2019-11-15 09:21:24 +01:00
|
|
|
if (++i == UINT_MAX) {
|
|
|
|
/* bitarithm_lsb must not be called with 0, but if all bits of i are
|
|
|
|
* ones, that ~i will be zero. Therefore, we jump back to start
|
|
|
|
* when all bits of i are ones
|
|
|
|
*/
|
2014-04-08 21:12:29 +02:00
|
|
|
i = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (unsigned j = 0; j < PER_ITERATION; ++j) {
|
2014-08-01 12:41:16 +02:00
|
|
|
volatile unsigned r;
|
2014-04-08 21:12:29 +02:00
|
|
|
r = test(i);
|
2014-08-01 12:41:16 +02:00
|
|
|
(void) r;
|
2014-04-08 21:12:29 +02:00
|
|
|
r = test(-1u - i);
|
2014-08-01 12:41:16 +02:00
|
|
|
(void) r;
|
2014-04-08 21:12:29 +02:00
|
|
|
r = test(~i);
|
2014-08-01 12:41:16 +02:00
|
|
|
(void) r;
|
2014-04-08 21:12:29 +02:00
|
|
|
r = test(~(-1u - i));
|
2014-08-01 12:41:16 +02:00
|
|
|
(void) r;
|
2014-04-08 21:12:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
++count;
|
2019-11-15 09:22:18 +01:00
|
|
|
} while (atomic_load(&done) == false);
|
2014-04-08 21:12:29 +02:00
|
|
|
|
|
|
|
printf("+ %s: %lu iterations per second\r\n", name, (4*PER_ITERATION) * count / TIMEOUT_S);
|
|
|
|
}
|
|
|
|
|
2020-07-19 22:39:06 +02:00
|
|
|
static unsigned do_test_and_clear(unsigned state)
|
|
|
|
{
|
|
|
|
uint8_t index = 0;
|
|
|
|
unsigned found = 0;
|
|
|
|
|
|
|
|
while (state) {
|
|
|
|
state = bitarithm_test_and_clear(state, &index);
|
|
|
|
found |= 1 << index;
|
|
|
|
}
|
|
|
|
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void run_test_test_and_clear(void)
|
|
|
|
{
|
|
|
|
unsigned long count = 0;
|
|
|
|
atomic_store(&done, false);
|
|
|
|
|
|
|
|
xtimer_t xtimer = { .callback = callback };
|
|
|
|
xtimer_set(&xtimer, TIMEOUT);
|
|
|
|
|
|
|
|
do {
|
|
|
|
assert(do_test_and_clear(TEST_AND_CLEAR_TEST_MASK_0) == TEST_AND_CLEAR_TEST_MASK_0);
|
|
|
|
assert(do_test_and_clear(TEST_AND_CLEAR_TEST_MASK_1) == TEST_AND_CLEAR_TEST_MASK_1);
|
|
|
|
++count;
|
|
|
|
} while (atomic_load(&done) == false);
|
|
|
|
|
|
|
|
printf("+ %s: %lu iterations per second\r\n", "bitarithm_test_and_clear", 2 * count / TIMEOUT_S);
|
|
|
|
}
|
|
|
|
|
2014-04-08 21:12:29 +02:00
|
|
|
#define run_test(test) run_test(#test, test)
|
|
|
|
|
|
|
|
int main(void)
|
|
|
|
{
|
|
|
|
printf("Start.\r\n");
|
|
|
|
|
|
|
|
run_test(bitarithm_msb);
|
|
|
|
run_test(bitarithm_lsb);
|
|
|
|
run_test(bitarithm_bits_set);
|
2020-07-19 22:39:06 +02:00
|
|
|
run_test_test_and_clear();
|
2014-04-08 21:12:29 +02:00
|
|
|
|
|
|
|
printf("Done.\r\n");
|
|
|
|
return 0;
|
|
|
|
}
|