xref: /aosp_15_r20/external/deqp/framework/common/tcuEither.hpp (revision 35238bce31c2a825756842865a792f8cf7f89930)
1*35238bceSAndroid Build Coastguard Worker #ifndef _TCUEITHER_HPP
2*35238bceSAndroid Build Coastguard Worker #define _TCUEITHER_HPP
3*35238bceSAndroid Build Coastguard Worker /*-------------------------------------------------------------------------
4*35238bceSAndroid Build Coastguard Worker  * drawElements Quality Program Tester Core
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 Template class that is either type of First or Second.
24*35238bceSAndroid Build Coastguard Worker  *//*--------------------------------------------------------------------*/
25*35238bceSAndroid Build Coastguard Worker 
26*35238bceSAndroid Build Coastguard Worker #include "tcuDefs.hpp"
27*35238bceSAndroid Build Coastguard Worker 
28*35238bceSAndroid Build Coastguard Worker namespace tcu
29*35238bceSAndroid Build Coastguard Worker {
30*35238bceSAndroid Build Coastguard Worker 
31*35238bceSAndroid Build Coastguard Worker /*--------------------------------------------------------------------*//*!
32*35238bceSAndroid Build Coastguard Worker  * \brief Object containing Either First or Second type of object
33*35238bceSAndroid Build Coastguard Worker  *
34*35238bceSAndroid Build Coastguard Worker  * \note Type First and Second are always aligned to same alignment as
35*35238bceSAndroid Build Coastguard Worker  *         uint64_t.
36*35238bceSAndroid Build Coastguard Worker  * \note This type always uses at least sizeof(bool) + max(sizeof(First*),
37*35238bceSAndroid Build Coastguard Worker  *         sizeof(Second*)) + sizeof(uint64_t) of memory.
38*35238bceSAndroid Build Coastguard Worker  *//*--------------------------------------------------------------------*/
39*35238bceSAndroid Build Coastguard Worker template <typename First, typename Second>
40*35238bceSAndroid Build Coastguard Worker class Either
41*35238bceSAndroid Build Coastguard Worker {
42*35238bceSAndroid Build Coastguard Worker public:
43*35238bceSAndroid Build Coastguard Worker     Either(const First &first);
44*35238bceSAndroid Build Coastguard Worker     Either(const Second &second);
45*35238bceSAndroid Build Coastguard Worker     ~Either(void);
46*35238bceSAndroid Build Coastguard Worker 
47*35238bceSAndroid Build Coastguard Worker     Either(const Either<First, Second> &other);
48*35238bceSAndroid Build Coastguard Worker     Either &operator=(const Either<First, Second> &other);
49*35238bceSAndroid Build Coastguard Worker 
50*35238bceSAndroid Build Coastguard Worker     Either &operator=(const First &first);
51*35238bceSAndroid Build Coastguard Worker     Either &operator=(const Second &second);
52*35238bceSAndroid Build Coastguard Worker 
53*35238bceSAndroid Build Coastguard Worker     bool isFirst(void) const;
54*35238bceSAndroid Build Coastguard Worker     bool isSecond(void) const;
55*35238bceSAndroid Build Coastguard Worker 
56*35238bceSAndroid Build Coastguard Worker     const First &getFirst(void) const;
57*35238bceSAndroid Build Coastguard Worker     const Second &getSecond(void) const;
58*35238bceSAndroid Build Coastguard Worker 
59*35238bceSAndroid Build Coastguard Worker     template <typename Type>
60*35238bceSAndroid Build Coastguard Worker     const Type &get(void) const;
61*35238bceSAndroid Build Coastguard Worker 
62*35238bceSAndroid Build Coastguard Worker     template <typename Type>
63*35238bceSAndroid Build Coastguard Worker     bool is(void) const;
64*35238bceSAndroid Build Coastguard Worker 
65*35238bceSAndroid Build Coastguard Worker private:
66*35238bceSAndroid Build Coastguard Worker     void release(void);
67*35238bceSAndroid Build Coastguard Worker 
68*35238bceSAndroid Build Coastguard Worker     bool m_isFirst;
69*35238bceSAndroid Build Coastguard Worker 
70*35238bceSAndroid Build Coastguard Worker     union
71*35238bceSAndroid Build Coastguard Worker     {
72*35238bceSAndroid Build Coastguard Worker         First *m_first;
73*35238bceSAndroid Build Coastguard Worker         Second *m_second;
74*35238bceSAndroid Build Coastguard Worker     };
75*35238bceSAndroid Build Coastguard Worker 
76*35238bceSAndroid Build Coastguard Worker     union
77*35238bceSAndroid Build Coastguard Worker     {
78*35238bceSAndroid Build Coastguard Worker         uint8_t m_data[sizeof(First) > sizeof(Second) ? sizeof(First) : sizeof(Second)];
79*35238bceSAndroid Build Coastguard Worker         uint64_t m_align;
80*35238bceSAndroid Build Coastguard Worker     };
81*35238bceSAndroid Build Coastguard Worker } DE_WARN_UNUSED_TYPE;
82*35238bceSAndroid Build Coastguard Worker 
83*35238bceSAndroid Build Coastguard Worker namespace EitherDetail
84*35238bceSAndroid Build Coastguard Worker {
85*35238bceSAndroid Build Coastguard Worker 
86*35238bceSAndroid Build Coastguard Worker template <typename Type, typename First, typename Second>
87*35238bceSAndroid Build Coastguard Worker struct Get;
88*35238bceSAndroid Build Coastguard Worker 
89*35238bceSAndroid Build Coastguard Worker template <typename First, typename Second>
90*35238bceSAndroid Build Coastguard Worker struct Get<First, First, Second>
91*35238bceSAndroid Build Coastguard Worker {
gettcu::EitherDetail::Get92*35238bceSAndroid Build Coastguard Worker     static const First &get(const Either<First, Second> &either)
93*35238bceSAndroid Build Coastguard Worker     {
94*35238bceSAndroid Build Coastguard Worker         return either.getFirst();
95*35238bceSAndroid Build Coastguard Worker     }
96*35238bceSAndroid Build Coastguard Worker };
97*35238bceSAndroid Build Coastguard Worker 
98*35238bceSAndroid Build Coastguard Worker template <typename First, typename Second>
99*35238bceSAndroid Build Coastguard Worker struct Get<Second, First, Second>
100*35238bceSAndroid Build Coastguard Worker {
gettcu::EitherDetail::Get101*35238bceSAndroid Build Coastguard Worker     static const Second &get(const Either<First, Second> &either)
102*35238bceSAndroid Build Coastguard Worker     {
103*35238bceSAndroid Build Coastguard Worker         return either.getSecond();
104*35238bceSAndroid Build Coastguard Worker     }
105*35238bceSAndroid Build Coastguard Worker };
106*35238bceSAndroid Build Coastguard Worker 
107*35238bceSAndroid Build Coastguard Worker template <typename Type, typename First, typename Second>
get(const Either<First,Second> & either)108*35238bceSAndroid Build Coastguard Worker const Type &get(const Either<First, Second> &either)
109*35238bceSAndroid Build Coastguard Worker {
110*35238bceSAndroid Build Coastguard Worker     return Get<Type, First, Second>::get(either);
111*35238bceSAndroid Build Coastguard Worker }
112*35238bceSAndroid Build Coastguard Worker 
113*35238bceSAndroid Build Coastguard Worker template <typename Type, typename First, typename Second>
114*35238bceSAndroid Build Coastguard Worker struct Is;
115*35238bceSAndroid Build Coastguard Worker 
116*35238bceSAndroid Build Coastguard Worker template <typename First, typename Second>
117*35238bceSAndroid Build Coastguard Worker struct Is<First, First, Second>
118*35238bceSAndroid Build Coastguard Worker {
istcu::EitherDetail::Is119*35238bceSAndroid Build Coastguard Worker     static bool is(const Either<First, Second> &either)
120*35238bceSAndroid Build Coastguard Worker     {
121*35238bceSAndroid Build Coastguard Worker         return either.isFirst();
122*35238bceSAndroid Build Coastguard Worker     }
123*35238bceSAndroid Build Coastguard Worker };
124*35238bceSAndroid Build Coastguard Worker 
125*35238bceSAndroid Build Coastguard Worker template <typename First, typename Second>
126*35238bceSAndroid Build Coastguard Worker struct Is<Second, First, Second>
127*35238bceSAndroid Build Coastguard Worker {
istcu::EitherDetail::Is128*35238bceSAndroid Build Coastguard Worker     static bool is(const Either<First, Second> &either)
129*35238bceSAndroid Build Coastguard Worker     {
130*35238bceSAndroid Build Coastguard Worker         return either.isSecond();
131*35238bceSAndroid Build Coastguard Worker     }
132*35238bceSAndroid Build Coastguard Worker };
133*35238bceSAndroid Build Coastguard Worker 
134*35238bceSAndroid Build Coastguard Worker template <typename Type, typename First, typename Second>
is(const Either<First,Second> & either)135*35238bceSAndroid Build Coastguard Worker bool is(const Either<First, Second> &either)
136*35238bceSAndroid Build Coastguard Worker {
137*35238bceSAndroid Build Coastguard Worker     return Is<Type, First, Second>::is(either);
138*35238bceSAndroid Build Coastguard Worker }
139*35238bceSAndroid Build Coastguard Worker 
140*35238bceSAndroid Build Coastguard Worker } // namespace EitherDetail
141*35238bceSAndroid Build Coastguard Worker 
142*35238bceSAndroid Build Coastguard Worker template <typename First, typename Second>
release(void)143*35238bceSAndroid Build Coastguard Worker void Either<First, Second>::release(void)
144*35238bceSAndroid Build Coastguard Worker {
145*35238bceSAndroid Build Coastguard Worker     if (m_isFirst)
146*35238bceSAndroid Build Coastguard Worker         m_first->~First();
147*35238bceSAndroid Build Coastguard Worker     else
148*35238bceSAndroid Build Coastguard Worker         m_second->~Second();
149*35238bceSAndroid Build Coastguard Worker 
150*35238bceSAndroid Build Coastguard Worker     m_isFirst = true;
151*35238bceSAndroid Build Coastguard Worker     m_first   = DE_NULL;
152*35238bceSAndroid Build Coastguard Worker }
153*35238bceSAndroid Build Coastguard Worker 
154*35238bceSAndroid Build Coastguard Worker template <typename First, typename Second>
Either(const First & first)155*35238bceSAndroid Build Coastguard Worker Either<First, Second>::Either(const First &first) : m_isFirst(true)
156*35238bceSAndroid Build Coastguard Worker {
157*35238bceSAndroid Build Coastguard Worker     m_first = new (m_data) First(first);
158*35238bceSAndroid Build Coastguard Worker }
159*35238bceSAndroid Build Coastguard Worker 
160*35238bceSAndroid Build Coastguard Worker template <typename First, typename Second>
Either(const Second & second)161*35238bceSAndroid Build Coastguard Worker Either<First, Second>::Either(const Second &second) : m_isFirst(false)
162*35238bceSAndroid Build Coastguard Worker {
163*35238bceSAndroid Build Coastguard Worker     m_second = new (m_data) Second(second);
164*35238bceSAndroid Build Coastguard Worker }
165*35238bceSAndroid Build Coastguard Worker 
166*35238bceSAndroid Build Coastguard Worker template <typename First, typename Second>
~Either(void)167*35238bceSAndroid Build Coastguard Worker Either<First, Second>::~Either(void)
168*35238bceSAndroid Build Coastguard Worker {
169*35238bceSAndroid Build Coastguard Worker     release();
170*35238bceSAndroid Build Coastguard Worker }
171*35238bceSAndroid Build Coastguard Worker 
172*35238bceSAndroid Build Coastguard Worker template <typename First, typename Second>
Either(const Either<First,Second> & other)173*35238bceSAndroid Build Coastguard Worker Either<First, Second>::Either(const Either<First, Second> &other) : m_isFirst(other.m_isFirst)
174*35238bceSAndroid Build Coastguard Worker {
175*35238bceSAndroid Build Coastguard Worker     if (m_isFirst)
176*35238bceSAndroid Build Coastguard Worker         m_first = new (m_data) First(*other.m_first);
177*35238bceSAndroid Build Coastguard Worker     else
178*35238bceSAndroid Build Coastguard Worker         m_second = new (m_data) Second(*other.m_second);
179*35238bceSAndroid Build Coastguard Worker }
180*35238bceSAndroid Build Coastguard Worker 
181*35238bceSAndroid Build Coastguard Worker template <typename First, typename Second>
operator =(const Either<First,Second> & other)182*35238bceSAndroid Build Coastguard Worker Either<First, Second> &Either<First, Second>::operator=(const Either<First, Second> &other)
183*35238bceSAndroid Build Coastguard Worker {
184*35238bceSAndroid Build Coastguard Worker     if (this == &other)
185*35238bceSAndroid Build Coastguard Worker         return *this;
186*35238bceSAndroid Build Coastguard Worker 
187*35238bceSAndroid Build Coastguard Worker     release();
188*35238bceSAndroid Build Coastguard Worker 
189*35238bceSAndroid Build Coastguard Worker     m_isFirst = other.m_isFirst;
190*35238bceSAndroid Build Coastguard Worker 
191*35238bceSAndroid Build Coastguard Worker     if (m_isFirst)
192*35238bceSAndroid Build Coastguard Worker         m_first = new (m_data) First(*other.m_first);
193*35238bceSAndroid Build Coastguard Worker     else
194*35238bceSAndroid Build Coastguard Worker         m_second = new (m_data) Second(*other.m_second);
195*35238bceSAndroid Build Coastguard Worker 
196*35238bceSAndroid Build Coastguard Worker     return *this;
197*35238bceSAndroid Build Coastguard Worker }
198*35238bceSAndroid Build Coastguard Worker 
199*35238bceSAndroid Build Coastguard Worker template <typename First, typename Second>
operator =(const First & first)200*35238bceSAndroid Build Coastguard Worker Either<First, Second> &Either<First, Second>::operator=(const First &first)
201*35238bceSAndroid Build Coastguard Worker {
202*35238bceSAndroid Build Coastguard Worker     release();
203*35238bceSAndroid Build Coastguard Worker 
204*35238bceSAndroid Build Coastguard Worker     m_isFirst = true;
205*35238bceSAndroid Build Coastguard Worker     m_first   = new (m_data) First(first);
206*35238bceSAndroid Build Coastguard Worker 
207*35238bceSAndroid Build Coastguard Worker     return *this;
208*35238bceSAndroid Build Coastguard Worker }
209*35238bceSAndroid Build Coastguard Worker 
210*35238bceSAndroid Build Coastguard Worker template <typename First, typename Second>
operator =(const Second & second)211*35238bceSAndroid Build Coastguard Worker Either<First, Second> &Either<First, Second>::operator=(const Second &second)
212*35238bceSAndroid Build Coastguard Worker {
213*35238bceSAndroid Build Coastguard Worker     release();
214*35238bceSAndroid Build Coastguard Worker 
215*35238bceSAndroid Build Coastguard Worker     m_isFirst = false;
216*35238bceSAndroid Build Coastguard Worker     m_second  = new (m_data) Second(second);
217*35238bceSAndroid Build Coastguard Worker 
218*35238bceSAndroid Build Coastguard Worker     return *this;
219*35238bceSAndroid Build Coastguard Worker }
220*35238bceSAndroid Build Coastguard Worker 
221*35238bceSAndroid Build Coastguard Worker template <typename First, typename Second>
isFirst(void) const222*35238bceSAndroid Build Coastguard Worker bool Either<First, Second>::isFirst(void) const
223*35238bceSAndroid Build Coastguard Worker {
224*35238bceSAndroid Build Coastguard Worker     return m_isFirst;
225*35238bceSAndroid Build Coastguard Worker }
226*35238bceSAndroid Build Coastguard Worker 
227*35238bceSAndroid Build Coastguard Worker template <typename First, typename Second>
isSecond(void) const228*35238bceSAndroid Build Coastguard Worker bool Either<First, Second>::isSecond(void) const
229*35238bceSAndroid Build Coastguard Worker {
230*35238bceSAndroid Build Coastguard Worker     return !m_isFirst;
231*35238bceSAndroid Build Coastguard Worker }
232*35238bceSAndroid Build Coastguard Worker 
233*35238bceSAndroid Build Coastguard Worker template <typename First, typename Second>
getFirst(void) const234*35238bceSAndroid Build Coastguard Worker const First &Either<First, Second>::getFirst(void) const
235*35238bceSAndroid Build Coastguard Worker {
236*35238bceSAndroid Build Coastguard Worker     DE_ASSERT(isFirst());
237*35238bceSAndroid Build Coastguard Worker     return *m_first;
238*35238bceSAndroid Build Coastguard Worker }
239*35238bceSAndroid Build Coastguard Worker 
240*35238bceSAndroid Build Coastguard Worker template <typename First, typename Second>
getSecond(void) const241*35238bceSAndroid Build Coastguard Worker const Second &Either<First, Second>::getSecond(void) const
242*35238bceSAndroid Build Coastguard Worker {
243*35238bceSAndroid Build Coastguard Worker     DE_ASSERT(isSecond());
244*35238bceSAndroid Build Coastguard Worker     return *m_second;
245*35238bceSAndroid Build Coastguard Worker }
246*35238bceSAndroid Build Coastguard Worker 
247*35238bceSAndroid Build Coastguard Worker template <typename First, typename Second>
248*35238bceSAndroid Build Coastguard Worker template <typename Type>
get(void) const249*35238bceSAndroid Build Coastguard Worker const Type &Either<First, Second>::get(void) const
250*35238bceSAndroid Build Coastguard Worker {
251*35238bceSAndroid Build Coastguard Worker     return EitherDetail::get<Type, First, Second>(*this);
252*35238bceSAndroid Build Coastguard Worker }
253*35238bceSAndroid Build Coastguard Worker 
254*35238bceSAndroid Build Coastguard Worker template <typename First, typename Second>
255*35238bceSAndroid Build Coastguard Worker template <typename Type>
is(void) const256*35238bceSAndroid Build Coastguard Worker bool Either<First, Second>::is(void) const
257*35238bceSAndroid Build Coastguard Worker {
258*35238bceSAndroid Build Coastguard Worker     return EitherDetail::is<Type, First, Second>(*this);
259*35238bceSAndroid Build Coastguard Worker }
260*35238bceSAndroid Build Coastguard Worker 
261*35238bceSAndroid Build Coastguard Worker void Either_selfTest(void);
262*35238bceSAndroid Build Coastguard Worker 
263*35238bceSAndroid Build Coastguard Worker } // namespace tcu
264*35238bceSAndroid Build Coastguard Worker 
265*35238bceSAndroid Build Coastguard Worker #endif // _TCUEITHER_HPP
266