2019-02-04 21:49:54 +09:00
|
|
|
use crate::deriving::generic::ty::*;
|
2019-12-22 17:42:04 -05:00
|
|
|
use crate::deriving::generic::*;
|
|
|
|
use crate::deriving::path_std;
|
2015-12-10 23:23:14 +09:00
|
|
|
|
2020-02-29 20:37:32 +03:00
|
|
|
use rustc_ast::ptr::P;
|
2020-04-27 23:26:11 +05:30
|
|
|
use rustc_ast::{self as ast, Expr, MetaItem};
|
2019-12-29 17:23:55 +03:00
|
|
|
use rustc_expand::base::{Annotatable, ExtCtxt};
|
2020-04-19 13:00:18 +02:00
|
|
|
use rustc_span::symbol::{sym, Ident};
|
2019-12-31 20:15:40 +03:00
|
|
|
use rustc_span::{Span, DUMMY_SP};
|
2014-02-06 23:02:28 +11:00
|
|
|
|
2019-12-22 17:42:04 -05:00
|
|
|
pub fn expand_deriving_debug(
|
|
|
|
cx: &mut ExtCtxt<'_>,
|
|
|
|
span: Span,
|
|
|
|
mitem: &MetaItem,
|
|
|
|
item: &Annotatable,
|
|
|
|
push: &mut dyn FnMut(Annotatable),
|
|
|
|
) {
|
2014-02-06 23:02:28 +11:00
|
|
|
// &mut ::std::fmt::Formatter
|
2019-12-22 17:42:04 -05:00
|
|
|
let fmtr =
|
2020-07-14 14:59:39 +10:00
|
|
|
Ptr(Box::new(Literal(path_std!(fmt::Formatter))), Borrowed(None, ast::Mutability::Mut));
|
2014-02-06 23:02:28 +11:00
|
|
|
|
|
|
|
let trait_def = TraitDef {
|
2017-08-06 22:54:09 -07:00
|
|
|
span,
|
2014-02-28 13:09:09 -08:00
|
|
|
attributes: Vec::new(),
|
2020-07-14 14:59:39 +10:00
|
|
|
path: path_std!(fmt::Debug),
|
2014-02-28 13:09:09 -08:00
|
|
|
additional_bounds: Vec::new(),
|
2020-07-14 16:19:44 +10:00
|
|
|
generics: Bounds::empty(),
|
2015-08-29 14:50:05 -04:00
|
|
|
is_unsafe: false,
|
2016-08-29 11:14:25 +00:00
|
|
|
supports_unions: false,
|
2016-07-19 23:02:06 +05:30
|
|
|
methods: vec![MethodDef {
|
2020-07-08 11:04:10 +10:00
|
|
|
name: sym::fmt,
|
2020-07-14 16:19:44 +10:00
|
|
|
generics: Bounds::empty(),
|
2019-12-22 17:42:04 -05:00
|
|
|
explicit_self: borrowed_explicit_self(),
|
2020-07-14 15:05:26 +10:00
|
|
|
args: vec![(fmtr, sym::f)],
|
2020-07-14 14:59:39 +10:00
|
|
|
ret_ty: Literal(path_std!(fmt::Result)),
|
2019-12-22 17:42:04 -05:00
|
|
|
attributes: Vec::new(),
|
|
|
|
is_unsafe: false,
|
|
|
|
unify_fieldless_variants: false,
|
|
|
|
combine_substructure: combine_substructure(Box::new(|a, b, c| {
|
|
|
|
show_substructure(a, b, c)
|
|
|
|
})),
|
|
|
|
}],
|
2015-01-25 00:29:24 -05:00
|
|
|
associated_types: Vec::new(),
|
2014-02-06 23:02:28 +11:00
|
|
|
};
|
2015-05-22 21:10:14 +05:30
|
|
|
trait_def.expand(cx, mitem, item, push)
|
2014-02-06 23:02:28 +11:00
|
|
|
}
|
|
|
|
|
2015-03-07 19:36:36 -08:00
|
|
|
/// We use the debug builders to do the heavy lifting here
|
2019-02-04 21:49:54 +09:00
|
|
|
fn show_substructure(cx: &mut ExtCtxt<'_>, span: Span, substr: &Substructure<'_>) -> P<Expr> {
|
2015-03-07 19:36:36 -08:00
|
|
|
// build fmt.debug_struct(<name>).field(<fieldname>, &<fieldval>)....build()
|
|
|
|
// or fmt.debug_tuple(<name>).field(&<fieldval>)....build()
|
|
|
|
// based on the "shape".
|
2019-03-24 00:06:58 +03:00
|
|
|
let (ident, vdata, fields) = match substr.fields {
|
|
|
|
Struct(vdata, fields) => (substr.type_ident, *vdata, fields),
|
2019-08-13 21:40:21 -03:00
|
|
|
EnumMatching(_, _, v, fields) => (v.ident, &v.data, fields),
|
2019-12-22 17:42:04 -05:00
|
|
|
EnumNonMatchingCollapsed(..) | StaticStruct(..) | StaticEnum(..) => {
|
|
|
|
cx.span_bug(span, "nonsensical .fields in `#[derive(Debug)]`")
|
|
|
|
}
|
2014-02-06 23:02:28 +11:00
|
|
|
};
|
|
|
|
|
2017-03-17 04:04:41 +00:00
|
|
|
// We want to make sure we have the ctxt set so that we can use unstable methods
|
2019-08-21 21:28:22 +03:00
|
|
|
let span = cx.with_def_site_ctxt(span);
|
2016-11-16 10:52:37 +00:00
|
|
|
let name = cx.expr_lit(span, ast::LitKind::Str(ident.name, ast::StrStyle::Cooked));
|
2020-07-14 15:05:26 +10:00
|
|
|
let builder = Ident::new(sym::debug_trait_builder, span);
|
2020-04-16 00:00:22 +02:00
|
|
|
let builder_expr = cx.expr_ident(span, builder);
|
2014-02-06 23:02:28 +11:00
|
|
|
|
2015-11-03 22:48:28 -08:00
|
|
|
let fmt = substr.nonself_args[0].clone();
|
2015-05-17 14:03:37 -07:00
|
|
|
|
2020-10-16 11:43:39 +02:00
|
|
|
let mut stmts = Vec::with_capacity(fields.len() + 2);
|
2019-03-24 00:06:58 +03:00
|
|
|
match vdata {
|
|
|
|
ast::VariantData::Tuple(..) | ast::VariantData::Unit(..) => {
|
|
|
|
// tuple struct/"normal" variant
|
2020-07-14 15:05:26 +10:00
|
|
|
let expr =
|
|
|
|
cx.expr_method_call(span, fmt, Ident::new(sym::debug_tuple, span), vec![name]);
|
2019-08-11 14:16:12 +01:00
|
|
|
stmts.push(cx.stmt_let(span, true, builder, expr));
|
2019-03-24 00:06:58 +03:00
|
|
|
|
|
|
|
for field in fields {
|
|
|
|
// Use double indirection to make sure this works for unsized types
|
|
|
|
let field = cx.expr_addr_of(field.span, field.self_.clone());
|
|
|
|
let field = cx.expr_addr_of(field.span, field);
|
|
|
|
|
2019-12-22 17:42:04 -05:00
|
|
|
let expr = cx.expr_method_call(
|
|
|
|
span,
|
|
|
|
builder_expr.clone(),
|
|
|
|
Ident::new(sym::field, span),
|
|
|
|
vec![field],
|
|
|
|
);
|
2019-03-24 00:06:58 +03:00
|
|
|
|
|
|
|
// Use `let _ = expr;` to avoid triggering the
|
|
|
|
// unused_results lint.
|
2020-05-04 18:27:23 -04:00
|
|
|
stmts.push(stmt_let_underscore(cx, span, expr));
|
2014-02-06 23:02:28 +11:00
|
|
|
}
|
|
|
|
}
|
2019-03-24 00:06:58 +03:00
|
|
|
ast::VariantData::Struct(..) => {
|
|
|
|
// normal struct/struct variant
|
|
|
|
let expr =
|
2020-07-14 15:05:26 +10:00
|
|
|
cx.expr_method_call(span, fmt, Ident::new(sym::debug_struct, span), vec![name]);
|
2019-03-24 00:06:58 +03:00
|
|
|
stmts.push(cx.stmt_let(DUMMY_SP, true, builder, expr));
|
|
|
|
|
|
|
|
for field in fields {
|
2019-12-22 17:42:04 -05:00
|
|
|
let name = cx.expr_lit(
|
|
|
|
field.span,
|
|
|
|
ast::LitKind::Str(field.name.unwrap().name, ast::StrStyle::Cooked),
|
|
|
|
);
|
2019-03-24 00:06:58 +03:00
|
|
|
|
|
|
|
// Use double indirection to make sure this works for unsized types
|
|
|
|
let field = cx.expr_addr_of(field.span, field.self_.clone());
|
|
|
|
let field = cx.expr_addr_of(field.span, field);
|
2019-12-22 17:42:04 -05:00
|
|
|
let expr = cx.expr_method_call(
|
|
|
|
span,
|
|
|
|
builder_expr.clone(),
|
|
|
|
Ident::new(sym::field, span),
|
|
|
|
vec![name, field],
|
|
|
|
);
|
2020-05-04 18:27:23 -04:00
|
|
|
stmts.push(stmt_let_underscore(cx, span, expr));
|
2019-03-24 00:06:58 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-11-03 22:48:28 -08:00
|
|
|
|
2020-07-14 15:05:26 +10:00
|
|
|
let expr = cx.expr_method_call(span, builder_expr, Ident::new(sym::finish, span), vec![]);
|
2014-02-06 23:02:28 +11:00
|
|
|
|
2016-06-23 09:51:18 +00:00
|
|
|
stmts.push(cx.stmt_expr(expr));
|
|
|
|
let block = cx.block(span, stmts);
|
2015-11-03 22:48:28 -08:00
|
|
|
cx.expr_block(block)
|
2014-02-06 23:02:28 +11:00
|
|
|
}
|
2015-11-09 10:02:39 -08:00
|
|
|
|
2020-05-04 18:27:23 -04:00
|
|
|
fn stmt_let_underscore(cx: &mut ExtCtxt<'_>, sp: Span, expr: P<ast::Expr>) -> ast::Stmt {
|
2015-11-09 10:02:39 -08:00
|
|
|
let local = P(ast::Local {
|
|
|
|
pat: cx.pat_wild(sp),
|
|
|
|
ty: None,
|
|
|
|
init: Some(expr),
|
|
|
|
id: ast::DUMMY_NODE_ID,
|
|
|
|
span: sp,
|
2019-12-03 16:38:34 +01:00
|
|
|
attrs: ast::AttrVec::new(),
|
2020-11-17 14:27:44 -05:00
|
|
|
tokens: None,
|
2015-11-09 10:02:39 -08:00
|
|
|
});
|
2020-11-17 14:27:44 -05:00
|
|
|
ast::Stmt { id: ast::DUMMY_NODE_ID, kind: ast::StmtKind::Local(local), span: sp }
|
2015-11-09 10:02:39 -08:00
|
|
|
}
|