blob: 2de584fab2351402064e07c264c260b992765e28 [file] [log] [blame]
ethannicholasd598f792016-07-25 10:08:54 -07001/*
2 * Copyright 2016 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
Ethan Nicholas0df1b042017-03-31 13:56:23 -04007
ethannicholasd598f792016-07-25 10:08:54 -07008#ifndef SKSL_CONTEXT
9#define SKSL_CONTEXT
10
11#include "ir/SkSLType.h"
ethannicholas22f939e2016-10-13 13:25:34 -070012#include "ir/SkSLExpression.h"
ethannicholasd598f792016-07-25 10:08:54 -070013
14namespace SkSL {
15
16/**
17 * Contains compiler-wide objects, which currently means the core types.
18 */
19class Context {
20public:
21 Context()
Ethan Nicholas0df1b042017-03-31 13:56:23 -040022 : fInvalid_Type(new Type(String("<INVALID>")))
23 , fVoid_Type(new Type(String("void")))
24 , fDouble_Type(new Type(String("double"), true))
Ethan Nicholas5af9ea32017-07-28 15:19:46 -040025 , fDouble2_Type(new Type(String("double2"), *fDouble_Type, 2))
26 , fDouble3_Type(new Type(String("double3"), *fDouble_Type, 3))
27 , fDouble4_Type(new Type(String("double4"), *fDouble_Type, 4))
Ethan Nicholas0df1b042017-03-31 13:56:23 -040028 , fFloat_Type(new Type(String("float"), true, { fDouble_Type.get() }))
Ethan Nicholas5af9ea32017-07-28 15:19:46 -040029 , fFloat2_Type(new Type(String("float2"), *fFloat_Type, 2))
30 , fFloat3_Type(new Type(String("float3"), *fFloat_Type, 3))
31 , fFloat4_Type(new Type(String("float4"), *fFloat_Type, 4))
Ethan Nicholas0df1b042017-03-31 13:56:23 -040032 , fUInt_Type(new Type(String("uint"), true, { fFloat_Type.get(), fDouble_Type.get() }))
Ethan Nicholas5af9ea32017-07-28 15:19:46 -040033 , fUInt2_Type(new Type(String("uint2"), *fUInt_Type, 2))
34 , fUInt3_Type(new Type(String("uint3"), *fUInt_Type, 3))
35 , fUInt4_Type(new Type(String("uint4"), *fUInt_Type, 4))
Ethan Nicholas0df1b042017-03-31 13:56:23 -040036 , fInt_Type(new Type(String("int"), true, { fUInt_Type.get(), fFloat_Type.get(),
Ethan Nicholas9e1138d2016-11-21 10:39:35 -050037 fDouble_Type.get() }))
Ethan Nicholas5af9ea32017-07-28 15:19:46 -040038 , fInt2_Type(new Type(String("int2"), *fInt_Type, 2))
39 , fInt3_Type(new Type(String("int3"), *fInt_Type, 3))
40 , fInt4_Type(new Type(String("int4"), *fInt_Type, 4))
Ethan Nicholas0df1b042017-03-31 13:56:23 -040041 , fBool_Type(new Type(String("bool"), false))
Ethan Nicholas5af9ea32017-07-28 15:19:46 -040042 , fBool2_Type(new Type(String("bool2"), *fBool_Type, 2))
43 , fBool3_Type(new Type(String("bool3"), *fBool_Type, 3))
44 , fBool4_Type(new Type(String("bool4"), *fBool_Type, 4))
45 , fFloat2x2_Type(new Type(String("float2x2"), *fFloat_Type, 2, 2))
46 , fFloat2x3_Type(new Type(String("float2x3"), *fFloat_Type, 2, 3))
47 , fFloat2x4_Type(new Type(String("float2x4"), *fFloat_Type, 2, 4))
48 , fFloat3x2_Type(new Type(String("float3x2"), *fFloat_Type, 3, 2))
49 , fFloat3x3_Type(new Type(String("float3x3"), *fFloat_Type, 3, 3))
50 , fFloat3x4_Type(new Type(String("float3x4"), *fFloat_Type, 3, 4))
51 , fFloat4x2_Type(new Type(String("float4x2"), *fFloat_Type, 4, 2))
52 , fFloat4x3_Type(new Type(String("float4x3"), *fFloat_Type, 4, 3))
53 , fFloat4x4_Type(new Type(String("float4x4"), *fFloat_Type, 4, 4))
54 , fDouble2x2_Type(new Type(String("double2x2"), *fFloat_Type, 2, 2))
55 , fDouble2x3_Type(new Type(String("double2x3"), *fFloat_Type, 2, 3))
56 , fDouble2x4_Type(new Type(String("double2x4"), *fFloat_Type, 2, 4))
57 , fDouble3x2_Type(new Type(String("double3x2"), *fFloat_Type, 3, 2))
58 , fDouble3x3_Type(new Type(String("double3x3"), *fFloat_Type, 3, 3))
59 , fDouble3x4_Type(new Type(String("double3x4"), *fFloat_Type, 3, 4))
60 , fDouble4x2_Type(new Type(String("double4x2"), *fFloat_Type, 4, 2))
61 , fDouble4x3_Type(new Type(String("double4x3"), *fFloat_Type, 4, 3))
62 , fDouble4x4_Type(new Type(String("double4x4"), *fFloat_Type, 4, 4))
Ethan Nicholas0df1b042017-03-31 13:56:23 -040063 , fSampler1D_Type(new Type(String("sampler1D"), SpvDim1D, false, false, false, true))
64 , fSampler2D_Type(new Type(String("sampler2D"), SpvDim2D, false, false, false, true))
65 , fSampler3D_Type(new Type(String("sampler3D"), SpvDim3D, false, false, false, true))
66 , fSamplerExternalOES_Type(new Type(String("samplerExternalOES"), SpvDim2D, false, false,
Ethan Nicholas9e1138d2016-11-21 10:39:35 -050067 false, true))
Ethan Nicholas0df1b042017-03-31 13:56:23 -040068 , fSamplerCube_Type(new Type(String("samplerCube"), SpvDimCube, false, false, false, true))
Ethan Nicholas762466e2017-06-29 10:03:38 -040069 , fSampler2DRect_Type(new Type(String("sampler2DRect"), SpvDimRect, false, false, false, true))
Ethan Nicholas0df1b042017-03-31 13:56:23 -040070 , fSampler1DArray_Type(new Type(String("sampler1DArray")))
71 , fSampler2DArray_Type(new Type(String("sampler2DArray")))
72 , fSamplerCubeArray_Type(new Type(String("samplerCubeArray")))
Ethan Nicholas0187ae62017-05-03 11:03:44 -040073 , fSamplerBuffer_Type(new Type(String("samplerBuffer"), SpvDimBuffer, false, false, false,
74 true))
Ethan Nicholas0df1b042017-03-31 13:56:23 -040075 , fSampler2DMS_Type(new Type(String("sampler2DMS")))
76 , fSampler2DMSArray_Type(new Type(String("sampler2DMSArray")))
77 , fSampler1DShadow_Type(new Type(String("sampler1DShadow")))
78 , fSampler2DShadow_Type(new Type(String("sampler2DShadow")))
79 , fSamplerCubeShadow_Type(new Type(String("samplerCubeShadow")))
80 , fSampler2DRectShadow_Type(new Type(String("sampler2DRectShadow")))
81 , fSampler1DArrayShadow_Type(new Type(String("sampler1DArrayShadow")))
82 , fSampler2DArrayShadow_Type(new Type(String("sampler2DArrayShadow")))
83 , fSamplerCubeArrayShadow_Type(new Type(String("samplerCubeArrayShadow")))
Brian Salomon2a51de82016-11-16 12:06:01 -050084
Brian Salomonbf7b6202016-11-11 16:08:03 -050085 // Related to below FIXME, gsampler*s don't currently expand to cover integer case.
Ethan Nicholas0df1b042017-03-31 13:56:23 -040086 , fISampler2D_Type(new Type(String("isampler2D"), SpvDim2D, false, false, false, true))
Brian Salomon2a51de82016-11-16 12:06:01 -050087
88 // FIXME express these as "gimage2D" that expand to image2D, iimage2D, and uimage2D.
Ethan Nicholas0df1b042017-03-31 13:56:23 -040089 , fImage2D_Type(new Type(String("image2D"), SpvDim2D, false, false, false, true))
90 , fIImage2D_Type(new Type(String("iimage2D"), SpvDim2D, false, false, false, true))
Brian Salomon2a51de82016-11-16 12:06:01 -050091
Greg Daniel64773e62016-11-22 09:44:03 -050092 // FIXME express these as "gsubpassInput" that expand to subpassInput, isubpassInput,
93 // and usubpassInput.
Ethan Nicholas0df1b042017-03-31 13:56:23 -040094 , fSubpassInput_Type(new Type(String("subpassInput"), SpvDimSubpassData, false, false,
Greg Daniel64773e62016-11-22 09:44:03 -050095 false, false))
Ethan Nicholas0df1b042017-03-31 13:56:23 -040096 , fSubpassInputMS_Type(new Type(String("subpassInputMS"), SpvDimSubpassData, false, false,
Greg Daniel64773e62016-11-22 09:44:03 -050097 true, false))
98
ethannicholasd598f792016-07-25 10:08:54 -070099 // FIXME figure out what we're supposed to do with the gsampler et al. types)
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400100 , fGSampler1D_Type(new Type(String("$gsampler1D"), static_type(*fSampler1D_Type)))
101 , fGSampler2D_Type(new Type(String("$gsampler2D"), static_type(*fSampler2D_Type)))
102 , fGSampler3D_Type(new Type(String("$gsampler3D"), static_type(*fSampler3D_Type)))
103 , fGSamplerCube_Type(new Type(String("$gsamplerCube"), static_type(*fSamplerCube_Type)))
104 , fGSampler2DRect_Type(new Type(String("$gsampler2DRect"), static_type(*fSampler2DRect_Type)))
105 , fGSampler1DArray_Type(new Type(String("$gsampler1DArray"),
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500106 static_type(*fSampler1DArray_Type)))
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400107 , fGSampler2DArray_Type(new Type(String("$gsampler2DArray"),
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500108 static_type(*fSampler2DArray_Type)))
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400109 , fGSamplerCubeArray_Type(new Type(String("$gsamplerCubeArray"),
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500110 static_type(*fSamplerCubeArray_Type)))
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400111 , fGSamplerBuffer_Type(new Type(String("$gsamplerBuffer"), static_type(*fSamplerBuffer_Type)))
112 , fGSampler2DMS_Type(new Type(String("$gsampler2DMS"), static_type(*fSampler2DMS_Type)))
113 , fGSampler2DMSArray_Type(new Type(String("$gsampler2DMSArray"),
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500114 static_type(*fSampler2DMSArray_Type)))
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400115 , fGSampler2DArrayShadow_Type(new Type(String("$gsampler2DArrayShadow"),
ethannicholasd598f792016-07-25 10:08:54 -0700116 static_type(*fSampler2DArrayShadow_Type)))
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400117 , fGSamplerCubeArrayShadow_Type(new Type(String("$gsamplerCubeArrayShadow"),
ethannicholasd598f792016-07-25 10:08:54 -0700118 static_type(*fSamplerCubeArrayShadow_Type)))
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400119 , fGenType_Type(new Type(String("$genType"), { fFloat_Type.get(), fFloat2_Type.get(),
120 fFloat3_Type.get(), fFloat4_Type.get() }))
121 , fGenDType_Type(new Type(String("$genDType"), { fDouble_Type.get(), fDouble2_Type.get(),
122 fDouble3_Type.get(), fDouble4_Type.get() }))
123 , fGenIType_Type(new Type(String("$genIType"), { fInt_Type.get(), fInt2_Type.get(),
124 fInt3_Type.get(), fInt4_Type.get() }))
125 , fGenUType_Type(new Type(String("$genUType"), { fUInt_Type.get(), fUInt2_Type.get(),
126 fUInt3_Type.get(), fUInt4_Type.get() }))
127 , fGenBType_Type(new Type(String("$genBType"), { fBool_Type.get(), fBool2_Type.get(),
128 fBool3_Type.get(), fBool4_Type.get() }))
129 , fMat_Type(new Type(String("$mat"), { fFloat2x2_Type.get(), fFloat2x3_Type.get(),
130 fFloat2x4_Type.get(), fFloat3x2_Type.get(),
131 fFloat3x3_Type.get(), fFloat3x4_Type.get(),
132 fFloat4x2_Type.get(), fFloat4x3_Type.get(),
133 fFloat4x4_Type.get(), fDouble2x2_Type.get(),
134 fDouble2x3_Type.get(), fDouble2x4_Type.get(),
135 fDouble3x2_Type.get(), fDouble3x3_Type.get(),
136 fDouble3x4_Type.get(), fDouble4x2_Type.get(),
137 fDouble4x3_Type.get(), fDouble4x4_Type.get() }))
138 , fVec_Type(new Type(String("$vec"), { fInvalid_Type.get(), fFloat2_Type.get(),
139 fFloat3_Type.get(), fFloat4_Type.get() }))
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400140 , fGVec_Type(new Type(String("$gvec")))
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400141 , fGVec2_Type(new Type(String("$gfloat2")))
142 , fGVec3_Type(new Type(String("$gfloat3")))
143 , fGVec4_Type(new Type(String("$gfloat4"), static_type(*fFloat4_Type)))
144 , fDVec_Type(new Type(String("$dvec"), { fInvalid_Type.get(), fDouble2_Type.get(),
145 fDouble3_Type.get(), fDouble4_Type.get() }))
146 , fIVec_Type(new Type(String("$ivec"), { fInvalid_Type.get(), fInt2_Type.get(),
147 fInt3_Type.get(), fInt4_Type.get() }))
148 , fUVec_Type(new Type(String("$uvec"), { fInvalid_Type.get(), fUInt2_Type.get(),
149 fUInt3_Type.get(), fUInt4_Type.get() }))
150 , fBVec_Type(new Type(String("$bvec"), { fInvalid_Type.get(), fBool2_Type.get(),
151 fBool3_Type.get(), fBool4_Type.get() }))
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400152 , fSkCaps_Type(new Type(String("$sk_Caps")))
Ethan Nicholas762466e2017-06-29 10:03:38 -0400153 , fSkArgs_Type(new Type(String("$sk_Args")))
154 , fColorSpaceXform_Type(new Type(String("colorSpaceXform"), *fFloat_Type, 4, 4))
ethannicholas22f939e2016-10-13 13:25:34 -0700155 , fDefined_Expression(new Defined(*fInvalid_Type)) {}
ethannicholasd598f792016-07-25 10:08:54 -0700156
157 static std::vector<const Type*> static_type(const Type& t) {
Greg Daniel64773e62016-11-22 09:44:03 -0500158 return { &t, &t, &t, &t };
ethannicholasd598f792016-07-25 10:08:54 -0700159 }
160
ethannicholas471e8942016-10-28 09:02:46 -0700161 const std::unique_ptr<Type> fInvalid_Type;
ethannicholasd598f792016-07-25 10:08:54 -0700162 const std::unique_ptr<Type> fVoid_Type;
163
164 const std::unique_ptr<Type> fDouble_Type;
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400165 const std::unique_ptr<Type> fDouble2_Type;
166 const std::unique_ptr<Type> fDouble3_Type;
167 const std::unique_ptr<Type> fDouble4_Type;
ethannicholasd598f792016-07-25 10:08:54 -0700168
169 const std::unique_ptr<Type> fFloat_Type;
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400170 const std::unique_ptr<Type> fFloat2_Type;
171 const std::unique_ptr<Type> fFloat3_Type;
172 const std::unique_ptr<Type> fFloat4_Type;
ethannicholasd598f792016-07-25 10:08:54 -0700173
174 const std::unique_ptr<Type> fUInt_Type;
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400175 const std::unique_ptr<Type> fUInt2_Type;
176 const std::unique_ptr<Type> fUInt3_Type;
177 const std::unique_ptr<Type> fUInt4_Type;
ethannicholasd598f792016-07-25 10:08:54 -0700178
179 const std::unique_ptr<Type> fInt_Type;
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400180 const std::unique_ptr<Type> fInt2_Type;
181 const std::unique_ptr<Type> fInt3_Type;
182 const std::unique_ptr<Type> fInt4_Type;
ethannicholasd598f792016-07-25 10:08:54 -0700183
184 const std::unique_ptr<Type> fBool_Type;
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400185 const std::unique_ptr<Type> fBool2_Type;
186 const std::unique_ptr<Type> fBool3_Type;
187 const std::unique_ptr<Type> fBool4_Type;
ethannicholasd598f792016-07-25 10:08:54 -0700188
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400189 const std::unique_ptr<Type> fFloat2x2_Type;
190 const std::unique_ptr<Type> fFloat2x3_Type;
191 const std::unique_ptr<Type> fFloat2x4_Type;
192 const std::unique_ptr<Type> fFloat3x2_Type;
193 const std::unique_ptr<Type> fFloat3x3_Type;
194 const std::unique_ptr<Type> fFloat3x4_Type;
195 const std::unique_ptr<Type> fFloat4x2_Type;
196 const std::unique_ptr<Type> fFloat4x3_Type;
197 const std::unique_ptr<Type> fFloat4x4_Type;
ethannicholasd598f792016-07-25 10:08:54 -0700198
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400199 const std::unique_ptr<Type> fDouble2x2_Type;
200 const std::unique_ptr<Type> fDouble2x3_Type;
201 const std::unique_ptr<Type> fDouble2x4_Type;
202 const std::unique_ptr<Type> fDouble3x2_Type;
203 const std::unique_ptr<Type> fDouble3x3_Type;
204 const std::unique_ptr<Type> fDouble3x4_Type;
205 const std::unique_ptr<Type> fDouble4x2_Type;
206 const std::unique_ptr<Type> fDouble4x3_Type;
207 const std::unique_ptr<Type> fDouble4x4_Type;
ethannicholasd598f792016-07-25 10:08:54 -0700208
209 const std::unique_ptr<Type> fSampler1D_Type;
210 const std::unique_ptr<Type> fSampler2D_Type;
211 const std::unique_ptr<Type> fSampler3D_Type;
ethannicholas5961bc92016-10-12 06:39:56 -0700212 const std::unique_ptr<Type> fSamplerExternalOES_Type;
ethannicholasd598f792016-07-25 10:08:54 -0700213 const std::unique_ptr<Type> fSamplerCube_Type;
214 const std::unique_ptr<Type> fSampler2DRect_Type;
215 const std::unique_ptr<Type> fSampler1DArray_Type;
216 const std::unique_ptr<Type> fSampler2DArray_Type;
217 const std::unique_ptr<Type> fSamplerCubeArray_Type;
218 const std::unique_ptr<Type> fSamplerBuffer_Type;
219 const std::unique_ptr<Type> fSampler2DMS_Type;
220 const std::unique_ptr<Type> fSampler2DMSArray_Type;
221 const std::unique_ptr<Type> fSampler1DShadow_Type;
222 const std::unique_ptr<Type> fSampler2DShadow_Type;
223 const std::unique_ptr<Type> fSamplerCubeShadow_Type;
224 const std::unique_ptr<Type> fSampler2DRectShadow_Type;
225 const std::unique_ptr<Type> fSampler1DArrayShadow_Type;
226 const std::unique_ptr<Type> fSampler2DArrayShadow_Type;
227 const std::unique_ptr<Type> fSamplerCubeArrayShadow_Type;
228
Brian Salomonbf7b6202016-11-11 16:08:03 -0500229 const std::unique_ptr<Type> fISampler2D_Type;
230
Brian Salomon2a51de82016-11-16 12:06:01 -0500231 const std::unique_ptr<Type> fImage2D_Type;
232 const std::unique_ptr<Type> fIImage2D_Type;
233
Greg Daniel64773e62016-11-22 09:44:03 -0500234 const std::unique_ptr<Type> fSubpassInput_Type;
235 const std::unique_ptr<Type> fSubpassInputMS_Type;
236
ethannicholasd598f792016-07-25 10:08:54 -0700237 const std::unique_ptr<Type> fGSampler1D_Type;
238 const std::unique_ptr<Type> fGSampler2D_Type;
239 const std::unique_ptr<Type> fGSampler3D_Type;
240 const std::unique_ptr<Type> fGSamplerCube_Type;
241 const std::unique_ptr<Type> fGSampler2DRect_Type;
242 const std::unique_ptr<Type> fGSampler1DArray_Type;
243 const std::unique_ptr<Type> fGSampler2DArray_Type;
244 const std::unique_ptr<Type> fGSamplerCubeArray_Type;
245 const std::unique_ptr<Type> fGSamplerBuffer_Type;
246 const std::unique_ptr<Type> fGSampler2DMS_Type;
247 const std::unique_ptr<Type> fGSampler2DMSArray_Type;
248 const std::unique_ptr<Type> fGSampler2DArrayShadow_Type;
249 const std::unique_ptr<Type> fGSamplerCubeArrayShadow_Type;
250
251 const std::unique_ptr<Type> fGenType_Type;
252 const std::unique_ptr<Type> fGenDType_Type;
253 const std::unique_ptr<Type> fGenIType_Type;
254 const std::unique_ptr<Type> fGenUType_Type;
255 const std::unique_ptr<Type> fGenBType_Type;
256
257 const std::unique_ptr<Type> fMat_Type;
258
259 const std::unique_ptr<Type> fVec_Type;
260
261 const std::unique_ptr<Type> fGVec_Type;
262 const std::unique_ptr<Type> fGVec2_Type;
263 const std::unique_ptr<Type> fGVec3_Type;
264 const std::unique_ptr<Type> fGVec4_Type;
265 const std::unique_ptr<Type> fDVec_Type;
266 const std::unique_ptr<Type> fIVec_Type;
267 const std::unique_ptr<Type> fUVec_Type;
268
269 const std::unique_ptr<Type> fBVec_Type;
270
Ethan Nicholas3605ace2016-11-21 15:59:48 -0500271 const std::unique_ptr<Type> fSkCaps_Type;
Ethan Nicholas762466e2017-06-29 10:03:38 -0400272 const std::unique_ptr<Type> fSkArgs_Type;
273 const std::unique_ptr<Type> fColorSpaceXform_Type;
Ethan Nicholas3605ace2016-11-21 15:59:48 -0500274
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400275 // dummy expression used to mark that a variable has a value during dataflow analysis (when it
ethannicholas22f939e2016-10-13 13:25:34 -0700276 // could have several different values, or the analyzer is otherwise unable to assign it a
277 // specific expression)
278 const std::unique_ptr<Expression> fDefined_Expression;
279
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400280private:
ethannicholas22f939e2016-10-13 13:25:34 -0700281 class Defined : public Expression {
282 public:
283 Defined(const Type& type)
284 : INHERITED(Position(), kDefined_Kind, type) {}
285
Ethan Nicholascb670962017-04-20 19:31:52 -0400286 bool hasSideEffects() const override {
287 return false;
288 }
ethannicholas22f939e2016-10-13 13:25:34 -0700289
Ethan Nicholas762466e2017-06-29 10:03:38 -0400290 String description() const override {
291 return String("<defined>");
292 }
293
ethannicholas22f939e2016-10-13 13:25:34 -0700294 typedef Expression INHERITED;
295 };
ethannicholasd598f792016-07-25 10:08:54 -0700296};
297
298} // namespace
299
300#endif