| // |
| // |
| // Copyright 2019 gRPC 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 |
| // |
| // 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. |
| // |
| // |
| |
| #ifndef GRPCPP_IMPL_SYNC_H |
| #define GRPCPP_IMPL_SYNC_H |
| |
| #include <grpc/support/port_platform.h> |
| |
| #ifdef GPR_HAS_PTHREAD_H |
| #include <pthread.h> |
| #endif |
| |
| #include <mutex> |
| |
| #include "absl/synchronization/mutex.h" |
| |
| #include <grpc/support/log.h> |
| #include <grpc/support/sync.h> |
| #include <grpc/support/time.h> |
| |
| // The core library is not accessible in C++ codegen headers, and vice versa. |
| // Thus, we need to have duplicate headers with similar functionality. |
| // Make sure any change to this file is also reflected in |
| // src/core/lib/gprpp/sync.h too. |
| // |
| // Whenever possible, prefer "src/core/lib/gprpp/sync.h" over this file, |
| // since in core we do not rely on g_core_codegen_interface and hence do not |
| // pay the costs of virtual function calls. |
| |
| namespace grpc { |
| namespace internal { |
| |
| #ifdef GPR_ABSEIL_SYNC |
| |
| using Mutex = absl::Mutex; |
| using MutexLock = absl::MutexLock; |
| using ReleasableMutexLock = absl::ReleasableMutexLock; |
| using CondVar = absl::CondVar; |
| |
| #else |
| |
| class ABSL_LOCKABLE Mutex { |
| public: |
| Mutex() { gpr_mu_init(&mu_); } |
| ~Mutex() { gpr_mu_destroy(&mu_); } |
| |
| Mutex(const Mutex&) = delete; |
| Mutex& operator=(const Mutex&) = delete; |
| |
| void Lock() ABSL_EXCLUSIVE_LOCK_FUNCTION() { gpr_mu_lock(&mu_); } |
| void Unlock() ABSL_UNLOCK_FUNCTION() { gpr_mu_unlock(&mu_); } |
| |
| private: |
| union { |
| gpr_mu mu_; |
| std::mutex do_not_use_sth_; |
| #ifdef GPR_HAS_PTHREAD_H |
| pthread_mutex_t do_not_use_pth_; |
| #endif |
| }; |
| |
| friend class CondVar; |
| }; |
| |
| class ABSL_SCOPED_LOCKABLE MutexLock { |
| public: |
| explicit MutexLock(Mutex* mu) ABSL_EXCLUSIVE_LOCK_FUNCTION(mu) : mu_(mu) { |
| mu_->Lock(); |
| } |
| ~MutexLock() ABSL_UNLOCK_FUNCTION() { mu_->Unlock(); } |
| |
| MutexLock(const MutexLock&) = delete; |
| MutexLock& operator=(const MutexLock&) = delete; |
| |
| private: |
| Mutex* const mu_; |
| }; |
| |
| class ABSL_SCOPED_LOCKABLE ReleasableMutexLock { |
| public: |
| explicit ReleasableMutexLock(Mutex* mu) ABSL_EXCLUSIVE_LOCK_FUNCTION(mu) |
| : mu_(mu) { |
| mu_->Lock(); |
| } |
| ~ReleasableMutexLock() ABSL_UNLOCK_FUNCTION() { |
| if (!released_) mu_->Unlock(); |
| } |
| |
| ReleasableMutexLock(const ReleasableMutexLock&) = delete; |
| ReleasableMutexLock& operator=(const ReleasableMutexLock&) = delete; |
| |
| void Release() ABSL_UNLOCK_FUNCTION() { |
| GPR_DEBUG_ASSERT(!released_); |
| released_ = true; |
| mu_->Unlock(); |
| } |
| |
| private: |
| Mutex* const mu_; |
| bool released_ = false; |
| }; |
| |
| class CondVar { |
| public: |
| CondVar() { gpr_cv_init(&cv_); } |
| ~CondVar() { gpr_cv_destroy(&cv_); } |
| |
| CondVar(const CondVar&) = delete; |
| CondVar& operator=(const CondVar&) = delete; |
| |
| void Signal() { gpr_cv_signal(&cv_); } |
| void SignalAll() { gpr_cv_broadcast(&cv_); } |
| |
| void Wait(Mutex* mu) { |
| gpr_cv_wait(&cv_, &mu->mu_, gpr_inf_future(GPR_CLOCK_REALTIME)); |
| } |
| |
| private: |
| gpr_cv cv_; |
| }; |
| |
| #endif // GPR_ABSEIL_SYNC |
| |
| template <typename Predicate> |
| GRPC_DEPRECATED("incompatible with thread safety analysis") |
| static void WaitUntil(CondVar* cv, Mutex* mu, Predicate pred) { |
| while (!pred()) { |
| cv->Wait(mu); |
| } |
| } |
| |
| } // namespace internal |
| } // namespace grpc |
| |
| #endif // GRPCPP_IMPL_SYNC_H |