microsoft/qdk

Public

mirrored fromhttps://github.com/microsoft/qdkAvailable

CodeCommitsIssuesPull requestsActionsInsightsSecurity
alex/pythontelem

Branches

Tags

  • No tags available.
0Branches0Tags
Go to file
Add file
Code

Clone

HTTPS

Download ZIP

compiler/qsc_ast/src/mut_visit.rs

414lines · modecode

1// Copyright (c) Microsoft Corporation.
2// Licensed under the MIT License.
3
4use crate::ast::{
5 Attr, Block, CallableBody, CallableDecl, Expr, ExprKind, FieldAssign, FieldDef, FunctorExpr,
6 FunctorExprKind, Ident, Item, ItemKind, Namespace, Package, Pat, PatKind, Path, PathKind,
7 QubitInit, QubitInitKind, SpecBody, SpecDecl, Stmt, StmtKind, StringComponent, StructDecl,
8 TopLevelNode, Ty, TyDef, TyDefKind, TyKind,
9};
10use qsc_data_structures::span::Span;
11
12pub trait MutVisitor: Sized {
13 fn visit_package(&mut self, package: &mut Package) {
14 walk_package(self, package);
15 }
16
17 fn visit_namespace(&mut self, namespace: &mut Namespace) {
18 walk_namespace(self, namespace);
19 }
20
21 fn visit_item(&mut self, item: &mut Item) {
22 walk_item(self, item);
23 }
24
25 fn visit_attr(&mut self, attr: &mut Attr) {
26 walk_attr(self, attr);
27 }
28
29 fn visit_ty_def(&mut self, def: &mut TyDef) {
30 walk_ty_def(self, def);
31 }
32
33 fn visit_callable_decl(&mut self, decl: &mut CallableDecl) {
34 walk_callable_decl(self, decl);
35 }
36
37 fn visit_struct_decl(&mut self, decl: &mut StructDecl) {
38 walk_struct_decl(self, decl);
39 }
40
41 fn visit_field_def(&mut self, def: &mut FieldDef) {
42 walk_field_def(self, def);
43 }
44
45 fn visit_spec_decl(&mut self, decl: &mut SpecDecl) {
46 walk_spec_decl(self, decl);
47 }
48
49 fn visit_functor_expr(&mut self, expr: &mut FunctorExpr) {
50 walk_functor_expr(self, expr);
51 }
52
53 fn visit_ty(&mut self, ty: &mut Ty) {
54 walk_ty(self, ty);
55 }
56
57 fn visit_block(&mut self, block: &mut Block) {
58 walk_block(self, block);
59 }
60
61 fn visit_stmt(&mut self, stmt: &mut Stmt) {
62 walk_stmt(self, stmt);
63 }
64
65 fn visit_expr(&mut self, expr: &mut Expr) {
66 walk_expr(self, expr);
67 }
68
69 fn visit_field_assign(&mut self, assign: &mut FieldAssign) {
70 walk_field_assign(self, assign);
71 }
72
73 fn visit_pat(&mut self, pat: &mut Pat) {
74 walk_pat(self, pat);
75 }
76
77 fn visit_qubit_init(&mut self, init: &mut QubitInit) {
78 walk_qubit_init(self, init);
79 }
80
81 fn visit_path(&mut self, path: &mut Path) {
82 walk_path(self, path);
83 }
84
85 fn visit_path_kind(&mut self, path: &mut PathKind) {
86 walk_path_kind(self, path);
87 }
88
89 fn visit_ident(&mut self, ident: &mut Ident) {
90 walk_ident(self, ident);
91 }
92
93 fn visit_idents(&mut self, ident: &mut [Ident]) {
94 walk_idents(self, ident);
95 }
96
97 fn visit_span(&mut self, _: &mut Span) {}
98}
99
100pub fn walk_package(vis: &mut impl MutVisitor, package: &mut Package) {
101 package.nodes.iter_mut().for_each(|n| match n {
102 TopLevelNode::Namespace(ns) => vis.visit_namespace(ns),
103 TopLevelNode::Stmt(stmt) => vis.visit_stmt(stmt),
104 });
105 package.entry.iter_mut().for_each(|e| vis.visit_expr(e));
106}
107
108pub fn walk_namespace(vis: &mut impl MutVisitor, namespace: &mut Namespace) {
109 vis.visit_span(&mut namespace.span);
110 vis.visit_idents(&mut namespace.name);
111
112 namespace.items.iter_mut().for_each(|i| vis.visit_item(i));
113}
114
115pub fn walk_item(vis: &mut impl MutVisitor, item: &mut Item) {
116 vis.visit_span(&mut item.span);
117 item.attrs.iter_mut().for_each(|a| vis.visit_attr(a));
118
119 match &mut *item.kind {
120 ItemKind::Callable(decl) => vis.visit_callable_decl(decl),
121 ItemKind::Err => {}
122 ItemKind::Open(ns, alias) => {
123 vis.visit_path_kind(ns);
124 alias.iter_mut().for_each(|a| vis.visit_ident(a));
125 }
126 ItemKind::Ty(ident, def) => {
127 vis.visit_ident(ident);
128 vis.visit_ty_def(def);
129 }
130 ItemKind::Struct(decl) => vis.visit_struct_decl(decl),
131 ItemKind::ImportOrExport(export) => {
132 vis.visit_span(&mut export.span);
133 for item in &mut *export.items {
134 vis.visit_span(&mut item.span);
135 vis.visit_path_kind(&mut item.path);
136 if let Some(ref mut alias) = item.alias {
137 vis.visit_ident(alias);
138 }
139 }
140 }
141 }
142}
143
144pub fn walk_attr(vis: &mut impl MutVisitor, attr: &mut Attr) {
145 vis.visit_span(&mut attr.span);
146 vis.visit_ident(&mut attr.name);
147 vis.visit_expr(&mut attr.arg);
148}
149
150pub fn walk_ty_def(vis: &mut impl MutVisitor, def: &mut TyDef) {
151 vis.visit_span(&mut def.span);
152
153 match &mut *def.kind {
154 TyDefKind::Field(name, ty) => {
155 name.iter_mut().for_each(|n| vis.visit_ident(n));
156 vis.visit_ty(ty);
157 }
158 TyDefKind::Paren(def) => vis.visit_ty_def(def),
159 TyDefKind::Tuple(defs) => defs.iter_mut().for_each(|d| vis.visit_ty_def(d)),
160 TyDefKind::Err => {}
161 }
162}
163
164pub fn walk_callable_decl(vis: &mut impl MutVisitor, decl: &mut CallableDecl) {
165 vis.visit_span(&mut decl.span);
166 vis.visit_ident(&mut decl.name);
167 decl.generics.iter_mut().for_each(|p| vis.visit_ident(p));
168 vis.visit_pat(&mut decl.input);
169 vis.visit_ty(&mut decl.output);
170 decl.functors
171 .iter_mut()
172 .for_each(|f| vis.visit_functor_expr(f));
173
174 match &mut *decl.body {
175 CallableBody::Block(block) => vis.visit_block(block),
176 CallableBody::Specs(specs) => specs.iter_mut().for_each(|s| vis.visit_spec_decl(s)),
177 }
178}
179
180pub fn walk_struct_decl(vis: &mut impl MutVisitor, decl: &mut StructDecl) {
181 vis.visit_span(&mut decl.span);
182 vis.visit_ident(&mut decl.name);
183 decl.fields.iter_mut().for_each(|f| vis.visit_field_def(f));
184}
185
186pub fn walk_field_def(vis: &mut impl MutVisitor, def: &mut FieldDef) {
187 vis.visit_span(&mut def.span);
188 vis.visit_ident(&mut def.name);
189 vis.visit_ty(&mut def.ty);
190}
191
192pub fn walk_spec_decl(vis: &mut impl MutVisitor, decl: &mut SpecDecl) {
193 vis.visit_span(&mut decl.span);
194
195 match &mut decl.body {
196 SpecBody::Gen(_) => {}
197 SpecBody::Impl(pat, block) => {
198 vis.visit_pat(pat);
199 vis.visit_block(block);
200 }
201 }
202}
203
204pub fn walk_functor_expr(vis: &mut impl MutVisitor, expr: &mut FunctorExpr) {
205 vis.visit_span(&mut expr.span);
206
207 match &mut *expr.kind {
208 FunctorExprKind::BinOp(_, lhs, rhs) => {
209 vis.visit_functor_expr(lhs);
210 vis.visit_functor_expr(rhs);
211 }
212 FunctorExprKind::Lit(_) => {}
213 FunctorExprKind::Paren(expr) => vis.visit_functor_expr(expr),
214 }
215}
216
217pub fn walk_ty(vis: &mut impl MutVisitor, ty: &mut Ty) {
218 vis.visit_span(&mut ty.span);
219
220 match &mut *ty.kind {
221 TyKind::Array(item) => vis.visit_ty(item),
222 TyKind::Arrow(_, lhs, rhs, functors) => {
223 vis.visit_ty(lhs);
224 vis.visit_ty(rhs);
225 functors.iter_mut().for_each(|f| vis.visit_functor_expr(f));
226 }
227 TyKind::Hole | TyKind::Err => {}
228 TyKind::Paren(ty) => vis.visit_ty(ty),
229 TyKind::Param(name) => vis.visit_ident(name),
230 TyKind::Path(path) => vis.visit_path_kind(path),
231 TyKind::Tuple(tys) => tys.iter_mut().for_each(|t| vis.visit_ty(t)),
232 }
233}
234
235pub fn walk_block(vis: &mut impl MutVisitor, block: &mut Block) {
236 vis.visit_span(&mut block.span);
237 block.stmts.iter_mut().for_each(|s| vis.visit_stmt(s));
238}
239
240pub fn walk_stmt(vis: &mut impl MutVisitor, stmt: &mut Stmt) {
241 vis.visit_span(&mut stmt.span);
242
243 match &mut *stmt.kind {
244 StmtKind::Empty | StmtKind::Err => {}
245 StmtKind::Expr(expr) | StmtKind::Semi(expr) => vis.visit_expr(expr),
246 StmtKind::Item(item) => vis.visit_item(item),
247 StmtKind::Local(_, pat, value) => {
248 vis.visit_pat(pat);
249 vis.visit_expr(value);
250 }
251 StmtKind::Qubit(_, pat, init, block) => {
252 vis.visit_pat(pat);
253 vis.visit_qubit_init(init);
254 block.iter_mut().for_each(|b| vis.visit_block(b));
255 }
256 }
257}
258
259pub fn walk_expr(vis: &mut impl MutVisitor, expr: &mut Expr) {
260 vis.visit_span(&mut expr.span);
261
262 match &mut *expr.kind {
263 ExprKind::Array(exprs) => exprs.iter_mut().for_each(|e| vis.visit_expr(e)),
264 ExprKind::ArrayRepeat(item, size) => {
265 vis.visit_expr(item);
266 vis.visit_expr(size);
267 }
268 ExprKind::Assign(lhs, rhs)
269 | ExprKind::AssignOp(_, lhs, rhs)
270 | ExprKind::BinOp(_, lhs, rhs) => {
271 vis.visit_expr(lhs);
272 vis.visit_expr(rhs);
273 }
274 ExprKind::AssignUpdate(record, index, value) => {
275 vis.visit_expr(record);
276 vis.visit_expr(index);
277 vis.visit_expr(value);
278 }
279 ExprKind::Block(block) => vis.visit_block(block),
280 ExprKind::Call(callee, arg) => {
281 vis.visit_expr(callee);
282 vis.visit_expr(arg);
283 }
284 ExprKind::Conjugate(within, apply) => {
285 vis.visit_block(within);
286 vis.visit_block(apply);
287 }
288 ExprKind::Fail(msg) => vis.visit_expr(msg),
289 ExprKind::Field(record, name) => {
290 vis.visit_expr(record);
291 vis.visit_ident(name);
292 }
293 ExprKind::For(pat, iter, block) => {
294 vis.visit_pat(pat);
295 vis.visit_expr(iter);
296 vis.visit_block(block);
297 }
298 ExprKind::If(cond, body, otherwise) => {
299 vis.visit_expr(cond);
300 vis.visit_block(body);
301 otherwise.iter_mut().for_each(|e| vis.visit_expr(e));
302 }
303 ExprKind::Index(array, index) => {
304 vis.visit_expr(array);
305 vis.visit_expr(index);
306 }
307 ExprKind::Interpolate(components) => {
308 for component in components.iter_mut() {
309 match component {
310 StringComponent::Expr(expr) => vis.visit_expr(expr.as_mut()),
311 StringComponent::Lit(_) => {}
312 }
313 }
314 }
315 ExprKind::Lambda(_, pat, expr) => {
316 vis.visit_pat(pat);
317 vis.visit_expr(expr);
318 }
319 ExprKind::Paren(expr) | ExprKind::Return(expr) | ExprKind::UnOp(_, expr) => {
320 vis.visit_expr(expr);
321 }
322 ExprKind::Path(path) => vis.visit_path_kind(path),
323 ExprKind::Range(start, step, end) => {
324 start.iter_mut().for_each(|s| vis.visit_expr(s));
325 step.iter_mut().for_each(|s| vis.visit_expr(s));
326 end.iter_mut().for_each(|e| vis.visit_expr(e));
327 }
328 ExprKind::Repeat(body, until, fixup) => {
329 vis.visit_block(body);
330 vis.visit_expr(until);
331 fixup.iter_mut().for_each(|f| vis.visit_block(f));
332 }
333 ExprKind::Struct(name, copy, fields) => {
334 vis.visit_path_kind(name);
335 copy.iter_mut().for_each(|c| vis.visit_expr(c));
336 fields.iter_mut().for_each(|f| vis.visit_field_assign(f));
337 }
338 ExprKind::TernOp(_, e1, e2, e3) => {
339 vis.visit_expr(e1);
340 vis.visit_expr(e2);
341 vis.visit_expr(e3);
342 }
343 ExprKind::Tuple(exprs) => exprs.iter_mut().for_each(|e| vis.visit_expr(e)),
344 ExprKind::While(cond, block) => {
345 vis.visit_expr(cond);
346 vis.visit_block(block);
347 }
348 ExprKind::Err | ExprKind::Hole | ExprKind::Lit(_) => {}
349 }
350}
351
352pub fn walk_field_assign(vis: &mut impl MutVisitor, assign: &mut FieldAssign) {
353 vis.visit_span(&mut assign.span);
354 vis.visit_ident(&mut assign.field);
355 vis.visit_expr(&mut assign.value);
356}
357
358pub fn walk_pat(vis: &mut impl MutVisitor, pat: &mut Pat) {
359 vis.visit_span(&mut pat.span);
360
361 match &mut *pat.kind {
362 PatKind::Bind(name, ty) => {
363 vis.visit_ident(name);
364 ty.iter_mut().for_each(|t| vis.visit_ty(t));
365 }
366 PatKind::Discard(ty) => ty.iter_mut().for_each(|t| vis.visit_ty(t)),
367 PatKind::Elided | PatKind::Err => {}
368 PatKind::Paren(pat) => vis.visit_pat(pat),
369 PatKind::Tuple(pats) => pats.iter_mut().for_each(|p| vis.visit_pat(p)),
370 }
371}
372
373pub fn walk_qubit_init(vis: &mut impl MutVisitor, init: &mut QubitInit) {
374 vis.visit_span(&mut init.span);
375
376 match &mut *init.kind {
377 QubitInitKind::Array(len) => vis.visit_expr(len),
378 QubitInitKind::Paren(init) => vis.visit_qubit_init(init),
379 QubitInitKind::Single | QubitInitKind::Err => {}
380 QubitInitKind::Tuple(inits) => inits.iter_mut().for_each(|i| vis.visit_qubit_init(i)),
381 }
382}
383
384pub fn walk_path(vis: &mut impl MutVisitor, path: &mut Path) {
385 vis.visit_span(&mut path.span);
386 if let Some(ref mut parts) = path.segments {
387 vis.visit_idents(parts);
388 }
389 vis.visit_ident(&mut path.name);
390}
391
392pub fn walk_path_kind(vis: &mut impl MutVisitor, path: &mut PathKind) {
393 match path {
394 PathKind::Ok(path) => vis.visit_path(path),
395 PathKind::Err(Some(incomplete_path)) => {
396 vis.visit_span(&mut incomplete_path.span);
397
398 for ref mut ident in &mut incomplete_path.segments {
399 vis.visit_ident(ident);
400 }
401 }
402 PathKind::Err(None) => {}
403 }
404}
405
406pub fn walk_ident(vis: &mut impl MutVisitor, ident: &mut Ident) {
407 vis.visit_span(&mut ident.span);
408}
409
410pub fn walk_idents(vis: &mut impl MutVisitor, idents: &mut [Ident]) {
411 for ref mut ident in idents {
412 vis.visit_ident(ident);
413 }
414}
415