mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
8854255d7a
The tests used the volatile qualifier for two this: 1. Prevent the compiler to optimize out calls to the inline-able functions bitarithm_msb, bitarithm_lsb, bitarithm_bits_set 2. Communication between IRQ context and thread context While the first use is valid, the second is dangerous, see [1], [2], [3], [4]. This commit replaces the second use with C11 atomics, which were explicitly added to the C standard to address this use case. [1]: https://www.kernel.org/doc/html/latest/process/volatile-considered-harmful.html [2]: http://c.isvolatileusefulwiththreads.com/ [3]: https://web.archive.org/web/20181124154026/http://software.intel.com/en-us/blogs/2007/11/30/volatile-almost-useless-for-multi-threaded-programming/ [4]: https://blog.regehr.org/archives/28
100 lines
2.5 KiB
C
100 lines
2.5 KiB
C
/*
|
|
* 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>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
#include <stdint.h>
|
|
#include <stdatomic.h>
|
|
#include <stdio.h>
|
|
|
|
#include "bitarithm.h"
|
|
#include "xtimer.h"
|
|
|
|
#define TIMEOUT_S (5ul)
|
|
#define TIMEOUT (TIMEOUT_S * US_PER_SEC)
|
|
#define PER_ITERATION (4)
|
|
|
|
static atomic_bool done;
|
|
|
|
static void callback(void *unused)
|
|
{
|
|
(void)unused;
|
|
atomic_store(&done, true);
|
|
}
|
|
|
|
static void run_test(const char *name, unsigned (*test)(unsigned))
|
|
{
|
|
unsigned i = 0;
|
|
unsigned long count = 0;
|
|
atomic_store(&done, false);
|
|
|
|
xtimer_t xtimer = { .callback = callback };
|
|
xtimer_set(&xtimer, TIMEOUT);
|
|
|
|
do {
|
|
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
|
|
*/
|
|
i = 1;
|
|
}
|
|
|
|
for (unsigned j = 0; j < PER_ITERATION; ++j) {
|
|
volatile unsigned r;
|
|
r = test(i);
|
|
(void) r;
|
|
r = test(-1u - i);
|
|
(void) r;
|
|
r = test(~i);
|
|
(void) r;
|
|
r = test(~(-1u - i));
|
|
(void) r;
|
|
}
|
|
|
|
++count;
|
|
} while (atomic_load(&done) == false);
|
|
|
|
printf("+ %s: %lu iterations per second\r\n", name, (4*PER_ITERATION) * count / TIMEOUT_S);
|
|
}
|
|
|
|
#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);
|
|
|
|
printf("Done.\r\n");
|
|
return 0;
|
|
}
|