aboutsummaryrefslogtreecommitdiff
path: root/pw_thread_threadx/test_threads.cc
blob: 24a18f8af3aa6fc1cf54308e7f463fd60e8ba0e3 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
// 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 <chrono>

#include "pw_assert/check.h"
#include "pw_chrono/system_clock.h"
#include "pw_log/log.h"
#include "pw_thread/non_portable_test_thread_options.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