// Copyright 2014 The Crashpad Authors. 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. #ifndef CRASHPAD_UTIL_SYNCHRONIZATION_SEMAPHORE_H_ #define CRASHPAD_UTIL_SYNCHRONIZATION_SEMAPHORE_H_ #include #include "build/build_config.h" #if BUILDFLAG(IS_APPLE) #include #elif BUILDFLAG(IS_WIN) #include #elif BUILDFLAG(IS_ANDROID) #include #include #else #include #endif namespace crashpad { //! \brief An anonymous in-process counting sempahore. class Semaphore { public: //! \brief A TimedWait() argument that causes an indefinite wait. static constexpr double kIndefiniteWait = std::numeric_limits::infinity(); //! \brief Initializes the semaphore. //! //! \param[in] value The initial value of the semaphore. //! //! If the semaphore cannot be created, execution is terminated. explicit Semaphore(int value); ~Semaphore(); //! \brief Performs the wait (or “procure”) operation on the semaphore. //! //! Atomically decrements the value of the semaphore by 1. If the new value is //! negative, this function blocks and will not return until the semaphore’s //! value is incremented to 0 by Signal(). //! //! \sa TimedWait() void Wait(); //! \brief Performs a timed wait (or “procure”) operation on the semaphore. //! //! \param[in] seconds The maximum number of seconds to wait for the operation //! to complete. If \a seconds is #kIndefiniteWait, this method behaves as //! Wait(), and will not time out. //! //! \return `false` if the wait timed out, `true` otherwise. //! //! This method is simlar to Wait(), except that the amount of time that it //! blocks is limited. bool TimedWait(double seconds); //! \brief Performs the signal (or “post”) operation on the semaphore. //! //! Atomically increments the value of the semaphore by 1. If the new value is //! 0, a caller blocked in Wait() will be awakened. void Signal(); private: #if BUILDFLAG(IS_APPLE) dispatch_semaphore_t semaphore_; #elif BUILDFLAG(IS_WIN) HANDLE semaphore_; #elif BUILDFLAG(IS_ANDROID) std::condition_variable cv_; std::mutex mutex_; int value_; #else sem_t semaphore_; #endif }; } // namespace crashpad #endif // CRASHPAD_UTIL_SYNCHRONIZATION_SEMAPHORE_H_