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