blob: deb10128cc9f2631b205a3138f91eaac17076fa2 [file] [log] [blame]
Austin Engcc2516a2023-10-17 20:57:54 +00001// Copyright 2020 The Dawn & Tint Authors
Ryan Harrisondbc13af2022-02-21 15:19:07 +00002//
Austin Engcc2516a2023-10-17 20:57:54 +00003// Redistribution and use in source and binary forms, with or without
4// modification, are permitted provided that the following conditions are met:
Ryan Harrisondbc13af2022-02-21 15:19:07 +00005//
Austin Engcc2516a2023-10-17 20:57:54 +00006// 1. Redistributions of source code must retain the above copyright notice, this
7// list of conditions and the following disclaimer.
Ryan Harrisondbc13af2022-02-21 15:19:07 +00008//
Austin Engcc2516a2023-10-17 20:57:54 +00009// 2. Redistributions in binary form must reproduce the above copyright notice,
10// this list of conditions and the following disclaimer in the documentation
11// and/or other materials provided with the distribution.
12//
13// 3. Neither the name of the copyright holder nor the names of its
14// contributors may be used to endorse or promote products derived from
15// this software without specific prior written permission.
16//
17// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
18// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
20// DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
21// FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22// DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
23// SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
24// CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
25// OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
Ryan Harrisondbc13af2022-02-21 15:19:07 +000027
dan sinclaircff24fc2023-07-24 21:04:39 +000028#ifndef SRC_TINT_LANG_WGSL_AST_CLONE_CONTEXT_H_
29#define SRC_TINT_LANG_WGSL_AST_CLONE_CONTEXT_H_
Ryan Harrisondbc13af2022-02-21 15:19:07 +000030
31#include <algorithm>
32#include <functional>
Peter Kastingd3921b82022-05-05 17:00:31 +000033#include <type_traits>
Ryan Harrisondbc13af2022-02-21 15:19:07 +000034#include <utility>
35#include <vector>
36
Ben Claytonae18c412023-07-29 13:00:40 +000037#include "src/tint/lang/wgsl/ast/builder.h"
dan sinclair22b4dd22023-07-21 00:40:07 +000038#include "src/tint/utils/containers/hashmap.h"
39#include "src/tint/utils/containers/hashset.h"
40#include "src/tint/utils/containers/vector.h"
Ben Claytonf848af22023-07-28 16:37:32 +000041#include "src/tint/utils/diagnostic/diagnostic.h"
42#include "src/tint/utils/diagnostic/source.h"
Ben Claytonf848af22023-07-28 16:37:32 +000043#include "src/tint/utils/ice/ice.h"
Ben Clayton67fc5902023-08-01 00:37:35 +000044#include "src/tint/utils/id/generation_id.h"
dan sinclair22b4dd22023-07-21 00:40:07 +000045#include "src/tint/utils/macros/compiler.h"
46#include "src/tint/utils/rtti/castable.h"
Ben Claytondc68d5c2023-08-01 00:37:35 +000047#include "src/tint/utils/symbol/symbol.h"
dan sinclair22b4dd22023-07-21 00:40:07 +000048#include "src/tint/utils/traits/traits.h"
Ryan Harrisondbc13af2022-02-21 15:19:07 +000049
dan sinclair8155b9d2022-04-07 19:10:25 +000050// Forward declarations
51namespace tint::ast {
52class FunctionList;
53class Node;
Ben Clayton971318f2023-02-14 13:52:43 +000054struct Type;
dan sinclair8155b9d2022-04-07 19:10:25 +000055} // namespace tint::ast
56
Ben Claytonae18c412023-07-29 13:00:40 +000057namespace tint::ast {
Ryan Harrisondbc13af2022-02-21 15:19:07 +000058
59/// CloneContext holds the state used while cloning AST nodes.
60class CloneContext {
dan sinclair41e4d9a2022-05-01 14:40:55 +000061 /// ParamTypeIsPtrOf<F, T> is true iff the first parameter of
62 /// F is a pointer of (or derives from) type T.
63 template <typename F, typename T>
dan sinclairbae54e72023-07-28 15:01:54 +000064 static constexpr bool ParamTypeIsPtrOf = tint::traits::
65 IsTypeOrDerived<typename std::remove_pointer<tint::traits::ParameterType<F, 0>>::type, T>;
Ryan Harrisondbc13af2022-02-21 15:19:07 +000066
dan sinclair41e4d9a2022-05-01 14:40:55 +000067 public:
68 /// SymbolTransform is a function that takes a symbol and returns a new
69 /// symbol.
70 using SymbolTransform = std::function<Symbol(Symbol)>;
Ryan Harrisondbc13af2022-02-21 15:19:07 +000071
dan sinclair41e4d9a2022-05-01 14:40:55 +000072 /// Constructor for cloning objects from `from` into `to`.
Ben Claytonae18c412023-07-29 13:00:40 +000073 /// @param to the target Builder to clone into
74 /// @param from the generation ID of the source program being cloned
75 CloneContext(Builder* to, GenerationID from);
Ryan Harrisondbc13af2022-02-21 15:19:07 +000076
dan sinclair41e4d9a2022-05-01 14:40:55 +000077 /// Destructor
78 ~CloneContext();
Ryan Harrisondbc13af2022-02-21 15:19:07 +000079
Ben Claytonae18c412023-07-29 13:00:40 +000080 /// Clones the Node or type::Type @p object into the Builder #dst if @p object is not null. If
81 /// @p object is null, then Clone() returns null.
dan sinclair41e4d9a2022-05-01 14:40:55 +000082 ///
Ben Claytonae18c412023-07-29 13:00:40 +000083 /// Clone() may use a function registered with ReplaceAll() to create a transformed version of
84 /// the object. See ReplaceAll() for more information.
dan sinclair41e4d9a2022-05-01 14:40:55 +000085 ///
Ben Claytonae18c412023-07-29 13:00:40 +000086 /// If the CloneContext is cloning from a Program to a Builder, then the Node or type::Type @p
87 /// object must be owned by the source program.
dan sinclair41e4d9a2022-05-01 14:40:55 +000088 ///
Ben Claytonae18c412023-07-29 13:00:40 +000089 /// @param object the type deriving from Node to clone
dan sinclair41e4d9a2022-05-01 14:40:55 +000090 /// @return the cloned node
91 template <typename T>
92 const T* Clone(const T* object) {
Ben Claytonae18c412023-07-29 13:00:40 +000093 TINT_ASSERT_GENERATION_IDS_EQUAL_IF_VALID(src_id, object);
94 if (auto* cloned = CloneNode(object)) {
dan sinclair41e4d9a2022-05-01 14:40:55 +000095 auto* out = CheckedCast<T>(cloned);
Ben Claytonf848af22023-07-28 16:37:32 +000096 TINT_ASSERT_GENERATION_IDS_EQUAL_IF_VALID(dst, out);
dan sinclair41e4d9a2022-05-01 14:40:55 +000097 return out;
Ryan Harrisondbc13af2022-02-21 15:19:07 +000098 }
dan sinclair41e4d9a2022-05-01 14:40:55 +000099 return nullptr;
100 }
Ryan Harrisondbc13af2022-02-21 15:19:07 +0000101
Ben Claytonae18c412023-07-29 13:00:40 +0000102 /// Clones the Node or type::Type @p object into the Builder #dst if @p object is not null. If
103 /// @p object is null, then Clone() returns null.
dan sinclair41e4d9a2022-05-01 14:40:55 +0000104 ///
Ben Claytonae18c412023-07-29 13:00:40 +0000105 /// Unlike Clone(), this method does not invoke or use any transformations registered by
106 /// ReplaceAll().
dan sinclair41e4d9a2022-05-01 14:40:55 +0000107 ///
Ben Claytonae18c412023-07-29 13:00:40 +0000108 /// If the CloneContext is cloning from a Program to a Builder, then the Node or type::Type @p
109 /// object must be owned by the source program.
dan sinclair41e4d9a2022-05-01 14:40:55 +0000110 ///
Ben Claytonae18c412023-07-29 13:00:40 +0000111 /// @param a the type deriving from Node to clone
dan sinclair41e4d9a2022-05-01 14:40:55 +0000112 /// @return the cloned node
113 template <typename T>
114 const T* CloneWithoutTransform(const T* a) {
115 // If the input is nullptr, there's nothing to clone - just return nullptr.
116 if (a == nullptr) {
117 return nullptr;
118 }
Ben Claytonae18c412023-07-29 13:00:40 +0000119 TINT_ASSERT_GENERATION_IDS_EQUAL_IF_VALID(src_id, a);
120 auto* c = a->Clone(*this);
dan sinclair41e4d9a2022-05-01 14:40:55 +0000121 return CheckedCast<T>(c);
122 }
123
Ben Claytonae18c412023-07-29 13:00:40 +0000124 /// Clones the ast::Type `ty` into the Builder #dst
Ben Clayton971318f2023-02-14 13:52:43 +0000125 /// @param ty the AST type.
126 /// @return the cloned node
127 ast::Type Clone(const ast::Type& ty);
128
dan sinclair41e4d9a2022-05-01 14:40:55 +0000129 /// Clones the Source `s` into #dst
Ben Claytone9ab1b62024-04-15 17:21:30 +0000130 /// @note this 'clone' is a shallow copy.
dan sinclair41e4d9a2022-05-01 14:40:55 +0000131 /// @param s the `Source` to clone
132 /// @return the cloned source
133 Source Clone(const Source& s) const { return s; }
134
135 /// Clones the Symbol `s` into #dst
136 ///
Ben Claytonae18c412023-07-29 13:00:40 +0000137 /// The Symbol `s` must be owned by the source program.
dan sinclair41e4d9a2022-05-01 14:40:55 +0000138 ///
139 /// @param s the Symbol to clone
140 /// @return the cloned source
141 Symbol Clone(Symbol s);
142
Ben Claytonae18c412023-07-29 13:00:40 +0000143 /// Clones each of the elements of the vector `v` into the Builder
dan sinclair41e4d9a2022-05-01 14:40:55 +0000144 /// #dst.
145 ///
Ben Claytonae18c412023-07-29 13:00:40 +0000146 /// All the elements of the vector `v` must be owned by the source program.
dan sinclair41e4d9a2022-05-01 14:40:55 +0000147 ///
148 /// @param v the vector to clone
149 /// @return the cloned vector
Ben Clayton783b1692022-08-02 17:03:35 +0000150 template <typename T, size_t N>
dan sinclairbae54e72023-07-28 15:01:54 +0000151 tint::Vector<T, N> Clone(const tint::Vector<T, N>& v) {
152 tint::Vector<T, N> out;
dan sinclair41e4d9a2022-05-01 14:40:55 +0000153 out.reserve(v.size());
154 for (auto& el : v) {
Ben Clayton783b1692022-08-02 17:03:35 +0000155 out.Push(Clone(el));
dan sinclair41e4d9a2022-05-01 14:40:55 +0000156 }
157 return out;
158 }
159
Ben Claytonae18c412023-07-29 13:00:40 +0000160 /// Clones each of the elements of the vector `v` using the Builder
dan sinclair41e4d9a2022-05-01 14:40:55 +0000161 /// #dst, inserting any additional elements into the list that were registered
162 /// with calls to InsertBefore().
163 ///
Ben Claytonae18c412023-07-29 13:00:40 +0000164 /// All the elements of the vector `v` must be owned by the source program.
dan sinclair41e4d9a2022-05-01 14:40:55 +0000165 ///
166 /// @param v the vector to clone
167 /// @return the cloned vector
Ben Clayton783b1692022-08-02 17:03:35 +0000168 template <typename T, size_t N>
dan sinclairbae54e72023-07-28 15:01:54 +0000169 tint::Vector<T*, N> Clone(const tint::Vector<T*, N>& v) {
170 tint::Vector<T*, N> out;
dan sinclair41e4d9a2022-05-01 14:40:55 +0000171 Clone(out, v);
172 return out;
173 }
174
175 /// Clones each of the elements of the vector `from` into the vector `to`,
176 /// inserting any additional elements into the list that were registered with
177 /// calls to InsertBefore().
178 ///
Ben Claytonae18c412023-07-29 13:00:40 +0000179 /// All the elements of the vector `from` must be owned by the source program.
dan sinclair41e4d9a2022-05-01 14:40:55 +0000180 ///
181 /// @param from the vector to clone
182 /// @param to the cloned result
Ben Clayton783b1692022-08-02 17:03:35 +0000183 template <typename T, size_t N>
dan sinclairbae54e72023-07-28 15:01:54 +0000184 void Clone(tint::Vector<T*, N>& to, const tint::Vector<T*, N>& from) {
Ben Clayton783b1692022-08-02 17:03:35 +0000185 to.Reserve(from.Length());
dan sinclair41e4d9a2022-05-01 14:40:55 +0000186
Ben Clayton7a27d6c2024-01-31 19:44:28 +0000187 auto transforms = list_transforms_.Get(&from);
Ben Clayton18dc3152022-08-23 18:38:35 +0000188
189 if (transforms) {
Ben Clayton2d63e322022-11-02 13:40:41 +0000190 for (auto& builder : transforms->insert_front_) {
191 to.Push(CheckedCast<T>(builder()));
Ryan Harrisondbc13af2022-02-21 15:19:07 +0000192 }
dan sinclair41e4d9a2022-05-01 14:40:55 +0000193 for (auto& el : from) {
Ben Clayton7a27d6c2024-01-31 19:44:28 +0000194 if (auto insert_before = transforms->insert_before_.Get(el)) {
Ben Clayton2d63e322022-11-02 13:40:41 +0000195 for (auto& builder : *insert_before) {
196 to.Push(CheckedCast<T>(builder()));
dan sinclair41e4d9a2022-05-01 14:40:55 +0000197 }
198 }
Ben Clayton18dc3152022-08-23 18:38:35 +0000199 if (!transforms->remove_.Contains(el)) {
Ben Clayton783b1692022-08-02 17:03:35 +0000200 to.Push(Clone(el));
dan sinclair41e4d9a2022-05-01 14:40:55 +0000201 }
Ben Clayton7a27d6c2024-01-31 19:44:28 +0000202 if (auto insert_after = transforms->insert_after_.Get(el)) {
Ben Clayton2d63e322022-11-02 13:40:41 +0000203 for (auto& builder : *insert_after) {
204 to.Push(CheckedCast<T>(builder()));
dan sinclair41e4d9a2022-05-01 14:40:55 +0000205 }
206 }
207 }
Ben Clayton2d63e322022-11-02 13:40:41 +0000208 for (auto& builder : transforms->insert_back_) {
209 to.Push(CheckedCast<T>(builder()));
dan sinclair41e4d9a2022-05-01 14:40:55 +0000210 }
211 } else {
212 for (auto& el : from) {
Ben Clayton783b1692022-08-02 17:03:35 +0000213 to.Push(Clone(el));
Ryan Harrisondbc13af2022-02-21 15:19:07 +0000214
Ben Clayton7a27d6c2024-01-31 19:44:28 +0000215 if (!transforms) {
216 // Clone(el) may have create a transformation list
217 transforms = list_transforms_.Get(&from);
218 }
Ben Clayton18dc3152022-08-23 18:38:35 +0000219 if (transforms) {
Ben Clayton7a27d6c2024-01-31 19:44:28 +0000220 if (auto insert_after = transforms->insert_after_.Get(el)) {
Ben Clayton2d63e322022-11-02 13:40:41 +0000221 for (auto& builder : *insert_after) {
222 to.Push(CheckedCast<T>(builder()));
dan sinclair41e4d9a2022-05-01 14:40:55 +0000223 }
224 }
225 }
226 }
227
Ben Clayton7a27d6c2024-01-31 19:44:28 +0000228 if (!transforms) {
229 // Clone(el) may have create a transformation list
230 transforms = list_transforms_.Get(&from);
231 }
Ben Clayton18dc3152022-08-23 18:38:35 +0000232 if (transforms) {
Ben Clayton2d63e322022-11-02 13:40:41 +0000233 for (auto& builder : transforms->insert_back_) {
234 to.Push(CheckedCast<T>(builder()));
dan sinclair41e4d9a2022-05-01 14:40:55 +0000235 }
236 }
Ryan Harrisondbc13af2022-02-21 15:19:07 +0000237 }
Ryan Harrisondbc13af2022-02-21 15:19:07 +0000238 }
Ryan Harrisondbc13af2022-02-21 15:19:07 +0000239
Ben Claytonae18c412023-07-29 13:00:40 +0000240 /// Clones each of the elements of the vector `v` into the Builder
dan sinclair41e4d9a2022-05-01 14:40:55 +0000241 /// #dst.
242 ///
Ben Claytonae18c412023-07-29 13:00:40 +0000243 /// All the elements of the vector `v` must be owned by the source program.
dan sinclair41e4d9a2022-05-01 14:40:55 +0000244 ///
245 /// @param v the vector to clone
246 /// @return the cloned vector
247 ast::FunctionList Clone(const ast::FunctionList& v);
Ryan Harrisondbc13af2022-02-21 15:19:07 +0000248
dan sinclair41e4d9a2022-05-01 14:40:55 +0000249 /// ReplaceAll() registers `replacer` to be called whenever the Clone() method
Ben Claytonae18c412023-07-29 13:00:40 +0000250 /// is called with a Node type that matches (or derives from) the type of
dan sinclair41e4d9a2022-05-01 14:40:55 +0000251 /// the single parameter of `replacer`.
Ben Claytonae18c412023-07-29 13:00:40 +0000252 /// The returned Node of `replacer` will be used as the replacement for
253 /// all references to the object that's being cloned. This returned Node
dan sinclair41e4d9a2022-05-01 14:40:55 +0000254 /// must be owned by the Program #dst.
255 ///
256 /// `replacer` must be function-like with the signature: `T* (T*)`
Ben Claytonae18c412023-07-29 13:00:40 +0000257 /// where `T` is a type deriving from Node.
dan sinclair41e4d9a2022-05-01 14:40:55 +0000258 ///
259 /// If `replacer` returns a nullptr then Clone() will call `T::Clone()` to
260 /// clone the object.
261 ///
262 /// Example:
263 ///
264 /// ```
265 /// // Replace all ast::UintLiteralExpressions with the number 42
266 /// CloneCtx ctx(&out, in);
267 /// ctx.ReplaceAll([&] (ast::UintLiteralExpression* l) {
Ben Claytonae18c412023-07-29 13:00:40 +0000268 /// return ctx.dst->create<ast::UintLiteralExpression>(
269 /// ctx.Clone(l->source),
270 /// ctx.Clone(l->type),
dan sinclair41e4d9a2022-05-01 14:40:55 +0000271 /// 42);
272 /// });
273 /// ctx.Clone();
274 /// ```
275 ///
276 /// @warning a single handler can only be registered for any given type.
277 /// Attempting to register two handlers for the same type will result in an
278 /// ICE.
279 /// @warning The replacement object must be of the correct type for all
280 /// references of the original object. A type mismatch will result in an
281 /// assertion in debug builds, and undefined behavior in release builds.
282 /// @param replacer a function or function-like object with the signature
Ben Claytonae18c412023-07-29 13:00:40 +0000283 /// `T* (T*)`, where `T` derives from Node
dan sinclair41e4d9a2022-05-01 14:40:55 +0000284 /// @returns this CloneContext so calls can be chained
285 template <typename F>
Ben Claytonae18c412023-07-29 13:00:40 +0000286 traits::EnableIf<ParamTypeIsPtrOf<F, ast::Node>, CloneContext>& ReplaceAll(F&& replacer) {
287 using TPtr = traits::ParameterType<F, 0>;
dan sinclair41e4d9a2022-05-01 14:40:55 +0000288 using T = typename std::remove_pointer<TPtr>::type;
289 for (auto& transform : transforms_) {
dan sinclairbae54e72023-07-28 15:01:54 +0000290 bool already_registered = transform.typeinfo->Is(&tint::TypeInfo::Of<T>()) ||
291 tint::TypeInfo::Of<T>().Is(transform.typeinfo);
Ben Clayton884f9522023-01-12 22:52:57 +0000292 if (TINT_UNLIKELY(already_registered)) {
Ben Claytonf848af22023-07-28 16:37:32 +0000293 TINT_ICE() << "ReplaceAll() called with a handler for type "
294 << TypeInfo::Of<T>().name
295 << " that is already handled by a handler for type "
296 << transform.typeinfo->name;
dan sinclair41e4d9a2022-05-01 14:40:55 +0000297 return *this;
298 }
299 }
300 CloneableTransform transform;
Ben Claytonae18c412023-07-29 13:00:40 +0000301 transform.typeinfo = &TypeInfo::Of<T>();
302 transform.function = [=](const ast::Node* in) { return replacer(in->As<T>()); };
Ben Clayton783b1692022-08-02 17:03:35 +0000303 transforms_.Push(std::move(transform));
Ryan Harrisondbc13af2022-02-21 15:19:07 +0000304 return *this;
Ryan Harrisondbc13af2022-02-21 15:19:07 +0000305 }
dan sinclair41e4d9a2022-05-01 14:40:55 +0000306
307 /// ReplaceAll() registers `replacer` to be called whenever the Clone() method
308 /// is called with a Symbol.
309 /// The returned symbol of `replacer` will be used as the replacement for
310 /// all references to the symbol that's being cloned. This returned Symbol
311 /// must be owned by the Program #dst.
312 /// @param replacer a function the signature `Symbol(Symbol)`.
313 /// @warning a SymbolTransform can only be registered once. Attempting to
314 /// register a SymbolTransform more than once will result in an ICE.
315 /// @returns this CloneContext so calls can be chained
316 CloneContext& ReplaceAll(const SymbolTransform& replacer) {
Ben Clayton884f9522023-01-12 22:52:57 +0000317 if (TINT_UNLIKELY(symbol_transform_)) {
Ben Claytonf848af22023-07-28 16:37:32 +0000318 TINT_ICE() << "ReplaceAll(const SymbolTransform&) called multiple times on the same "
319 "CloneContext";
dan sinclair41e4d9a2022-05-01 14:40:55 +0000320 }
321 symbol_transform_ = replacer;
322 return *this;
323 }
324
Ben Claytonae18c412023-07-29 13:00:40 +0000325 /// Replace replaces all occurrences of `what` in the source program with the pointer `with`
dan sinclair41e4d9a2022-05-01 14:40:55 +0000326 /// in #dst when calling Clone().
327 /// [DEPRECATED]: This function cannot handle nested replacements. Use the
328 /// overload of Replace() that take a function for the `WITH` argument.
Ben Claytonae18c412023-07-29 13:00:40 +0000329 /// @param what a pointer to the object in the source program that will be replaced with
dan sinclair41e4d9a2022-05-01 14:40:55 +0000330 /// `with`
331 /// @param with a pointer to the replacement object owned by #dst that will be
332 /// used as a replacement for `what`
333 /// @warning The replacement object must be of the correct type for all
334 /// references of the original object. A type mismatch will result in an
335 /// assertion in debug builds, and undefined behavior in release builds.
336 /// @returns this CloneContext so calls can be chained
Ben Claytonae18c412023-07-29 13:00:40 +0000337 template <typename WHAT, typename WITH, typename = traits::EnableIfIsType<WITH, ast::Node>>
dan sinclair41e4d9a2022-05-01 14:40:55 +0000338 CloneContext& Replace(const WHAT* what, const WITH* with) {
Ben Claytonae18c412023-07-29 13:00:40 +0000339 TINT_ASSERT_GENERATION_IDS_EQUAL_IF_VALID(src_id, what);
Ben Claytonf848af22023-07-28 16:37:32 +0000340 TINT_ASSERT_GENERATION_IDS_EQUAL_IF_VALID(dst, with);
Ben Claytonae18c412023-07-29 13:00:40 +0000341 replacements_.Replace(what, [with]() -> const ast::Node* { return with; });
dan sinclair41e4d9a2022-05-01 14:40:55 +0000342 return *this;
343 }
344
Ben Claytonae18c412023-07-29 13:00:40 +0000345 /// Replace replaces all occurrences of `what` in the source program with the result of the
dan sinclair41e4d9a2022-05-01 14:40:55 +0000346 /// function `with` in #dst when calling Clone(). `with` will be called each
347 /// time `what` is cloned by this context. If `what` is not cloned, then
348 /// `with` may never be called.
Ben Claytonae18c412023-07-29 13:00:40 +0000349 /// @param what a pointer to the object in the source program that will be replaced with
dan sinclair41e4d9a2022-05-01 14:40:55 +0000350 /// `with`
351 /// @param with a function that takes no arguments and returns a pointer to
352 /// the replacement object owned by #dst. The returned pointer will be used as
353 /// a replacement for `what`.
354 /// @warning The replacement object must be of the correct type for all
355 /// references of the original object. A type mismatch will result in an
356 /// assertion in debug builds, and undefined behavior in release builds.
357 /// @returns this CloneContext so calls can be chained
Peter Kastingd3921b82022-05-05 17:00:31 +0000358 template <typename WHAT, typename WITH, typename = std::invoke_result_t<WITH>>
dan sinclair41e4d9a2022-05-01 14:40:55 +0000359 CloneContext& Replace(const WHAT* what, WITH&& with) {
Ben Claytonae18c412023-07-29 13:00:40 +0000360 TINT_ASSERT_GENERATION_IDS_EQUAL_IF_VALID(src_id, what);
Ben Clayton57be2ff2023-03-02 17:45:21 +0000361 replacements_.Replace(what, with);
dan sinclair41e4d9a2022-05-01 14:40:55 +0000362 return *this;
363 }
364
Ben Clayton2d63e322022-11-02 13:40:41 +0000365 /// Removes @p object from the cloned copy of @p vector.
Ben Claytonae18c412023-07-29 13:00:40 +0000366 /// @param vector the vector in the source program
367 /// @param object a pointer to the object in the source program that will be omitted from
dan sinclair41e4d9a2022-05-01 14:40:55 +0000368 /// the cloned vector.
369 /// @returns this CloneContext so calls can be chained
Ben Clayton783b1692022-08-02 17:03:35 +0000370 template <typename T, size_t N, typename OBJECT>
Ben Claytonf848af22023-07-28 16:37:32 +0000371 CloneContext& Remove(const Vector<T, N>& vector, OBJECT* object) {
Ben Claytonae18c412023-07-29 13:00:40 +0000372 TINT_ASSERT_GENERATION_IDS_EQUAL_IF_VALID(src_id, object);
Ben Clayton884f9522023-01-12 22:52:57 +0000373 if (TINT_UNLIKELY((std::find(vector.begin(), vector.end(), object) == vector.end()))) {
Ben Claytonf848af22023-07-28 16:37:32 +0000374 TINT_ICE() << "CloneContext::Remove() vector does not contain object";
dan sinclair41e4d9a2022-05-01 14:40:55 +0000375 return *this;
376 }
377
Ben Clayton7a27d6c2024-01-31 19:44:28 +0000378 list_transforms_.GetOrAddZero(&vector).remove_.Add(object);
dan sinclair41e4d9a2022-05-01 14:40:55 +0000379 return *this;
380 }
381
Ben Clayton2d63e322022-11-02 13:40:41 +0000382 /// Inserts @p object before any other objects of @p vector, when the vector is cloned.
Ben Claytonae18c412023-07-29 13:00:40 +0000383 /// @param vector the vector in the source program
dan sinclair41e4d9a2022-05-01 14:40:55 +0000384 /// @param object a pointer to the object in #dst that will be inserted at the
385 /// front of the vector
386 /// @returns this CloneContext so calls can be chained
Ben Clayton783b1692022-08-02 17:03:35 +0000387 template <typename T, size_t N, typename OBJECT>
Ben Claytonf848af22023-07-28 16:37:32 +0000388 CloneContext& InsertFront(const Vector<T, N>& vector, OBJECT* object) {
389 TINT_ASSERT_GENERATION_IDS_EQUAL_IF_VALID(dst, object);
Ben Clayton2d63e322022-11-02 13:40:41 +0000390 return InsertFront(vector, [object] { return object; });
391 }
392
393 /// Inserts a lazily built object before any other objects of @p vector, when the vector is
394 /// cloned.
Ben Claytonae18c412023-07-29 13:00:40 +0000395 /// @param vector the vector in the source program
Ben Clayton2d63e322022-11-02 13:40:41 +0000396 /// @param builder a builder of the object that will be inserted at the front of the vector.
397 /// @returns this CloneContext so calls can be chained
398 template <typename T, size_t N, typename BUILDER>
dan sinclairbae54e72023-07-28 15:01:54 +0000399 CloneContext& InsertFront(const tint::Vector<T, N>& vector, BUILDER&& builder) {
Ben Clayton7a27d6c2024-01-31 19:44:28 +0000400 list_transforms_.GetOrAddZero(&vector).insert_front_.Push(std::forward<BUILDER>(builder));
dan sinclair41e4d9a2022-05-01 14:40:55 +0000401 return *this;
402 }
403
Ben Clayton2d63e322022-11-02 13:40:41 +0000404 /// Inserts @p object after any other objects of @p vector, when the vector is cloned.
Ben Claytonae18c412023-07-29 13:00:40 +0000405 /// @param vector the vector in the source program
dan sinclair41e4d9a2022-05-01 14:40:55 +0000406 /// @param object a pointer to the object in #dst that will be inserted at the
407 /// end of the vector
408 /// @returns this CloneContext so calls can be chained
Ben Clayton783b1692022-08-02 17:03:35 +0000409 template <typename T, size_t N, typename OBJECT>
Ben Claytonf848af22023-07-28 16:37:32 +0000410 CloneContext& InsertBack(const Vector<T, N>& vector, OBJECT* object) {
411 TINT_ASSERT_GENERATION_IDS_EQUAL_IF_VALID(dst, object);
Ben Clayton2d63e322022-11-02 13:40:41 +0000412 return InsertBack(vector, [object] { return object; });
413 }
414
415 /// Inserts a lazily built object after any other objects of @p vector, when the vector is
416 /// cloned.
Ben Claytonae18c412023-07-29 13:00:40 +0000417 /// @param vector the vector in the source program
Ben Clayton2d63e322022-11-02 13:40:41 +0000418 /// @param builder the builder of the object in #dst that will be inserted at the end of the
419 /// vector.
420 /// @returns this CloneContext so calls can be chained
421 template <typename T, size_t N, typename BUILDER>
dan sinclairbae54e72023-07-28 15:01:54 +0000422 CloneContext& InsertBack(const tint::Vector<T, N>& vector, BUILDER&& builder) {
Ben Clayton7a27d6c2024-01-31 19:44:28 +0000423 list_transforms_.GetOrAddZero(&vector).insert_back_.Push(std::forward<BUILDER>(builder));
dan sinclair41e4d9a2022-05-01 14:40:55 +0000424 return *this;
425 }
426
Ben Clayton2d63e322022-11-02 13:40:41 +0000427 /// Inserts @p object before @p before whenever @p vector is cloned.
Ben Claytonae18c412023-07-29 13:00:40 +0000428 /// @param vector the vector in the source program
429 /// @param before a pointer to the object in the source program
dan sinclair41e4d9a2022-05-01 14:40:55 +0000430 /// @param object a pointer to the object in #dst that will be inserted before
Ben Clayton2d63e322022-11-02 13:40:41 +0000431 /// any occurrence of the clone of @p before
dan sinclair41e4d9a2022-05-01 14:40:55 +0000432 /// @returns this CloneContext so calls can be chained
Ben Clayton783b1692022-08-02 17:03:35 +0000433 template <typename T, size_t N, typename BEFORE, typename OBJECT>
dan sinclairbae54e72023-07-28 15:01:54 +0000434 CloneContext& InsertBefore(const tint::Vector<T, N>& vector,
dan sinclair41e4d9a2022-05-01 14:40:55 +0000435 const BEFORE* before,
436 const OBJECT* object) {
Ben Claytonae18c412023-07-29 13:00:40 +0000437 TINT_ASSERT_GENERATION_IDS_EQUAL_IF_VALID(src_id, before);
Ben Claytonf848af22023-07-28 16:37:32 +0000438 TINT_ASSERT_GENERATION_IDS_EQUAL_IF_VALID(dst, object);
Ben Clayton884f9522023-01-12 22:52:57 +0000439 if (TINT_UNLIKELY((std::find(vector.begin(), vector.end(), before) == vector.end()))) {
Ben Claytonf848af22023-07-28 16:37:32 +0000440 TINT_ICE() << "CloneContext::InsertBefore() vector does not contain before";
dan sinclair41e4d9a2022-05-01 14:40:55 +0000441 return *this;
442 }
443
Ben Clayton7a27d6c2024-01-31 19:44:28 +0000444 list_transforms_.GetOrAddZero(&vector).insert_before_.GetOrAddZero(before).Push(
Ben Clayton2d63e322022-11-02 13:40:41 +0000445 [object] { return object; });
dan sinclair41e4d9a2022-05-01 14:40:55 +0000446 return *this;
447 }
448
Ben Clayton2d63e322022-11-02 13:40:41 +0000449 /// Inserts a lazily created object before @p before whenever @p vector is cloned.
Ben Claytonae18c412023-07-29 13:00:40 +0000450 /// @param vector the vector in the source program
451 /// @param before a pointer to the object in the source program
Ben Clayton2d63e322022-11-02 13:40:41 +0000452 /// @param builder the builder of the object in #dst that will be inserted before any occurrence
453 /// of the clone of @p before
454 /// @returns this CloneContext so calls can be chained
455 template <typename T,
456 size_t N,
457 typename BEFORE,
458 typename BUILDER,
459 typename _ = std::enable_if_t<!std::is_pointer_v<std::decay_t<BUILDER>>>>
dan sinclairbae54e72023-07-28 15:01:54 +0000460 CloneContext& InsertBefore(const tint::Vector<T, N>& vector,
Ben Clayton2d63e322022-11-02 13:40:41 +0000461 const BEFORE* before,
462 BUILDER&& builder) {
Ben Clayton7a27d6c2024-01-31 19:44:28 +0000463 list_transforms_.GetOrAddZero(&vector).insert_before_.GetOrAddZero(before).Push(
Ben Clayton2d63e322022-11-02 13:40:41 +0000464 std::forward<BUILDER>(builder));
465 return *this;
466 }
467
468 /// Inserts @p object after @p after whenever @p vector is cloned.
Ben Claytonae18c412023-07-29 13:00:40 +0000469 /// @param vector the vector in the source program
470 /// @param after a pointer to the object in the source program
dan sinclair41e4d9a2022-05-01 14:40:55 +0000471 /// @param object a pointer to the object in #dst that will be inserted after
Ben Clayton2d63e322022-11-02 13:40:41 +0000472 /// any occurrence of the clone of @p after
dan sinclair41e4d9a2022-05-01 14:40:55 +0000473 /// @returns this CloneContext so calls can be chained
Ben Clayton783b1692022-08-02 17:03:35 +0000474 template <typename T, size_t N, typename AFTER, typename OBJECT>
dan sinclairbae54e72023-07-28 15:01:54 +0000475 CloneContext& InsertAfter(const tint::Vector<T, N>& vector,
dan sinclair41e4d9a2022-05-01 14:40:55 +0000476 const AFTER* after,
477 const OBJECT* object) {
Ben Claytonae18c412023-07-29 13:00:40 +0000478 TINT_ASSERT_GENERATION_IDS_EQUAL_IF_VALID(src_id, after);
Ben Claytonf848af22023-07-28 16:37:32 +0000479 TINT_ASSERT_GENERATION_IDS_EQUAL_IF_VALID(dst, object);
Ben Clayton884f9522023-01-12 22:52:57 +0000480 if (TINT_UNLIKELY((std::find(vector.begin(), vector.end(), after) == vector.end()))) {
Ben Claytonf848af22023-07-28 16:37:32 +0000481 TINT_ICE() << "CloneContext::InsertAfter() vector does not contain after";
dan sinclair41e4d9a2022-05-01 14:40:55 +0000482 return *this;
483 }
484
Ben Clayton7a27d6c2024-01-31 19:44:28 +0000485 list_transforms_.GetOrAddZero(&vector).insert_after_.GetOrAddZero(after).Push(
Ben Clayton2d63e322022-11-02 13:40:41 +0000486 [object] { return object; });
487 return *this;
488 }
489
490 /// Inserts a lazily created object after @p after whenever @p vector is cloned.
Ben Claytonae18c412023-07-29 13:00:40 +0000491 /// @param vector the vector in the source program
492 /// @param after a pointer to the object in the source program
Ben Clayton2d63e322022-11-02 13:40:41 +0000493 /// @param builder the builder of the object in #dst that will be inserted after any occurrence
494 /// of the clone of @p after
495 /// @returns this CloneContext so calls can be chained
496 template <typename T,
497 size_t N,
498 typename AFTER,
499 typename BUILDER,
500 typename _ = std::enable_if_t<!std::is_pointer_v<std::decay_t<BUILDER>>>>
dan sinclairbae54e72023-07-28 15:01:54 +0000501 CloneContext& InsertAfter(const tint::Vector<T, N>& vector,
Ben Clayton2d63e322022-11-02 13:40:41 +0000502 const AFTER* after,
503 BUILDER&& builder) {
Ben Clayton7a27d6c2024-01-31 19:44:28 +0000504 list_transforms_.GetOrAddZero(&vector).insert_after_.GetOrAddZero(after).Push(
Ben Clayton2d63e322022-11-02 13:40:41 +0000505 std::forward<BUILDER>(builder));
dan sinclair41e4d9a2022-05-01 14:40:55 +0000506 return *this;
507 }
508
Ben Claytonae18c412023-07-29 13:00:40 +0000509 /// The target Builder to clone into.
510 Builder* const dst;
dan sinclair41e4d9a2022-05-01 14:40:55 +0000511
Ben Claytonae18c412023-07-29 13:00:40 +0000512 /// The source Program generation identifier.
513 GenerationID const src_id;
dan sinclair41e4d9a2022-05-01 14:40:55 +0000514
515 private:
516 struct CloneableTransform {
517 /// Constructor
518 CloneableTransform();
519 /// Copy constructor
520 /// @param other the CloneableTransform to copy
521 CloneableTransform(const CloneableTransform& other);
522 /// Destructor
523 ~CloneableTransform();
524
Ben Claytonae18c412023-07-29 13:00:40 +0000525 // TypeInfo of the Node that the transform operates on
526 const TypeInfo* typeinfo;
527 std::function<const ast::Node*(const ast::Node*)> function;
Ryan Harrisondbc13af2022-02-21 15:19:07 +0000528 };
Ryan Harrisondbc13af2022-02-21 15:19:07 +0000529
Ben Claytonae18c412023-07-29 13:00:40 +0000530 /// A vector of const ast::Node*
531 using NodeBuilderList = Vector<std::function<const ast::Node*()>, 4>;
Ben Clayton18dc3152022-08-23 18:38:35 +0000532
533 /// Transformations to be applied to a list (vector)
534 struct ListTransforms {
Ben Claytonae18c412023-07-29 13:00:40 +0000535 /// A map of object in the source program to omit when cloned into #dst.
536 Hashset<const ast::Node*, 4> remove_;
Ben Clayton18dc3152022-08-23 18:38:35 +0000537
538 /// A list of objects in #dst to insert before any others when the vector is cloned.
Ben Claytonae18c412023-07-29 13:00:40 +0000539 NodeBuilderList insert_front_;
Ben Clayton18dc3152022-08-23 18:38:35 +0000540
541 /// A list of objects in #dst to insert after all others when the vector is cloned.
Ben Claytonae18c412023-07-29 13:00:40 +0000542 NodeBuilderList insert_back_;
Ben Clayton18dc3152022-08-23 18:38:35 +0000543
Ben Claytonae18c412023-07-29 13:00:40 +0000544 /// A map of object in the source program to the list of cloned objects in #dst.
545 /// Clone(const Vector<T*>& v) will use this to insert the map-value
Ben Clayton18dc3152022-08-23 18:38:35 +0000546 /// list into the target vector before cloning and inserting the map-key.
Ben Claytonae18c412023-07-29 13:00:40 +0000547 Hashmap<const ast::Node*, NodeBuilderList, 4> insert_before_;
Ben Clayton18dc3152022-08-23 18:38:35 +0000548
Ben Claytonae18c412023-07-29 13:00:40 +0000549 /// A map of object in the source program to the list of cloned objects in #dst.
550 /// Clone(const Vector<T*>& v) will use this to insert the map-value
Ben Clayton18dc3152022-08-23 18:38:35 +0000551 /// list into the target vector after cloning and inserting the map-key.
Ben Claytonae18c412023-07-29 13:00:40 +0000552 Hashmap<const ast::Node*, NodeBuilderList, 4> insert_after_;
Ben Clayton18dc3152022-08-23 18:38:35 +0000553 };
554
dan sinclair41e4d9a2022-05-01 14:40:55 +0000555 CloneContext(const CloneContext&) = delete;
556 CloneContext& operator=(const CloneContext&) = delete;
Ryan Harrisondbc13af2022-02-21 15:19:07 +0000557
dan sinclair41e4d9a2022-05-01 14:40:55 +0000558 /// Cast `obj` from type `FROM` to type `TO`, returning the cast object.
559 /// Reports an internal compiler error if the cast failed.
560 template <typename TO, typename FROM>
561 const TO* CheckedCast(const FROM* obj) {
562 if (obj == nullptr) {
563 return nullptr;
564 }
Ben Clayton884f9522023-01-12 22:52:57 +0000565 const TO* cast = obj->template As<TO>();
566 if (TINT_LIKELY(cast)) {
dan sinclair41e4d9a2022-05-01 14:40:55 +0000567 return cast;
568 }
dan sinclairbae54e72023-07-28 15:01:54 +0000569 CheckedCastFailure(obj, tint::TypeInfo::Of<TO>());
Ryan Harrisondbc13af2022-02-21 15:19:07 +0000570 }
571
Ben Claytonae18c412023-07-29 13:00:40 +0000572 /// Clones a Node object, using any replacements or transforms that have
dan sinclair41e4d9a2022-05-01 14:40:55 +0000573 /// been configured.
Ben Claytonae18c412023-07-29 13:00:40 +0000574 const ast::Node* CloneNode(const ast::Node* object);
Ryan Harrisondbc13af2022-02-21 15:19:07 +0000575
Ben Clayton7cdaffe2024-05-08 16:24:07 +0000576 /// Aborts with an ICE describing that the cloned object type was not as required.
577 [[noreturn]] void CheckedCastFailure(const ast::Node* got, const TypeInfo& expected);
Ryan Harrisondbc13af2022-02-21 15:19:07 +0000578
dan sinclair41e4d9a2022-05-01 14:40:55 +0000579 /// @returns the diagnostic list of #dst
580 diag::List& Diagnostics() const;
Ryan Harrisondbc13af2022-02-21 15:19:07 +0000581
Ben Claytonae18c412023-07-29 13:00:40 +0000582 /// A map of object in the source program to functions that create their replacement in #dst
583 Hashmap<const ast::Node*, std::function<const ast::Node*()>, 8> replacements_;
Ryan Harrisondbc13af2022-02-21 15:19:07 +0000584
Ben Claytonae18c412023-07-29 13:00:40 +0000585 /// A map of symbol in the source program to their cloned equivalent in #dst
dan sinclairbae54e72023-07-28 15:01:54 +0000586 Hashmap<Symbol, Symbol, 32> cloned_symbols_;
Ryan Harrisondbc13af2022-02-21 15:19:07 +0000587
Ben Claytonae18c412023-07-29 13:00:40 +0000588 /// Node transform functions registered with ReplaceAll()
589 Vector<CloneableTransform, 8> transforms_;
Ryan Harrisondbc13af2022-02-21 15:19:07 +0000590
Ben Clayton18dc3152022-08-23 18:38:35 +0000591 /// Transformations to apply to vectors
dan sinclairbae54e72023-07-28 15:01:54 +0000592 Hashmap<const void*, ListTransforms, 4> list_transforms_;
Ryan Harrisondbc13af2022-02-21 15:19:07 +0000593
dan sinclair41e4d9a2022-05-01 14:40:55 +0000594 /// Symbol transform registered with ReplaceAll()
595 SymbolTransform symbol_transform_;
Ryan Harrisondbc13af2022-02-21 15:19:07 +0000596};
597
Ben Claytonae18c412023-07-29 13:00:40 +0000598} // namespace tint::ast
Ryan Harrisondbc13af2022-02-21 15:19:07 +0000599
dan sinclaircff24fc2023-07-24 21:04:39 +0000600#endif // SRC_TINT_LANG_WGSL_AST_CLONE_CONTEXT_H_