1use ast::StaticItem;
2use itertools::{Itertools, Position};
3use rustc_ast as ast;
4use rustc_ast::ModKind;
5use rustc_span::Ident;
6
7use crate::pp::BoxMarker;
8use crate::pp::Breaks::Inconsistent;
9use crate::pprust::state::fixup::FixupContext;
10use crate::pprust::state::{AnnNode, INDENT_UNIT, PrintState, State};
11
12enum DelegationKind<'a> {
13 Single,
14 List(&'a [(Ident, Option<Ident>)]),
15 Glob,
16}
17
18fn visibility_qualified(vis: &ast::Visibility, s: &str) -> String {
19 format!("{}{}", State::to_string(|s| s.print_visibility(vis)), s)
20}
21
22impl<'a> State<'a> {
23 fn print_foreign_mod(&mut self, nmod: &ast::ForeignMod, attrs: &[ast::Attribute]) {
24 self.print_inner_attributes(attrs);
25 for item in &nmod.items {
26 self.print_foreign_item(item);
27 }
28 }
29
30 pub(crate) fn print_foreign_item(&mut self, item: &ast::ForeignItem) {
31 let ast::Item { id, span, ref attrs, ref kind, ref vis, tokens: _ } = *item;
32 self.ann.pre(self, AnnNode::SubItem(id));
33 self.hardbreak_if_not_bol();
34 self.maybe_print_comment(span.lo());
35 self.print_outer_attributes(attrs);
36 match kind {
37 ast::ForeignItemKind::Fn(func) => {
38 self.print_fn_full(vis, attrs, &*func);
39 }
40 ast::ForeignItemKind::Static(box ast::StaticItem {
41 ident,
42 ty,
43 mutability,
44 expr,
45 safety,
46 define_opaque,
47 }) => self.print_item_const(
48 *ident,
49 Some(*mutability),
50 &ast::Generics::default(),
51 ty,
52 expr.as_deref(),
53 vis,
54 *safety,
55 ast::Defaultness::Final,
56 define_opaque.as_deref(),
57 ),
58 ast::ForeignItemKind::TyAlias(box ast::TyAlias {
59 defaultness,
60 ident,
61 generics,
62 where_clauses,
63 bounds,
64 ty,
65 }) => {
66 self.print_associated_type(
67 *ident,
68 generics,
69 *where_clauses,
70 bounds,
71 ty.as_deref(),
72 vis,
73 *defaultness,
74 );
75 }
76 ast::ForeignItemKind::MacCall(m) => {
77 self.print_mac(m);
78 if m.args.need_semicolon() {
79 self.word(";");
80 }
81 }
82 }
83 self.ann.post(self, AnnNode::SubItem(id))
84 }
85
86 fn print_item_const(
87 &mut self,
88 ident: Ident,
89 mutbl: Option<ast::Mutability>,
90 generics: &ast::Generics,
91 ty: &ast::Ty,
92 body: Option<&ast::Expr>,
93 vis: &ast::Visibility,
94 safety: ast::Safety,
95 defaultness: ast::Defaultness,
96 define_opaque: Option<&[(ast::NodeId, ast::Path)]>,
97 ) {
98 self.print_define_opaques(define_opaque);
99 let (cb, ib) = self.head("");
100 self.print_visibility(vis);
101 self.print_safety(safety);
102 self.print_defaultness(defaultness);
103 let leading = match mutbl {
104 None => "const",
105 Some(ast::Mutability::Not) => "static",
106 Some(ast::Mutability::Mut) => "static mut",
107 };
108 self.word_space(leading);
109 self.print_ident(ident);
110 self.print_generic_params(&generics.params);
111 self.word_space(":");
112 self.print_type(ty);
113 if body.is_some() {
114 self.space();
115 }
116 self.end(ib);
117 if let Some(body) = body {
118 self.word_space("=");
119 self.print_expr(body, FixupContext::default());
120 }
121 self.print_where_clause(&generics.where_clause);
122 self.word(";");
123 self.end(cb);
124 }
125
126 fn print_associated_type(
127 &mut self,
128 ident: Ident,
129 generics: &ast::Generics,
130 where_clauses: ast::TyAliasWhereClauses,
131 bounds: &ast::GenericBounds,
132 ty: Option<&ast::Ty>,
133 vis: &ast::Visibility,
134 defaultness: ast::Defaultness,
135 ) {
136 let (before_predicates, after_predicates) =
137 generics.where_clause.predicates.split_at(where_clauses.split);
138 let (cb, ib) = self.head("");
139 self.print_visibility(vis);
140 self.print_defaultness(defaultness);
141 self.word_space("type");
142 self.print_ident(ident);
143 self.print_generic_params(&generics.params);
144 if !bounds.is_empty() {
145 self.word_nbsp(":");
146 self.print_type_bounds(bounds);
147 }
148 self.print_where_clause_parts(where_clauses.before.has_where_token, before_predicates);
149 if let Some(ty) = ty {
150 self.space();
151 self.word_space("=");
152 self.print_type(ty);
153 }
154 self.print_where_clause_parts(where_clauses.after.has_where_token, after_predicates);
155 self.word(";");
156 self.end(ib);
157 self.end(cb);
158 }
159
160 pub(crate) fn print_item(&mut self, item: &ast::Item) {
162 if self.is_sdylib_interface && item.span.is_dummy() {
163 return;
165 }
166 self.hardbreak_if_not_bol();
167 self.maybe_print_comment(item.span.lo());
168 self.print_outer_attributes(&item.attrs);
169 self.ann.pre(self, AnnNode::Item(item));
170 match &item.kind {
171 ast::ItemKind::ExternCrate(orig_name, ident) => {
172 let (cb, ib) = self.head(visibility_qualified(&item.vis, "extern crate"));
173 if let &Some(orig_name) = orig_name {
174 self.print_name(orig_name);
175 self.space();
176 self.word("as");
177 self.space();
178 }
179 self.print_ident(*ident);
180 self.word(";");
181 self.end(ib);
182 self.end(cb);
183 }
184 ast::ItemKind::Use(tree) => {
185 self.print_visibility(&item.vis);
186 self.word_nbsp("use");
187 self.print_use_tree(tree);
188 self.word(";");
189 }
190 ast::ItemKind::Static(box StaticItem {
191 ident,
192 ty,
193 safety,
194 mutability: mutbl,
195 expr: body,
196 define_opaque,
197 }) => {
198 self.print_safety(*safety);
199 self.print_item_const(
200 *ident,
201 Some(*mutbl),
202 &ast::Generics::default(),
203 ty,
204 body.as_deref(),
205 &item.vis,
206 ast::Safety::Default,
207 ast::Defaultness::Final,
208 define_opaque.as_deref(),
209 );
210 }
211 ast::ItemKind::Const(box ast::ConstItem {
212 defaultness,
213 ident,
214 generics,
215 ty,
216 expr,
217 define_opaque,
218 }) => {
219 self.print_item_const(
220 *ident,
221 None,
222 generics,
223 ty,
224 expr.as_deref(),
225 &item.vis,
226 ast::Safety::Default,
227 *defaultness,
228 define_opaque.as_deref(),
229 );
230 }
231 ast::ItemKind::Fn(func) => {
232 self.print_fn_full(&item.vis, &item.attrs, &*func);
233 }
234 ast::ItemKind::Mod(safety, ident, mod_kind) => {
235 let (cb, ib) = self.head(Self::to_string(|s| {
236 s.print_visibility(&item.vis);
237 s.print_safety(*safety);
238 s.word("mod");
239 }));
240 self.print_ident(*ident);
241
242 match mod_kind {
243 ModKind::Loaded(items, ..) => {
244 self.nbsp();
245 self.bopen(ib);
246 self.print_inner_attributes(&item.attrs);
247 for item in items {
248 self.print_item(item);
249 }
250 let empty = item.attrs.is_empty() && items.is_empty();
251 self.bclose(item.span, empty, cb);
252 }
253 ModKind::Unloaded => {
254 self.word(";");
255 self.end(ib);
256 self.end(cb);
257 }
258 }
259 }
260 ast::ItemKind::ForeignMod(nmod) => {
261 let (cb, ib) = self.head(Self::to_string(|s| {
262 s.print_safety(nmod.safety);
263 s.word("extern");
264 }));
265 if let Some(abi) = nmod.abi {
266 self.print_token_literal(abi.as_token_lit(), abi.span);
267 self.nbsp();
268 }
269 self.bopen(ib);
270 self.print_foreign_mod(nmod, &item.attrs);
271 let empty = item.attrs.is_empty() && nmod.items.is_empty();
272 self.bclose(item.span, empty, cb);
273 }
274 ast::ItemKind::GlobalAsm(asm) => {
275 let (cb, ib) = self.head(visibility_qualified(&item.vis, "global_asm!"));
277 self.print_inline_asm(asm);
278 self.word(";");
279 self.end(ib);
280 self.end(cb);
281 }
282 ast::ItemKind::TyAlias(box ast::TyAlias {
283 defaultness,
284 ident,
285 generics,
286 where_clauses,
287 bounds,
288 ty,
289 }) => {
290 self.print_associated_type(
291 *ident,
292 generics,
293 *where_clauses,
294 bounds,
295 ty.as_deref(),
296 &item.vis,
297 *defaultness,
298 );
299 }
300 ast::ItemKind::Enum(ident, generics, enum_definition) => {
301 self.print_enum_def(enum_definition, generics, *ident, item.span, &item.vis);
302 }
303 ast::ItemKind::Struct(ident, generics, struct_def) => {
304 let (cb, ib) = self.head(visibility_qualified(&item.vis, "struct"));
305 self.print_struct(struct_def, generics, *ident, item.span, true, cb, ib);
306 }
307 ast::ItemKind::Union(ident, generics, struct_def) => {
308 let (cb, ib) = self.head(visibility_qualified(&item.vis, "union"));
309 self.print_struct(struct_def, generics, *ident, item.span, true, cb, ib);
310 }
311 ast::ItemKind::Impl(ast::Impl { generics, of_trait, self_ty, items }) => {
312 let (cb, ib) = self.head("");
313 self.print_visibility(&item.vis);
314
315 let impl_generics = |this: &mut Self| {
316 this.word("impl");
317
318 if generics.params.is_empty() {
319 this.nbsp();
320 } else {
321 this.print_generic_params(&generics.params);
322 this.space();
323 }
324 };
325
326 if let Some(box of_trait) = of_trait {
327 let ast::TraitImplHeader {
328 defaultness,
329 safety,
330 constness,
331 polarity,
332 ref trait_ref,
333 } = *of_trait;
334 self.print_defaultness(defaultness);
335 self.print_safety(safety);
336 impl_generics(self);
337 self.print_constness(constness);
338 if let ast::ImplPolarity::Negative(_) = polarity {
339 self.word("!");
340 }
341 self.print_trait_ref(trait_ref);
342 self.space();
343 self.word_space("for");
344 } else {
345 impl_generics(self);
346 }
347
348 self.print_type(self_ty);
349 self.print_where_clause(&generics.where_clause);
350
351 self.space();
352 self.bopen(ib);
353 self.print_inner_attributes(&item.attrs);
354 for impl_item in items {
355 self.print_assoc_item(impl_item);
356 }
357 let empty = item.attrs.is_empty() && items.is_empty();
358 self.bclose(item.span, empty, cb);
359 }
360 ast::ItemKind::Trait(box ast::Trait {
361 constness,
362 safety,
363 is_auto,
364 ident,
365 generics,
366 bounds,
367 items,
368 }) => {
369 let (cb, ib) = self.head("");
370 self.print_visibility(&item.vis);
371 self.print_constness(*constness);
372 self.print_safety(*safety);
373 self.print_is_auto(*is_auto);
374 self.word_nbsp("trait");
375 self.print_ident(*ident);
376 self.print_generic_params(&generics.params);
377 if !bounds.is_empty() {
378 self.word_nbsp(":");
379 self.print_type_bounds(bounds);
380 }
381 self.print_where_clause(&generics.where_clause);
382 self.word(" ");
383 self.bopen(ib);
384 self.print_inner_attributes(&item.attrs);
385 for trait_item in items {
386 self.print_assoc_item(trait_item);
387 }
388 let empty = item.attrs.is_empty() && items.is_empty();
389 self.bclose(item.span, empty, cb);
390 }
391 ast::ItemKind::TraitAlias(ident, generics, bounds) => {
392 let (cb, ib) = self.head(visibility_qualified(&item.vis, "trait"));
393 self.print_ident(*ident);
394 self.print_generic_params(&generics.params);
395 self.nbsp();
396 if !bounds.is_empty() {
397 self.word_nbsp("=");
398 self.print_type_bounds(bounds);
399 }
400 self.print_where_clause(&generics.where_clause);
401 self.word(";");
402 self.end(ib);
403 self.end(cb);
404 }
405 ast::ItemKind::MacCall(mac) => {
406 self.print_mac(mac);
407 if mac.args.need_semicolon() {
408 self.word(";");
409 }
410 }
411 ast::ItemKind::MacroDef(ident, macro_def) => {
412 self.print_mac_def(macro_def, &ident, item.span, |state| {
413 state.print_visibility(&item.vis)
414 });
415 }
416 ast::ItemKind::Delegation(deleg) => self.print_delegation(
417 &item.attrs,
418 &item.vis,
419 &deleg.qself,
420 &deleg.path,
421 DelegationKind::Single,
422 &deleg.body,
423 ),
424 ast::ItemKind::DelegationMac(deleg) => self.print_delegation(
425 &item.attrs,
426 &item.vis,
427 &deleg.qself,
428 &deleg.prefix,
429 deleg.suffixes.as_ref().map_or(DelegationKind::Glob, |s| DelegationKind::List(s)),
430 &deleg.body,
431 ),
432 }
433 self.ann.post(self, AnnNode::Item(item))
434 }
435
436 fn print_enum_def(
437 &mut self,
438 enum_definition: &ast::EnumDef,
439 generics: &ast::Generics,
440 ident: Ident,
441 span: rustc_span::Span,
442 visibility: &ast::Visibility,
443 ) {
444 let (cb, ib) = self.head(visibility_qualified(visibility, "enum"));
445 self.print_ident(ident);
446 self.print_generic_params(&generics.params);
447 self.print_where_clause(&generics.where_clause);
448 self.space();
449 self.bopen(ib);
450 for v in enum_definition.variants.iter() {
451 self.space_if_not_bol();
452 self.maybe_print_comment(v.span.lo());
453 self.print_outer_attributes(&v.attrs);
454 let ib = self.ibox(0);
455 self.print_variant(v);
456 self.word(",");
457 self.end(ib);
458 self.maybe_print_trailing_comment(v.span, None);
459 }
460 let empty = enum_definition.variants.is_empty();
461 self.bclose(span, empty, cb)
462 }
463
464 pub(crate) fn print_visibility(&mut self, vis: &ast::Visibility) {
465 match &vis.kind {
466 ast::VisibilityKind::Public => self.word_nbsp("pub"),
467 ast::VisibilityKind::Restricted { path, shorthand, .. } => {
468 let path = Self::to_string(|s| s.print_path(path, false, 0));
469 if *shorthand && (path == "crate" || path == "self" || path == "super") {
470 self.word_nbsp(format!("pub({path})"))
471 } else {
472 self.word_nbsp(format!("pub(in {path})"))
473 }
474 }
475 ast::VisibilityKind::Inherited => {}
476 }
477 }
478
479 fn print_defaultness(&mut self, defaultness: ast::Defaultness) {
480 if let ast::Defaultness::Default(_) = defaultness {
481 self.word_nbsp("default");
482 }
483 }
484
485 fn print_struct(
486 &mut self,
487 struct_def: &ast::VariantData,
488 generics: &ast::Generics,
489 ident: Ident,
490 span: rustc_span::Span,
491 print_finalizer: bool,
492 cb: BoxMarker,
493 ib: BoxMarker,
494 ) {
495 self.print_ident(ident);
496 self.print_generic_params(&generics.params);
497 match &struct_def {
498 ast::VariantData::Tuple(..) | ast::VariantData::Unit(..) => {
499 if let ast::VariantData::Tuple(..) = struct_def {
500 self.popen();
501 self.commasep(Inconsistent, struct_def.fields(), |s, field| {
502 s.maybe_print_comment(field.span.lo());
503 s.print_outer_attributes(&field.attrs);
504 s.print_visibility(&field.vis);
505 s.print_type(&field.ty)
506 });
507 self.pclose();
508 }
509 self.print_where_clause(&generics.where_clause);
510 if print_finalizer {
511 self.word(";");
512 }
513 self.end(ib);
514 self.end(cb);
515 }
516 ast::VariantData::Struct { fields, .. } => {
517 self.print_where_clause(&generics.where_clause);
518 self.nbsp();
519 self.bopen(ib);
520
521 let empty = fields.is_empty();
522 if !empty {
523 self.hardbreak_if_not_bol();
524
525 for field in fields {
526 self.hardbreak_if_not_bol();
527 self.maybe_print_comment(field.span.lo());
528 self.print_outer_attributes(&field.attrs);
529 self.print_visibility(&field.vis);
530 self.print_ident(field.ident.unwrap());
531 self.word_nbsp(":");
532 self.print_type(&field.ty);
533 self.word(",");
534 }
535 }
536
537 self.bclose(span, empty, cb);
538 }
539 }
540 }
541
542 pub(crate) fn print_variant(&mut self, v: &ast::Variant) {
543 let (cb, ib) = self.head("");
544 self.print_visibility(&v.vis);
545 let generics = ast::Generics::default();
546 self.print_struct(&v.data, &generics, v.ident, v.span, false, cb, ib);
547 if let Some(d) = &v.disr_expr {
548 self.space();
549 self.word_space("=");
550 self.print_expr(&d.value, FixupContext::default())
551 }
552 }
553
554 pub(crate) fn print_assoc_item(&mut self, item: &ast::AssocItem) {
555 let ast::Item { id, span, ref attrs, ref kind, ref vis, tokens: _ } = *item;
556 self.ann.pre(self, AnnNode::SubItem(id));
557 self.hardbreak_if_not_bol();
558 self.maybe_print_comment(span.lo());
559 self.print_outer_attributes(attrs);
560 match kind {
561 ast::AssocItemKind::Fn(func) => {
562 self.print_fn_full(vis, attrs, &*func);
563 }
564 ast::AssocItemKind::Const(box ast::ConstItem {
565 defaultness,
566 ident,
567 generics,
568 ty,
569 expr,
570 define_opaque,
571 }) => {
572 self.print_item_const(
573 *ident,
574 None,
575 generics,
576 ty,
577 expr.as_deref(),
578 vis,
579 ast::Safety::Default,
580 *defaultness,
581 define_opaque.as_deref(),
582 );
583 }
584 ast::AssocItemKind::Type(box ast::TyAlias {
585 defaultness,
586 ident,
587 generics,
588 where_clauses,
589 bounds,
590 ty,
591 }) => {
592 self.print_associated_type(
593 *ident,
594 generics,
595 *where_clauses,
596 bounds,
597 ty.as_deref(),
598 vis,
599 *defaultness,
600 );
601 }
602 ast::AssocItemKind::MacCall(m) => {
603 self.print_mac(m);
604 if m.args.need_semicolon() {
605 self.word(";");
606 }
607 }
608 ast::AssocItemKind::Delegation(deleg) => self.print_delegation(
609 &item.attrs,
610 vis,
611 &deleg.qself,
612 &deleg.path,
613 DelegationKind::Single,
614 &deleg.body,
615 ),
616 ast::AssocItemKind::DelegationMac(deleg) => self.print_delegation(
617 &item.attrs,
618 vis,
619 &deleg.qself,
620 &deleg.prefix,
621 deleg.suffixes.as_ref().map_or(DelegationKind::Glob, |s| DelegationKind::List(s)),
622 &deleg.body,
623 ),
624 }
625 self.ann.post(self, AnnNode::SubItem(id))
626 }
627
628 fn print_delegation(
629 &mut self,
630 attrs: &[ast::Attribute],
631 vis: &ast::Visibility,
632 qself: &Option<Box<ast::QSelf>>,
633 path: &ast::Path,
634 kind: DelegationKind<'_>,
635 body: &Option<Box<ast::Block>>,
636 ) {
637 let body_cb_ib = body.as_ref().map(|body| (body, self.head("")));
638 self.print_visibility(vis);
639 self.word_nbsp("reuse");
640
641 if let Some(qself) = qself {
642 self.print_qpath(path, qself, false);
643 } else {
644 self.print_path(path, false, 0);
645 }
646 match kind {
647 DelegationKind::Single => {}
648 DelegationKind::List(suffixes) => {
649 self.word("::");
650 self.word("{");
651 for (i, (ident, rename)) in suffixes.iter().enumerate() {
652 self.print_ident(*ident);
653 if let Some(rename) = rename {
654 self.nbsp();
655 self.word_nbsp("as");
656 self.print_ident(*rename);
657 }
658 if i != suffixes.len() - 1 {
659 self.word_space(",");
660 }
661 }
662 self.word("}");
663 }
664 DelegationKind::Glob => {
665 self.word("::");
666 self.word("*");
667 }
668 }
669 if let Some((body, (cb, ib))) = body_cb_ib {
670 self.nbsp();
671 self.print_block_with_attrs(body, attrs, cb, ib);
672 } else {
673 self.word(";");
674 }
675 }
676
677 fn print_fn_full(&mut self, vis: &ast::Visibility, attrs: &[ast::Attribute], func: &ast::Fn) {
678 let ast::Fn { defaultness, ident, generics, sig, contract, body, define_opaque } = func;
679
680 self.print_define_opaques(define_opaque.as_deref());
681
682 let body_cb_ib = body.as_ref().map(|body| (body, self.head("")));
683
684 self.print_visibility(vis);
685 self.print_defaultness(*defaultness);
686 self.print_fn(&sig.decl, sig.header, Some(*ident), generics);
687 if let Some(contract) = &contract {
688 self.nbsp();
689 self.print_contract(contract);
690 }
691 if let Some((body, (cb, ib))) = body_cb_ib {
692 if self.is_sdylib_interface {
693 self.word(";");
694 self.end(ib); self.end(cb); return;
697 }
698
699 self.nbsp();
700 self.print_block_with_attrs(body, attrs, cb, ib);
701 } else {
702 self.word(";");
703 }
704 }
705
706 fn print_define_opaques(&mut self, define_opaque: Option<&[(ast::NodeId, ast::Path)]>) {
707 if let Some(define_opaque) = define_opaque {
708 self.word("#[define_opaque(");
709 for (i, (_, path)) in define_opaque.iter().enumerate() {
710 if i != 0 {
711 self.word_space(",");
712 }
713
714 self.print_path(path, false, 0);
715 }
716 self.word(")]");
717 }
718 self.hardbreak_if_not_bol();
719 }
720
721 fn print_contract(&mut self, contract: &ast::FnContract) {
722 if let Some(pred) = &contract.requires {
723 self.word("rustc_requires");
724 self.popen();
725 self.print_expr(pred, FixupContext::default());
726 self.pclose();
727 }
728 if let Some(pred) = &contract.ensures {
729 self.word("rustc_ensures");
730 self.popen();
731 self.print_expr(pred, FixupContext::default());
732 self.pclose();
733 }
734 }
735
736 pub(crate) fn print_fn(
737 &mut self,
738 decl: &ast::FnDecl,
739 header: ast::FnHeader,
740 ident: Option<Ident>,
741 generics: &ast::Generics,
742 ) {
743 self.print_fn_header_info(header);
744 if let Some(ident) = ident {
745 self.nbsp();
746 self.print_ident(ident);
747 }
748 self.print_generic_params(&generics.params);
749 self.print_fn_params_and_ret(decl, false);
750 self.print_where_clause(&generics.where_clause);
751 }
752
753 pub(crate) fn print_fn_params_and_ret(&mut self, decl: &ast::FnDecl, is_closure: bool) {
754 let (open, close) = if is_closure { ("|", "|") } else { ("(", ")") };
755 self.word(open);
756 self.commasep(Inconsistent, &decl.inputs, |s, param| s.print_param(param, is_closure));
757 self.word(close);
758 self.print_fn_ret_ty(&decl.output)
759 }
760
761 fn print_where_clause(&mut self, where_clause: &ast::WhereClause) {
762 self.print_where_clause_parts(where_clause.has_where_token, &where_clause.predicates);
763 }
764
765 fn print_where_clause_parts(
766 &mut self,
767 has_where_token: bool,
768 predicates: &[ast::WherePredicate],
769 ) {
770 if predicates.is_empty() && !has_where_token {
771 return;
772 }
773
774 self.space();
775 self.word_space("where");
776
777 for (i, predicate) in predicates.iter().enumerate() {
778 if i != 0 {
779 self.word_space(",");
780 }
781
782 self.print_where_predicate(predicate);
783 }
784 }
785
786 pub fn print_where_predicate(&mut self, predicate: &ast::WherePredicate) {
787 let ast::WherePredicate { attrs, kind, id: _, span: _, is_placeholder: _ } = predicate;
788 self.print_outer_attributes(attrs);
789 match kind {
790 ast::WherePredicateKind::BoundPredicate(where_bound_predicate) => {
791 self.print_where_bound_predicate(where_bound_predicate);
792 }
793 ast::WherePredicateKind::RegionPredicate(ast::WhereRegionPredicate {
794 lifetime,
795 bounds,
796 ..
797 }) => {
798 self.print_lifetime(*lifetime);
799 self.word(":");
800 if !bounds.is_empty() {
801 self.nbsp();
802 self.print_lifetime_bounds(bounds);
803 }
804 }
805 ast::WherePredicateKind::EqPredicate(ast::WhereEqPredicate {
806 lhs_ty, rhs_ty, ..
807 }) => {
808 self.print_type(lhs_ty);
809 self.space();
810 self.word_space("=");
811 self.print_type(rhs_ty);
812 }
813 }
814 }
815
816 pub(crate) fn print_where_bound_predicate(
817 &mut self,
818 where_bound_predicate: &ast::WhereBoundPredicate,
819 ) {
820 self.print_formal_generic_params(&where_bound_predicate.bound_generic_params);
821 self.print_type(&where_bound_predicate.bounded_ty);
822 self.word(":");
823 if !where_bound_predicate.bounds.is_empty() {
824 self.nbsp();
825 self.print_type_bounds(&where_bound_predicate.bounds);
826 }
827 }
828
829 fn print_use_tree(&mut self, tree: &ast::UseTree) {
830 match &tree.kind {
831 ast::UseTreeKind::Simple(rename) => {
832 self.print_path(&tree.prefix, false, 0);
833 if let &Some(rename) = rename {
834 self.nbsp();
835 self.word_nbsp("as");
836 self.print_ident(rename);
837 }
838 }
839 ast::UseTreeKind::Glob => {
840 if !tree.prefix.segments.is_empty() {
841 self.print_path(&tree.prefix, false, 0);
842 self.word("::");
843 }
844 self.word("*");
845 }
846 ast::UseTreeKind::Nested { items, .. } => {
847 if !tree.prefix.segments.is_empty() {
848 self.print_path(&tree.prefix, false, 0);
849 self.word("::");
850 }
851 if items.is_empty() {
852 self.word("{}");
853 } else if let [(item, _)] = items.as_slice() {
854 self.print_use_tree(item);
855 } else {
856 let cb = self.cbox(INDENT_UNIT);
857 self.word("{");
858 self.zerobreak();
859 let ib = self.ibox(0);
860 for (pos, use_tree) in items.iter().with_position() {
861 let is_last = matches!(pos, Position::Last | Position::Only);
862 self.print_use_tree(&use_tree.0);
863 if !is_last {
864 self.word(",");
865 if let ast::UseTreeKind::Nested { .. } = use_tree.0.kind {
866 self.hardbreak();
867 } else {
868 self.space();
869 }
870 }
871 }
872 self.end(ib);
873 self.trailing_comma();
874 self.offset(-INDENT_UNIT);
875 self.word("}");
876 self.end(cb);
877 }
878 }
879 }
880 }
881}