forked from eclipse-iceoryx/iceoryx
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
iox-eclipse-iceoryx#2301 Add SpinSemaphore
- Loading branch information
1 parent
c983dcc
commit d3182ac
Showing
13 changed files
with
371 additions
and
53 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
86 changes: 86 additions & 0 deletions
86
iceoryx_hoofs/concurrent/sync/include/iox/spin_semaphore.hpp
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,86 @@ | ||
|
||
// Copyright (c) 2024 by ekxide IO GmbH. All rights reserved. | ||
// | ||
// 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. | ||
// | ||
// SPDX-License-Identifier: Apache-2.0 | ||
|
||
#ifndef IOX_HOOFS_CONCURRENT_SYNC_SPIN_SEMAPHORE_HPP | ||
#define IOX_HOOFS_CONCURRENT_SYNC_SPIN_SEMAPHORE_HPP | ||
|
||
#include "iox/atomic.hpp" | ||
#include "iox/deadline_timer.hpp" | ||
#include "iox/detail/adaptive_wait.hpp" | ||
#include "iox/optional.hpp" | ||
#include "iox/semaphore_interface.hpp" | ||
#include "iox/spin_lock.hpp" | ||
|
||
namespace iox | ||
{ | ||
namespace concurrent | ||
{ | ||
class SpinSemaphoreBuilder; | ||
|
||
class SpinSemaphore : public detail::SemaphoreInterface<SpinSemaphore> | ||
{ | ||
public: | ||
using Builder = SpinSemaphoreBuilder; | ||
|
||
SpinSemaphore(const SpinSemaphore&) = delete; | ||
SpinSemaphore(SpinSemaphore&&) = delete; | ||
SpinSemaphore& operator=(const SpinSemaphore&) = delete; | ||
SpinSemaphore& operator=(SpinSemaphore&&) = delete; | ||
|
||
~SpinSemaphore() noexcept; | ||
|
||
private: | ||
friend class optional<SpinSemaphore>; | ||
friend class detail::SemaphoreInterface<SpinSemaphore>; | ||
|
||
explicit SpinSemaphore(int32_t initial_value) noexcept; | ||
|
||
expected<void, SemaphoreError> post_impl() noexcept; | ||
|
||
expected<void, SemaphoreError> wait_impl() noexcept; | ||
|
||
expected<bool, SemaphoreError> try_wait_impl() noexcept; | ||
|
||
expected<SemaphoreWaitState, SemaphoreError> timed_wait_impl(const units::Duration& timeout) noexcept; | ||
|
||
private: | ||
concurrent::Atomic<int32_t> m_count{0}; | ||
concurrent::Atomic<bool> m_to_be_destroyed{false}; | ||
optional<concurrent::SpinLock> m_spinlock; | ||
}; | ||
|
||
class SpinSemaphoreBuilder | ||
{ | ||
/// @brief Set the initial value of the spin semaphore | ||
IOX_BUILDER_PARAMETER(uint32_t, initialValue, 0U) | ||
|
||
/// @brief Set if the spin semaphore can be stored in the shared memory | ||
/// for inter process usage | ||
IOX_BUILDER_PARAMETER(bool, isInterProcessCapable, true) | ||
|
||
public: | ||
/// @brief Create a spin semaphore | ||
/// @param[in] uninitializedSemaphore since the semaphore is not movable the user has to provide | ||
/// memory to store the semaphore into - packed in an optional | ||
/// @return an error describing the failure or success | ||
expected<void, SemaphoreError> create(optional<SpinSemaphore>& uninitializedSemaphore) const noexcept; | ||
}; | ||
|
||
} // namespace concurrent | ||
} // namespace iox | ||
|
||
#endif // IOX_HOOFS_CONCURRENT_SYNC_SPIN_LOCK_HPP |
112 changes: 112 additions & 0 deletions
112
iceoryx_hoofs/concurrent/sync/source/spin_semaphore.cpp
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,112 @@ | ||
// Copyright (c) 2024 by ekxide IO GmbH. All rights reserved. | ||
// | ||
// 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. | ||
// | ||
// SPDX-License-Identifier: Apache-2.0 | ||
|
||
#include "iox/spin_semaphore.hpp" | ||
#include "iox/detail/adaptive_wait.hpp" | ||
|
||
namespace iox | ||
{ | ||
namespace concurrent | ||
{ | ||
expected<void, SemaphoreError> | ||
SpinSemaphoreBuilder::create(optional<SpinSemaphore>& uninitializedSemaphore) const noexcept | ||
{ | ||
if (m_initialValue > IOX_SEM_VALUE_MAX) | ||
{ | ||
IOX_LOG(ERROR, | ||
"The spin semaphore initial value of " << m_initialValue << " exceeds the maximum semaphore value " | ||
<< IOX_SEM_VALUE_MAX); | ||
return err(SemaphoreError::SEMAPHORE_OVERFLOW); | ||
} | ||
|
||
uninitializedSemaphore.emplace(static_cast<int32_t>(m_initialValue)); | ||
return ok(); | ||
} | ||
|
||
SpinSemaphore::SpinSemaphore(int32_t initial_value) noexcept | ||
: m_count(initial_value) | ||
{ | ||
SpinLockBuilder() | ||
.is_inter_process_capable(true) | ||
.lock_behavior(LockBehavior::NORMAL) | ||
.create(m_spinlock) | ||
.expect("Failed to create Lock"); | ||
} | ||
|
||
SpinSemaphore::~SpinSemaphore() noexcept | ||
{ | ||
m_to_be_destroyed = true; | ||
} | ||
|
||
expected<void, SemaphoreError> SpinSemaphore::post_impl() noexcept | ||
{ | ||
std::lock_guard<concurrent::SpinLock> lock(*m_spinlock); | ||
|
||
if (m_count.load(std::memory_order_relaxed) == IOX_SEM_VALUE_MAX) | ||
{ | ||
return err(SemaphoreError::SEMAPHORE_OVERFLOW); | ||
} | ||
|
||
++m_count; | ||
return ok(); | ||
} | ||
|
||
expected<void, SemaphoreError> SpinSemaphore::wait_impl() noexcept | ||
{ | ||
detail::adaptive_wait spinner; | ||
spinner.wait_loop([this] { | ||
auto wait_result = this->tryWait(); | ||
return wait_result.has_value() && !wait_result.value(); | ||
}); | ||
return ok(); | ||
} | ||
|
||
expected<bool, SemaphoreError> SpinSemaphore::try_wait_impl() noexcept | ||
{ | ||
std::lock_guard<concurrent::SpinLock> lock(*m_spinlock); | ||
if (m_to_be_destroyed.load(std::memory_order_relaxed)) | ||
{ | ||
return ok(true); | ||
} | ||
if (m_count.load(std::memory_order_relaxed) > 0) | ||
{ | ||
--m_count; | ||
return ok(true); | ||
} | ||
return ok(false); | ||
} | ||
|
||
expected<SemaphoreWaitState, SemaphoreError> SpinSemaphore::timed_wait_impl(const units::Duration& timeout) noexcept | ||
{ | ||
iox::deadline_timer deadline_timer(timeout); | ||
detail::adaptive_wait spinner; | ||
|
||
auto ret_val = SemaphoreWaitState::TIMEOUT; | ||
spinner.wait_loop([this, &deadline_timer, &ret_val] { | ||
auto wait_result = this->tryWait(); | ||
|
||
if (wait_result.has_value() && wait_result.value()) | ||
{ | ||
ret_val = SemaphoreWaitState::NO_TIMEOUT; | ||
return false; | ||
} | ||
return !deadline_timer.hasExpired(); | ||
}); | ||
|
||
return ok(ret_val); | ||
} | ||
} // namespace concurrent | ||
} // namespace iox |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
55 changes: 55 additions & 0 deletions
55
iceoryx_hoofs/posix/sync/include/iox/detail/semaphore_helper.hpp
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,55 @@ | ||
// Copyright (c) 2022 by Apex.AI Inc. All rights reserved. | ||
// Copyright (c) 2024 by ekxide IO GmbH. All rights reserved. | ||
// | ||
// 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. | ||
// | ||
// SPDX-License-Identifier: Apache-2.0 | ||
|
||
#ifndef IOX_HOOFS_POSIX_SYNC_SEMAPHORE_HELPER_HPP | ||
#define IOX_HOOFS_POSIX_SYNC_SEMAPHORE_HELPER_HPP | ||
|
||
#include "iceoryx_platform/semaphore.hpp" | ||
#include "iox/duration.hpp" | ||
#include "iox/expected.hpp" | ||
#include "iox/semaphore_interface.hpp" | ||
|
||
namespace iox | ||
{ | ||
namespace detail | ||
{ | ||
/// @brief Increments the semaphore by one | ||
/// @return Fails when the value of the semaphore overflows or when the | ||
/// semaphore was removed from outside the process | ||
expected<void, SemaphoreError> sem_post(iox_sem_t* handle) noexcept; | ||
|
||
/// @brief Decrements the semaphore by one. When the semaphore value is zero | ||
/// it blocks until the semaphore value is greater zero | ||
/// @return Fails when semaphore was removed from outside the process | ||
expected<void, SemaphoreError> sem_wait(iox_sem_t* handle) noexcept; | ||
|
||
/// @brief Tries to decrement the semaphore by one. When the semaphore value is zero | ||
/// it returns false otherwise it returns true and decrement the value by one. | ||
/// @return Fails when semaphore was removed from outside the process | ||
expected<bool, SemaphoreError> sem_try_wait(iox_sem_t* handle) noexcept; | ||
|
||
/// @brief Tries to decrement the semaphore by one. When the semaphore value is zero | ||
/// it waits until the timeout has passed. | ||
/// @return If during the timeout time the semaphore value increases to non zero | ||
/// it returns SemaphoreWaitState::NO_TIMEOUT and decreases the semaphore by one | ||
/// otherwise returns SemaphoreWaitState::TIMEOUT | ||
expected<SemaphoreWaitState, SemaphoreError> sem_timed_wait(iox_sem_t* handle, const units::Duration& timeout) noexcept; | ||
|
||
} // namespace detail | ||
} // namespace iox | ||
|
||
#endif // IOX_HOOFS_POSIX_SYNC_SEMAPHORE_HELPER_HPP |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.