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