@@ -139,13 +139,20 @@ struct MatcherPos {
139
139
sep : Option < Token > ,
140
140
idx : usize ,
141
141
up : Option < Box < MatcherPos > > ,
142
- matches : Vec < Vec < Rc < NamedMatch > > > ,
142
+ matches : Vec < Rc < Vec < NamedMatch > > > ,
143
143
match_lo : usize ,
144
144
match_cur : usize ,
145
145
match_hi : usize ,
146
146
sp_lo : BytePos ,
147
147
}
148
148
149
+ impl MatcherPos {
150
+ fn push_match ( & mut self , idx : usize , m : NamedMatch ) {
151
+ let matches = Rc :: make_mut ( & mut self . matches [ idx] ) ;
152
+ matches. push ( m) ;
153
+ }
154
+ }
155
+
149
156
pub type NamedParseResult = ParseResult < HashMap < Ident , Rc < NamedMatch > > > ;
150
157
151
158
pub fn count_names ( ms : & [ TokenTree ] ) -> usize {
@@ -199,14 +206,15 @@ fn initial_matcher_pos(ms: Vec<TokenTree>, lo: BytePos) -> Box<MatcherPos> {
199
206
/// only on the nesting depth of `ast::TTSeq`s in the originating
200
207
/// token tree it was derived from.
201
208
209
+ #[ derive( Debug , Clone ) ]
202
210
pub enum NamedMatch {
203
- MatchedSeq ( Vec < Rc < NamedMatch > > , syntax_pos:: Span ) ,
211
+ MatchedSeq ( Rc < Vec < NamedMatch > > , syntax_pos:: Span ) ,
204
212
MatchedNonterminal ( Rc < Nonterminal > )
205
213
}
206
214
207
- fn nameize < I : Iterator < Item =Rc < NamedMatch > > > ( sess : & ParseSess , ms : & [ TokenTree ] , mut res : I )
215
+ fn nameize < I : Iterator < Item =NamedMatch > > ( sess : & ParseSess , ms : & [ TokenTree ] , mut res : I )
208
216
-> NamedParseResult {
209
- fn n_rec < I : Iterator < Item =Rc < NamedMatch > > > ( sess : & ParseSess , m : & TokenTree , mut res : & mut I ,
217
+ fn n_rec < I : Iterator < Item =NamedMatch > > ( sess : & ParseSess , m : & TokenTree , mut res : & mut I ,
210
218
ret_val : & mut HashMap < Ident , Rc < NamedMatch > > )
211
219
-> Result < ( ) , ( syntax_pos:: Span , String ) > {
212
220
match * m {
@@ -228,7 +236,8 @@ fn nameize<I: Iterator<Item=Rc<NamedMatch>>>(sess: &ParseSess, ms: &[TokenTree],
228
236
TokenTree :: MetaVarDecl ( sp, bind_name, _) => {
229
237
match ret_val. entry ( bind_name) {
230
238
Vacant ( spot) => {
231
- spot. insert ( res. next ( ) . unwrap ( ) ) ;
239
+ // FIXME(simulacrum): Don't construct Rc here
240
+ spot. insert ( Rc :: new ( res. next ( ) . unwrap ( ) ) ) ;
232
241
}
233
242
Occupied ( ..) => {
234
243
return Err ( ( sp, format ! ( "duplicated bind name: {}" , bind_name) ) )
@@ -280,8 +289,8 @@ fn token_name_eq(t1 : &Token, t2 : &Token) -> bool {
280
289
}
281
290
}
282
291
283
- fn create_matches ( len : usize ) -> Vec < Vec < Rc < NamedMatch > > > {
284
- ( 0 ..len) . into_iter ( ) . map ( |_| Vec :: new ( ) ) . collect ( )
292
+ fn create_matches ( len : usize ) -> Vec < Rc < Vec < NamedMatch > > > {
293
+ ( 0 ..len) . into_iter ( ) . map ( |_| Rc :: new ( Vec :: new ( ) ) ) . collect ( )
285
294
}
286
295
287
296
fn inner_parse_loop ( sess : & ParseSess ,
@@ -320,15 +329,10 @@ fn inner_parse_loop(sess: &ParseSess,
320
329
// update matches (the MBE "parse tree") by appending
321
330
// each tree as a subtree.
322
331
323
- // I bet this is a perf problem: we're preemptively
324
- // doing a lot of array work that will get thrown away
325
- // most of the time.
326
-
327
332
// Only touch the binders we have actually bound
328
333
for idx in ei. match_lo ..ei. match_hi {
329
334
let sub = ei. matches [ idx] . clone ( ) ;
330
- new_pos. matches [ idx]
331
- . push ( Rc :: new ( MatchedSeq ( sub, Span { lo : ei. sp_lo , ..span } ) ) ) ;
335
+ new_pos. push_match ( idx, MatchedSeq ( sub, Span { lo : ei. sp_lo , ..span } ) ) ;
332
336
}
333
337
334
338
new_pos. match_cur = ei. match_hi ;
@@ -362,7 +366,7 @@ fn inner_parse_loop(sess: &ParseSess,
362
366
new_ei. match_cur += seq. num_captures ;
363
367
new_ei. idx += 1 ;
364
368
for idx in ei. match_cur ..ei. match_cur + seq. num_captures {
365
- new_ei. matches [ idx] . push ( Rc :: new ( MatchedSeq ( vec ! [ ] , sp) ) ) ;
369
+ new_ei. push_match ( idx, MatchedSeq ( Rc :: new ( vec ! [ ] ) , sp) ) ;
366
370
}
367
371
cur_eis. push ( new_ei) ;
368
372
}
@@ -446,7 +450,9 @@ pub fn parse(sess: &ParseSess,
446
450
/* error messages here could be improved with links to orig. rules */
447
451
if token_name_eq ( & parser. token , & token:: Eof ) {
448
452
if eof_eis. len ( ) == 1 {
449
- let matches = eof_eis[ 0 ] . matches . iter_mut ( ) . map ( |mut dv| dv. pop ( ) . unwrap ( ) ) ;
453
+ let matches = eof_eis[ 0 ] . matches . iter_mut ( ) . map ( |mut dv| {
454
+ Rc :: make_mut ( dv) . pop ( ) . unwrap ( )
455
+ } ) ;
450
456
return nameize ( sess, ms, matches) ;
451
457
} else if eof_eis. len ( ) > 1 {
452
458
return Error ( parser. span , "ambiguity: multiple successful parses" . to_string ( ) ) ;
@@ -479,8 +485,8 @@ pub fn parse(sess: &ParseSess,
479
485
let mut ei = bb_eis. pop ( ) . unwrap ( ) ;
480
486
if let TokenTree :: MetaVarDecl ( span, _, ident) = ei. top_elts . get_tt ( ei. idx ) {
481
487
let match_cur = ei. match_cur ;
482
- ei. matches [ match_cur ] . push ( Rc :: new ( MatchedNonterminal (
483
- Rc :: new ( parse_nt ( & mut parser, span, & ident. name . as_str ( ) ) ) ) ) ) ;
488
+ ei. push_match ( match_cur ,
489
+ MatchedNonterminal ( Rc :: new ( parse_nt ( & mut parser, span, & ident. name . as_str ( ) ) ) ) ) ;
484
490
ei. idx += 1 ;
485
491
ei. match_cur += 1 ;
486
492
} else {
0 commit comments