2014-03-31 14:02:47 +02:00
|
|
|
/*
|
2015-02-08 19:19:22 +01:00
|
|
|
* Copyright (C) 2014 Martine Lenders <mlenders@inf.fu-berlin.de>
|
2015-04-28 23:13:07 +02:00
|
|
|
* Copyright (C) 2015 Eistec AB
|
2014-03-31 14:02:47 +02:00
|
|
|
*
|
2014-07-31 19:45:27 +02:00
|
|
|
* 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.
|
2014-03-31 14:02:47 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <limits.h>
|
|
|
|
|
2014-11-27 22:30:14 +01:00
|
|
|
#include "embUnit.h"
|
2014-03-31 14:02:47 +02:00
|
|
|
|
|
|
|
#include "atomic.h"
|
|
|
|
|
|
|
|
#include "tests-core.h"
|
|
|
|
|
2015-04-28 23:13:07 +02:00
|
|
|
/* Test atomic_set_to_one on a variable set to 0 */
|
|
|
|
static void test_atomic_set_to_one_zero(void)
|
|
|
|
{
|
|
|
|
atomic_int_t res = ATOMIC_INIT(0);
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL_INT(1, atomic_set_to_one(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(1, ATOMIC_VALUE(res));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Test atomic_set_to_one on a variable set to 1 */
|
|
|
|
static void test_atomic_set_to_one_one(void)
|
|
|
|
{
|
|
|
|
atomic_int_t res = ATOMIC_INIT(1);
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, atomic_set_to_one(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(1, ATOMIC_VALUE(res));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Test atomic_set_to_one twice */
|
|
|
|
static void test_atomic_set_to_one_twice(void)
|
|
|
|
{
|
|
|
|
atomic_int_t res = ATOMIC_INIT(0);
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL_INT(1, atomic_set_to_one(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(1, ATOMIC_VALUE(res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, atomic_set_to_one(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(1, ATOMIC_VALUE(res));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Test atomic_set_to_zero on a variable set to 0 */
|
|
|
|
static void test_atomic_set_to_zero_zero(void)
|
|
|
|
{
|
|
|
|
atomic_int_t res = ATOMIC_INIT(0);
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, atomic_set_to_zero(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, ATOMIC_VALUE(res));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Test atomic_set_to_zero on a variable set to 1 */
|
|
|
|
static void test_atomic_set_to_zero_one(void)
|
|
|
|
{
|
|
|
|
atomic_int_t res = ATOMIC_INIT(1);
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL_INT(1, atomic_set_to_zero(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, ATOMIC_VALUE(res));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Test atomic_set_to_zero twice */
|
|
|
|
static void test_atomic_set_to_zero_twice(void)
|
|
|
|
{
|
|
|
|
atomic_int_t res = ATOMIC_INIT(1);
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL_INT(1, atomic_set_to_zero(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, ATOMIC_VALUE(res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, atomic_set_to_zero(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, ATOMIC_VALUE(res));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Test atomic_inc */
|
|
|
|
static void test_atomic_inc_positive(void)
|
|
|
|
{
|
|
|
|
atomic_int_t res = ATOMIC_INIT(0);
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, atomic_inc(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(1, ATOMIC_VALUE(res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(1, atomic_inc(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(2, ATOMIC_VALUE(res));
|
|
|
|
ATOMIC_VALUE(res) = 0;
|
|
|
|
for (int i = 0; i < 512; ++i) {
|
|
|
|
TEST_ASSERT_EQUAL_INT(i, atomic_inc(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(i + 1, ATOMIC_VALUE(res));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void test_atomic_inc_negative(void)
|
|
|
|
{
|
|
|
|
atomic_int_t res = ATOMIC_INIT(-99);
|
|
|
|
|
|
|
|
for (int i = -99; i < 123; ++i) {
|
|
|
|
TEST_ASSERT_EQUAL_INT(i, atomic_inc(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(i + 1, ATOMIC_VALUE(res));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void test_atomic_inc_rollover(void)
|
|
|
|
{
|
|
|
|
atomic_int_t res = ATOMIC_INIT(INT_MAX - 30);
|
|
|
|
|
|
|
|
for (int i = 0; i < 30; ++i) {
|
|
|
|
TEST_ASSERT_EQUAL_INT(INT_MAX - 30 + i, atomic_inc(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(INT_MAX - 30 + i + 1, ATOMIC_VALUE(res));
|
|
|
|
}
|
|
|
|
TEST_ASSERT_EQUAL_INT(INT_MAX, atomic_inc(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(INT_MIN, ATOMIC_VALUE(res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(INT_MIN, atomic_inc(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(INT_MIN + 1, ATOMIC_VALUE(res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(INT_MIN + 1, atomic_inc(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(INT_MIN + 2, ATOMIC_VALUE(res));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Test atomic_dec */
|
|
|
|
static void test_atomic_dec_negative(void)
|
|
|
|
{
|
|
|
|
atomic_int_t res = ATOMIC_INIT(0);
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, atomic_dec(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(-1, ATOMIC_VALUE(res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(-1, atomic_dec(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(-2, ATOMIC_VALUE(res));
|
|
|
|
ATOMIC_VALUE(res) = 0;
|
|
|
|
for (int i = 0; i < 512; ++i) {
|
|
|
|
TEST_ASSERT_EQUAL_INT(-i, atomic_dec(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(-i - 1, ATOMIC_VALUE(res));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void test_atomic_dec_positive(void)
|
|
|
|
{
|
|
|
|
atomic_int_t res = ATOMIC_INIT(99);
|
|
|
|
|
|
|
|
for (int i = 99; i < -123; --i) {
|
|
|
|
TEST_ASSERT_EQUAL_INT(i, atomic_dec(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(i - 1, ATOMIC_VALUE(res));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void test_atomic_dec_rollover(void)
|
|
|
|
{
|
|
|
|
atomic_int_t res = ATOMIC_INIT(INT_MIN + 30);
|
|
|
|
|
|
|
|
for (int i = 0; i < 30; ++i) {
|
|
|
|
TEST_ASSERT_EQUAL_INT(INT_MIN + 30 - i, atomic_dec(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(INT_MIN + 30 - i - 1, ATOMIC_VALUE(res));
|
|
|
|
}
|
|
|
|
TEST_ASSERT_EQUAL_INT(INT_MIN, atomic_dec(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(INT_MAX, ATOMIC_VALUE(res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(INT_MAX, atomic_dec(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(INT_MAX - 1, ATOMIC_VALUE(res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(INT_MAX - 1, atomic_dec(&res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(INT_MAX - 2, ATOMIC_VALUE(res));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Test atomic_cas with a correct old value */
|
|
|
|
static void test_atomic_cas_same(void)
|
|
|
|
{
|
|
|
|
atomic_int_t res = ATOMIC_INIT(0);
|
|
|
|
|
2015-07-06 00:42:06 +02:00
|
|
|
TEST_ASSERT_EQUAL_INT(1, atomic_cas(&res, 0, 12345));
|
|
|
|
TEST_ASSERT_EQUAL_INT(12345, ATOMIC_VALUE(res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(1, atomic_cas(&res, 12345, -9876));
|
|
|
|
TEST_ASSERT_EQUAL_INT(-9876, ATOMIC_VALUE(res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(1, atomic_cas(&res, -9876, -9876));
|
|
|
|
TEST_ASSERT_EQUAL_INT(-9876, ATOMIC_VALUE(res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(1, atomic_cas(&res, -9876, 0));
|
2015-04-28 23:13:07 +02:00
|
|
|
TEST_ASSERT_EQUAL_INT(0, ATOMIC_VALUE(res));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Test atomic_cas with a non-matching old value */
|
|
|
|
static void test_atomic_cas_diff(void)
|
|
|
|
{
|
|
|
|
atomic_int_t res = ATOMIC_INIT(32767);
|
|
|
|
|
2015-07-06 00:42:06 +02:00
|
|
|
TEST_ASSERT_EQUAL_INT(0, atomic_cas(&res, 22222, 12345));
|
2015-04-28 23:13:07 +02:00
|
|
|
TEST_ASSERT_EQUAL_INT(32767, ATOMIC_VALUE(res));
|
2015-07-06 00:42:06 +02:00
|
|
|
ATOMIC_VALUE(res) = -12345;
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, atomic_cas(&res, 12345, 12345));
|
|
|
|
TEST_ASSERT_EQUAL_INT(-12345, ATOMIC_VALUE(res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, atomic_cas(&res, 12345, 12345));
|
|
|
|
TEST_ASSERT_EQUAL_INT(-12345, ATOMIC_VALUE(res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(0, atomic_cas(&res, 12345, -12345));
|
|
|
|
TEST_ASSERT_EQUAL_INT(-12345, ATOMIC_VALUE(res));
|
2015-04-28 23:13:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Test ATOMIC_VALUE */
|
|
|
|
static void test_atomic_value(void)
|
|
|
|
{
|
|
|
|
atomic_int_t res = ATOMIC_INIT(12345);
|
|
|
|
atomic_int_t *ptr = &res;
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL_INT(12345, ATOMIC_VALUE(res));
|
2015-07-06 00:42:06 +02:00
|
|
|
ATOMIC_VALUE(res) = 24332;
|
|
|
|
TEST_ASSERT_EQUAL_INT(24332, ATOMIC_VALUE(res));
|
|
|
|
TEST_ASSERT_EQUAL_INT(24332, res.value);
|
|
|
|
res.value = 12323;
|
2015-04-28 23:13:07 +02:00
|
|
|
TEST_ASSERT_EQUAL_INT(ATOMIC_VALUE(res), res.value);
|
|
|
|
TEST_ASSERT_EQUAL_INT(ATOMIC_VALUE(*ptr), res.value);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ATOMIC_INIT is implicitly tested by the other tests */
|
|
|
|
|
2014-03-31 14:02:47 +02:00
|
|
|
Test *tests_core_atomic_tests(void)
|
|
|
|
{
|
|
|
|
EMB_UNIT_TESTFIXTURES(fixtures) {
|
2015-04-28 23:13:07 +02:00
|
|
|
new_TestFixture(test_atomic_set_to_one_one),
|
|
|
|
new_TestFixture(test_atomic_set_to_one_zero),
|
|
|
|
new_TestFixture(test_atomic_set_to_one_twice),
|
|
|
|
new_TestFixture(test_atomic_set_to_zero_one),
|
|
|
|
new_TestFixture(test_atomic_set_to_zero_zero),
|
|
|
|
new_TestFixture(test_atomic_set_to_zero_twice),
|
|
|
|
new_TestFixture(test_atomic_inc_positive),
|
|
|
|
new_TestFixture(test_atomic_inc_negative),
|
|
|
|
new_TestFixture(test_atomic_inc_rollover),
|
|
|
|
new_TestFixture(test_atomic_dec_positive),
|
|
|
|
new_TestFixture(test_atomic_dec_negative),
|
|
|
|
new_TestFixture(test_atomic_dec_rollover),
|
|
|
|
new_TestFixture(test_atomic_cas_same),
|
|
|
|
new_TestFixture(test_atomic_cas_diff),
|
|
|
|
new_TestFixture(test_atomic_value),
|
2014-03-31 14:02:47 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
EMB_UNIT_TESTCALLER(core_atomic_tests, NULL, NULL,
|
|
|
|
fixtures);
|
|
|
|
|
|
|
|
return (Test *)&core_atomic_tests;
|
|
|
|
}
|