2017-08-07 17:02:33 +00:00
|
|
|
/* mbed Microcontroller Library
|
|
|
|
* Copyright (c) 2017 ARM Limited
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
2016-07-19 18:39:53 +00:00
|
|
|
#include "mbed.h"
|
|
|
|
#include "greentea-client/test_env.h"
|
|
|
|
#include "unity.h"
|
|
|
|
#include "utest.h"
|
|
|
|
#include "rtos.h"
|
|
|
|
#include "SynchronizedIntegral.h"
|
|
|
|
#include "LockGuard.h"
|
|
|
|
|
2016-07-25 10:41:00 +00:00
|
|
|
#if defined(MBED_RTOS_SINGLE_THREAD)
|
|
|
|
#error [NOT_SUPPORTED] test not supported
|
|
|
|
#endif
|
2016-07-19 18:39:53 +00:00
|
|
|
|
2017-10-23 07:39:48 +00:00
|
|
|
#define THREAD_STACK_SIZE 512
|
2017-12-14 11:41:28 +00:00
|
|
|
#if defined(__CORTEX_A9)
|
|
|
|
#define PARALLEL_THREAD_STACK_SIZE 512
|
|
|
|
#else
|
2017-11-22 11:54:43 +00:00
|
|
|
#define PARALLEL_THREAD_STACK_SIZE 384
|
2017-12-14 11:41:28 +00:00
|
|
|
#endif
|
2017-11-22 11:54:43 +00:00
|
|
|
#define CHILD_THREAD_STACK_SIZE 384
|
2016-07-25 22:56:14 +00:00
|
|
|
|
2016-07-19 18:39:53 +00:00
|
|
|
using namespace utest::v1;
|
|
|
|
|
|
|
|
// The counter type used accross all the tests
|
|
|
|
// It is internall ysynchronized so read
|
|
|
|
typedef SynchronizedIntegral<int> counter_t;
|
|
|
|
|
2017-10-23 07:39:48 +00:00
|
|
|
template<osPriority P, uint32_t S>
|
|
|
|
class ParallelThread : public Thread {
|
|
|
|
public:
|
|
|
|
ParallelThread() : Thread(P, S) { }
|
|
|
|
};
|
|
|
|
|
2016-07-19 18:39:53 +00:00
|
|
|
// Tasks with different functions to test on threads
|
|
|
|
void increment(counter_t* counter) {
|
|
|
|
(*counter)++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void increment_with_yield(counter_t* counter) {
|
|
|
|
Thread::yield();
|
|
|
|
(*counter)++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void increment_with_wait(counter_t* counter) {
|
|
|
|
Thread::wait(100);
|
|
|
|
(*counter)++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void increment_with_child(counter_t* counter) {
|
2017-10-23 07:39:48 +00:00
|
|
|
Thread *child = new Thread(osPriorityNormal, CHILD_THREAD_STACK_SIZE);
|
|
|
|
child->start(callback(increment, counter));
|
|
|
|
child->join();
|
|
|
|
delete child;
|
2016-07-19 18:39:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void increment_with_murder(counter_t* counter) {
|
|
|
|
{
|
|
|
|
// take ownership of the counter mutex so it prevent the child to
|
|
|
|
// modify counter.
|
|
|
|
LockGuard lock(counter->internal_mutex());
|
2017-10-23 07:39:48 +00:00
|
|
|
Thread *child = new Thread(osPriorityNormal, CHILD_THREAD_STACK_SIZE);
|
|
|
|
child->start(callback(increment, counter));
|
|
|
|
child->terminate();
|
|
|
|
delete child;
|
2016-07-19 18:39:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
(*counter)++;
|
|
|
|
}
|
|
|
|
|
2016-09-07 22:33:33 +00:00
|
|
|
void self_terminate(Thread *self) {
|
|
|
|
self->terminate();
|
|
|
|
// Code should not get here
|
|
|
|
TEST_ASSERT(0);
|
|
|
|
}
|
|
|
|
|
2016-07-19 18:39:53 +00:00
|
|
|
// Tests that spawn tasks in different configurations
|
2017-06-21 10:37:09 +00:00
|
|
|
|
|
|
|
/** Template for tests: single thread, with yield, with wait, with child, with murder
|
|
|
|
|
|
|
|
Testing single thread
|
|
|
|
Given single thread is started
|
|
|
|
when the thread increments the counter
|
|
|
|
then the final value of the counter is equal to 1
|
|
|
|
|
|
|
|
Testing single thread with yield
|
|
|
|
Given single thread is started
|
|
|
|
when the thread yields and then increments the counter
|
|
|
|
then the final value of the counter is equal to 1
|
|
|
|
|
|
|
|
Testing single thread with wait
|
|
|
|
Given single thread is started
|
|
|
|
when the thread waits for 100ms and then increments the counter
|
|
|
|
then the final value of the counter is equal to 1
|
|
|
|
|
|
|
|
Testing single thread with child
|
|
|
|
Given single thread is started
|
|
|
|
when the thread spawns another thread that increments the counter
|
|
|
|
then the final value of the counter is equal to 1
|
|
|
|
|
|
|
|
Testing serial threads with murder
|
|
|
|
Given single thread is started
|
|
|
|
when the parent thread is holding a lock
|
|
|
|
and the parent thread spawns a child thread that waits for the lock before incrementing the counter
|
|
|
|
and the parent terminates the child before releasing the lock
|
|
|
|
and the parent increments the counter
|
|
|
|
then the final value of the counter is equal to 1
|
|
|
|
*/
|
2016-07-19 18:39:53 +00:00
|
|
|
template <void (*F)(counter_t *)>
|
|
|
|
void test_single_thread() {
|
|
|
|
counter_t counter(0);
|
2017-06-21 10:37:09 +00:00
|
|
|
Thread thread(osPriorityNormal, THREAD_STACK_SIZE);
|
2016-12-21 13:46:09 +00:00
|
|
|
thread.start(callback(F, &counter));
|
2016-07-19 18:39:53 +00:00
|
|
|
thread.join();
|
|
|
|
TEST_ASSERT_EQUAL(counter, 1);
|
|
|
|
}
|
|
|
|
|
2017-06-21 10:37:09 +00:00
|
|
|
/** Template for tests: parallel threads, with yield, with wait, with child, with murder
|
|
|
|
|
|
|
|
Testing parallel threads
|
|
|
|
Given multiple threads are started in parallel
|
|
|
|
when each of the threads increments the counter
|
|
|
|
then the final value of the counter is equal to number of threads
|
|
|
|
|
|
|
|
Testing parallel threads with yield
|
|
|
|
Given multiple threads are started in parallel
|
|
|
|
when each of the threads yields and then increments the counter
|
|
|
|
then the final value of the counter is equal to number of threads
|
|
|
|
|
|
|
|
Testing parallel threads with wait
|
|
|
|
Given multiple threads are started in parallel
|
|
|
|
when each of the threads waits for 100ms and then increments the counter
|
|
|
|
then the final value of the counter is equal to number of threads
|
|
|
|
|
|
|
|
Testing parallel threads with child
|
|
|
|
Given multiple threads are started in parallel
|
|
|
|
when each of the threads spawns another thread that increments the counter
|
|
|
|
then the final value of the counter is equal to number of parallel threads
|
|
|
|
|
|
|
|
Testing parallel threads with murder
|
|
|
|
Given multiple threads are started in parallel
|
|
|
|
when the parent thread is holding a lock
|
|
|
|
and the parent thread spawns a child thread that waits for the lock before incrementing the counter
|
|
|
|
and the parent terminates the child before releasing the lock
|
|
|
|
and the parent increments the counter
|
|
|
|
then the final value of the counter is equal to number of parallel threads
|
|
|
|
*/
|
2016-07-19 18:39:53 +00:00
|
|
|
template <int N, void (*F)(counter_t *)>
|
|
|
|
void test_parallel_threads() {
|
|
|
|
counter_t counter(0);
|
2017-10-23 07:39:48 +00:00
|
|
|
ParallelThread<osPriorityNormal, PARALLEL_THREAD_STACK_SIZE> threads[N];
|
2016-07-19 18:39:53 +00:00
|
|
|
|
|
|
|
for (int i = 0; i < N; i++) {
|
2017-10-23 07:39:48 +00:00
|
|
|
threads[i].start(callback(F, &counter));
|
2016-07-19 18:39:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < N; i++) {
|
2017-10-23 07:39:48 +00:00
|
|
|
threads[i].join();
|
2016-07-19 18:39:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL(counter, N);
|
|
|
|
}
|
|
|
|
|
2017-06-21 10:37:09 +00:00
|
|
|
/** Template for tests: serial threads, with yield, with wait, with child, with murder
|
|
|
|
|
|
|
|
Testing serial threads
|
|
|
|
Given multiple threads are started serially
|
|
|
|
when each of the threads increments the counter
|
|
|
|
then the final value of the counter is equal to number of threads
|
|
|
|
|
|
|
|
Testing serial threads with yield
|
|
|
|
Given multiple threads are started serially
|
|
|
|
when each of the threads yields and then increments the counter
|
|
|
|
then the final value of the counter is equal to number of threads
|
|
|
|
|
|
|
|
Testing serial threads with wait
|
|
|
|
Given multiple threads are started serially
|
|
|
|
when each of the threads waits for 100ms and then increments the counter
|
|
|
|
then the final value of the counter is equal to number of threads
|
|
|
|
|
|
|
|
Testing serial threads with child
|
|
|
|
Given multiple threads are started serially
|
|
|
|
when each of the threads spawns another thread that increments the counter
|
|
|
|
then the final value of the counter is equal to number of serial threads
|
|
|
|
|
|
|
|
Testing serial threads with murder
|
|
|
|
Given multiple threads are started serially
|
|
|
|
when the parent thread is holding a lock
|
|
|
|
and the parent thread spawns a child thread that waits for the lock before incrementing the counter
|
|
|
|
and the parent terminates the child before releasing the lock
|
|
|
|
and the parent increments the counter
|
|
|
|
then the final value of the counter is equal to number of serial threads
|
|
|
|
*/
|
2016-07-19 18:39:53 +00:00
|
|
|
template <int N, void (*F)(counter_t *)>
|
|
|
|
void test_serial_threads() {
|
|
|
|
counter_t counter(0);
|
|
|
|
|
|
|
|
for (int i = 0; i < N; i++) {
|
2017-05-14 23:55:19 +00:00
|
|
|
Thread thread(osPriorityNormal, THREAD_STACK_SIZE);
|
2016-12-21 13:46:09 +00:00
|
|
|
thread.start(callback(F, &counter));
|
2016-07-19 18:39:53 +00:00
|
|
|
thread.join();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL(counter, N);
|
|
|
|
}
|
|
|
|
|
2017-06-21 10:37:09 +00:00
|
|
|
/** Testing thread self terminate
|
|
|
|
|
|
|
|
Given the thread is running
|
|
|
|
when the thread calls @a terminate on its self
|
|
|
|
then the thread terminates execution cleanly
|
|
|
|
*/
|
2016-09-07 22:33:33 +00:00
|
|
|
void test_self_terminate() {
|
2017-05-14 23:55:19 +00:00
|
|
|
Thread *thread = new Thread(osPriorityNormal, THREAD_STACK_SIZE);
|
2016-12-21 13:46:09 +00:00
|
|
|
thread->start(callback(self_terminate, thread));
|
2016-09-07 22:33:33 +00:00
|
|
|
thread->join();
|
|
|
|
delete thread;
|
|
|
|
}
|
|
|
|
|
2017-06-21 10:37:09 +00:00
|
|
|
void signal_wait()
|
|
|
|
{
|
|
|
|
osEvent evt = Thread::signal_wait(0x1);
|
|
|
|
TEST_ASSERT_EQUAL(osEventSignal, evt.status);
|
|
|
|
TEST_ASSERT_EQUAL(0x1, evt.value.signals);
|
|
|
|
}
|
|
|
|
|
|
|
|
void signal_wait_tout()
|
|
|
|
{
|
|
|
|
osEvent evt = Thread::signal_wait(0x2, 50);
|
|
|
|
TEST_ASSERT_EQUAL(osEventTimeout, evt.status);
|
|
|
|
}
|
|
|
|
|
|
|
|
void signal_wait_multibit()
|
|
|
|
{
|
|
|
|
osEvent evt = Thread::signal_wait(0x1 | 0x2, 50);
|
|
|
|
TEST_ASSERT_EQUAL(osEventSignal, evt.status);
|
|
|
|
TEST_ASSERT_EQUAL(0x3, evt.value.signals);
|
|
|
|
}
|
|
|
|
|
|
|
|
void signal_wait_multibit_tout()
|
|
|
|
{
|
|
|
|
osEvent evt = Thread::signal_wait(0x1 | 0x2, 50);
|
|
|
|
TEST_ASSERT_EQUAL(osEventTimeout, evt.status);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Testing thread signal: wait
|
|
|
|
Given two threads (A & B) are started
|
|
|
|
when thread A executes @a signal_wait(0x1)
|
|
|
|
and thread B execute @a signal_set(0x1)
|
|
|
|
then thread A exits the wait and continues execution
|
|
|
|
|
|
|
|
Testing thread signal: timeout
|
|
|
|
Given two threads (A & B) are started
|
|
|
|
when thread A executes @a signal_wait(0x1 | 0x2, 50) with a timeout of 50ms
|
|
|
|
and thread B execute @a signal_set(0x2)
|
|
|
|
then thread A keeps waiting for correct signal until it timeouts
|
|
|
|
|
|
|
|
Testing thread signal: multi-bit
|
|
|
|
Given two threads (A & B) are started
|
|
|
|
when thread A executes @a signal_wait(0x1 | 0x2)
|
|
|
|
and thread B execute @a signal_set(0x1 | 0x2)
|
|
|
|
then thread A exits the wait and continues execution
|
|
|
|
|
|
|
|
Testing thread signal: multi-bit timeout
|
|
|
|
Given two threads (A & B) are started
|
|
|
|
when thread A executes @a signal_wait(0x1, 50) with a timeout of 50ms
|
|
|
|
and thread B execute @a signal_set(0x2)
|
|
|
|
then thread A keeps waiting for correct signal until it timeouts
|
|
|
|
*/
|
|
|
|
template <int S, void (*F)()>
|
|
|
|
void test_thread_signal()
|
|
|
|
{
|
2017-10-23 07:39:48 +00:00
|
|
|
Thread t_wait(osPriorityNormal, THREAD_STACK_SIZE);
|
2017-06-21 10:37:09 +00:00
|
|
|
|
|
|
|
t_wait.start(callback(F));
|
|
|
|
|
|
|
|
Thread::yield();
|
|
|
|
|
|
|
|
Thread::State state = t_wait.get_state();
|
|
|
|
TEST_ASSERT_EQUAL(Thread::WaitingThreadFlag, state);
|
|
|
|
|
|
|
|
int32_t res = t_wait.signal_set(S);
|
|
|
|
|
|
|
|
t_wait.join();
|
|
|
|
}
|
|
|
|
|
|
|
|
void signal_clr()
|
|
|
|
{
|
|
|
|
Thread::yield();
|
|
|
|
|
|
|
|
int32_t sig = Thread::signal_clr(0x1);
|
|
|
|
TEST_ASSERT_EQUAL(0x1, sig);
|
|
|
|
|
|
|
|
/* Signal cleared we should get timeout */
|
|
|
|
osEvent evt = Thread::signal_wait(0x1, 0);
|
|
|
|
TEST_ASSERT_EQUAL(osOK, evt.status);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Testing thread signals: signal clear
|
|
|
|
|
|
|
|
Given two threads (A & B) are started
|
|
|
|
when thread A executes @a signal_set(0x1)
|
|
|
|
and thread B execute @a signal_clr(0x1)
|
|
|
|
and thread B execute @a signal_wait(0x1, 0)
|
|
|
|
then thread B @a signal_wait status should be osOK indicating a timeout
|
|
|
|
*/
|
|
|
|
void test_thread_signal_clr()
|
|
|
|
{
|
2017-10-23 07:39:48 +00:00
|
|
|
Thread t_wait(osPriorityNormal, THREAD_STACK_SIZE);
|
2017-06-21 10:37:09 +00:00
|
|
|
|
|
|
|
t_wait.start(callback(signal_clr));
|
|
|
|
|
|
|
|
int32_t res = t_wait.signal_set(0x1);
|
|
|
|
TEST_ASSERT_EQUAL(0x1, res);
|
|
|
|
|
|
|
|
t_wait.join();
|
|
|
|
}
|
|
|
|
|
|
|
|
void thread_wait_signal() {
|
|
|
|
Thread::signal_wait(0x1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void stack_info() {
|
|
|
|
Thread::signal_wait(0x1);
|
|
|
|
|
|
|
|
thread_wait_signal();
|
|
|
|
|
|
|
|
Thread::signal_wait(0x1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Testing thread stack info
|
|
|
|
|
|
|
|
Given the thread is running
|
|
|
|
when a function is called from the thread context
|
|
|
|
then the stack usage goes up
|
|
|
|
and the reported stack size is as requested in the constructor
|
|
|
|
and the sum of free and used stack sizes is equal to the total stack size
|
|
|
|
when the function returns
|
|
|
|
then the stack usage goes down
|
|
|
|
and the reported stack size is as requested in the constructor
|
|
|
|
and the sum of free and used stack sizes is equal to the total stack size
|
|
|
|
*/
|
|
|
|
void test_thread_stack_info() {
|
|
|
|
Thread t(osPriorityNormal, THREAD_STACK_SIZE);
|
|
|
|
t.start(callback(stack_info));
|
|
|
|
|
|
|
|
Thread::yield();
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL(THREAD_STACK_SIZE, t.stack_size());
|
|
|
|
TEST_ASSERT_EQUAL(THREAD_STACK_SIZE, t.free_stack() + t.used_stack());
|
|
|
|
uint32_t last_stack = t.used_stack();
|
|
|
|
|
|
|
|
t.signal_set(0x1);
|
|
|
|
Thread::yield();
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL(THREAD_STACK_SIZE, t.free_stack() + t.used_stack());
|
|
|
|
TEST_ASSERT(last_stack <= t.used_stack());
|
|
|
|
last_stack = t.used_stack();
|
|
|
|
|
|
|
|
t.signal_set(0x1);
|
|
|
|
Thread::yield();
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL(THREAD_STACK_SIZE, t.free_stack() + t.used_stack());
|
|
|
|
TEST_ASSERT(last_stack >= t.used_stack());
|
|
|
|
|
|
|
|
t.signal_set(0x1);
|
|
|
|
|
|
|
|
t.join();
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Testing thread wait aka delay
|
|
|
|
|
|
|
|
Given the thread is running
|
|
|
|
when the @a wait function is called
|
|
|
|
then the thread sleeps for given amount of time
|
|
|
|
*/
|
|
|
|
void test_thread_wait() {
|
2017-09-14 00:48:33 +00:00
|
|
|
Timer timer;
|
|
|
|
timer.start();
|
2017-06-21 10:37:09 +00:00
|
|
|
|
|
|
|
Thread::wait(150);
|
|
|
|
|
2017-09-14 00:48:33 +00:00
|
|
|
TEST_ASSERT_UINT32_WITHIN(50000, 150000, timer.read_us());
|
2017-06-21 10:37:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/** Testing thread name
|
|
|
|
|
|
|
|
Given a thread is started with a specified name
|
|
|
|
when the name is queried using @a get_name
|
|
|
|
then the returned name is as set
|
|
|
|
*/
|
|
|
|
void test_thread_name() {
|
|
|
|
const char tname[] = "Amazing thread";
|
|
|
|
Thread t(osPriorityNormal, THREAD_STACK_SIZE, NULL, tname);
|
|
|
|
t.start(callback(thread_wait_signal));
|
|
|
|
TEST_ASSERT_EQUAL(strcmp(tname, t.get_name()), 0);
|
|
|
|
t.signal_set(0x1);
|
|
|
|
t.join();
|
|
|
|
}
|
|
|
|
|
|
|
|
void test_deleted_thread()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Testing thread states: deleted
|
|
|
|
|
|
|
|
Given the thread is not started
|
|
|
|
then its state, as reported by @a get_state, is @a Deleted
|
|
|
|
when the thread is started and finishes executing
|
|
|
|
then its state, as reported by @a get_state, is @a Deleted
|
|
|
|
*/
|
|
|
|
void test_deleted()
|
|
|
|
{
|
2017-10-23 07:39:48 +00:00
|
|
|
Thread t(osPriorityNormal, THREAD_STACK_SIZE);
|
2017-06-21 10:37:09 +00:00
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL(Thread::Deleted, t.get_state());
|
|
|
|
|
|
|
|
t.start(callback(test_deleted_thread));
|
|
|
|
|
|
|
|
t.join();
|
|
|
|
TEST_ASSERT_EQUAL(Thread::Deleted, t.get_state());
|
|
|
|
}
|
|
|
|
|
|
|
|
void test_delay_thread()
|
|
|
|
{
|
|
|
|
Thread::wait(50);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Testing thread states: wait delay
|
|
|
|
|
|
|
|
Given the thread is running
|
|
|
|
when thread calls @a wait
|
|
|
|
then its state, as reported by @a get_state, is @a WaitingDelay
|
|
|
|
*/
|
|
|
|
void test_delay()
|
|
|
|
{
|
2017-10-23 07:39:48 +00:00
|
|
|
Thread t(osPriorityNormal, THREAD_STACK_SIZE);
|
2017-06-21 10:37:09 +00:00
|
|
|
|
|
|
|
t.start(callback(test_delay_thread));
|
|
|
|
|
|
|
|
Thread::yield();
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL(Thread::WaitingDelay, t.get_state());
|
|
|
|
|
|
|
|
t.join();
|
|
|
|
TEST_ASSERT_EQUAL(Thread::Deleted, t.get_state());
|
|
|
|
}
|
|
|
|
|
|
|
|
void test_signal_thread()
|
|
|
|
{
|
|
|
|
Thread::signal_wait(0x1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Testing thread states: wait signal
|
|
|
|
|
|
|
|
Given the thread is running
|
|
|
|
when thread waits for a signal
|
|
|
|
then its state, as reported by @a get_state, is @a WaitingSignal
|
|
|
|
*/
|
|
|
|
void test_signal()
|
|
|
|
{
|
2017-10-23 07:39:48 +00:00
|
|
|
Thread t(osPriorityNormal, THREAD_STACK_SIZE);
|
2017-06-21 10:37:09 +00:00
|
|
|
|
|
|
|
t.start(callback(test_signal_thread));
|
|
|
|
|
|
|
|
Thread::yield();
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL(Thread::WaitingThreadFlag, t.get_state());
|
|
|
|
|
|
|
|
t.signal_set(0x1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void test_evt_flag_thread(osEventFlagsId_t evtflg)
|
|
|
|
{
|
|
|
|
osEventFlagsWait(evtflg, 0x1, osFlagsWaitAny, osWaitForever);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Testing thread states: wait evt flag
|
|
|
|
|
|
|
|
Given the thread is running
|
|
|
|
when thread waits for an event flag
|
|
|
|
then its state, as reported by @a get_state, is @a WaitingEventFlag
|
|
|
|
*/
|
|
|
|
void test_evt_flag()
|
|
|
|
{
|
2017-10-23 07:39:48 +00:00
|
|
|
Thread t(osPriorityNormal, THREAD_STACK_SIZE);
|
2017-06-21 10:37:09 +00:00
|
|
|
mbed_rtos_storage_event_flags_t evtflg_mem;
|
|
|
|
osEventFlagsAttr_t evtflg_attr;
|
|
|
|
osEventFlagsId_t evtflg;
|
|
|
|
|
|
|
|
evtflg_attr.cb_mem = &evtflg_mem;
|
|
|
|
evtflg_attr.cb_size = sizeof(evtflg_mem);
|
|
|
|
evtflg = osEventFlagsNew(&evtflg_attr);
|
|
|
|
TEST_ASSERT_NOT_EQUAL(NULL, evtflg);
|
|
|
|
|
|
|
|
t.start(callback(test_evt_flag_thread, evtflg));
|
|
|
|
|
|
|
|
Thread::yield();
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL(Thread::WaitingEventFlag, t.get_state());
|
|
|
|
|
|
|
|
osEventFlagsSet(evtflg, 0x1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void test_mutex_thread(Mutex *mutex)
|
|
|
|
{
|
|
|
|
mutex->lock();
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Testing thread states: wait mutex
|
|
|
|
|
|
|
|
Given the thread is running
|
|
|
|
when thread waits for a mutex
|
|
|
|
then its state, as reported by @a get_state, is @a WaitingMutex
|
|
|
|
*/
|
|
|
|
void test_mutex()
|
|
|
|
{
|
2017-10-23 07:39:48 +00:00
|
|
|
Thread t(osPriorityNormal, THREAD_STACK_SIZE);
|
2017-06-21 10:37:09 +00:00
|
|
|
Mutex mutex;
|
|
|
|
|
|
|
|
mutex.lock();
|
|
|
|
|
|
|
|
t.start(callback(test_mutex_thread, &mutex));
|
|
|
|
|
|
|
|
Thread::yield();
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL(Thread::WaitingMutex, t.get_state());
|
|
|
|
|
|
|
|
mutex.unlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
void test_semaphore_thread(Semaphore *sem)
|
|
|
|
{
|
|
|
|
sem->wait();
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Testing thread states: wait semaphore
|
|
|
|
|
|
|
|
Given the thread is running
|
|
|
|
when thread waits for a semaphore
|
|
|
|
then its state, as reported by @a get_state, is @a WaitingSemaphore
|
|
|
|
*/
|
|
|
|
void test_semaphore()
|
|
|
|
{
|
2017-10-23 07:39:48 +00:00
|
|
|
Thread t(osPriorityNormal, THREAD_STACK_SIZE);
|
2017-06-21 10:37:09 +00:00
|
|
|
Semaphore sem;
|
|
|
|
|
|
|
|
t.start(callback(test_semaphore_thread, &sem));
|
|
|
|
|
|
|
|
Thread::yield();
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL(Thread::WaitingSemaphore, t.get_state());
|
|
|
|
|
|
|
|
sem.release();
|
|
|
|
}
|
|
|
|
|
|
|
|
void test_msg_get_thread(Queue<int32_t, 1> *queue)
|
|
|
|
{
|
|
|
|
queue->get();
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Testing thread states: wait message get
|
|
|
|
|
|
|
|
Given the thread is running
|
|
|
|
when thread tries to get a message from an empty queue
|
|
|
|
then its state, as reported by @a get_state, is @a WaitingMessageGet
|
|
|
|
*/
|
|
|
|
void test_msg_get()
|
|
|
|
{
|
2017-10-23 07:39:48 +00:00
|
|
|
Thread t(osPriorityNormal, THREAD_STACK_SIZE);
|
2017-06-21 10:37:09 +00:00
|
|
|
Queue<int32_t, 1> queue;
|
|
|
|
|
|
|
|
t.start(callback(test_msg_get_thread, &queue));
|
|
|
|
|
|
|
|
Thread::yield();
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL(Thread::WaitingMessageGet, t.get_state());
|
|
|
|
|
|
|
|
queue.put((int32_t *)0xE1EE7);
|
|
|
|
}
|
|
|
|
|
|
|
|
void test_msg_put_thread(Queue<int32_t, 1> *queue)
|
|
|
|
{
|
|
|
|
queue->put((int32_t *)0xDEADBEEF, osWaitForever);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Testing thread states: wait message put
|
|
|
|
|
|
|
|
Given the thread is running
|
|
|
|
when thread tries to put a message into a full queue
|
|
|
|
then its state, as reported by @a get_state, is @a WaitingMessagePut
|
|
|
|
*/
|
|
|
|
void test_msg_put()
|
|
|
|
{
|
2017-10-23 07:39:48 +00:00
|
|
|
Thread t(osPriorityNormal, THREAD_STACK_SIZE);
|
2017-06-21 10:37:09 +00:00
|
|
|
Queue<int32_t, 1> queue;
|
|
|
|
|
|
|
|
queue.put((int32_t *)0xE1EE7);
|
|
|
|
|
|
|
|
t.start(callback(test_msg_put_thread, &queue));
|
|
|
|
|
|
|
|
Thread::yield();
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL(Thread::WaitingMessagePut, t.get_state());
|
|
|
|
queue.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Utility function that places some date on the stack */
|
|
|
|
void use_some_stack () {
|
|
|
|
volatile uint32_t stack_filler[10] = {0xDEADBEEF};
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Testing thread with external stack memory
|
|
|
|
|
|
|
|
Given external buffer is supplied as stack to a thread
|
|
|
|
when the thread executes
|
|
|
|
then the supplies buffer is used as a stack
|
|
|
|
*/
|
|
|
|
void test_thread_ext_stack() {
|
|
|
|
char stack[512];
|
|
|
|
Thread t(osPriorityNormal, sizeof(stack), (unsigned char*)stack);
|
|
|
|
|
|
|
|
memset(&stack, 0, sizeof(stack));
|
|
|
|
t.start(callback(use_some_stack));
|
|
|
|
t.join();
|
|
|
|
|
|
|
|
/* If buffer was used as a stack it was cleared with pattern and some data were placed in it */
|
|
|
|
for(unsigned i = 0; i < sizeof(stack); i++) {
|
|
|
|
if (stack[i] != 0)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_FAIL_MESSAGE("External stack was not used.");
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Testing thread priority operations
|
|
|
|
|
|
|
|
Given thread running with osPriorityNormal
|
|
|
|
when new priority is set using @a set_priority
|
|
|
|
then priority is changed and can be retrieved using @a get_priority
|
|
|
|
*/
|
|
|
|
void test_thread_prio() {
|
2017-10-23 07:39:48 +00:00
|
|
|
Thread t(osPriorityNormal, THREAD_STACK_SIZE);
|
2017-06-21 10:37:09 +00:00
|
|
|
t.start(callback(thread_wait_signal));
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL(osPriorityNormal, t.get_priority());
|
|
|
|
|
|
|
|
t.set_priority(osPriorityHigh);
|
|
|
|
|
|
|
|
TEST_ASSERT_EQUAL(osPriorityHigh, t.get_priority());
|
|
|
|
|
|
|
|
t.signal_set(0x1);
|
|
|
|
t.join();
|
|
|
|
}
|
|
|
|
|
2016-07-19 18:39:53 +00:00
|
|
|
utest::v1::status_t test_setup(const size_t number_of_cases) {
|
2017-09-15 16:39:47 +00:00
|
|
|
GREENTEA_SETUP(20, "default_auto");
|
2016-07-19 18:39:53 +00:00
|
|
|
return verbose_test_setup_handler(number_of_cases);
|
|
|
|
}
|
|
|
|
|
2017-06-21 10:37:09 +00:00
|
|
|
#define DEFAULT_HANDLERS NULL,NULL,greentea_case_setup_handler,greentea_case_teardown_handler,greentea_case_failure_abort_handler
|
|
|
|
|
|
|
|
// Test cases. It's spelled out rather than constructed with macro because
|
|
|
|
// macros don't play nicely with the templates (extra comma).
|
|
|
|
static const case_t cases[] = {
|
|
|
|
{"Testing single thread", test_single_thread<increment>, DEFAULT_HANDLERS},
|
|
|
|
{"Testing parallel threads", test_parallel_threads<3, increment> , DEFAULT_HANDLERS},
|
|
|
|
{"Testing serial threads", test_serial_threads<10, increment> , DEFAULT_HANDLERS},
|
|
|
|
|
|
|
|
{"Testing single thread with yield", test_single_thread<increment_with_yield>, DEFAULT_HANDLERS},
|
|
|
|
{"Testing parallel threads with yield", test_parallel_threads<3, increment_with_yield>, DEFAULT_HANDLERS},
|
|
|
|
{"Testing serial threads with yield", test_serial_threads<10, increment_with_yield>, DEFAULT_HANDLERS},
|
|
|
|
|
|
|
|
{"Testing single thread with wait", test_single_thread<increment_with_wait>, DEFAULT_HANDLERS},
|
|
|
|
{"Testing parallel threads with wait", test_parallel_threads<3, increment_with_wait>, DEFAULT_HANDLERS},
|
|
|
|
{"Testing serial threads with wait", test_serial_threads<10, increment_with_wait>, DEFAULT_HANDLERS},
|
|
|
|
|
|
|
|
{"Testing single thread with child", test_single_thread<increment_with_child>, DEFAULT_HANDLERS},
|
2017-10-23 07:39:48 +00:00
|
|
|
{"Testing parallel threads with child", test_parallel_threads<2, increment_with_child>, DEFAULT_HANDLERS},
|
2017-06-21 10:37:09 +00:00
|
|
|
{"Testing serial threads with child", test_serial_threads<10, increment_with_child>, DEFAULT_HANDLERS},
|
|
|
|
|
|
|
|
{"Testing single thread with murder", test_single_thread<increment_with_murder>, DEFAULT_HANDLERS},
|
2017-10-23 07:39:48 +00:00
|
|
|
{"Testing parallel threads with murder", test_parallel_threads<2, increment_with_murder>, DEFAULT_HANDLERS},
|
2017-06-21 10:37:09 +00:00
|
|
|
{"Testing serial threads with murder", test_serial_threads<10, increment_with_murder>, DEFAULT_HANDLERS},
|
|
|
|
|
|
|
|
{"Testing thread self terminate", test_self_terminate, DEFAULT_HANDLERS},
|
2016-07-19 18:39:53 +00:00
|
|
|
|
2017-06-21 10:37:09 +00:00
|
|
|
{"Testing thread signals: wait", test_thread_signal<0x1, signal_wait>, DEFAULT_HANDLERS},
|
|
|
|
{"Testing thread signals: timeout", test_thread_signal<0x1, signal_wait_tout>, DEFAULT_HANDLERS},
|
|
|
|
{"Testing thread signals: multi-bit", test_thread_signal<0x3, signal_wait_multibit>, DEFAULT_HANDLERS},
|
|
|
|
{"Testing thread signals: multi-bit timeout", test_thread_signal<0x1, signal_wait_multibit_tout>, DEFAULT_HANDLERS},
|
|
|
|
{"Testing thread signals: signal clear", test_thread_signal_clr, DEFAULT_HANDLERS},
|
2016-07-19 18:39:53 +00:00
|
|
|
|
|
|
|
|
2017-06-21 10:37:09 +00:00
|
|
|
{"Testing thread stack info", test_thread_stack_info, DEFAULT_HANDLERS},
|
|
|
|
{"Testing thread wait", test_thread_wait, DEFAULT_HANDLERS},
|
|
|
|
{"Testing thread name", test_thread_name, DEFAULT_HANDLERS},
|
2016-07-19 18:39:53 +00:00
|
|
|
|
2017-06-21 10:37:09 +00:00
|
|
|
{"Testing thread states: deleted", test_deleted, DEFAULT_HANDLERS},
|
|
|
|
{"Testing thread states: wait delay", test_delay, DEFAULT_HANDLERS},
|
|
|
|
{"Testing thread states: wait signal", test_signal, DEFAULT_HANDLERS},
|
|
|
|
{"Testing thread states: wait event flag", test_evt_flag, DEFAULT_HANDLERS},
|
|
|
|
{"Testing thread states: wait mutex", test_mutex, DEFAULT_HANDLERS},
|
|
|
|
{"Testing thread states: wait semaphore", test_semaphore, DEFAULT_HANDLERS},
|
|
|
|
{"Testing thread states: wait message get", test_msg_get, DEFAULT_HANDLERS},
|
|
|
|
{"Testing thread states: wait message put", test_msg_put, DEFAULT_HANDLERS},
|
2016-09-07 22:33:33 +00:00
|
|
|
|
2017-06-21 10:37:09 +00:00
|
|
|
{"Testing thread with external stack memory", test_thread_ext_stack, DEFAULT_HANDLERS},
|
|
|
|
{"Testing thread priority ops", test_thread_prio, DEFAULT_HANDLERS}
|
2017-10-23 07:39:48 +00:00
|
|
|
|
2016-07-19 18:39:53 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
Specification specification(test_setup, cases);
|
|
|
|
|
|
|
|
int main() {
|
|
|
|
return !Harness::run(specification);
|
|
|
|
}
|