2018-04-10 17:03:59 +02:00
|
|
|
/*
|
|
|
|
* Copyright(c) 2006 to 2018 ADLINK Technology Limited and others
|
|
|
|
*
|
|
|
|
* This program and the accompanying materials are made available under the
|
|
|
|
* terms of the Eclipse Public License v. 2.0 which is available at
|
|
|
|
* http://www.eclipse.org/legal/epl-2.0, or the Eclipse Distribution License
|
|
|
|
* v. 1.0 which is available at
|
|
|
|
* http://www.eclipse.org/org/documents/edl-v10.php.
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: EPL-2.0 OR BSD-3-Clause
|
|
|
|
*/
|
|
|
|
#include <assert.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
|
|
|
|
#include "os/os.h"
|
2018-11-01 19:32:24 +01:00
|
|
|
#include "CUnit/Test.h"
|
2018-04-10 17:03:59 +02:00
|
|
|
|
2018-11-01 19:32:24 +01:00
|
|
|
CU_Init(os_thread_cleanup)
|
2018-04-10 17:03:59 +02:00
|
|
|
{
|
|
|
|
os_osInit();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-11-01 19:32:24 +01:00
|
|
|
CU_Clean(os_thread_cleanup)
|
2018-04-10 17:03:59 +02:00
|
|
|
{
|
|
|
|
os_osExit();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define THREAD_RESET_1 (1<<0)
|
|
|
|
#define THREAD_RESET_2 (1<<1)
|
|
|
|
#define THREAD_RUN_OFFSET (4)
|
|
|
|
#define THREAD_RUN_1 (1<<(THREAD_RUN_OFFSET))
|
|
|
|
#define THREAD_RUN_2 (1<<(THREAD_RUN_OFFSET + 1))
|
|
|
|
|
|
|
|
struct thread_argument {
|
|
|
|
int flags;
|
|
|
|
int pop;
|
|
|
|
int one;
|
|
|
|
int two;
|
|
|
|
int executed;
|
|
|
|
int cancelled;
|
|
|
|
int block;
|
|
|
|
os_mutex *mutex;
|
|
|
|
os_threadId thread;
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct thread_argument *
|
|
|
|
make_thread_argument(
|
|
|
|
int flags, int pop, int one, int two)
|
|
|
|
{
|
|
|
|
struct thread_argument *targ = os_malloc(sizeof(*targ));
|
|
|
|
memset(targ, 0, sizeof(*targ));
|
|
|
|
targ->flags = flags;
|
|
|
|
targ->pop = pop;
|
|
|
|
targ->one = one;
|
|
|
|
targ->two = two;
|
|
|
|
|
|
|
|
return targ;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
reset_one(
|
|
|
|
void *arg)
|
|
|
|
{
|
|
|
|
struct thread_argument *targ = (struct thread_argument *)arg;
|
|
|
|
targ->one = 0;
|
|
|
|
targ->executed++;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
reset_two(
|
|
|
|
void *arg)
|
|
|
|
{
|
|
|
|
struct thread_argument *targ = (struct thread_argument *)arg;
|
|
|
|
targ->two = 0;
|
|
|
|
targ->executed++;
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint32_t
|
|
|
|
thread_main(
|
|
|
|
void *arg)
|
|
|
|
{
|
|
|
|
int pushed = 0;
|
|
|
|
int popped = 0;
|
|
|
|
int execute = 0;
|
|
|
|
struct thread_argument *targ = (struct thread_argument *)arg;
|
|
|
|
|
|
|
|
if (targ->flags & THREAD_RESET_1) {
|
|
|
|
os_threadCleanupPush(&reset_one, arg);
|
|
|
|
pushed++;
|
|
|
|
}
|
|
|
|
if (targ->flags & THREAD_RESET_2) {
|
|
|
|
os_threadCleanupPush(&reset_two, arg);
|
|
|
|
pushed++;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(targ->pop <= pushed);
|
|
|
|
|
|
|
|
if (targ->block) {
|
|
|
|
os_mutexLock(targ->mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (popped < targ->pop) {
|
|
|
|
execute = 1 << (THREAD_RUN_OFFSET + (targ->pop - (popped + 1)));
|
|
|
|
os_threadCleanupPop(targ->flags & execute);
|
|
|
|
targ->cancelled++;
|
|
|
|
popped++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (targ->block) {
|
|
|
|
os_mutexUnlock(targ->mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
setup(
|
|
|
|
struct thread_argument *targ)
|
|
|
|
{
|
|
|
|
os_result res;
|
|
|
|
os_threadId tid;
|
|
|
|
os_threadAttr tattr;
|
|
|
|
|
|
|
|
uint32_t tres = 0;
|
|
|
|
|
|
|
|
os_threadAttrInit(&tattr);
|
|
|
|
res = os_threadCreate(&tid, "", &tattr, &thread_main, (void *)targ);
|
|
|
|
CU_ASSERT_EQUAL_FATAL(res, os_resultSuccess);
|
|
|
|
|
|
|
|
targ->thread = tid;
|
|
|
|
if (!targ->block) {
|
|
|
|
res = os_threadWaitExit(tid, &tres);
|
|
|
|
CU_ASSERT_EQUAL_FATAL(res, os_resultSuccess);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* verify the cleanup routine is called */
|
2018-11-01 19:32:24 +01:00
|
|
|
CU_Test(os_thread_cleanup, push_one)
|
2018-04-10 17:03:59 +02:00
|
|
|
{
|
|
|
|
int flags = THREAD_RESET_1;
|
|
|
|
struct thread_argument *targ = make_thread_argument(flags, 0, 1, 2);
|
|
|
|
setup(targ);
|
|
|
|
|
|
|
|
CU_ASSERT_EQUAL(targ->one, 0);
|
|
|
|
CU_ASSERT_EQUAL(targ->two, 2);
|
|
|
|
CU_ASSERT_EQUAL(targ->executed, 1);
|
|
|
|
CU_ASSERT_EQUAL(targ->cancelled, 0);
|
|
|
|
|
|
|
|
free(targ);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* verify all cleanup routines are called if multiple are registered */
|
2018-11-01 19:32:24 +01:00
|
|
|
CU_Test(os_thread_cleanup, push_two)
|
2018-04-10 17:03:59 +02:00
|
|
|
{
|
|
|
|
int flags = THREAD_RESET_1 | THREAD_RESET_2;
|
|
|
|
struct thread_argument *targ = make_thread_argument(flags, 0, 1, 2);
|
|
|
|
setup(targ);
|
|
|
|
|
|
|
|
CU_ASSERT_EQUAL(targ->one, 0);
|
|
|
|
CU_ASSERT_EQUAL(targ->two, 0);
|
|
|
|
CU_ASSERT_EQUAL(targ->executed, 2);
|
|
|
|
CU_ASSERT_EQUAL(targ->cancelled, 0);
|
|
|
|
|
|
|
|
free(targ);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* verify the first cleanup routine is still called if second got popped */
|
2018-11-01 19:32:24 +01:00
|
|
|
CU_Test(os_thread_cleanup, push_two_pop_one_no_exec)
|
2018-04-10 17:03:59 +02:00
|
|
|
{
|
|
|
|
int flags = THREAD_RESET_1 | THREAD_RESET_2;
|
|
|
|
struct thread_argument *targ = make_thread_argument(flags, 1, 1, 2);
|
|
|
|
setup(targ);
|
|
|
|
|
|
|
|
CU_ASSERT_EQUAL(targ->one, 0);
|
|
|
|
CU_ASSERT_EQUAL(targ->two, 2);
|
|
|
|
CU_ASSERT_EQUAL(targ->executed, 1);
|
|
|
|
CU_ASSERT_EQUAL(targ->cancelled, 1);
|
|
|
|
|
|
|
|
free(targ);
|
|
|
|
}
|
|
|
|
|
2018-11-01 19:32:24 +01:00
|
|
|
CU_Test(os_thread_cleanup, push_two_pop_one_exec)
|
2018-04-10 17:03:59 +02:00
|
|
|
{
|
|
|
|
int flags = THREAD_RESET_1 | THREAD_RESET_2 | THREAD_RUN_1;
|
|
|
|
struct thread_argument *targ = make_thread_argument(flags, 1, 1, 2);
|
|
|
|
setup(targ);
|
|
|
|
|
|
|
|
CU_ASSERT_EQUAL(targ->one, 0);
|
|
|
|
CU_ASSERT_EQUAL(targ->two, 0);
|
|
|
|
CU_ASSERT_EQUAL(targ->executed, 2);
|
|
|
|
CU_ASSERT_EQUAL(targ->cancelled, 1);
|
|
|
|
|
|
|
|
free(targ);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* verify no cleanup routines are called if all got popped */
|
2018-11-01 19:32:24 +01:00
|
|
|
CU_Test(os_thread_cleanup, push_two_pop_two_no_exec)
|
2018-04-10 17:03:59 +02:00
|
|
|
{
|
|
|
|
int flags = THREAD_RESET_1 | THREAD_RESET_2;
|
|
|
|
struct thread_argument *targ = make_thread_argument(flags, 2, 1, 2);
|
|
|
|
setup(targ);
|
|
|
|
|
|
|
|
CU_ASSERT_EQUAL(targ->one, 1);
|
|
|
|
CU_ASSERT_EQUAL(targ->two, 2);
|
|
|
|
CU_ASSERT_EQUAL(targ->executed, 0);
|
|
|
|
CU_ASSERT_EQUAL(targ->cancelled, 2);
|
|
|
|
|
|
|
|
free(targ);
|
|
|
|
}
|
|
|
|
|
2018-11-01 19:32:24 +01:00
|
|
|
CU_Test(os_thread_cleanup, push_two_pop_two_exec_one)
|
2018-04-10 17:03:59 +02:00
|
|
|
{
|
|
|
|
int flags = THREAD_RESET_1 | THREAD_RESET_2 | THREAD_RUN_1;
|
|
|
|
struct thread_argument *targ = make_thread_argument(flags, 2, 1, 2);
|
|
|
|
setup(targ);
|
|
|
|
|
|
|
|
CU_ASSERT_EQUAL(targ->one, 0);
|
|
|
|
CU_ASSERT_EQUAL(targ->two, 2);
|
|
|
|
CU_ASSERT_EQUAL(targ->executed, 1);
|
|
|
|
CU_ASSERT_EQUAL(targ->cancelled, 2);
|
|
|
|
|
|
|
|
free(targ);
|
|
|
|
}
|
|
|
|
|
2018-11-01 19:32:24 +01:00
|
|
|
CU_Test(os_thread_cleanup, push_two_pop_two_exec_both)
|
2018-04-10 17:03:59 +02:00
|
|
|
{
|
|
|
|
int flags = THREAD_RESET_1 | THREAD_RESET_2 | THREAD_RUN_1 | THREAD_RUN_2;
|
|
|
|
struct thread_argument *targ = make_thread_argument(flags, 2, 1, 2);
|
|
|
|
setup(targ);
|
|
|
|
|
|
|
|
CU_ASSERT_EQUAL(targ->one, 0);
|
|
|
|
CU_ASSERT_EQUAL(targ->two, 0);
|
|
|
|
CU_ASSERT_EQUAL(targ->executed, 2);
|
|
|
|
CU_ASSERT_EQUAL(targ->cancelled, 2);
|
|
|
|
|
|
|
|
free(targ);
|
|
|
|
}
|
|
|
|
|
2018-11-01 19:32:24 +01:00
|
|
|
CU_Test(os_thread_cleanup, no_interference)
|
2018-04-10 17:03:59 +02:00
|
|
|
{
|
|
|
|
int flags = THREAD_RESET_1 | THREAD_RESET_2;
|
|
|
|
struct thread_argument *targ1 = make_thread_argument(flags, 0, 1, 2);
|
|
|
|
struct thread_argument *targ2 = make_thread_argument(flags, 2, 1, 2);
|
|
|
|
os_mutex mutex1, mutex2;
|
|
|
|
|
|
|
|
os_mutexInit(&mutex1);
|
|
|
|
os_mutexInit(&mutex2);
|
|
|
|
os_mutexLock(&mutex1);
|
|
|
|
os_mutexLock(&mutex2);
|
|
|
|
|
|
|
|
targ1->mutex = &mutex1;
|
|
|
|
targ1->block = 1;
|
|
|
|
targ2->mutex = &mutex2;
|
|
|
|
targ2->block = 1;
|
|
|
|
|
|
|
|
setup(targ1);
|
|
|
|
setup(targ2);
|
|
|
|
|
|
|
|
/* ensure thread 2 pops it's cleanup routines while thread 1 blocks */
|
|
|
|
os_mutexUnlock(&mutex2);
|
|
|
|
os_threadWaitExit(targ2->thread, NULL);
|
|
|
|
|
|
|
|
CU_ASSERT_EQUAL(targ2->one, 1);
|
|
|
|
CU_ASSERT_EQUAL(targ2->two, 2);
|
|
|
|
CU_ASSERT_EQUAL(targ2->executed, 0);
|
|
|
|
CU_ASSERT_EQUAL(targ2->cancelled, 2);
|
|
|
|
|
|
|
|
/* instruct thread 1 to continue */
|
|
|
|
os_mutexUnlock(&mutex1);
|
|
|
|
os_threadWaitExit(targ1->thread, NULL);
|
|
|
|
|
|
|
|
CU_ASSERT_EQUAL(targ1->one, 0);
|
|
|
|
CU_ASSERT_EQUAL(targ1->two, 0);
|
|
|
|
CU_ASSERT_EQUAL(targ1->executed, 2);
|
|
|
|
CU_ASSERT_EQUAL(targ1->cancelled, 0);
|
|
|
|
|
|
|
|
os_mutexDestroy(&mutex1);
|
|
|
|
os_mutexDestroy(&mutex2);
|
|
|
|
free(targ1);
|
|
|
|
free(targ2);
|
|
|
|
}
|