microsoft/qdk
Publicmirrored fromhttps://github.com/microsoft/qdkAvailable
compiler/qsc_ast/src/mut_visit.rs
414lines · modecode
| 1 | // Copyright (c) Microsoft Corporation. |
| 2 | // Licensed under the MIT License. |
| 3 | |
| 4 | use 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 | }; |
| 10 | use qsc_data_structures::span::Span; |
| 11 | |
| 12 | pub 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 | |
| 100 | pub 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 | |
| 108 | pub 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 | |
| 115 | pub 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 | |
| 144 | pub 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 | |
| 150 | pub 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 | |
| 164 | pub 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 | |
| 180 | pub 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 | |
| 186 | pub 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 | |
| 192 | pub 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 | |
| 204 | pub 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 | |
| 217 | pub 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 | |
| 235 | pub 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 | |
| 240 | pub 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 | |
| 259 | pub 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 | |
| 352 | pub 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 | |
| 358 | pub 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 | |
| 373 | pub 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 | |
| 384 | pub 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 | |
| 392 | pub 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 | |
| 406 | pub fn walk_ident(vis: &mut impl MutVisitor, ident: &mut Ident) { |
| 407 | vis.visit_span(&mut ident.span); |
| 408 | } |
| 409 | |
| 410 | pub 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 | |