| // Copyright 2021 The Pigweed Authors |
| // |
| // 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 |
| // |
| // https://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. |
| |
| #include "pw_thread/test_threads.h" |
| |
| #include <chrono> |
| |
| #include "pw_assert/check.h" |
| #include "pw_chrono/system_clock.h" |
| #include "pw_log/log.h" |
| #include "pw_thread/sleep.h" |
| #include "pw_thread_threadx/context.h" |
| #include "pw_thread_threadx/options.h" |
| |
| namespace pw::thread::test { |
| namespace { |
| |
| std::array<threadx::ContextWithStack<>, 2> thread_contexts; |
| |
| } // namespace |
| |
| const Options& TestOptionsThread0() { |
| static constexpr threadx::Options thread_0_options = |
| threadx::Options() |
| .set_name("pw::TestThread0") |
| .set_context(thread_contexts[0]); |
| return thread_0_options; |
| } |
| |
| const Options& TestOptionsThread1() { |
| static constexpr threadx::Options thread_1_options = |
| threadx::Options() |
| .set_name("pw::TestThread1") |
| .set_context(thread_contexts[1]); |
| return thread_1_options; |
| } |
| |
| void WaitUntilDetachedThreadsCleanedUp() { |
| // ThreadX does not permit the running thread to delete itself, which means |
| // we have to do this to re-use a TCB as otherwise we will be leaking stale |
| // references in the kernel. |
| for (auto& context : thread_contexts) { |
| if (context.tcb().tx_thread_id != TX_THREAD_ID) { |
| // The TCB was either not used or was already deleted. Note that |
| // tx_thread_terminate does NOT clear this state by design. |
| continue; |
| } |
| |
| // If the thread was created but has not been deleted, it means that the |
| // thread was detached before it finished. Wait until it is completed. |
| while (context.tcb().tx_thread_state != TX_COMPLETED) { |
| pw::this_thread::sleep_for( |
| chrono::SystemClock::for_at_least(std::chrono::milliseconds(1))); |
| } |
| |
| const UINT result = tx_thread_delete(&context.tcb()); |
| PW_CHECK_UINT_EQ(TX_SUCCESS, result, "Failed to delete thread"); |
| } |
| } |
| |
| } // namespace pw::thread::test |