mirror of https://github.com/ARMmbed/mbed-os.git
				
				
				
			
		
			
				
	
	
		
			195 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			C++
		
	
	
			
		
		
	
	
			195 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			C++
		
	
	
/* mbed Microcontroller Library
 | 
						|
 * Copyright (c) 2017 ARM Limited
 | 
						|
 * SPDX-License-Identifier: Apache-2.0
 | 
						|
 *
 | 
						|
 * 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.
 | 
						|
 */
 | 
						|
#if defined(MBED_RTOS_SINGLE_THREAD) || !defined(MBED_CONF_RTOS_PRESENT)
 | 
						|
#error [NOT_SUPPORTED] Condition variable test cases require RTOS with multithread to run
 | 
						|
#else
 | 
						|
 | 
						|
#if !DEVICE_USTICKER
 | 
						|
#error [NOT_SUPPORTED] UsTicker need to be enabled for this test.
 | 
						|
#else
 | 
						|
 | 
						|
#include "mbed.h"
 | 
						|
#include "greentea-client/test_env.h"
 | 
						|
#include "unity.h"
 | 
						|
#include "utest.h"
 | 
						|
#include "rtos.h"
 | 
						|
 | 
						|
using namespace utest::v1;
 | 
						|
 | 
						|
#define TEST_STACK_SIZE 512
 | 
						|
#define TEST_DELAY 10
 | 
						|
 | 
						|
static int change_counter = 0;
 | 
						|
static Mutex mutex;
 | 
						|
static ConditionVariable cond(mutex);
 | 
						|
 | 
						|
void increment_on_signal()
 | 
						|
{
 | 
						|
    mutex.lock();
 | 
						|
 | 
						|
    cond.wait();
 | 
						|
    change_counter++;
 | 
						|
 | 
						|
    mutex.unlock();
 | 
						|
}
 | 
						|
 | 
						|
void test_notify_one()
 | 
						|
{
 | 
						|
    Thread t1(osPriorityNormal, TEST_STACK_SIZE);
 | 
						|
    Thread t2(osPriorityNormal, TEST_STACK_SIZE);
 | 
						|
 | 
						|
    change_counter = 0;
 | 
						|
    t1.start(increment_on_signal);
 | 
						|
    t2.start(increment_on_signal);
 | 
						|
 | 
						|
    ThisThread::sleep_for(TEST_DELAY);
 | 
						|
    TEST_ASSERT_EQUAL(0, change_counter);
 | 
						|
 | 
						|
    mutex.lock();
 | 
						|
    cond.notify_one();
 | 
						|
    mutex.unlock();
 | 
						|
 | 
						|
    ThisThread::sleep_for(TEST_DELAY);
 | 
						|
    TEST_ASSERT_EQUAL(1, change_counter);
 | 
						|
 | 
						|
    mutex.lock();
 | 
						|
    cond.notify_one();
 | 
						|
    mutex.unlock();
 | 
						|
 | 
						|
    t1.join();
 | 
						|
    t2.join();
 | 
						|
}
 | 
						|
 | 
						|
void test_notify_all()
 | 
						|
{
 | 
						|
    Thread t1(osPriorityNormal, TEST_STACK_SIZE);
 | 
						|
    Thread t2(osPriorityNormal, TEST_STACK_SIZE);
 | 
						|
 | 
						|
    change_counter = 0;
 | 
						|
    t1.start(increment_on_signal);
 | 
						|
    t2.start(increment_on_signal);
 | 
						|
 | 
						|
    ThisThread::sleep_for(TEST_DELAY);
 | 
						|
    TEST_ASSERT_EQUAL(0, change_counter);
 | 
						|
 | 
						|
    mutex.lock();
 | 
						|
    cond.notify_all();
 | 
						|
    mutex.unlock();
 | 
						|
 | 
						|
    ThisThread::sleep_for(TEST_DELAY);
 | 
						|
    TEST_ASSERT_EQUAL(2, change_counter);
 | 
						|
 | 
						|
    t1.join();
 | 
						|
    t2.join();
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
class TestConditionVariable : public ConditionVariable {
 | 
						|
 | 
						|
public:
 | 
						|
    static void test_linked_list(void)
 | 
						|
    {
 | 
						|
        Waiter *list = NULL;
 | 
						|
        Waiter w1;
 | 
						|
        Waiter w2;
 | 
						|
        Waiter w3;
 | 
						|
        Waiter w4;
 | 
						|
 | 
						|
        TEST_ASSERT_EQUAL(0, validate_and_get_size(&list));
 | 
						|
 | 
						|
        // Add 4 nodes
 | 
						|
        _add_wait_list(&list, &w1);
 | 
						|
        TEST_ASSERT_EQUAL(1, validate_and_get_size(&list));
 | 
						|
        _add_wait_list(&list, &w2);
 | 
						|
        TEST_ASSERT_EQUAL(2, validate_and_get_size(&list));
 | 
						|
        _add_wait_list(&list, &w3);
 | 
						|
        TEST_ASSERT_EQUAL(3, validate_and_get_size(&list));
 | 
						|
        _add_wait_list(&list, &w4);
 | 
						|
        TEST_ASSERT_EQUAL(4, validate_and_get_size(&list));
 | 
						|
 | 
						|
        // Remove a middle node
 | 
						|
        _remove_wait_list(&list, &w2);
 | 
						|
        TEST_ASSERT_EQUAL(3, validate_and_get_size(&list));
 | 
						|
 | 
						|
        // Remove front node
 | 
						|
        _remove_wait_list(&list, &w1);
 | 
						|
        TEST_ASSERT_EQUAL(2, validate_and_get_size(&list));
 | 
						|
 | 
						|
        // remove back node
 | 
						|
        _remove_wait_list(&list, &w4);
 | 
						|
        TEST_ASSERT_EQUAL(1, validate_and_get_size(&list));
 | 
						|
 | 
						|
        // remove last node
 | 
						|
        _remove_wait_list(&list, &w3);
 | 
						|
        TEST_ASSERT_EQUAL(0, validate_and_get_size(&list));
 | 
						|
 | 
						|
        TEST_ASSERT_EQUAL_PTR(NULL, list);
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Validate the linked list an return the number of elements
 | 
						|
     *
 | 
						|
     * If this list is invalid then this function asserts and does not
 | 
						|
     * return.
 | 
						|
     *
 | 
						|
     * Every node in a valid linked list has the properties:
 | 
						|
     * 1. node->prev->next == node
 | 
						|
     * 2. node->next->prev == node
 | 
						|
     */
 | 
						|
    static int validate_and_get_size(Waiter **list)
 | 
						|
    {
 | 
						|
        Waiter *first = *list;
 | 
						|
        if (NULL == first) {
 | 
						|
            // List is empty
 | 
						|
            return 0;
 | 
						|
        }
 | 
						|
 | 
						|
        int size = 0;
 | 
						|
        Waiter *current = first;
 | 
						|
        do {
 | 
						|
            TEST_ASSERT_EQUAL_PTR(current, current->prev->next);
 | 
						|
            TEST_ASSERT_EQUAL_PTR(current, current->next->prev);
 | 
						|
            current = current->next;
 | 
						|
            size++;
 | 
						|
        } while (current != first);
 | 
						|
        return size;
 | 
						|
    }
 | 
						|
 | 
						|
};
 | 
						|
 | 
						|
utest::v1::status_t test_setup(const size_t number_of_cases)
 | 
						|
{
 | 
						|
    GREENTEA_SETUP(10, "default_auto");
 | 
						|
    return verbose_test_setup_handler(number_of_cases);
 | 
						|
}
 | 
						|
 | 
						|
Case cases[] = {
 | 
						|
    Case("Test notify one", test_notify_one),
 | 
						|
    Case("Test notify all", test_notify_all),
 | 
						|
    Case("Test linked list", TestConditionVariable::test_linked_list),
 | 
						|
};
 | 
						|
 | 
						|
Specification specification(test_setup, cases);
 | 
						|
 | 
						|
int main()
 | 
						|
{
 | 
						|
    return !Harness::run(specification);
 | 
						|
}
 | 
						|
 | 
						|
#endif // !DEVICE_USTICKER
 | 
						|
#endif // defined(MBED_RTOS_SINGLE_THREAD) || !defined(MBED_CONF_RTOS_PRESENT)
 |