1use rustc_ast::mut_visit::*;
2use rustc_ast::token::Delimiter;
3use rustc_ast::visit::AssocCtxt;
4use rustc_ast::{self as ast, Safety};
5use rustc_data_structures::fx::FxHashMap;
6use rustc_span::{DUMMY_SP, Ident};
7use smallvec::{SmallVec, smallvec};
8use thin_vec::ThinVec;
9
10use crate::expand::{AstFragment, AstFragmentKind};
11
12pub(crate) fn placeholder(
13 kind: AstFragmentKind,
14 id: ast::NodeId,
15 vis: Option<ast::Visibility>,
16) -> AstFragment {
17 fn mac_placeholder() -> Box<ast::MacCall> {
18 Box::new(ast::MacCall {
19 path: ast::Path { span: DUMMY_SP, segments: ThinVec::new(), tokens: None },
20 args: Box::new(ast::DelimArgs {
21 dspan: ast::tokenstream::DelimSpan::dummy(),
22 delim: Delimiter::Parenthesis,
23 tokens: ast::tokenstream::TokenStream::new(Vec::new()),
24 }),
25 })
26 }
27
28 let ident = Ident::dummy();
29 let attrs = ast::AttrVec::new();
30 let vis = vis.unwrap_or(ast::Visibility {
31 span: DUMMY_SP,
32 kind: ast::VisibilityKind::Inherited,
33 tokens: None,
34 });
35 let span = DUMMY_SP;
36 let expr_placeholder = || {
37 Box::new(ast::Expr {
38 id,
39 span,
40 attrs: ast::AttrVec::new(),
41 kind: ast::ExprKind::MacCall(mac_placeholder()),
42 tokens: None,
43 })
44 };
45 let ty = || {
46 Box::new(ast::Ty { id, kind: ast::TyKind::MacCall(mac_placeholder()), span, tokens: None })
47 };
48 let pat = || {
49 Box::new(ast::Pat {
50 id,
51 kind: ast::PatKind::MacCall(mac_placeholder()),
52 span,
53 tokens: None,
54 })
55 };
56
57 match kind {
58 AstFragmentKind::Crate => AstFragment::Crate(ast::Crate {
59 attrs: Default::default(),
60 items: Default::default(),
61 spans: ast::ModSpans { inner_span: span, ..Default::default() },
62 id,
63 is_placeholder: true,
64 }),
65 AstFragmentKind::Expr => AstFragment::Expr(expr_placeholder()),
66 AstFragmentKind::OptExpr => AstFragment::OptExpr(Some(expr_placeholder())),
67 AstFragmentKind::MethodReceiverExpr => AstFragment::MethodReceiverExpr(expr_placeholder()),
68 AstFragmentKind::Items => AstFragment::Items({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(Box::new(ast::Item {
id,
span,
vis,
attrs,
kind: ast::ItemKind::MacCall(mac_placeholder()),
tokens: None,
}));
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[Box::new(ast::Item {
id,
span,
vis,
attrs,
kind: ast::ItemKind::MacCall(mac_placeholder()),
tokens: None,
})])))
}
}smallvec![Box::new(ast::Item {
69 id,
70 span,
71 vis,
72 attrs,
73 kind: ast::ItemKind::MacCall(mac_placeholder()),
74 tokens: None,
75 })]),
76 AstFragmentKind::TraitItems => {
77 AstFragment::TraitItems({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(Box::new(ast::AssocItem {
id,
span,
vis,
attrs,
kind: ast::AssocItemKind::MacCall(mac_placeholder()),
tokens: None,
}));
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[Box::new(ast::AssocItem {
id,
span,
vis,
attrs,
kind: ast::AssocItemKind::MacCall(mac_placeholder()),
tokens: None,
})])))
}
}smallvec![Box::new(ast::AssocItem {
78 id,
79 span,
80 vis,
81 attrs,
82 kind: ast::AssocItemKind::MacCall(mac_placeholder()),
83 tokens: None,
84 })])
85 }
86 AstFragmentKind::ImplItems => AstFragment::ImplItems({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(Box::new(ast::AssocItem {
id,
span,
vis,
attrs,
kind: ast::AssocItemKind::MacCall(mac_placeholder()),
tokens: None,
}));
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[Box::new(ast::AssocItem {
id,
span,
vis,
attrs,
kind: ast::AssocItemKind::MacCall(mac_placeholder()),
tokens: None,
})])))
}
}smallvec![Box::new(ast::AssocItem {
87 id,
88 span,
89 vis,
90 attrs,
91 kind: ast::AssocItemKind::MacCall(mac_placeholder()),
92 tokens: None,
93 })]),
94 AstFragmentKind::TraitImplItems => {
95 AstFragment::TraitImplItems({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(Box::new(ast::AssocItem {
id,
span,
vis,
attrs,
kind: ast::AssocItemKind::MacCall(mac_placeholder()),
tokens: None,
}));
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[Box::new(ast::AssocItem {
id,
span,
vis,
attrs,
kind: ast::AssocItemKind::MacCall(mac_placeholder()),
tokens: None,
})])))
}
}smallvec![Box::new(ast::AssocItem {
96 id,
97 span,
98 vis,
99 attrs,
100 kind: ast::AssocItemKind::MacCall(mac_placeholder()),
101 tokens: None,
102 })])
103 }
104 AstFragmentKind::ForeignItems => {
105 AstFragment::ForeignItems({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(Box::new(ast::ForeignItem {
id,
span,
vis,
attrs,
kind: ast::ForeignItemKind::MacCall(mac_placeholder()),
tokens: None,
}));
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[Box::new(ast::ForeignItem {
id,
span,
vis,
attrs,
kind: ast::ForeignItemKind::MacCall(mac_placeholder()),
tokens: None,
})])))
}
}smallvec![Box::new(ast::ForeignItem {
106 id,
107 span,
108 vis,
109 attrs,
110 kind: ast::ForeignItemKind::MacCall(mac_placeholder()),
111 tokens: None,
112 })])
113 }
114 AstFragmentKind::Pat => AstFragment::Pat(Box::new(ast::Pat {
115 id,
116 span,
117 kind: ast::PatKind::MacCall(mac_placeholder()),
118 tokens: None,
119 })),
120 AstFragmentKind::Ty => AstFragment::Ty(Box::new(ast::Ty {
121 id,
122 span,
123 kind: ast::TyKind::MacCall(mac_placeholder()),
124 tokens: None,
125 })),
126 AstFragmentKind::Stmts => AstFragment::Stmts({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push({
let mac =
Box::new(ast::MacCallStmt {
mac: mac_placeholder(),
style: ast::MacStmtStyle::Braces,
attrs: ast::AttrVec::new(),
tokens: None,
});
ast::Stmt { id, span, kind: ast::StmtKind::MacCall(mac) }
});
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[{
let mac =
Box::new(ast::MacCallStmt {
mac: mac_placeholder(),
style: ast::MacStmtStyle::Braces,
attrs: ast::AttrVec::new(),
tokens: None,
});
ast::Stmt { id, span, kind: ast::StmtKind::MacCall(mac) }
}])))
}
}smallvec![{
127 let mac = Box::new(ast::MacCallStmt {
128 mac: mac_placeholder(),
129 style: ast::MacStmtStyle::Braces,
130 attrs: ast::AttrVec::new(),
131 tokens: None,
132 });
133 ast::Stmt { id, span, kind: ast::StmtKind::MacCall(mac) }
134 }]),
135 AstFragmentKind::Arms => AstFragment::Arms({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(ast::Arm {
attrs: Default::default(),
body: Some(expr_placeholder()),
guard: None,
id,
pat: pat(),
span,
is_placeholder: true,
});
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[ast::Arm {
attrs: Default::default(),
body: Some(expr_placeholder()),
guard: None,
id,
pat: pat(),
span,
is_placeholder: true,
}])))
}
}smallvec![ast::Arm {
136 attrs: Default::default(),
137 body: Some(expr_placeholder()),
138 guard: None,
139 id,
140 pat: pat(),
141 span,
142 is_placeholder: true,
143 }]),
144 AstFragmentKind::ExprFields => AstFragment::ExprFields({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(ast::ExprField {
attrs: Default::default(),
expr: expr_placeholder(),
id,
ident,
is_shorthand: false,
span,
is_placeholder: true,
});
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[ast::ExprField {
attrs: Default::default(),
expr: expr_placeholder(),
id,
ident,
is_shorthand: false,
span,
is_placeholder: true,
}])))
}
}smallvec![ast::ExprField {
145 attrs: Default::default(),
146 expr: expr_placeholder(),
147 id,
148 ident,
149 is_shorthand: false,
150 span,
151 is_placeholder: true,
152 }]),
153 AstFragmentKind::PatFields => AstFragment::PatFields({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(ast::PatField {
attrs: Default::default(),
id,
ident,
is_shorthand: false,
pat: pat(),
span,
is_placeholder: true,
});
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[ast::PatField {
attrs: Default::default(),
id,
ident,
is_shorthand: false,
pat: pat(),
span,
is_placeholder: true,
}])))
}
}smallvec![ast::PatField {
154 attrs: Default::default(),
155 id,
156 ident,
157 is_shorthand: false,
158 pat: pat(),
159 span,
160 is_placeholder: true,
161 }]),
162 AstFragmentKind::GenericParams => AstFragment::GenericParams({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push({
ast::GenericParam {
attrs: Default::default(),
bounds: Default::default(),
id,
ident,
is_placeholder: true,
kind: ast::GenericParamKind::Lifetime,
colon_span: None,
}
});
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[{
ast::GenericParam {
attrs: Default::default(),
bounds: Default::default(),
id,
ident,
is_placeholder: true,
kind: ast::GenericParamKind::Lifetime,
colon_span: None,
}
}])))
}
}smallvec![{
163 ast::GenericParam {
164 attrs: Default::default(),
165 bounds: Default::default(),
166 id,
167 ident,
168 is_placeholder: true,
169 kind: ast::GenericParamKind::Lifetime,
170 colon_span: None,
171 }
172 }]),
173 AstFragmentKind::Params => AstFragment::Params({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(ast::Param {
attrs: Default::default(),
id,
pat: pat(),
span,
ty: ty(),
is_placeholder: true,
});
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[ast::Param {
attrs: Default::default(),
id,
pat: pat(),
span,
ty: ty(),
is_placeholder: true,
}])))
}
}smallvec![ast::Param {
174 attrs: Default::default(),
175 id,
176 pat: pat(),
177 span,
178 ty: ty(),
179 is_placeholder: true,
180 }]),
181 AstFragmentKind::FieldDefs => AstFragment::FieldDefs({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(ast::FieldDef {
attrs: Default::default(),
id,
ident: None,
span,
ty: ty(),
vis,
is_placeholder: true,
safety: Safety::Default,
default: None,
});
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[ast::FieldDef {
attrs: Default::default(),
id,
ident: None,
span,
ty: ty(),
vis,
is_placeholder: true,
safety: Safety::Default,
default: None,
}])))
}
}smallvec![ast::FieldDef {
182 attrs: Default::default(),
183 id,
184 ident: None,
185 span,
186 ty: ty(),
187 vis,
188 is_placeholder: true,
189 safety: Safety::Default,
190 default: None,
191 }]),
192 AstFragmentKind::Variants => AstFragment::Variants({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(ast::Variant {
attrs: Default::default(),
data: ast::VariantData::Struct {
fields: Default::default(),
recovered: ast::Recovered::No,
},
disr_expr: None,
id,
ident,
span,
vis,
is_placeholder: true,
});
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[ast::Variant {
attrs: Default::default(),
data: ast::VariantData::Struct {
fields: Default::default(),
recovered: ast::Recovered::No,
},
disr_expr: None,
id,
ident,
span,
vis,
is_placeholder: true,
}])))
}
}smallvec![ast::Variant {
193 attrs: Default::default(),
194 data: ast::VariantData::Struct {
195 fields: Default::default(),
196 recovered: ast::Recovered::No
197 },
198 disr_expr: None,
199 id,
200 ident,
201 span,
202 vis,
203 is_placeholder: true,
204 }]),
205 AstFragmentKind::WherePredicates => {
206 AstFragment::WherePredicates({
let count = 0usize + 1usize;
let mut vec = ::smallvec::SmallVec::new();
if count <= vec.inline_size() {
vec.push(ast::WherePredicate {
attrs: Default::default(),
id,
span,
kind: ast::WherePredicateKind::BoundPredicate(ast::WhereBoundPredicate {
bound_generic_params: Default::default(),
bounded_ty: ty(),
bounds: Default::default(),
}),
is_placeholder: true,
});
vec
} else {
::smallvec::SmallVec::from_vec(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[ast::WherePredicate {
attrs: Default::default(),
id,
span,
kind: ast::WherePredicateKind::BoundPredicate(ast::WhereBoundPredicate {
bound_generic_params: Default::default(),
bounded_ty: ty(),
bounds: Default::default(),
}),
is_placeholder: true,
}])))
}
}smallvec![ast::WherePredicate {
207 attrs: Default::default(),
208 id,
209 span,
210 kind: ast::WherePredicateKind::BoundPredicate(ast::WhereBoundPredicate {
211 bound_generic_params: Default::default(),
212 bounded_ty: ty(),
213 bounds: Default::default(),
214 }),
215 is_placeholder: true,
216 }])
217 }
218 }
219}
220
221pub(crate) struct PlaceholderExpander {
222 expanded_fragments: FxHashMap<ast::NodeId, AstFragment>,
223}
224
225impl PlaceholderExpander {
226 pub(crate) fn with_capacity(capacity: usize) -> Self {
227 PlaceholderExpander {
228 expanded_fragments: FxHashMap::with_capacity_and_hasher(capacity, Default::default()),
229 }
230 }
231
232 pub(crate) fn add(&mut self, id: ast::NodeId, mut fragment: AstFragment) {
233 fragment.mut_visit_with(self);
234 self.expanded_fragments.insert(id, fragment);
235 }
236
237 fn remove(&mut self, id: ast::NodeId) -> AstFragment {
238 self.expanded_fragments.remove(&id).unwrap()
239 }
240}
241
242impl MutVisitor for PlaceholderExpander {
243 fn flat_map_arm(&mut self, arm: ast::Arm) -> SmallVec<[ast::Arm; 1]> {
244 if arm.is_placeholder {
245 self.remove(arm.id).make_arms()
246 } else {
247 walk_flat_map_arm(self, arm)
248 }
249 }
250
251 fn flat_map_expr_field(&mut self, field: ast::ExprField) -> SmallVec<[ast::ExprField; 1]> {
252 if field.is_placeholder {
253 self.remove(field.id).make_expr_fields()
254 } else {
255 walk_flat_map_expr_field(self, field)
256 }
257 }
258
259 fn flat_map_pat_field(&mut self, fp: ast::PatField) -> SmallVec<[ast::PatField; 1]> {
260 if fp.is_placeholder {
261 self.remove(fp.id).make_pat_fields()
262 } else {
263 walk_flat_map_pat_field(self, fp)
264 }
265 }
266
267 fn flat_map_generic_param(
268 &mut self,
269 param: ast::GenericParam,
270 ) -> SmallVec<[ast::GenericParam; 1]> {
271 if param.is_placeholder {
272 self.remove(param.id).make_generic_params()
273 } else {
274 walk_flat_map_generic_param(self, param)
275 }
276 }
277
278 fn flat_map_param(&mut self, p: ast::Param) -> SmallVec<[ast::Param; 1]> {
279 if p.is_placeholder {
280 self.remove(p.id).make_params()
281 } else {
282 walk_flat_map_param(self, p)
283 }
284 }
285
286 fn flat_map_field_def(&mut self, sf: ast::FieldDef) -> SmallVec<[ast::FieldDef; 1]> {
287 if sf.is_placeholder {
288 self.remove(sf.id).make_field_defs()
289 } else {
290 walk_flat_map_field_def(self, sf)
291 }
292 }
293
294 fn flat_map_variant(&mut self, variant: ast::Variant) -> SmallVec<[ast::Variant; 1]> {
295 if variant.is_placeholder {
296 self.remove(variant.id).make_variants()
297 } else {
298 walk_flat_map_variant(self, variant)
299 }
300 }
301
302 fn flat_map_where_predicate(
303 &mut self,
304 predicate: ast::WherePredicate,
305 ) -> SmallVec<[ast::WherePredicate; 1]> {
306 if predicate.is_placeholder {
307 self.remove(predicate.id).make_where_predicates()
308 } else {
309 walk_flat_map_where_predicate(self, predicate)
310 }
311 }
312
313 fn flat_map_item(&mut self, item: Box<ast::Item>) -> SmallVec<[Box<ast::Item>; 1]> {
314 match item.kind {
315 ast::ItemKind::MacCall(_) => self.remove(item.id).make_items(),
316 _ => walk_flat_map_item(self, item),
317 }
318 }
319
320 fn flat_map_assoc_item(
321 &mut self,
322 item: Box<ast::AssocItem>,
323 ctxt: AssocCtxt,
324 ) -> SmallVec<[Box<ast::AssocItem>; 1]> {
325 match item.kind {
326 ast::AssocItemKind::MacCall(_) => {
327 let it = self.remove(item.id);
328 match ctxt {
329 AssocCtxt::Trait => it.make_trait_items(),
330 AssocCtxt::Impl { of_trait: false } => it.make_impl_items(),
331 AssocCtxt::Impl { of_trait: true } => it.make_trait_impl_items(),
332 }
333 }
334 _ => walk_flat_map_assoc_item(self, item, ctxt),
335 }
336 }
337
338 fn flat_map_foreign_item(
339 &mut self,
340 item: Box<ast::ForeignItem>,
341 ) -> SmallVec<[Box<ast::ForeignItem>; 1]> {
342 match item.kind {
343 ast::ForeignItemKind::MacCall(_) => self.remove(item.id).make_foreign_items(),
344 _ => walk_flat_map_foreign_item(self, item),
345 }
346 }
347
348 fn visit_expr(&mut self, expr: &mut ast::Expr) {
349 match expr.kind {
350 ast::ExprKind::MacCall(_) => *expr = *self.remove(expr.id).make_expr(),
351 _ => walk_expr(self, expr),
352 }
353 }
354
355 fn visit_method_receiver_expr(&mut self, expr: &mut ast::Expr) {
356 match expr.kind {
357 ast::ExprKind::MacCall(_) => *expr = *self.remove(expr.id).make_method_receiver_expr(),
358 _ => walk_expr(self, expr),
359 }
360 }
361
362 fn filter_map_expr(&mut self, expr: Box<ast::Expr>) -> Option<Box<ast::Expr>> {
363 match expr.kind {
364 ast::ExprKind::MacCall(_) => self.remove(expr.id).make_opt_expr(),
365 _ => walk_filter_map_expr(self, expr),
366 }
367 }
368
369 fn flat_map_stmt(&mut self, stmt: ast::Stmt) -> SmallVec<[ast::Stmt; 1]> {
370 let (style, mut stmts) = match stmt.kind {
371 ast::StmtKind::MacCall(mac) => (mac.style, self.remove(stmt.id).make_stmts()),
372 _ => return walk_flat_map_stmt(self, stmt),
373 };
374
375 if style == ast::MacStmtStyle::Semicolon {
376 let empty_stmt =
398 ast::Stmt { id: ast::DUMMY_NODE_ID, kind: ast::StmtKind::Empty, span: DUMMY_SP };
399
400 if let Some(stmt) = stmts.pop() {
401 if stmt.has_trailing_semicolon() {
402 stmts.push(stmt);
403 stmts.push(empty_stmt);
404 } else {
405 stmts.push(stmt.add_trailing_semicolon());
406 }
407 } else {
408 stmts.push(empty_stmt);
409 }
410 }
411
412 stmts
413 }
414
415 fn visit_pat(&mut self, pat: &mut ast::Pat) {
416 match pat.kind {
417 ast::PatKind::MacCall(_) => *pat = *self.remove(pat.id).make_pat(),
418 _ => walk_pat(self, pat),
419 }
420 }
421
422 fn visit_ty(&mut self, ty: &mut ast::Ty) {
423 match ty.kind {
424 ast::TyKind::MacCall(_) => *ty = *self.remove(ty.id).make_ty(),
425 _ => walk_ty(self, ty),
426 }
427 }
428
429 fn visit_crate(&mut self, krate: &mut ast::Crate) {
430 if krate.is_placeholder {
431 *krate = self.remove(krate.id).make_crate();
432 } else {
433 walk_crate(self, krate)
434 }
435 }
436}