third_party.pigweed.src/pw_thread_threadx/sleep.cc
Ewout van Bekkum 1dea2c8e0b threadx: fix for_at_least contract to add one tick
Fixes the ThreadX backends for pw::sync::Mutex,
pw::sync::BinarySemaphore, pw::sync::CountingSemaphore, and
pw::this_thread::sleep_for to add one tick when invoking the native
API to comply with the for_at_least contract as we do not know how
far we are into the current tick.

Note this is not observable without the use of an independent clock.

This also adds explicit downcasting from int64_t to ULONG ticks when
invoking native APIs.

Change-Id: Ia28e0149a5c6426f7363c0467815dc3dcac0a010
Reviewed-on: https://pigweed-review.googlesource.com/c/pigweed/pigweed/+/36923
Reviewed-by: Wyatt Hepler <hepler@google.com>
Pigweed-Auto-Submit: Ewout van Bekkum <ewout@google.com>
Commit-Queue: Auto-Submit <auto-submit@pigweed.google.com.iam.gserviceaccount.com>
2021-03-11 02:23:26 +00:00

54 lines
1.8 KiB
C++

// Copyright 2020 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/sleep.h"
#include <algorithm>
#include "pw_assert/assert.h"
#include "pw_chrono/system_clock.h"
#include "pw_chrono_threadx/system_clock_constants.h"
#include "pw_thread/id.h"
#include "tx_api.h"
using pw::chrono::SystemClock;
namespace pw::this_thread {
void sleep_for(chrono::SystemClock::duration for_at_least) {
PW_DCHECK(get_id() != thread::Id());
// Yield for negative and zero length durations.
if (for_at_least <= chrono::SystemClock::duration::zero()) {
tx_thread_relinquish();
return;
}
// On a tick based kernel we cannot tell how far along we are on the current
// tick, ergo we add one whole tick to the final duration.
constexpr SystemClock::duration kMaxTimeoutMinusOne =
pw::chrono::threadx::kMaxTimeout - SystemClock::duration(1);
while (for_at_least > kMaxTimeoutMinusOne) {
const UINT result =
tx_thread_sleep(static_cast<ULONG>(kMaxTimeoutMinusOne.count()));
PW_CHECK_UINT_EQ(TX_SUCCESS, result);
for_at_least -= kMaxTimeoutMinusOne;
}
const UINT result =
tx_thread_sleep(static_cast<ULONG>(for_at_least.count() + 1));
PW_CHECK_UINT_EQ(TX_SUCCESS, result);
}
} // namespace pw::this_thread