2022-08-15 13:51:45 +02:00
|
|
|
|
//! A pretty-printer for HIR.
|
|
|
|
|
|
|
|
|
|
use std::fmt::{self, Write};
|
|
|
|
|
|
2022-09-12 21:02:30 +01:00
|
|
|
|
use syntax::ast::HasName;
|
|
|
|
|
|
2022-08-15 13:51:45 +02:00
|
|
|
|
use crate::{
|
2023-04-06 16:14:38 +03:30
|
|
|
|
hir::{
|
|
|
|
|
Array, BindingAnnotation, BindingId, CaptureBy, ClosureKind, Literal, Movability, Statement,
|
|
|
|
|
},
|
2022-08-15 13:51:45 +02:00
|
|
|
|
pretty::{print_generic_args, print_path, print_type_ref},
|
|
|
|
|
type_ref::TypeRef,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
|
|
pub(super) fn print_body_hir(db: &dyn DefDatabase, body: &Body, owner: DefWithBodyId) -> String {
|
|
|
|
|
let header = match owner {
|
|
|
|
|
DefWithBodyId::FunctionId(it) => {
|
|
|
|
|
let item_tree_id = it.lookup(db).id;
|
2023-04-07 09:34:04 +02:00
|
|
|
|
format!("fn {}", item_tree_id.item_tree(db)[item_tree_id.value].name)
|
2022-08-15 13:51:45 +02:00
|
|
|
|
}
|
|
|
|
|
DefWithBodyId::StaticId(it) => {
|
|
|
|
|
let item_tree_id = it.lookup(db).id;
|
|
|
|
|
format!("static {} = ", item_tree_id.item_tree(db)[item_tree_id.value].name)
|
|
|
|
|
}
|
|
|
|
|
DefWithBodyId::ConstId(it) => {
|
|
|
|
|
let item_tree_id = it.lookup(db).id;
|
|
|
|
|
let name = match &item_tree_id.item_tree(db)[item_tree_id.value].name {
|
|
|
|
|
Some(name) => name.to_string(),
|
|
|
|
|
None => "_".to_string(),
|
|
|
|
|
};
|
2022-12-23 13:42:58 -05:00
|
|
|
|
format!("const {name} = ")
|
2022-09-12 21:02:30 +01:00
|
|
|
|
}
|
|
|
|
|
DefWithBodyId::VariantId(it) => {
|
|
|
|
|
let src = it.parent.child_source(db);
|
|
|
|
|
let variant = &src.value[it.local_id];
|
|
|
|
|
let name = match &variant.name() {
|
|
|
|
|
Some(name) => name.to_string(),
|
|
|
|
|
None => "_".to_string(),
|
|
|
|
|
};
|
2022-12-23 13:42:58 -05:00
|
|
|
|
format!("{name}")
|
2022-08-15 13:51:45 +02:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let mut p = Printer { body, buf: header, indent_level: 0, needs_indent: false };
|
2023-04-07 09:34:04 +02:00
|
|
|
|
if let DefWithBodyId::FunctionId(it) = owner {
|
|
|
|
|
p.buf.push('(');
|
|
|
|
|
body.params.iter().zip(&db.function_data(it).params).for_each(|(¶m, ty)| {
|
|
|
|
|
p.print_pat(param);
|
|
|
|
|
p.buf.push(':');
|
|
|
|
|
p.print_type_ref(ty);
|
|
|
|
|
});
|
|
|
|
|
p.buf.push(')');
|
|
|
|
|
p.buf.push(' ');
|
|
|
|
|
}
|
2022-08-15 13:51:45 +02:00
|
|
|
|
p.print_expr(body.body_expr);
|
2023-04-07 09:34:04 +02:00
|
|
|
|
if matches!(owner, DefWithBodyId::StaticId(_) | DefWithBodyId::ConstId(_)) {
|
2022-08-15 13:51:45 +02:00
|
|
|
|
p.buf.push(';');
|
|
|
|
|
}
|
|
|
|
|
p.buf
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
macro_rules! w {
|
|
|
|
|
($dst:expr, $($arg:tt)*) => {
|
|
|
|
|
{ let _ = write!($dst, $($arg)*); }
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
macro_rules! wln {
|
|
|
|
|
($dst:expr) => {
|
|
|
|
|
{ let _ = writeln!($dst); }
|
|
|
|
|
};
|
|
|
|
|
($dst:expr, $($arg:tt)*) => {
|
|
|
|
|
{ let _ = writeln!($dst, $($arg)*); }
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct Printer<'a> {
|
|
|
|
|
body: &'a Body,
|
|
|
|
|
buf: String,
|
|
|
|
|
indent_level: usize,
|
|
|
|
|
needs_indent: bool,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl<'a> Write for Printer<'a> {
|
|
|
|
|
fn write_str(&mut self, s: &str) -> fmt::Result {
|
|
|
|
|
for line in s.split_inclusive('\n') {
|
|
|
|
|
if self.needs_indent {
|
2023-01-10 18:48:51 +00:00
|
|
|
|
match self.buf.chars().rev().find(|ch| *ch != ' ') {
|
2022-08-15 13:51:45 +02:00
|
|
|
|
Some('\n') | None => {}
|
|
|
|
|
_ => self.buf.push('\n'),
|
|
|
|
|
}
|
|
|
|
|
self.buf.push_str(&" ".repeat(self.indent_level));
|
|
|
|
|
self.needs_indent = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
self.buf.push_str(line);
|
|
|
|
|
self.needs_indent = line.ends_with('\n');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl<'a> Printer<'a> {
|
|
|
|
|
fn indented(&mut self, f: impl FnOnce(&mut Self)) {
|
|
|
|
|
self.indent_level += 1;
|
|
|
|
|
wln!(self);
|
|
|
|
|
f(self);
|
|
|
|
|
self.indent_level -= 1;
|
|
|
|
|
self.buf = self.buf.trim_end_matches('\n').to_string();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn whitespace(&mut self) {
|
|
|
|
|
match self.buf.chars().next_back() {
|
|
|
|
|
None | Some('\n' | ' ') => {}
|
|
|
|
|
_ => self.buf.push(' '),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn newline(&mut self) {
|
2023-01-10 19:51:21 +00:00
|
|
|
|
match self.buf.chars().rev().find(|ch| *ch != ' ') {
|
2022-08-15 13:51:45 +02:00
|
|
|
|
Some('\n') | None => {}
|
|
|
|
|
_ => writeln!(self).unwrap(),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn print_expr(&mut self, expr: ExprId) {
|
|
|
|
|
let expr = &self.body[expr];
|
|
|
|
|
|
|
|
|
|
match expr {
|
|
|
|
|
Expr::Missing => w!(self, "<EFBFBD>"),
|
|
|
|
|
Expr::Underscore => w!(self, "_"),
|
|
|
|
|
Expr::Path(path) => self.print_path(path),
|
|
|
|
|
Expr::If { condition, then_branch, else_branch } => {
|
|
|
|
|
w!(self, "if ");
|
|
|
|
|
self.print_expr(*condition);
|
|
|
|
|
w!(self, " ");
|
|
|
|
|
self.print_expr(*then_branch);
|
|
|
|
|
if let Some(els) = *else_branch {
|
|
|
|
|
w!(self, " else ");
|
|
|
|
|
self.print_expr(els);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Expr::Let { pat, expr } => {
|
|
|
|
|
w!(self, "let ");
|
|
|
|
|
self.print_pat(*pat);
|
|
|
|
|
w!(self, " = ");
|
|
|
|
|
self.print_expr(*expr);
|
|
|
|
|
}
|
|
|
|
|
Expr::Loop { body, label } => {
|
|
|
|
|
if let Some(lbl) = label {
|
|
|
|
|
w!(self, "{}: ", self.body[*lbl].name);
|
|
|
|
|
}
|
|
|
|
|
w!(self, "loop ");
|
|
|
|
|
self.print_expr(*body);
|
|
|
|
|
}
|
|
|
|
|
Expr::While { condition, body, label } => {
|
|
|
|
|
if let Some(lbl) = label {
|
|
|
|
|
w!(self, "{}: ", self.body[*lbl].name);
|
|
|
|
|
}
|
|
|
|
|
w!(self, "while ");
|
|
|
|
|
self.print_expr(*condition);
|
|
|
|
|
self.print_expr(*body);
|
|
|
|
|
}
|
|
|
|
|
Expr::For { iterable, pat, body, label } => {
|
|
|
|
|
if let Some(lbl) = label {
|
|
|
|
|
w!(self, "{}: ", self.body[*lbl].name);
|
|
|
|
|
}
|
|
|
|
|
w!(self, "for ");
|
|
|
|
|
self.print_pat(*pat);
|
|
|
|
|
w!(self, " in ");
|
|
|
|
|
self.print_expr(*iterable);
|
|
|
|
|
self.print_expr(*body);
|
|
|
|
|
}
|
|
|
|
|
Expr::Call { callee, args, is_assignee_expr: _ } => {
|
|
|
|
|
self.print_expr(*callee);
|
|
|
|
|
w!(self, "(");
|
|
|
|
|
if !args.is_empty() {
|
|
|
|
|
self.indented(|p| {
|
|
|
|
|
for arg in &**args {
|
|
|
|
|
p.print_expr(*arg);
|
|
|
|
|
wln!(p, ",");
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
w!(self, ")");
|
|
|
|
|
}
|
|
|
|
|
Expr::MethodCall { receiver, method_name, args, generic_args } => {
|
|
|
|
|
self.print_expr(*receiver);
|
|
|
|
|
w!(self, ".{}", method_name);
|
|
|
|
|
if let Some(args) = generic_args {
|
|
|
|
|
w!(self, "::<");
|
|
|
|
|
print_generic_args(args, self).unwrap();
|
|
|
|
|
w!(self, ">");
|
|
|
|
|
}
|
|
|
|
|
w!(self, "(");
|
|
|
|
|
if !args.is_empty() {
|
|
|
|
|
self.indented(|p| {
|
|
|
|
|
for arg in &**args {
|
|
|
|
|
p.print_expr(*arg);
|
|
|
|
|
wln!(p, ",");
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
w!(self, ")");
|
|
|
|
|
}
|
|
|
|
|
Expr::Match { expr, arms } => {
|
|
|
|
|
w!(self, "match ");
|
|
|
|
|
self.print_expr(*expr);
|
|
|
|
|
w!(self, " {{");
|
|
|
|
|
self.indented(|p| {
|
|
|
|
|
for arm in &**arms {
|
|
|
|
|
p.print_pat(arm.pat);
|
|
|
|
|
if let Some(guard) = arm.guard {
|
|
|
|
|
w!(p, " if ");
|
|
|
|
|
p.print_expr(guard);
|
|
|
|
|
}
|
|
|
|
|
w!(p, " => ");
|
|
|
|
|
p.print_expr(arm.expr);
|
|
|
|
|
wln!(p, ",");
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
wln!(self, "}}");
|
|
|
|
|
}
|
|
|
|
|
Expr::Continue { label } => {
|
|
|
|
|
w!(self, "continue");
|
2023-04-06 12:50:16 +02:00
|
|
|
|
if let Some(lbl) = label {
|
|
|
|
|
w!(self, " {}", self.body[*lbl].name);
|
2022-08-15 13:51:45 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Expr::Break { expr, label } => {
|
|
|
|
|
w!(self, "break");
|
2023-04-06 12:50:16 +02:00
|
|
|
|
if let Some(lbl) = label {
|
|
|
|
|
w!(self, " {}", self.body[*lbl].name);
|
2022-08-15 13:51:45 +02:00
|
|
|
|
}
|
|
|
|
|
if let Some(expr) = expr {
|
|
|
|
|
self.whitespace();
|
|
|
|
|
self.print_expr(*expr);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Expr::Return { expr } => {
|
|
|
|
|
w!(self, "return");
|
|
|
|
|
if let Some(expr) = expr {
|
|
|
|
|
self.whitespace();
|
|
|
|
|
self.print_expr(*expr);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Expr::Yield { expr } => {
|
|
|
|
|
w!(self, "yield");
|
|
|
|
|
if let Some(expr) = expr {
|
|
|
|
|
self.whitespace();
|
|
|
|
|
self.print_expr(*expr);
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-12-28 23:17:13 +00:00
|
|
|
|
Expr::Yeet { expr } => {
|
|
|
|
|
w!(self, "do");
|
|
|
|
|
self.whitespace();
|
|
|
|
|
w!(self, "yeet");
|
|
|
|
|
if let Some(expr) = expr {
|
|
|
|
|
self.whitespace();
|
|
|
|
|
self.print_expr(*expr);
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-08-15 13:51:45 +02:00
|
|
|
|
Expr::RecordLit { path, fields, spread, ellipsis, is_assignee_expr: _ } => {
|
|
|
|
|
match path {
|
|
|
|
|
Some(path) => self.print_path(path),
|
|
|
|
|
None => w!(self, "<EFBFBD>"),
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
w!(self, "{{");
|
|
|
|
|
self.indented(|p| {
|
|
|
|
|
for field in &**fields {
|
|
|
|
|
w!(p, "{}: ", field.name);
|
|
|
|
|
p.print_expr(field.expr);
|
|
|
|
|
wln!(p, ",");
|
|
|
|
|
}
|
|
|
|
|
if let Some(spread) = spread {
|
|
|
|
|
w!(p, "..");
|
|
|
|
|
p.print_expr(*spread);
|
|
|
|
|
wln!(p);
|
|
|
|
|
}
|
|
|
|
|
if *ellipsis {
|
|
|
|
|
wln!(p, "..");
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
w!(self, "}}");
|
|
|
|
|
}
|
|
|
|
|
Expr::Field { expr, name } => {
|
|
|
|
|
self.print_expr(*expr);
|
|
|
|
|
w!(self, ".{}", name);
|
|
|
|
|
}
|
|
|
|
|
Expr::Await { expr } => {
|
|
|
|
|
self.print_expr(*expr);
|
|
|
|
|
w!(self, ".await");
|
|
|
|
|
}
|
|
|
|
|
Expr::Cast { expr, type_ref } => {
|
|
|
|
|
self.print_expr(*expr);
|
|
|
|
|
w!(self, " as ");
|
|
|
|
|
self.print_type_ref(type_ref);
|
|
|
|
|
}
|
|
|
|
|
Expr::Ref { expr, rawness, mutability } => {
|
|
|
|
|
w!(self, "&");
|
|
|
|
|
if rawness.is_raw() {
|
|
|
|
|
w!(self, "raw ");
|
|
|
|
|
}
|
|
|
|
|
if mutability.is_mut() {
|
|
|
|
|
w!(self, "mut ");
|
|
|
|
|
}
|
|
|
|
|
self.print_expr(*expr);
|
|
|
|
|
}
|
|
|
|
|
Expr::Box { expr } => {
|
|
|
|
|
w!(self, "box ");
|
|
|
|
|
self.print_expr(*expr);
|
|
|
|
|
}
|
|
|
|
|
Expr::UnaryOp { expr, op } => {
|
|
|
|
|
let op = match op {
|
|
|
|
|
ast::UnaryOp::Deref => "*",
|
|
|
|
|
ast::UnaryOp::Not => "!",
|
|
|
|
|
ast::UnaryOp::Neg => "-",
|
|
|
|
|
};
|
|
|
|
|
w!(self, "{}", op);
|
|
|
|
|
self.print_expr(*expr);
|
|
|
|
|
}
|
|
|
|
|
Expr::BinaryOp { lhs, rhs, op } => {
|
|
|
|
|
let (bra, ket) = match op {
|
|
|
|
|
None | Some(ast::BinaryOp::Assignment { .. }) => ("", ""),
|
|
|
|
|
_ => ("(", ")"),
|
|
|
|
|
};
|
|
|
|
|
w!(self, "{}", bra);
|
|
|
|
|
self.print_expr(*lhs);
|
|
|
|
|
w!(self, "{} ", ket);
|
|
|
|
|
match op {
|
|
|
|
|
Some(op) => w!(self, "{}", op),
|
|
|
|
|
None => w!(self, "<EFBFBD>"), // :)
|
|
|
|
|
}
|
|
|
|
|
w!(self, " {}", bra);
|
|
|
|
|
self.print_expr(*rhs);
|
|
|
|
|
w!(self, "{}", ket);
|
|
|
|
|
}
|
|
|
|
|
Expr::Range { lhs, rhs, range_type } => {
|
|
|
|
|
if let Some(lhs) = lhs {
|
|
|
|
|
w!(self, "(");
|
|
|
|
|
self.print_expr(*lhs);
|
|
|
|
|
w!(self, ") ");
|
|
|
|
|
}
|
|
|
|
|
let range = match range_type {
|
|
|
|
|
ast::RangeOp::Exclusive => "..",
|
|
|
|
|
ast::RangeOp::Inclusive => "..=",
|
|
|
|
|
};
|
|
|
|
|
w!(self, "{}", range);
|
|
|
|
|
if let Some(rhs) = rhs {
|
|
|
|
|
w!(self, "(");
|
|
|
|
|
self.print_expr(*rhs);
|
|
|
|
|
w!(self, ") ");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Expr::Index { base, index } => {
|
|
|
|
|
self.print_expr(*base);
|
|
|
|
|
w!(self, "[");
|
|
|
|
|
self.print_expr(*index);
|
|
|
|
|
w!(self, "]");
|
|
|
|
|
}
|
2023-04-06 16:14:38 +03:30
|
|
|
|
Expr::Closure { args, arg_types, ret_type, body, closure_kind, capture_by } => {
|
2023-03-15 13:49:57 +01:00
|
|
|
|
match closure_kind {
|
|
|
|
|
ClosureKind::Generator(Movability::Static) => {
|
|
|
|
|
w!(self, "static ");
|
|
|
|
|
}
|
|
|
|
|
ClosureKind::Async => {
|
|
|
|
|
w!(self, "async ");
|
|
|
|
|
}
|
|
|
|
|
_ => (),
|
2022-09-05 22:43:26 +09:00
|
|
|
|
}
|
2023-04-06 16:14:38 +03:30
|
|
|
|
match capture_by {
|
|
|
|
|
CaptureBy::Value => {
|
|
|
|
|
w!(self, "move ");
|
|
|
|
|
}
|
|
|
|
|
CaptureBy::Ref => (),
|
|
|
|
|
}
|
2022-08-15 13:51:45 +02:00
|
|
|
|
w!(self, "|");
|
|
|
|
|
for (i, (pat, ty)) in args.iter().zip(arg_types.iter()).enumerate() {
|
|
|
|
|
if i != 0 {
|
|
|
|
|
w!(self, ", ");
|
|
|
|
|
}
|
|
|
|
|
self.print_pat(*pat);
|
|
|
|
|
if let Some(ty) = ty {
|
|
|
|
|
w!(self, ": ");
|
|
|
|
|
self.print_type_ref(ty);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
w!(self, "|");
|
2023-03-15 13:49:57 +01:00
|
|
|
|
if let Some(ret_ty) = ret_type {
|
|
|
|
|
w!(self, " -> ");
|
|
|
|
|
self.print_type_ref(ret_ty);
|
2022-08-15 13:51:45 +02:00
|
|
|
|
}
|
|
|
|
|
self.whitespace();
|
|
|
|
|
self.print_expr(*body);
|
|
|
|
|
}
|
|
|
|
|
Expr::Tuple { exprs, is_assignee_expr: _ } => {
|
|
|
|
|
w!(self, "(");
|
|
|
|
|
for expr in exprs.iter() {
|
|
|
|
|
self.print_expr(*expr);
|
|
|
|
|
w!(self, ", ");
|
|
|
|
|
}
|
|
|
|
|
w!(self, ")");
|
|
|
|
|
}
|
|
|
|
|
Expr::Array(arr) => {
|
|
|
|
|
w!(self, "[");
|
|
|
|
|
if !matches!(arr, Array::ElementList { elements, .. } if elements.is_empty()) {
|
|
|
|
|
self.indented(|p| match arr {
|
|
|
|
|
Array::ElementList { elements, is_assignee_expr: _ } => {
|
|
|
|
|
for elem in elements.iter() {
|
|
|
|
|
p.print_expr(*elem);
|
|
|
|
|
w!(p, ", ");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Array::Repeat { initializer, repeat } => {
|
|
|
|
|
p.print_expr(*initializer);
|
|
|
|
|
w!(p, "; ");
|
|
|
|
|
p.print_expr(*repeat);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
self.newline();
|
|
|
|
|
}
|
|
|
|
|
w!(self, "]");
|
|
|
|
|
}
|
|
|
|
|
Expr::Literal(lit) => self.print_literal(lit),
|
|
|
|
|
Expr::Block { id: _, statements, tail, label } => {
|
2023-03-04 14:45:57 +01:00
|
|
|
|
let label = label.map(|lbl| format!("{}: ", self.body[lbl].name));
|
|
|
|
|
self.print_block(label.as_deref(), statements, tail);
|
|
|
|
|
}
|
|
|
|
|
Expr::Unsafe { id: _, statements, tail } => {
|
|
|
|
|
self.print_block(Some("unsafe "), statements, tail);
|
|
|
|
|
}
|
|
|
|
|
Expr::Async { id: _, statements, tail } => {
|
|
|
|
|
self.print_block(Some("async "), statements, tail);
|
|
|
|
|
}
|
2023-05-12 18:17:15 +03:30
|
|
|
|
Expr::Const(id) => {
|
|
|
|
|
w!(self, "const {{ /* {id:?} */ }}");
|
2023-03-04 14:45:57 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn print_block(
|
|
|
|
|
&mut self,
|
|
|
|
|
label: Option<&str>,
|
|
|
|
|
statements: &Box<[Statement]>,
|
|
|
|
|
tail: &Option<la_arena::Idx<Expr>>,
|
|
|
|
|
) {
|
|
|
|
|
self.whitespace();
|
|
|
|
|
if let Some(lbl) = label {
|
|
|
|
|
w!(self, "{}", lbl);
|
|
|
|
|
}
|
|
|
|
|
w!(self, "{{");
|
|
|
|
|
if !statements.is_empty() || tail.is_some() {
|
|
|
|
|
self.indented(|p| {
|
|
|
|
|
for stmt in &**statements {
|
|
|
|
|
p.print_stmt(stmt);
|
2022-08-15 13:51:45 +02:00
|
|
|
|
}
|
2023-03-04 14:45:57 +01:00
|
|
|
|
if let Some(tail) = tail {
|
|
|
|
|
p.print_expr(*tail);
|
2022-08-15 13:51:45 +02:00
|
|
|
|
}
|
2023-03-04 14:45:57 +01:00
|
|
|
|
p.newline();
|
|
|
|
|
});
|
2022-08-15 13:51:45 +02:00
|
|
|
|
}
|
2023-03-04 14:45:57 +01:00
|
|
|
|
w!(self, "}}");
|
2022-08-15 13:51:45 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn print_pat(&mut self, pat: PatId) {
|
|
|
|
|
let pat = &self.body[pat];
|
|
|
|
|
|
|
|
|
|
match pat {
|
|
|
|
|
Pat::Missing => w!(self, "<EFBFBD>"),
|
|
|
|
|
Pat::Wild => w!(self, "_"),
|
|
|
|
|
Pat::Tuple { args, ellipsis } => {
|
|
|
|
|
w!(self, "(");
|
|
|
|
|
for (i, pat) in args.iter().enumerate() {
|
|
|
|
|
if i != 0 {
|
|
|
|
|
w!(self, ", ");
|
|
|
|
|
}
|
|
|
|
|
if *ellipsis == Some(i) {
|
|
|
|
|
w!(self, ".., ");
|
|
|
|
|
}
|
|
|
|
|
self.print_pat(*pat);
|
|
|
|
|
}
|
|
|
|
|
w!(self, ")");
|
|
|
|
|
}
|
|
|
|
|
Pat::Or(pats) => {
|
|
|
|
|
for (i, pat) in pats.iter().enumerate() {
|
|
|
|
|
if i != 0 {
|
|
|
|
|
w!(self, " | ");
|
|
|
|
|
}
|
|
|
|
|
self.print_pat(*pat);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Pat::Record { path, args, ellipsis } => {
|
|
|
|
|
match path {
|
|
|
|
|
Some(path) => self.print_path(path),
|
|
|
|
|
None => w!(self, "<EFBFBD>"),
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
w!(self, " {{");
|
|
|
|
|
self.indented(|p| {
|
|
|
|
|
for arg in args.iter() {
|
|
|
|
|
w!(p, "{}: ", arg.name);
|
|
|
|
|
p.print_pat(arg.pat);
|
|
|
|
|
wln!(p, ",");
|
|
|
|
|
}
|
|
|
|
|
if *ellipsis {
|
|
|
|
|
wln!(p, "..");
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
w!(self, "}}");
|
|
|
|
|
}
|
|
|
|
|
Pat::Range { start, end } => {
|
|
|
|
|
self.print_expr(*start);
|
|
|
|
|
w!(self, "...");
|
|
|
|
|
self.print_expr(*end);
|
|
|
|
|
}
|
|
|
|
|
Pat::Slice { prefix, slice, suffix } => {
|
|
|
|
|
w!(self, "[");
|
|
|
|
|
for pat in prefix.iter() {
|
|
|
|
|
self.print_pat(*pat);
|
|
|
|
|
w!(self, ", ");
|
|
|
|
|
}
|
|
|
|
|
if let Some(pat) = slice {
|
|
|
|
|
self.print_pat(*pat);
|
|
|
|
|
w!(self, ", ");
|
|
|
|
|
}
|
|
|
|
|
for pat in suffix.iter() {
|
|
|
|
|
self.print_pat(*pat);
|
|
|
|
|
w!(self, ", ");
|
|
|
|
|
}
|
|
|
|
|
w!(self, "]");
|
|
|
|
|
}
|
|
|
|
|
Pat::Path(path) => self.print_path(path),
|
|
|
|
|
Pat::Lit(expr) => self.print_expr(*expr),
|
2023-02-19 00:02:55 +03:30
|
|
|
|
Pat::Bind { id, subpat } => {
|
|
|
|
|
self.print_binding(*id);
|
2022-08-15 13:51:45 +02:00
|
|
|
|
if let Some(pat) = subpat {
|
|
|
|
|
self.whitespace();
|
|
|
|
|
self.print_pat(*pat);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Pat::TupleStruct { path, args, ellipsis } => {
|
|
|
|
|
match path {
|
|
|
|
|
Some(path) => self.print_path(path),
|
|
|
|
|
None => w!(self, "<EFBFBD>"),
|
|
|
|
|
}
|
|
|
|
|
w!(self, "(");
|
|
|
|
|
for (i, arg) in args.iter().enumerate() {
|
|
|
|
|
if i != 0 {
|
|
|
|
|
w!(self, ", ");
|
|
|
|
|
}
|
|
|
|
|
if *ellipsis == Some(i) {
|
|
|
|
|
w!(self, ", ..");
|
|
|
|
|
}
|
|
|
|
|
self.print_pat(*arg);
|
|
|
|
|
}
|
|
|
|
|
w!(self, ")");
|
|
|
|
|
}
|
|
|
|
|
Pat::Ref { pat, mutability } => {
|
|
|
|
|
w!(self, "&");
|
|
|
|
|
if mutability.is_mut() {
|
|
|
|
|
w!(self, "mut ");
|
|
|
|
|
}
|
|
|
|
|
self.print_pat(*pat);
|
|
|
|
|
}
|
|
|
|
|
Pat::Box { inner } => {
|
|
|
|
|
w!(self, "box ");
|
|
|
|
|
self.print_pat(*inner);
|
|
|
|
|
}
|
|
|
|
|
Pat::ConstBlock(c) => {
|
|
|
|
|
w!(self, "const ");
|
|
|
|
|
self.print_expr(*c);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn print_stmt(&mut self, stmt: &Statement) {
|
|
|
|
|
match stmt {
|
|
|
|
|
Statement::Let { pat, type_ref, initializer, else_branch } => {
|
|
|
|
|
w!(self, "let ");
|
|
|
|
|
self.print_pat(*pat);
|
|
|
|
|
if let Some(ty) = type_ref {
|
|
|
|
|
w!(self, ": ");
|
|
|
|
|
self.print_type_ref(ty);
|
|
|
|
|
}
|
|
|
|
|
if let Some(init) = initializer {
|
|
|
|
|
w!(self, " = ");
|
|
|
|
|
self.print_expr(*init);
|
|
|
|
|
}
|
|
|
|
|
if let Some(els) = else_branch {
|
|
|
|
|
w!(self, " else ");
|
|
|
|
|
self.print_expr(*els);
|
|
|
|
|
}
|
|
|
|
|
wln!(self, ";");
|
|
|
|
|
}
|
|
|
|
|
Statement::Expr { expr, has_semi } => {
|
|
|
|
|
self.print_expr(*expr);
|
|
|
|
|
if *has_semi {
|
|
|
|
|
w!(self, ";");
|
|
|
|
|
}
|
|
|
|
|
wln!(self);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn print_literal(&mut self, literal: &Literal) {
|
|
|
|
|
match literal {
|
|
|
|
|
Literal::String(it) => w!(self, "{:?}", it),
|
|
|
|
|
Literal::ByteString(it) => w!(self, "\"{}\"", it.escape_ascii()),
|
2023-05-18 11:06:05 +02:00
|
|
|
|
Literal::CString(it) => w!(self, "\"{}\\0\"", it),
|
2022-08-15 13:51:45 +02:00
|
|
|
|
Literal::Char(it) => w!(self, "'{}'", it.escape_debug()),
|
|
|
|
|
Literal::Bool(it) => w!(self, "{}", it),
|
|
|
|
|
Literal::Int(i, suffix) => {
|
|
|
|
|
w!(self, "{}", i);
|
|
|
|
|
if let Some(suffix) = suffix {
|
|
|
|
|
w!(self, "{}", suffix);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Literal::Uint(i, suffix) => {
|
|
|
|
|
w!(self, "{}", i);
|
|
|
|
|
if let Some(suffix) = suffix {
|
|
|
|
|
w!(self, "{}", suffix);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Literal::Float(f, suffix) => {
|
|
|
|
|
w!(self, "{}", f);
|
|
|
|
|
if let Some(suffix) = suffix {
|
|
|
|
|
w!(self, "{}", suffix);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn print_type_ref(&mut self, ty: &TypeRef) {
|
|
|
|
|
print_type_ref(ty, self).unwrap();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn print_path(&mut self, path: &Path) {
|
|
|
|
|
print_path(path, self).unwrap();
|
|
|
|
|
}
|
2023-02-19 00:02:55 +03:30
|
|
|
|
|
|
|
|
|
fn print_binding(&mut self, id: BindingId) {
|
|
|
|
|
let Binding { name, mode, .. } = &self.body.bindings[id];
|
|
|
|
|
let mode = match mode {
|
|
|
|
|
BindingAnnotation::Unannotated => "",
|
|
|
|
|
BindingAnnotation::Mutable => "mut ",
|
|
|
|
|
BindingAnnotation::Ref => "ref ",
|
|
|
|
|
BindingAnnotation::RefMut => "ref mut ",
|
|
|
|
|
};
|
|
|
|
|
w!(self, "{}{}", mode, name);
|
|
|
|
|
}
|
2022-08-15 13:51:45 +02:00
|
|
|
|
}
|