1*35238bceSAndroid Build Coastguard Worker #ifndef _DESPINBARRIER_HPP 2*35238bceSAndroid Build Coastguard Worker #define _DESPINBARRIER_HPP 3*35238bceSAndroid Build Coastguard Worker /*------------------------------------------------------------------------- 4*35238bceSAndroid Build Coastguard Worker * drawElements C++ Base Library 5*35238bceSAndroid Build Coastguard Worker * ----------------------------- 6*35238bceSAndroid Build Coastguard Worker * 7*35238bceSAndroid Build Coastguard Worker * Copyright 2015 The Android Open Source Project 8*35238bceSAndroid Build Coastguard Worker * 9*35238bceSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License"); 10*35238bceSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License. 11*35238bceSAndroid Build Coastguard Worker * You may obtain a copy of the License at 12*35238bceSAndroid Build Coastguard Worker * 13*35238bceSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0 14*35238bceSAndroid Build Coastguard Worker * 15*35238bceSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software 16*35238bceSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS, 17*35238bceSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 18*35238bceSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and 19*35238bceSAndroid Build Coastguard Worker * limitations under the License. 20*35238bceSAndroid Build Coastguard Worker * 21*35238bceSAndroid Build Coastguard Worker *//*! 22*35238bceSAndroid Build Coastguard Worker * \file 23*35238bceSAndroid Build Coastguard Worker * \brief Cross-thread barrier. 24*35238bceSAndroid Build Coastguard Worker *//*--------------------------------------------------------------------*/ 25*35238bceSAndroid Build Coastguard Worker 26*35238bceSAndroid Build Coastguard Worker #include "deDefs.hpp" 27*35238bceSAndroid Build Coastguard Worker #include "deAtomic.h" 28*35238bceSAndroid Build Coastguard Worker 29*35238bceSAndroid Build Coastguard Worker namespace de 30*35238bceSAndroid Build Coastguard Worker { 31*35238bceSAndroid Build Coastguard Worker 32*35238bceSAndroid Build Coastguard Worker /*--------------------------------------------------------------------*//*! 33*35238bceSAndroid Build Coastguard Worker * \brief Cross-thread barrier 34*35238bceSAndroid Build Coastguard Worker * 35*35238bceSAndroid Build Coastguard Worker * SpinBarrier provides barrier implementation that uses spin loop for 36*35238bceSAndroid Build Coastguard Worker * waiting for other threads. Threads may choose to wait in tight loop 37*35238bceSAndroid Build Coastguard Worker * (WAIT_MODE_BUSY) or yield between iterations (WAIT_MODE_YIELD). 38*35238bceSAndroid Build Coastguard Worker * 39*35238bceSAndroid Build Coastguard Worker * It is not recommended to use WAIT_MODE_BUSY when there are more threads 40*35238bceSAndroid Build Coastguard Worker * than number of cores participating in the barrier as it will lead to 41*35238bceSAndroid Build Coastguard Worker * priority inversion and dramatic slowdown. For that reason WAIT_MODE_AUTO 42*35238bceSAndroid Build Coastguard Worker * is provided, which selects between busy and yielding waiting based on 43*35238bceSAndroid Build Coastguard Worker * number of threads. 44*35238bceSAndroid Build Coastguard Worker *//*--------------------------------------------------------------------*/ 45*35238bceSAndroid Build Coastguard Worker class SpinBarrier 46*35238bceSAndroid Build Coastguard Worker { 47*35238bceSAndroid Build Coastguard Worker public: 48*35238bceSAndroid Build Coastguard Worker enum WaitMode 49*35238bceSAndroid Build Coastguard Worker { 50*35238bceSAndroid Build Coastguard Worker WAIT_MODE_BUSY = 0, //! Wait in tight spin loop. 51*35238bceSAndroid Build Coastguard Worker WAIT_MODE_YIELD, //! Call deYield() between spin loop iterations. 52*35238bceSAndroid Build Coastguard Worker WAIT_MODE_AUTO, //! Use WAIT_MODE_BUSY loop if #threads <= #cores, otherwise WAIT_MODE_YIELD. 53*35238bceSAndroid Build Coastguard Worker 54*35238bceSAndroid Build Coastguard Worker WAIT_MODE_LAST 55*35238bceSAndroid Build Coastguard Worker }; 56*35238bceSAndroid Build Coastguard Worker 57*35238bceSAndroid Build Coastguard Worker SpinBarrier(int32_t numThreads); 58*35238bceSAndroid Build Coastguard Worker ~SpinBarrier(void); 59*35238bceSAndroid Build Coastguard Worker 60*35238bceSAndroid Build Coastguard Worker //! Reset barrier. Not thread-safe, e.g. no other thread can 61*35238bceSAndroid Build Coastguard Worker //! be calling sync() or removeThread() at the same time. 62*35238bceSAndroid Build Coastguard Worker void reset(uint32_t numThreads); 63*35238bceSAndroid Build Coastguard Worker 64*35238bceSAndroid Build Coastguard Worker //! Wait until all threads (determined by active thread count) 65*35238bceSAndroid Build Coastguard Worker //! have entered sync(). 66*35238bceSAndroid Build Coastguard Worker void sync(WaitMode mode); 67*35238bceSAndroid Build Coastguard Worker 68*35238bceSAndroid Build Coastguard Worker //! Remove thread from barrier (decrements active thread count). 69*35238bceSAndroid Build Coastguard Worker //! Can be called concurrently with sync() or removeThread(). 70*35238bceSAndroid Build Coastguard Worker void removeThread(WaitMode mode); 71*35238bceSAndroid Build Coastguard Worker 72*35238bceSAndroid Build Coastguard Worker private: 73*35238bceSAndroid Build Coastguard Worker SpinBarrier(const SpinBarrier &); 74*35238bceSAndroid Build Coastguard Worker SpinBarrier operator=(const SpinBarrier &); 75*35238bceSAndroid Build Coastguard Worker 76*35238bceSAndroid Build Coastguard Worker const uint32_t m_numCores; 77*35238bceSAndroid Build Coastguard Worker 78*35238bceSAndroid Build Coastguard Worker volatile int32_t m_numThreads; 79*35238bceSAndroid Build Coastguard Worker volatile int32_t m_numEntered; 80*35238bceSAndroid Build Coastguard Worker volatile int32_t m_numLeaving; 81*35238bceSAndroid Build Coastguard Worker volatile int32_t m_numRemoved; 82*35238bceSAndroid Build Coastguard Worker }; 83*35238bceSAndroid Build Coastguard Worker 84*35238bceSAndroid Build Coastguard Worker void SpinBarrier_selfTest(void); 85*35238bceSAndroid Build Coastguard Worker 86*35238bceSAndroid Build Coastguard Worker } // namespace de 87*35238bceSAndroid Build Coastguard Worker 88*35238bceSAndroid Build Coastguard Worker #endif // _DESPINBARRIER_HPP 89