/* * Copyright (c) 2014-2015, Siemens AG. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ #include #include namespace embb { namespace base { namespace test { SharedMutexTest::SharedMutexTest() : counter_(0), num_threads_(partest::TestSuite::GetDefaultNumThreads()), num_iterations_(partest::TestSuite::GetDefaultNumIterations()) { CreateUnit("Shared read") .Pre(&SharedMutexTest::TestSharedReadPre, this) .Add(&SharedMutexTest::TestSharedReadThreadMethod, this, num_threads_, num_iterations_) .Post(&SharedMutexTest::TestSharedReadPost, this); CreateUnit("Multiple writer") .Pre(&SharedMutexTest::TestMultipleWriterPre, this) .Add(&SharedMutexTest::TestMultipleWriterReaderMethod, this, num_threads_ / 2, num_iterations_) .Add(&SharedMutexTest::TestMultipleWriterWriterMethod, this, num_threads_ / 2, num_iterations_) .Post(&SharedMutexTest::TestMultipleWriterPost, this); CreateUnit("Single writer") .Pre(&SharedMutexTest::TestSingleWriterPre, this) .Add(&SharedMutexTest::TestSingleWriterReaderMethod, this, num_threads_, num_iterations_) .Add(&SharedMutexTest::TestSingleWriterWriterMethod, this, 1, num_iterations_) .Post(&SharedMutexTest::TestSingleWriterPost, this); } void SharedMutexTest::TestSharedReadPre() { int success = embb_shared_mutex_init(&shared_mutex_); PT_ASSERT_EQ_MSG(success, EMBB_SUCCESS, "Failed to initialize shared mutex."); } void SharedMutexTest::TestSharedReadThreadMethod() { int success = embb_shared_mutex_try_lock_shared(&shared_mutex_); PT_ASSERT_EQ_MSG(success, EMBB_SUCCESS, "Failed to lock for reading."); int spin = 10000; while (--spin != 0); success = embb_shared_mutex_unlock_shared(&shared_mutex_); PT_ASSERT_EQ_MSG(success, EMBB_SUCCESS, "Failed to unlock (reading)."); } void SharedMutexTest::TestSharedReadPost() { embb_shared_mutex_destroy(&shared_mutex_); } void SharedMutexTest::TestMultipleWriterPre() { int success = embb_shared_mutex_init(&shared_mutex_); PT_ASSERT_EQ_MSG(success, EMBB_SUCCESS, "Failed to initialize shared mutex."); counter_ = 0; } void SharedMutexTest::TestMultipleWriterReaderMethod() { // Just add some contention int success = embb_shared_mutex_lock_shared(&shared_mutex_); if (success != EMBB_SUCCESS) return; success = embb_shared_mutex_unlock_shared(&shared_mutex_); PT_ASSERT_EQ_MSG(success, EMBB_SUCCESS, "Failed to unlock (reading)."); } void SharedMutexTest::TestMultipleWriterWriterMethod() { int success = embb_shared_mutex_lock(&shared_mutex_); PT_ASSERT_EQ_MSG(success, EMBB_SUCCESS, "Failed to lock for writing."); ++counter_; success = embb_shared_mutex_unlock(&shared_mutex_); PT_ASSERT_EQ_MSG(success, EMBB_SUCCESS, "Failed to unlock (writing)."); } void SharedMutexTest::TestMultipleWriterPost() { PT_ASSERT_EQ_MSG(counter_, num_iterations_ * (num_threads_ / 2), "Counter value is inconsistent."); embb_shared_mutex_destroy(&shared_mutex_); } void SharedMutexTest::TestSingleWriterPre() { int success = embb_shared_mutex_init(&shared_mutex_); PT_ASSERT_EQ_MSG(success, EMBB_SUCCESS, "Failed to initialize shared mutex."); counter_ = 0; } void SharedMutexTest::TestSingleWriterReaderMethod() { // Just add some contention int success = embb_shared_mutex_lock_shared(&shared_mutex_); PT_ASSERT_EQ_MSG(success, EMBB_SUCCESS, "Failed to lock for reading."); success = embb_shared_mutex_unlock_shared(&shared_mutex_); PT_ASSERT_EQ_MSG(success, EMBB_SUCCESS, "Failed to unlock (reading)."); } void SharedMutexTest::TestSingleWriterWriterMethod() { int success = embb_shared_mutex_try_lock(&shared_mutex_); PT_ASSERT_EQ_MSG(success, EMBB_SUCCESS, "Failed to lock for writing."); ++counter_; success = embb_shared_mutex_unlock(&shared_mutex_); PT_ASSERT_EQ_MSG(success, EMBB_SUCCESS, "Failed to unlock (writing)."); } void SharedMutexTest::TestSingleWriterPost() { PT_ASSERT_EQ_MSG(counter_, num_iterations_, "Counter value is inconsistent."); embb_shared_mutex_destroy(&shared_mutex_); } } // namespace test } // namespace base } // namespace embb