blob: 0f497ef666d2a7ad5f6ca0eabb26885d1a42d16c [file] [log] [blame]
Ben Clayton59d24732021-02-08 22:42:54 +00001// Copyright 2021 The Tint Authors.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15#include "src/intrinsic_table.h"
16
17#include <algorithm>
18#include <limits>
Ben Clayton59d24732021-02-08 22:42:54 +000019#include <unordered_map>
20#include <utility>
21
Ben Clayton59d24732021-02-08 22:42:54 +000022#include "src/program_builder.h"
Ben Clayton6a772362021-06-18 18:56:13 +000023#include "src/sem/atomic_type.h"
Ben Claytonfd35aa82021-07-26 22:19:48 +000024#include "src/sem/depth_multisampled_texture_type.h"
Antonio Maioranoaea9c682021-04-19 22:54:43 +000025#include "src/sem/depth_texture_type.h"
Brandon Jones145f8652021-04-22 22:47:03 +000026#include "src/sem/external_texture_type.h"
Antonio Maioranoaea9c682021-04-19 22:54:43 +000027#include "src/sem/multisampled_texture_type.h"
Ben Clayton71786c92021-06-03 16:07:34 +000028#include "src/sem/pipeline_stage_set.h"
Antonio Maioranoaea9c682021-04-19 22:54:43 +000029#include "src/sem/sampled_texture_type.h"
30#include "src/sem/storage_texture_type.h"
Ben Claytonb478f972021-07-15 20:34:21 +000031#include "src/utils/get_or_create.h"
32#include "src/utils/hash.h"
Ben Clayton053559d2021-07-23 16:43:01 +000033#include "src/utils/math.h"
Ben Claytonb29a59d2021-06-01 19:06:31 +000034#include "src/utils/scoped_assignment.h"
Ben Clayton59d24732021-02-08 22:42:54 +000035
36namespace tint {
37namespace {
38
Ben Claytonb29a59d2021-06-01 19:06:31 +000039// Forward declarations
40struct OverloadInfo;
41class Matchers;
42class NumberMatcher;
43class TypeMatcher;
Ben Clayton59d24732021-02-08 22:42:54 +000044
Ben Claytonb29a59d2021-06-01 19:06:31 +000045/// A special type that matches all TypeMatchers
46class Any : public Castable<Any, sem::Type> {
Ben Clayton59d24732021-02-08 22:42:54 +000047 public:
Ben Claytonb29a59d2021-06-01 19:06:31 +000048 Any() = default;
49 ~Any() override = default;
50 std::string type_name() const override { return "<any>"; }
51 std::string FriendlyName(const SymbolTable&) const override {
52 return "<any>";
53 }
54};
55
56/// Number is an 32 bit unsigned integer, which can be in one of three states:
57/// * Invalid - Number has not been assigned a value
58/// * Valid - a fixed integer value
59/// * Any - matches any other non-invalid number
60struct Number {
61 static const Number any;
62 static const Number invalid;
63
64 /// Constructed as a valid number with the value v
65 explicit Number(uint32_t v) : value_(v), state_(kValid) {}
66
67 /// @returns the value of the number
68 inline uint32_t Value() const { return value_; }
69
70 /// @returns the true if the number is valid
71 inline bool IsValid() const { return state_ == kValid; }
72
73 /// @returns the true if the number is any
74 inline bool IsAny() const { return state_ == kAny; }
75
76 /// Assignment operator.
77 /// The number becomes valid, with the value n
78 inline Number& operator=(uint32_t n) {
79 value_ = n;
80 state_ = kValid;
81 return *this;
82 }
83
84 private:
85 enum State {
86 kInvalid,
87 kValid,
88 kAny,
Ben Clayton59d24732021-02-08 22:42:54 +000089 };
90
Ben Claytonb29a59d2021-06-01 19:06:31 +000091 constexpr explicit Number(State state) : state_(state) {}
92
93 uint32_t value_ = 0;
94 State state_ = kInvalid;
95};
96
97const Number Number::any{Number::kAny};
98const Number Number::invalid{Number::kInvalid};
99
100/// ClosedState holds the state of the open / closed numbers and types.
101/// Used by the MatchState.
102class ClosedState {
103 public:
104 explicit ClosedState(ProgramBuilder& b) : builder(b) {}
105
106 /// If the type with index `idx` is open, then it is closed with type `ty` and
107 /// Type() returns true. If the type is closed, then `Type()` returns true iff
108 /// it is equal to `ty`.
109 bool Type(uint32_t idx, const sem::Type* ty) {
110 auto res = types_.emplace(idx, ty);
111 return res.second || res.first->second == ty;
112 }
113
114 /// If the number with index `idx` is open, then it is closed with number
115 /// `number` and Num() returns true. If the number is closed, then `Num()`
116 /// returns true iff it is equal to `ty`.
117 bool Num(uint32_t idx, Number number) {
118 auto res = numbers_.emplace(idx, number.Value());
119 return res.second || res.first->second == number.Value();
120 }
121
122 /// Type returns the closed type with index `idx`.
123 /// An ICE is raised if the type is not closed.
124 const sem::Type* Type(uint32_t idx) const {
125 auto it = types_.find(idx);
126 if (it == types_.end()) {
Ben Claytonffd28e22021-06-24 11:27:36 +0000127 TINT_ICE(Resolver, builder.Diagnostics())
Ben Claytonb29a59d2021-06-01 19:06:31 +0000128 << "type with index " << idx << " is not closed";
129 return nullptr;
130 }
Ben Claytonffd28e22021-06-24 11:27:36 +0000131 TINT_ASSERT(Resolver, it != types_.end());
Ben Claytonb29a59d2021-06-01 19:06:31 +0000132 return it->second;
133 }
134
135 /// Type returns the number type with index `idx`.
136 /// An ICE is raised if the number is not closed.
137 Number Num(uint32_t idx) const {
138 auto it = numbers_.find(idx);
139 if (it == numbers_.end()) {
Ben Claytonffd28e22021-06-24 11:27:36 +0000140 TINT_ICE(Resolver, builder.Diagnostics())
Ben Claytonb29a59d2021-06-01 19:06:31 +0000141 << "number with index " << idx << " is not closed";
142 return Number::invalid;
143 }
144 return Number(it->second);
145 }
146
147 private:
148 ProgramBuilder& builder;
149 std::unordered_map<uint32_t, const sem::Type*> types_;
150 std::unordered_map<uint32_t, uint32_t> numbers_;
151};
152
153/// Index type used for matcher indices
154using MatcherIndex = uint8_t;
155
156/// Index value used for open types / numbers that do not have a constraint
157constexpr MatcherIndex kNoMatcher = std::numeric_limits<MatcherIndex>::max();
158
159/// MatchState holds the state used to match an overload.
160class MatchState {
161 public:
162 MatchState(ProgramBuilder& b,
163 ClosedState& c,
164 const Matchers& m,
165 const OverloadInfo& o,
166 MatcherIndex const* matcher_indices)
167 : builder(b),
168 closed(c),
169 matchers(m),
170 overload(o),
171 matcher_indices_(matcher_indices) {}
172
173 /// The program builder
174 ProgramBuilder& builder;
175 /// The open / closed types and numbers
176 ClosedState& closed;
177 /// The type and number matchers
178 Matchers const& matchers;
179 /// The current overload being evaluated
180 OverloadInfo const& overload;
181
182 /// Type uses the next TypeMatcher from the matcher indices to match the type
183 /// `ty`. If the type matches, the canonical expected type is returned. If the
184 /// type `ty` does not match, then nullptr is returned.
185 /// @note: The matcher indices are progressed on calling.
186 const sem::Type* Type(const sem::Type* ty);
187
188 /// Num uses the next NumMatcher from the matcher indices to match the number
189 /// `num`. If the number matches, the canonical expected number is returned.
190 /// If the number `num` does not match, then an invalid number is returned.
191 /// @note: The matcher indices are progressed on calling.
192 Number Num(Number num);
193
194 /// @returns a string representation of the next TypeMatcher from the matcher
195 /// indices.
196 /// @note: The matcher indices are progressed on calling.
197 std::string TypeName();
198
199 /// @returns a string representation of the next NumberMatcher from the
200 /// matcher indices.
201 /// @note: The matcher indices are progressed on calling.
202 std::string NumName();
203
204 private:
205 MatcherIndex const* matcher_indices_ = nullptr;
206};
207
208/// A TypeMatcher is the interface used to match an type used as part of an
209/// overload's parameter or return type.
210class TypeMatcher {
211 public:
Ben Clayton59d24732021-02-08 22:42:54 +0000212 /// Destructor
Ben Claytonb29a59d2021-06-01 19:06:31 +0000213 virtual ~TypeMatcher() = default;
Ben Clayton59d24732021-02-08 22:42:54 +0000214
Ben Claytonb29a59d2021-06-01 19:06:31 +0000215 /// Checks whether the given type matches the matcher rules, and returns the
216 /// expected, canonicalized type on success.
217 /// Match may close open types and numbers in state.
218 /// @param type the type to match
219 /// @returns the canonicalized type on match, otherwise nullptr
220 virtual const sem::Type* Match(MatchState& state,
221 const sem::Type* type) const = 0;
Ben Clayton59d24732021-02-08 22:42:54 +0000222
223 /// @return a string representation of the matcher. Used for printing error
224 /// messages when no overload is found.
Ben Claytonb29a59d2021-06-01 19:06:31 +0000225 virtual std::string String(MatchState& state) const = 0;
Ben Clayton59d24732021-02-08 22:42:54 +0000226};
227
Ben Claytonb29a59d2021-06-01 19:06:31 +0000228/// A NumberMatcher is the interface used to match a number or enumerator used
229/// as part of an overload's parameter or return type.
230class NumberMatcher {
Ben Clayton59d24732021-02-08 22:42:54 +0000231 public:
Ben Clayton59d24732021-02-08 22:42:54 +0000232 /// Destructor
Ben Claytonb29a59d2021-06-01 19:06:31 +0000233 virtual ~NumberMatcher() = default;
Ben Clayton59d24732021-02-08 22:42:54 +0000234
Ben Claytonb29a59d2021-06-01 19:06:31 +0000235 /// Checks whether the given number matches the matcher rules.
236 /// Match may close open numbers in state.
237 /// @param number the number to match
238 /// @returns true if the argument type is as expected.
239 virtual Number Match(MatchState& state, Number number) const = 0;
240
241 /// @return a string representation of the matcher. Used for printing error
242 /// messages when no overload is found.
243 virtual std::string String(MatchState& state) const = 0;
Ben Clayton59d24732021-02-08 22:42:54 +0000244};
245
Ben Claytonb29a59d2021-06-01 19:06:31 +0000246/// OpenTypeMatcher is a Matcher for an open type.
247/// The OpenTypeMatcher will match against any type (so long as it is consistent
248/// across all uses in the overload)
249class OpenTypeMatcher : public TypeMatcher {
Ben Clayton59d24732021-02-08 22:42:54 +0000250 public:
Ben Claytonb29a59d2021-06-01 19:06:31 +0000251 /// Constructor
252 explicit OpenTypeMatcher(uint32_t index) : index_(index) {}
Ben Clayton59d24732021-02-08 22:42:54 +0000253
Ben Claytonb29a59d2021-06-01 19:06:31 +0000254 const sem::Type* Match(MatchState& state,
255 const sem::Type* type) const override {
256 if (type->Is<Any>()) {
257 return state.closed.Type(index_);
258 }
259 return state.closed.Type(index_, type) ? type : nullptr;
Ben Clayton59d24732021-02-08 22:42:54 +0000260 }
261
Ben Claytonb29a59d2021-06-01 19:06:31 +0000262 std::string String(MatchState& state) const override;
Ben Clayton59d24732021-02-08 22:42:54 +0000263
264 private:
Ben Claytonb29a59d2021-06-01 19:06:31 +0000265 uint32_t index_;
Ben Clayton59d24732021-02-08 22:42:54 +0000266};
267
Ben Claytonb29a59d2021-06-01 19:06:31 +0000268/// OpenNumberMatcher is a Matcher for an open number.
269/// The OpenNumberMatcher will match against any number (so long as it is
270/// consistent for the overload)
271class OpenNumberMatcher : public NumberMatcher {
Ben Clayton06d8db22021-02-08 22:59:44 +0000272 public:
Ben Claytonb29a59d2021-06-01 19:06:31 +0000273 explicit OpenNumberMatcher(uint32_t index) : index_(index) {}
Ben Clayton06d8db22021-02-08 22:59:44 +0000274
Ben Claytonb29a59d2021-06-01 19:06:31 +0000275 Number Match(MatchState& state, Number number) const override {
276 if (number.IsAny()) {
277 return state.closed.Num(index_);
Ben Clayton59d24732021-02-08 22:42:54 +0000278 }
Ben Claytonb29a59d2021-06-01 19:06:31 +0000279 return state.closed.Num(index_, number) ? number : Number::invalid;
Ben Clayton59d24732021-02-08 22:42:54 +0000280 }
281
Ben Claytonb29a59d2021-06-01 19:06:31 +0000282 std::string String(MatchState& state) const override;
Ben Clayton59d24732021-02-08 22:42:54 +0000283
284 private:
Ben Claytonb29a59d2021-06-01 19:06:31 +0000285 uint32_t index_;
Ben Clayton59d24732021-02-08 22:42:54 +0000286};
287
Ben Claytonb29a59d2021-06-01 19:06:31 +0000288////////////////////////////////////////////////////////////////////////////////
289// Binding functions for use in the generated intrinsic_table.inl
290// TODO(bclayton): See if we can move more of this hand-rolled code to the
291// template
292////////////////////////////////////////////////////////////////////////////////
293using TexelFormat = ast::ImageFormat;
Ben Clayton93e8f522021-06-04 20:41:47 +0000294using Access = ast::Access;
Ben Claytonb29a59d2021-06-01 19:06:31 +0000295using StorageClass = ast::StorageClass;
296using ParameterUsage = sem::ParameterUsage;
Ben Clayton71786c92021-06-03 16:07:34 +0000297using PipelineStageSet = sem::PipelineStageSet;
298using PipelineStage = ast::PipelineStage;
Ben Clayton59d24732021-02-08 22:42:54 +0000299
Ben Claytonb29a59d2021-06-01 19:06:31 +0000300bool match_bool(const sem::Type* ty) {
301 return ty->IsAnyOf<Any, sem::Bool>();
302}
303
304const sem::Bool* build_bool(MatchState& state) {
305 return state.builder.create<sem::Bool>();
306}
307
308bool match_f32(const sem::Type* ty) {
309 return ty->IsAnyOf<Any, sem::F32>();
310}
311
312const sem::I32* build_i32(MatchState& state) {
313 return state.builder.create<sem::I32>();
314}
315
316bool match_i32(const sem::Type* ty) {
317 return ty->IsAnyOf<Any, sem::I32>();
318}
319
320const sem::U32* build_u32(MatchState& state) {
321 return state.builder.create<sem::U32>();
322}
323
324bool match_u32(const sem::Type* ty) {
325 return ty->IsAnyOf<Any, sem::U32>();
326}
327
328const sem::F32* build_f32(MatchState& state) {
329 return state.builder.create<sem::F32>();
330}
331
332bool match_vec(const sem::Type* ty, Number& N, const sem::Type*& T) {
333 if (ty->Is<Any>()) {
334 N = Number::any;
335 T = ty;
336 return true;
337 }
338
339 if (auto* v = ty->As<sem::Vector>()) {
Ben Claytonf5ed2ba2021-07-22 18:31:34 +0000340 N = v->Width();
Ben Claytonb29a59d2021-06-01 19:06:31 +0000341 T = v->type();
342 return true;
343 }
344 return false;
345}
346
347const sem::Vector* build_vec(MatchState& state, Number N, const sem::Type* el) {
348 return state.builder.create<sem::Vector>(el, N.Value());
349}
350
351template <int N>
352bool match_vec(const sem::Type* ty, const sem::Type*& T) {
353 if (ty->Is<Any>()) {
354 T = ty;
355 return true;
356 }
357
358 if (auto* v = ty->As<sem::Vector>()) {
Ben Claytonf5ed2ba2021-07-22 18:31:34 +0000359 if (v->Width() == N) {
Ben Claytonb29a59d2021-06-01 19:06:31 +0000360 T = v->type();
361 return true;
Ben Clayton59d24732021-02-08 22:42:54 +0000362 }
Ben Claytonb29a59d2021-06-01 19:06:31 +0000363 }
364 return false;
365}
366
367bool match_vec2(const sem::Type* ty, const sem::Type*& T) {
368 return match_vec<2>(ty, T);
369}
370
371const sem::Vector* build_vec2(MatchState& state, const sem::Type* T) {
372 return build_vec(state, Number(2), T);
373}
374
375bool match_vec3(const sem::Type* ty, const sem::Type*& T) {
376 return match_vec<3>(ty, T);
377}
378
379const sem::Vector* build_vec3(MatchState& state, const sem::Type* T) {
380 return build_vec(state, Number(3), T);
381}
382
383bool match_vec4(const sem::Type* ty, const sem::Type*& T) {
384 return match_vec<4>(ty, T);
385}
386
387const sem::Vector* build_vec4(MatchState& state, const sem::Type* T) {
388 return build_vec(state, Number(4), T);
389}
390
391bool match_mat(const sem::Type* ty, Number& M, Number& N, const sem::Type*& T) {
392 if (ty->Is<Any>()) {
393 M = Number::any;
394 N = Number::any;
395 T = ty;
396 return true;
397 }
398 if (auto* m = ty->As<sem::Matrix>()) {
399 M = m->columns();
Ben Claytonf5ed2ba2021-07-22 18:31:34 +0000400 N = m->ColumnType()->Width();
Ben Claytonb29a59d2021-06-01 19:06:31 +0000401 T = m->type();
402 return true;
403 }
404 return false;
405}
406
407const sem::Matrix* build_mat(MatchState& state,
408 Number N,
409 Number M,
410 const sem::Type* T) {
Ben Clayton0a2b5f22021-06-09 14:32:14 +0000411 auto* column_type = state.builder.create<sem::Vector>(T, M.Value());
412 return state.builder.create<sem::Matrix>(column_type, N.Value());
Ben Claytonb29a59d2021-06-01 19:06:31 +0000413}
414
415bool match_array(const sem::Type* ty, const sem::Type*& T) {
416 if (ty->Is<Any>()) {
417 T = ty;
418 return true;
Ben Clayton59d24732021-02-08 22:42:54 +0000419 }
420
Ben Claytonb29a59d2021-06-01 19:06:31 +0000421 if (auto* a = ty->As<sem::Array>()) {
422 if (a->Count() == 0) {
423 T = a->ElemType();
424 return true;
Ben Clayton06d8db22021-02-08 22:59:44 +0000425 }
Ben Claytonb29a59d2021-06-01 19:06:31 +0000426 }
427 return false;
428}
429
430const sem::Array* build_array(MatchState& state, const sem::Type* el) {
431 return state.builder.create<sem::Array>(el,
432 /* count */ 0,
433 /* align */ 0,
434 /* size */ 0,
435 /* stride */ 0,
436 /* stride_implicit */ 0);
437}
438
Ben Clayton18588542021-06-04 22:17:37 +0000439bool match_ptr(const sem::Type* ty, Number& S, const sem::Type*& T, Number& A) {
Ben Claytonb29a59d2021-06-01 19:06:31 +0000440 if (ty->Is<Any>()) {
441 S = Number::any;
442 T = ty;
Ben Clayton18588542021-06-04 22:17:37 +0000443 A = Number::any;
Ben Claytonb29a59d2021-06-01 19:06:31 +0000444 return true;
Ben Clayton06d8db22021-02-08 22:59:44 +0000445 }
446
Ben Claytonb29a59d2021-06-01 19:06:31 +0000447 if (auto* p = ty->As<sem::Pointer>()) {
448 S = Number(static_cast<uint32_t>(p->StorageClass()));
449 T = p->StoreType();
Ben Clayton18588542021-06-04 22:17:37 +0000450 A = Number(static_cast<uint32_t>(p->Access()));
Ben Claytonb29a59d2021-06-01 19:06:31 +0000451 return true;
Ben Clayton06d8db22021-02-08 22:59:44 +0000452 }
Ben Claytonb29a59d2021-06-01 19:06:31 +0000453 return false;
454}
Ben Clayton06d8db22021-02-08 22:59:44 +0000455
Ben Clayton18588542021-06-04 22:17:37 +0000456const sem::Pointer* build_ptr(MatchState& state,
457 Number S,
458 const sem::Type* T,
459 Number& A) {
Ben Claytonb29a59d2021-06-01 19:06:31 +0000460 return state.builder.create<sem::Pointer>(
Ben Clayton18588542021-06-04 22:17:37 +0000461 T, static_cast<ast::StorageClass>(S.Value()),
462 static_cast<ast::Access>(A.Value()));
Ben Claytonb29a59d2021-06-01 19:06:31 +0000463}
464
Ben Clayton6a772362021-06-18 18:56:13 +0000465bool match_atomic(const sem::Type* ty, const sem::Type*& T) {
466 if (ty->Is<Any>()) {
467 T = ty;
468 return true;
469 }
470
471 if (auto* a = ty->As<sem::Atomic>()) {
472 T = a->Type();
473 return true;
474 }
475 return false;
476}
477
478const sem::Atomic* build_atomic(MatchState& state, const sem::Type* T) {
479 return state.builder.create<sem::Atomic>(T);
480}
481
Ben Claytonb29a59d2021-06-01 19:06:31 +0000482bool match_sampler(const sem::Type* ty) {
483 if (ty->Is<Any>()) {
484 return true;
Ben Clayton06d8db22021-02-08 22:59:44 +0000485 }
Ben Clayton367f5df2021-08-23 15:16:42 +0000486 return ty->Is([](const sem::Sampler* s) {
Ben Claytonb29a59d2021-06-01 19:06:31 +0000487 return s->kind() == ast::SamplerKind::kSampler;
488 });
489}
Ben Clayton06d8db22021-02-08 22:59:44 +0000490
Ben Claytonb29a59d2021-06-01 19:06:31 +0000491const sem::Sampler* build_sampler(MatchState& state) {
492 return state.builder.create<sem::Sampler>(ast::SamplerKind::kSampler);
493}
Ben Clayton06d8db22021-02-08 22:59:44 +0000494
Ben Claytonb29a59d2021-06-01 19:06:31 +0000495bool match_sampler_comparison(const sem::Type* ty) {
496 if (ty->Is<Any>()) {
497 return true;
498 }
Ben Clayton367f5df2021-08-23 15:16:42 +0000499 return ty->Is([](const sem::Sampler* s) {
Ben Claytonb29a59d2021-06-01 19:06:31 +0000500 return s->kind() == ast::SamplerKind::kComparisonSampler;
501 });
502}
Ben Clayton06d8db22021-02-08 22:59:44 +0000503
Ben Claytonb29a59d2021-06-01 19:06:31 +0000504const sem::Sampler* build_sampler_comparison(MatchState& state) {
505 return state.builder.create<sem::Sampler>(
506 ast::SamplerKind::kComparisonSampler);
507}
508
509bool match_texture(const sem::Type* ty,
510 ast::TextureDimension dim,
511 const sem::Type*& T) {
512 if (ty->Is<Any>()) {
513 T = ty;
514 return true;
515 }
516 if (auto* v = ty->As<sem::SampledTexture>()) {
517 if (v->dim() == dim) {
518 T = v->type();
519 return true;
Ben Clayton06d8db22021-02-08 22:59:44 +0000520 }
Ben Claytonb29a59d2021-06-01 19:06:31 +0000521 }
522 return false;
523}
524
525#define JOIN(a, b) a##b
526
527#define DECLARE_SAMPLED_TEXTURE(suffix, dim) \
528 bool JOIN(match_texture_, suffix)(const sem::Type* ty, \
529 const sem::Type*& T) { \
530 return match_texture(ty, dim, T); \
531 } \
532 const sem::SampledTexture* JOIN(build_texture_, suffix)( \
533 MatchState & state, const sem::Type* T) { \
534 return state.builder.create<sem::SampledTexture>(dim, T); \
Ben Clayton06d8db22021-02-08 22:59:44 +0000535 }
536
Ben Claytonb29a59d2021-06-01 19:06:31 +0000537DECLARE_SAMPLED_TEXTURE(1d, ast::TextureDimension::k1d)
538DECLARE_SAMPLED_TEXTURE(2d, ast::TextureDimension::k2d)
539DECLARE_SAMPLED_TEXTURE(2d_array, ast::TextureDimension::k2dArray)
540DECLARE_SAMPLED_TEXTURE(3d, ast::TextureDimension::k3d)
541DECLARE_SAMPLED_TEXTURE(cube, ast::TextureDimension::kCube)
542DECLARE_SAMPLED_TEXTURE(cube_array, ast::TextureDimension::kCubeArray)
543#undef DECLARE_SAMPLED_TEXTURE
544
545bool match_texture_multisampled(const sem::Type* ty,
546 ast::TextureDimension dim,
547 const sem::Type*& T) {
548 if (ty->Is<Any>()) {
549 T = ty;
550 return true;
Ben Clayton06d8db22021-02-08 22:59:44 +0000551 }
Ben Claytonb29a59d2021-06-01 19:06:31 +0000552 if (auto* v = ty->As<sem::MultisampledTexture>()) {
553 if (v->dim() == dim) {
554 T = v->type();
555 return true;
Ben Clayton06d8db22021-02-08 22:59:44 +0000556 }
Ben Claytonb29a59d2021-06-01 19:06:31 +0000557 }
558 return false;
559}
560
561#define DECLARE_MULTISAMPLED_TEXTURE(suffix, dim) \
562 bool JOIN(match_texture_multisampled_, suffix)(const sem::Type* ty, \
563 const sem::Type*& T) { \
564 return match_texture_multisampled(ty, dim, T); \
565 } \
566 const sem::MultisampledTexture* JOIN(build_texture_multisampled_, suffix)( \
567 MatchState & state, const sem::Type* T) { \
568 return state.builder.create<sem::MultisampledTexture>(dim, T); \
Ben Clayton06d8db22021-02-08 22:59:44 +0000569 }
570
Ben Claytonb29a59d2021-06-01 19:06:31 +0000571DECLARE_MULTISAMPLED_TEXTURE(2d, ast::TextureDimension::k2d)
572#undef DECLARE_MULTISAMPLED_TEXTURE
573
574bool match_texture_depth(const sem::Type* ty, ast::TextureDimension dim) {
575 if (ty->Is<Any>()) {
576 return true;
577 }
Ben Clayton367f5df2021-08-23 15:16:42 +0000578 return ty->Is([&](const sem::DepthTexture* t) { return t->dim() == dim; });
Ben Claytonb29a59d2021-06-01 19:06:31 +0000579}
580
581#define DECLARE_DEPTH_TEXTURE(suffix, dim) \
582 bool JOIN(match_texture_depth_, suffix)(const sem::Type* ty) { \
583 return match_texture_depth(ty, dim); \
584 } \
585 const sem::DepthTexture* JOIN(build_texture_depth_, \
586 suffix)(MatchState & state) { \
587 return state.builder.create<sem::DepthTexture>(dim); \
Ben Clayton06d8db22021-02-08 22:59:44 +0000588 }
589
Ben Claytonb29a59d2021-06-01 19:06:31 +0000590DECLARE_DEPTH_TEXTURE(2d, ast::TextureDimension::k2d)
591DECLARE_DEPTH_TEXTURE(2d_array, ast::TextureDimension::k2dArray)
592DECLARE_DEPTH_TEXTURE(cube, ast::TextureDimension::kCube)
593DECLARE_DEPTH_TEXTURE(cube_array, ast::TextureDimension::kCubeArray)
594#undef DECLARE_DEPTH_TEXTURE
595
Ben Claytonfd35aa82021-07-26 22:19:48 +0000596bool match_texture_depth_multisampled_2d(const sem::Type* ty) {
597 if (ty->Is<Any>()) {
598 return true;
599 }
Ben Clayton367f5df2021-08-23 15:16:42 +0000600 return ty->Is([&](const sem::DepthMultisampledTexture* t) {
601 return t->dim() == ast::TextureDimension::k2d;
602 });
Ben Claytonfd35aa82021-07-26 22:19:48 +0000603}
604
605sem::DepthMultisampledTexture* build_texture_depth_multisampled_2d(
606 MatchState& state) {
607 return state.builder.create<sem::DepthMultisampledTexture>(
608 ast::TextureDimension::k2d);
609}
610
Ben Claytonb29a59d2021-06-01 19:06:31 +0000611bool match_texture_storage(const sem::Type* ty,
612 ast::TextureDimension dim,
613 Number& F,
614 Number& A) {
615 if (ty->Is<Any>()) {
616 F = Number::any;
617 A = Number::any;
618 return true;
Ben Clayton06d8db22021-02-08 22:59:44 +0000619 }
Ben Claytonb29a59d2021-06-01 19:06:31 +0000620 if (auto* v = ty->As<sem::StorageTexture>()) {
621 if (v->dim() == dim) {
622 F = Number(static_cast<uint32_t>(v->image_format()));
Ben Clayton93e8f522021-06-04 20:41:47 +0000623 A = Number(static_cast<uint32_t>(v->access()));
Ben Claytonb29a59d2021-06-01 19:06:31 +0000624 return true;
Ben Clayton06d8db22021-02-08 22:59:44 +0000625 }
Ben Claytonb29a59d2021-06-01 19:06:31 +0000626 }
627 return false;
628}
629
630#define DECLARE_STORAGE_TEXTURE(suffix, dim) \
631 bool JOIN(match_texture_storage_, suffix)(const sem::Type* ty, Number& F, \
632 Number& A) { \
633 return match_texture_storage(ty, dim, F, A); \
634 } \
635 const sem::StorageTexture* JOIN(build_texture_storage_, suffix)( \
636 MatchState & state, Number F, Number A) { \
637 auto format = static_cast<TexelFormat>(F.Value()); \
Ben Clayton93e8f522021-06-04 20:41:47 +0000638 auto access = static_cast<Access>(A.Value()); \
Ben Claytonb29a59d2021-06-01 19:06:31 +0000639 auto* T = sem::StorageTexture::SubtypeFor(format, state.builder.Types()); \
640 return state.builder.create<sem::StorageTexture>(dim, format, access, T); \
Ben Clayton06d8db22021-02-08 22:59:44 +0000641 }
642
Ben Claytonb29a59d2021-06-01 19:06:31 +0000643DECLARE_STORAGE_TEXTURE(1d, ast::TextureDimension::k1d)
644DECLARE_STORAGE_TEXTURE(2d, ast::TextureDimension::k2d)
645DECLARE_STORAGE_TEXTURE(2d_array, ast::TextureDimension::k2dArray)
646DECLARE_STORAGE_TEXTURE(3d, ast::TextureDimension::k3d)
647#undef DECLARE_STORAGE_TEXTURE
Ben Clayton06d8db22021-02-08 22:59:44 +0000648
Ben Claytonb29a59d2021-06-01 19:06:31 +0000649bool match_texture_external(const sem::Type* ty) {
650 return ty->IsAnyOf<Any, sem::ExternalTexture>();
651}
Antonio Maioranodc4e6c12021-05-14 17:51:13 +0000652
Ben Claytonb29a59d2021-06-01 19:06:31 +0000653const sem::ExternalTexture* build_texture_external(MatchState& state) {
654 return state.builder.create<sem::ExternalTexture>();
655}
Antonio Maioranodc4e6c12021-05-14 17:51:13 +0000656
Ben Clayton053559d2021-07-23 16:43:01 +0000657// Builtin types starting with a _ prefix cannot be declared in WGSL, so they
658// can only be used as return types. Because of this, they must only match Any,
659// which is used as the return type matcher.
660bool match_modf_result(const sem::Type* ty) {
661 return ty->Is<Any>();
662}
663bool match_modf_result_vec(const sem::Type* ty, Number& N) {
664 if (!ty->Is<Any>()) {
665 return false;
666 }
667 N = Number::any;
668 return true;
669}
670bool match_frexp_result(const sem::Type* ty) {
671 return ty->Is<Any>();
672}
673bool match_frexp_result_vec(const sem::Type* ty, Number& N) {
674 if (!ty->Is<Any>()) {
675 return false;
676 }
677 N = Number::any;
678 return true;
679}
680
681struct NameAndType {
682 std::string name;
683 sem::Type* type;
684};
685const sem::Struct* build_struct(
686 MatchState& state,
687 std::string name,
688 std::initializer_list<NameAndType> member_names_and_types) {
689 uint32_t offset = 0;
690 uint32_t max_align = 0;
691 sem::StructMemberList members;
692 for (auto& m : member_names_and_types) {
693 uint32_t align = m.type->Align();
694 uint32_t size = m.type->Size();
695 offset = utils::RoundUp(align, offset);
696 max_align = std::max(max_align, align);
697 members.emplace_back(state.builder.create<sem::StructMember>(
698 /* declaration */ nullptr,
699 /* name */ state.builder.Sym(m.name),
700 /* type */ m.type,
701 /* index */ static_cast<uint32_t>(members.size()),
702 /* offset */ offset,
703 /* align */ align,
704 /* size */ size));
705 offset += size;
706 }
707 uint32_t size_without_padding = offset;
708 uint32_t size_with_padding = utils::RoundUp(max_align, offset);
709 return state.builder.create<sem::Struct>(
710 /* declaration */ nullptr,
711 /* name */ state.builder.Sym(name),
712 /* members */ members,
713 /* align */ max_align,
714 /* size */ size_with_padding,
715 /* size_no_padding */ size_without_padding);
716}
717
718const sem::Struct* build_modf_result(MatchState& state) {
719 auto* f32 = state.builder.create<sem::F32>();
720 return build_struct(state, "_modf_result", {{"fract", f32}, {"whole", f32}});
721}
722const sem::Struct* build_modf_result_vec(MatchState& state, Number& n) {
723 auto* vec_f32 = state.builder.create<sem::Vector>(
724 state.builder.create<sem::F32>(), n.Value());
725 return build_struct(state, "_modf_result_vec" + std::to_string(n.Value()),
726 {{"fract", vec_f32}, {"whole", vec_f32}});
727}
728const sem::Struct* build_frexp_result(MatchState& state) {
729 auto* f32 = state.builder.create<sem::F32>();
730 auto* i32 = state.builder.create<sem::I32>();
731 return build_struct(state, "_frexp_result", {{"sig", f32}, {"exp", i32}});
732}
733const sem::Struct* build_frexp_result_vec(MatchState& state, Number& n) {
734 auto* vec_f32 = state.builder.create<sem::Vector>(
735 state.builder.create<sem::F32>(), n.Value());
736 auto* vec_i32 = state.builder.create<sem::Vector>(
737 state.builder.create<sem::I32>(), n.Value());
738 return build_struct(state, "_frexp_result_vec" + std::to_string(n.Value()),
739 {{"sig", vec_f32}, {"exp", vec_i32}});
740}
741
Ben Clayton71786c92021-06-03 16:07:34 +0000742/// ParameterInfo describes a parameter
743struct ParameterInfo {
744 /// The parameter usage (parameter name in definition file)
745 ParameterUsage const usage;
746
747 /// Pointer to a list of indices that are used to match the parameter type.
748 /// The matcher indices index on Matchers::type and / or Matchers::number.
749 /// These indices are consumed by the matchers themselves.
750 /// The first index is always a TypeMatcher.
751 MatcherIndex const* const matcher_indices;
752};
753
754/// OpenTypeInfo describes an open type
755struct OpenTypeInfo {
756 /// Name of the open type (e.g. 'T')
757 const char* name;
758 /// Optional type matcher constraint.
759 /// Either an index in Matchers::type, or kNoMatcher
760 MatcherIndex const matcher_index;
761};
762
763/// OpenNumberInfo describes an open number
764struct OpenNumberInfo {
765 /// Name of the open number (e.g. 'N')
766 const char* name;
767 /// Optional number matcher constraint.
768 /// Either an index in Matchers::number, or kNoMatcher
769 MatcherIndex const matcher_index;
770};
771
772/// OverloadInfo describes a single function overload
773struct OverloadInfo {
774 /// Total number of parameters for the overload
775 uint8_t const num_parameters;
776 /// Total number of open types for the overload
777 uint8_t const num_open_types;
778 /// Total number of open numbers for the overload
779 uint8_t const num_open_numbers;
780 /// Pointer to the first open type
781 OpenTypeInfo const* const open_types;
782 /// Pointer to the first open number
783 OpenNumberInfo const* const open_numbers;
784 /// Pointer to the first parameter
785 ParameterInfo const* const parameters;
786 /// Pointer to a list of matcher indices that index on Matchers::type and
787 /// Matchers::number, used to build the return type. If the function has no
Ben Clayton3e59eb02021-06-10 17:31:54 +0000788 /// return type then this is null
Ben Clayton71786c92021-06-03 16:07:34 +0000789 MatcherIndex const* const return_matcher_indices;
Ben Clayton3e59eb02021-06-10 17:31:54 +0000790 /// The pipeline stages that this overload can be used in
Ben Clayton71786c92021-06-03 16:07:34 +0000791 PipelineStageSet supported_stages;
Ben Clayton3e59eb02021-06-10 17:31:54 +0000792 /// True if the overload is marked as deprecated
793 bool is_deprecated;
Ben Clayton71786c92021-06-03 16:07:34 +0000794};
795
796/// IntrinsicInfo describes an intrinsic function
797struct IntrinsicInfo {
798 /// Number of overloads of the intrinsic function
799 uint8_t const num_overloads;
800 /// Pointer to the start of the overloads for the function
801 OverloadInfo const* const overloads;
802};
803
Ben Claytonb29a59d2021-06-01 19:06:31 +0000804#include "intrinsic_table.inl"
Ben Clayton06d8db22021-02-08 22:59:44 +0000805
Ben Clayton0f2d95d2021-07-22 13:24:59 +0000806/// IntrinsicPrototype describes a fully matched intrinsic function, which is
807/// used as a lookup for building unique sem::Intrinsic instances.
808struct IntrinsicPrototype {
809 /// Parameter describes a single parameter
810 struct Parameter {
811 /// Parameter type
812 sem::Type* const type;
813 /// Parameter usage
814 ParameterUsage const usage = ParameterUsage::kNone;
815 };
816
817 /// Hasher provides a hash function for the IntrinsicPrototype
818 struct Hasher {
819 /// @param i the IntrinsicPrototype to create a hash for
820 /// @return the hash value
821 inline std::size_t operator()(const IntrinsicPrototype& i) const {
822 size_t hash = utils::Hash(i.parameters.size());
823 for (auto& p : i.parameters) {
824 utils::HashCombine(&hash, p.type, p.usage);
825 }
826 return utils::Hash(hash, i.type, i.return_type, i.supported_stages,
827 i.is_deprecated);
828 }
829 };
830
831 sem::IntrinsicType type = sem::IntrinsicType::kNone;
832 std::vector<Parameter> parameters;
833 sem::Type const* return_type = nullptr;
834 PipelineStageSet supported_stages;
835 bool is_deprecated = false;
836};
837
838/// Equality operator for IntrinsicPrototype
839bool operator==(const IntrinsicPrototype& a, const IntrinsicPrototype& b) {
840 if (a.type != b.type || a.supported_stages != b.supported_stages ||
841 a.return_type != b.return_type || a.is_deprecated != b.is_deprecated ||
842 a.parameters.size() != b.parameters.size()) {
843 return false;
844 }
845 for (size_t i = 0; i < a.parameters.size(); i++) {
846 auto& pa = a.parameters[i];
847 auto& pb = b.parameters[i];
848 if (pa.type != pb.type || pa.usage != pb.usage) {
849 return false;
850 }
851 }
852 return true;
853}
854
Ben Clayton59d24732021-02-08 22:42:54 +0000855/// Impl is the private implementation of the IntrinsicTable interface.
856class Impl : public IntrinsicTable {
857 public:
Ben Claytonb29a59d2021-06-01 19:06:31 +0000858 explicit Impl(ProgramBuilder& builder);
Ben Clayton59d24732021-02-08 22:42:54 +0000859
Ben Claytonb29a59d2021-06-01 19:06:31 +0000860 const sem::Intrinsic* Lookup(sem::IntrinsicType intrinsic_type,
861 const std::vector<const sem::Type*>& args,
Ben Claytonb478f972021-07-15 20:34:21 +0000862 const Source& source) override;
Ben Clayton59d24732021-02-08 22:42:54 +0000863
864 private:
Ben Claytonb29a59d2021-06-01 19:06:31 +0000865 const sem::Intrinsic* Match(sem::IntrinsicType intrinsic_type,
866 const OverloadInfo& overload,
867 const std::vector<const sem::Type*>& args,
Ben Claytonb478f972021-07-15 20:34:21 +0000868 int& match_score);
Ben Clayton59d24732021-02-08 22:42:54 +0000869
Ben Claytonb29a59d2021-06-01 19:06:31 +0000870 MatchState Match(ClosedState& closed,
871 const OverloadInfo& overload,
872 MatcherIndex const* matcher_indices) const;
Ben Clayton59d24732021-02-08 22:42:54 +0000873
Ben Claytonb29a59d2021-06-01 19:06:31 +0000874 void PrintOverload(std::ostream& ss,
875 const OverloadInfo& overload,
876 sem::IntrinsicType intrinsic_type) const;
Ben Clayton59d24732021-02-08 22:42:54 +0000877
Ben Claytonb29a59d2021-06-01 19:06:31 +0000878 ProgramBuilder& builder;
879 Matchers matchers;
Ben Clayton0f2d95d2021-07-22 13:24:59 +0000880 std::unordered_map<IntrinsicPrototype,
881 sem::Intrinsic*,
882 IntrinsicPrototype::Hasher>
883 intrinsics;
Ben Clayton59d24732021-02-08 22:42:54 +0000884};
885
Ben Clayton6b4924f2021-02-17 20:13:34 +0000886/// @return a string representing a call to an intrinsic with the given argument
887/// types.
888std::string CallSignature(ProgramBuilder& builder,
Ben Claytonb29a59d2021-06-01 19:06:31 +0000889 sem::IntrinsicType intrinsic_type,
Ben Clayton12ed13d2021-04-28 12:38:13 +0000890 const std::vector<const sem::Type*>& args) {
Ben Clayton6b4924f2021-02-17 20:13:34 +0000891 std::stringstream ss;
Ben Claytonb29a59d2021-06-01 19:06:31 +0000892 ss << sem::str(intrinsic_type) << "(";
Ben Clayton6b4924f2021-02-17 20:13:34 +0000893 {
894 bool first = true;
895 for (auto* arg : args) {
896 if (!first) {
897 ss << ", ";
898 }
899 first = false;
Ben Clayton9b54a2e2021-05-18 10:28:48 +0000900 ss << arg->UnwrapRef()->FriendlyName(builder.Symbols());
Ben Clayton6b4924f2021-02-17 20:13:34 +0000901 }
902 }
903 ss << ")";
904
905 return ss.str();
906}
907
Ben Claytonb29a59d2021-06-01 19:06:31 +0000908std::string OpenTypeMatcher::String(MatchState& state) const {
909 return state.overload.open_types[index_].name;
910}
911
912std::string OpenNumberMatcher::String(MatchState& state) const {
913 return state.overload.open_numbers[index_].name;
914}
915
916Impl::Impl(ProgramBuilder& b) : builder(b) {}
917
918const sem::Intrinsic* Impl::Lookup(sem::IntrinsicType intrinsic_type,
919 const std::vector<const sem::Type*>& args,
Ben Claytonb478f972021-07-15 20:34:21 +0000920 const Source& source) {
Ben Clayton59d24732021-02-08 22:42:54 +0000921 // Candidate holds information about a mismatched overload that could be what
922 // the user intended to call.
923 struct Candidate {
Ben Claytonb29a59d2021-06-01 19:06:31 +0000924 const OverloadInfo* overload;
Ben Clayton59d24732021-02-08 22:42:54 +0000925 int score;
926 };
927
928 // The list of failed matches that had promise.
929 std::vector<Candidate> candidates;
930
Ben Claytonb29a59d2021-06-01 19:06:31 +0000931 auto& intrinsic = kIntrinsics[static_cast<uint32_t>(intrinsic_type)];
932 for (uint32_t o = 0; o < intrinsic.num_overloads; o++) {
933 int match_score = 1000;
934 auto& overload = intrinsic.overloads[o];
935 if (auto* match = Match(intrinsic_type, overload, args, match_score)) {
936 return match;
Ben Clayton59d24732021-02-08 22:42:54 +0000937 }
938 if (match_score > 0) {
939 candidates.emplace_back(Candidate{&overload, match_score});
940 }
941 }
942
943 // Sort the candidates with the most promising first
944 std::stable_sort(
945 candidates.begin(), candidates.end(),
946 [](const Candidate& a, const Candidate& b) { return a.score > b.score; });
947
948 // Generate an error message
949 std::stringstream ss;
Ben Claytonb29a59d2021-06-01 19:06:31 +0000950 ss << "no matching call to " << CallSignature(builder, intrinsic_type, args)
Ben Clayton6b4924f2021-02-17 20:13:34 +0000951 << std::endl;
Ben Clayton59d24732021-02-08 22:42:54 +0000952 if (!candidates.empty()) {
953 ss << std::endl;
954 ss << candidates.size() << " candidate function"
955 << (candidates.size() > 1 ? "s:" : ":") << std::endl;
956 for (auto& candidate : candidates) {
Ben Claytonb29a59d2021-06-01 19:06:31 +0000957 ss << " ";
958 PrintOverload(ss, *candidate.overload, intrinsic_type);
959 ss << std::endl;
Ben Clayton59d24732021-02-08 22:42:54 +0000960 }
961 }
Ben Claytonffd28e22021-06-24 11:27:36 +0000962 builder.Diagnostics().add_error(diag::System::Resolver, ss.str(), source);
Ben Claytonb29a59d2021-06-01 19:06:31 +0000963 return nullptr;
Ben Clayton59d24732021-02-08 22:42:54 +0000964}
965
Ben Claytonb29a59d2021-06-01 19:06:31 +0000966const sem::Intrinsic* Impl::Match(sem::IntrinsicType intrinsic_type,
967 const OverloadInfo& overload,
968 const std::vector<const sem::Type*>& args,
Ben Claytonb478f972021-07-15 20:34:21 +0000969 int& match_score) {
Ben Claytonb29a59d2021-06-01 19:06:31 +0000970 // Score wait for argument <-> parameter count matches / mismatches
971 constexpr int kScorePerParamArgMismatch = -1;
972 constexpr int kScorePerMatchedParam = 2;
973 constexpr int kScorePerMatchedOpenType = 1;
974 constexpr int kScorePerMatchedOpenNumber = 1;
975
976 auto num_parameters = overload.num_parameters;
977 auto num_arguments = static_cast<decltype(num_parameters)>(args.size());
978
979 bool overload_matched = true;
980
981 if (num_parameters != num_arguments) {
982 match_score +=
983 kScorePerParamArgMismatch * (std::max(num_parameters, num_arguments) -
984 std::min(num_parameters, num_arguments));
985 overload_matched = false;
Ben Clayton59d24732021-02-08 22:42:54 +0000986 }
987
Ben Claytonb29a59d2021-06-01 19:06:31 +0000988 ClosedState closed(builder);
Ben Clayton59d24732021-02-08 22:42:54 +0000989
Ben Clayton0f2d95d2021-07-22 13:24:59 +0000990 std::vector<IntrinsicPrototype::Parameter> parameters;
Ben Clayton59d24732021-02-08 22:42:54 +0000991
Ben Claytonb29a59d2021-06-01 19:06:31 +0000992 auto num_params = std::min(num_parameters, num_arguments);
993 for (uint32_t p = 0; p < num_params; p++) {
994 auto& parameter = overload.parameters[p];
995 auto* indices = parameter.matcher_indices;
996 auto* type = Match(closed, overload, indices).Type(args[p]->UnwrapRef());
997 if (type) {
Ben Clayton0f2d95d2021-07-22 13:24:59 +0000998 parameters.emplace_back(IntrinsicPrototype::Parameter{
999 const_cast<sem::Type*>(type), parameter.usage});
Ben Claytonb29a59d2021-06-01 19:06:31 +00001000 match_score += kScorePerMatchedParam;
Ben Claytonfaca02d2021-02-10 21:34:25 +00001001 } else {
Ben Claytonb29a59d2021-06-01 19:06:31 +00001002 overload_matched = false;
Ben Claytonfaca02d2021-02-10 21:34:25 +00001003 }
Ben Clayton59d24732021-02-08 22:42:54 +00001004 }
Ben Clayton59d24732021-02-08 22:42:54 +00001005
Ben Claytonb29a59d2021-06-01 19:06:31 +00001006 if (overload_matched) {
1007 // Check all constrained open types matched
1008 for (uint32_t ot = 0; ot < overload.num_open_types; ot++) {
1009 auto& open_type = overload.open_types[ot];
1010 if (open_type.matcher_index != kNoMatcher) {
1011 auto* index = &open_type.matcher_index;
1012 if (Match(closed, overload, index).Type(closed.Type(ot))) {
1013 match_score += kScorePerMatchedOpenType;
1014 } else {
1015 overload_matched = false;
1016 }
1017 }
Ben Clayton59d24732021-02-08 22:42:54 +00001018 }
Ben Clayton59d24732021-02-08 22:42:54 +00001019 }
1020
Ben Claytonb29a59d2021-06-01 19:06:31 +00001021 if (overload_matched) {
1022 // Check all constrained open numbers matched
1023 for (uint32_t on = 0; on < overload.num_open_numbers; on++) {
1024 auto& open_number = overload.open_numbers[on];
1025 if (open_number.matcher_index != kNoMatcher) {
1026 auto* index = &open_number.matcher_index;
1027 if (Match(closed, overload, index).Num(closed.Num(on)).IsValid()) {
1028 match_score += kScorePerMatchedOpenNumber;
1029 } else {
1030 overload_matched = false;
1031 }
1032 }
1033 }
1034 }
1035
1036 if (!overload_matched) {
1037 return nullptr;
1038 }
Ben Clayton59d24732021-02-08 22:42:54 +00001039
1040 // Build the return type
Ben Claytonb29a59d2021-06-01 19:06:31 +00001041 const sem::Type* return_type = nullptr;
1042 if (auto* indices = overload.return_matcher_indices) {
1043 Any any;
1044 return_type = Match(closed, overload, indices).Type(&any);
1045 if (!return_type) {
1046 std::stringstream ss;
1047 PrintOverload(ss, overload, intrinsic_type);
Ben Claytonffd28e22021-06-24 11:27:36 +00001048 TINT_ICE(Resolver, builder.Diagnostics())
Ben Claytonb29a59d2021-06-01 19:06:31 +00001049 << "MatchState.Match() returned null for " << ss.str();
1050 return nullptr;
1051 }
1052 } else {
1053 return_type = builder.create<sem::Void>();
Ben Clayton90f43cf2021-03-31 20:43:26 +00001054 }
Ben Clayton59d24732021-02-08 22:42:54 +00001055
Ben Clayton0f2d95d2021-07-22 13:24:59 +00001056 IntrinsicPrototype intrinsic;
1057 intrinsic.type = intrinsic_type;
1058 intrinsic.return_type = return_type;
1059 intrinsic.parameters = std::move(parameters);
1060 intrinsic.supported_stages = overload.supported_stages;
1061 intrinsic.is_deprecated = overload.is_deprecated;
Ben Claytonb478f972021-07-15 20:34:21 +00001062
1063 // De-duplicate intrinsics that are identical.
1064 return utils::GetOrCreate(intrinsics, intrinsic, [&] {
Ben Clayton4ffcf062021-07-22 13:24:59 +00001065 std::vector<sem::Parameter*> params;
Ben Clayton0f2d95d2021-07-22 13:24:59 +00001066 params.reserve(intrinsic.parameters.size());
1067 for (auto& p : intrinsic.parameters) {
1068 params.emplace_back(builder.create<sem::Parameter>(
Ben Clayton4ffcf062021-07-22 13:24:59 +00001069 nullptr, static_cast<uint32_t>(params.size()), p.type,
1070 ast::StorageClass::kNone, ast::Access::kUndefined, p.usage));
Ben Clayton0f2d95d2021-07-22 13:24:59 +00001071 }
1072 return builder.create<sem::Intrinsic>(
1073 intrinsic.type, const_cast<sem::Type*>(intrinsic.return_type),
1074 std::move(params), intrinsic.supported_stages, intrinsic.is_deprecated);
Ben Claytonb478f972021-07-15 20:34:21 +00001075 });
Ben Claytonb29a59d2021-06-01 19:06:31 +00001076}
1077
1078MatchState Impl::Match(ClosedState& closed,
1079 const OverloadInfo& overload,
1080 MatcherIndex const* matcher_indices) const {
1081 return MatchState(builder, closed, matchers, overload, matcher_indices);
1082}
1083
1084void Impl::PrintOverload(std::ostream& ss,
1085 const OverloadInfo& overload,
1086 sem::IntrinsicType intrinsic_type) const {
1087 ClosedState closed(builder);
1088
1089 ss << intrinsic_type << "(";
1090 for (uint32_t p = 0; p < overload.num_parameters; p++) {
1091 auto& parameter = overload.parameters[p];
1092 if (p > 0) {
1093 ss << ", ";
1094 }
1095 if (parameter.usage != ParameterUsage::kNone) {
1096 ss << sem::str(parameter.usage) << ": ";
1097 }
1098 auto* indices = parameter.matcher_indices;
1099 ss << Match(closed, overload, indices).TypeName();
1100 }
1101 ss << ")";
1102 if (overload.return_matcher_indices) {
1103 ss << " -> ";
1104 auto* indices = overload.return_matcher_indices;
1105 ss << Match(closed, overload, indices).TypeName();
Ben Clayton59d24732021-02-08 22:42:54 +00001106 }
1107
Ben Claytonb29a59d2021-06-01 19:06:31 +00001108 bool first = true;
1109 auto separator = [&] {
1110 ss << (first ? " where: " : ", ");
1111 first = false;
1112 };
1113 for (uint32_t i = 0; i < overload.num_open_types; i++) {
1114 auto& open_type = overload.open_types[i];
1115 if (open_type.matcher_index != kNoMatcher) {
1116 separator();
1117 ss << open_type.name;
1118 auto* index = &open_type.matcher_index;
1119 ss << " is " << Match(closed, overload, index).TypeName();
1120 }
1121 }
1122 for (uint32_t i = 0; i < overload.num_open_numbers; i++) {
1123 auto& open_number = overload.open_numbers[i];
1124 if (open_number.matcher_index != kNoMatcher) {
1125 separator();
1126 ss << open_number.name;
1127 auto* index = &open_number.matcher_index;
1128 ss << " is " << Match(closed, overload, index).NumName();
1129 }
1130 }
1131}
1132
1133const sem::Type* MatchState::Type(const sem::Type* ty) {
1134 MatcherIndex matcher_index = *matcher_indices_++;
1135 auto* matcher = matchers.type[matcher_index];
1136 return matcher->Match(*this, ty);
1137}
1138
1139Number MatchState::Num(Number number) {
1140 MatcherIndex matcher_index = *matcher_indices_++;
1141 auto* matcher = matchers.number[matcher_index];
1142 return matcher->Match(*this, number);
1143}
1144
1145std::string MatchState::TypeName() {
1146 MatcherIndex matcher_index = *matcher_indices_++;
1147 auto* matcher = matchers.type[matcher_index];
1148 return matcher->String(*this);
1149}
1150
1151std::string MatchState::NumName() {
1152 MatcherIndex matcher_index = *matcher_indices_++;
1153 auto* matcher = matchers.number[matcher_index];
1154 return matcher->String(*this);
Ben Clayton59d24732021-02-08 22:42:54 +00001155}
1156
1157} // namespace
1158
Ben Claytonb29a59d2021-06-01 19:06:31 +00001159std::unique_ptr<IntrinsicTable> IntrinsicTable::Create(
1160 ProgramBuilder& builder) {
1161 return std::make_unique<Impl>(builder);
Ben Clayton59d24732021-02-08 22:42:54 +00001162}
1163
1164IntrinsicTable::~IntrinsicTable() = default;
1165
Ben Claytonb29a59d2021-06-01 19:06:31 +00001166/// TypeInfo for the Any type declared in the anonymous namespace above
1167TINT_INSTANTIATE_TYPEINFO(Any);
1168
Ben Clayton59d24732021-02-08 22:42:54 +00001169} // namespace tint