@@ -561,6 +561,45 @@ impl Parser {
561
561
}
562
562
}
563
563
564
+ // Expect and consume a `|`. If `||` is seen, replace it with a single
565
+ // `|` and continue. If a `|` is not seen, signal an error.
566
+ fn expect_or ( & self ) {
567
+ match * self . token {
568
+ token:: BINOP ( token:: OR ) => self . bump ( ) ,
569
+ token:: OROR => {
570
+ self . replace_token ( token:: BINOP ( token:: OR ) ,
571
+ self . span . lo + BytePos ( 1 ) ,
572
+ self . span . hi )
573
+ }
574
+ _ => {
575
+ let found_token = self . token_to_str ( & token:: BINOP ( token:: OR ) ) ;
576
+ self . fatal ( format ! ( "expected `{}`, found `{}`" ,
577
+ found_token,
578
+ self . this_token_to_str( ) ) )
579
+ }
580
+ }
581
+ }
582
+
583
+ // Parse a sequence bracketed by `|` and `|`, stopping before the `|`.
584
+ fn parse_seq_to_before_or < T > ( & self ,
585
+ sep : & token:: Token ,
586
+ f : & fn ( & Parser ) -> T )
587
+ -> ~[ T ] {
588
+ let mut first = true ;
589
+ let mut vector = ~[ ] ;
590
+ while * self . token != token:: BINOP ( token:: OR ) &&
591
+ * self . token != token:: OROR {
592
+ if first {
593
+ first = false
594
+ } else {
595
+ self . expect ( sep)
596
+ }
597
+
598
+ vector. push ( f ( self ) )
599
+ }
600
+ vector
601
+ }
602
+
564
603
// expect and consume a GT. if a >> is seen, replace it
565
604
// with a single > and continue. If a GT is not seen,
566
605
// signal an error.
@@ -761,11 +800,33 @@ impl Parser {
761
800
get_ident_interner ( ) . get ( id. name )
762
801
}
763
802
764
- // is this one of the keywords that signals a closure type?
765
- pub fn token_is_closure_keyword ( & self , tok : & token:: Token ) -> bool {
766
- token:: is_keyword ( keywords:: Unsafe , tok) ||
767
- token:: is_keyword ( keywords:: Once , tok) ||
768
- token:: is_keyword ( keywords:: Fn , tok)
803
+ // Is the current token one of the keywords that signals a bare function
804
+ // type?
805
+ pub fn token_is_bare_fn_keyword ( & self ) -> bool {
806
+ if token:: is_keyword ( keywords:: Fn , self . token ) {
807
+ return true
808
+ }
809
+
810
+ if token:: is_keyword ( keywords:: Unsafe , self . token ) ||
811
+ token:: is_keyword ( keywords:: Once , self . token ) {
812
+ return self . look_ahead ( 1 , |t| token:: is_keyword ( keywords:: Fn , t) )
813
+ }
814
+
815
+ false
816
+ }
817
+
818
+ // Is the current token one of the keywords that signals a closure type?
819
+ pub fn token_is_closure_keyword ( & self ) -> bool {
820
+ token:: is_keyword ( keywords:: Unsafe , self . token ) ||
821
+ token:: is_keyword ( keywords:: Once , self . token )
822
+ }
823
+
824
+ // Is the current token one of the keywords that signals an old-style
825
+ // closure type (with explicit sigil)?
826
+ pub fn token_is_old_style_closure_keyword ( & self ) -> bool {
827
+ token:: is_keyword ( keywords:: Unsafe , self . token ) ||
828
+ token:: is_keyword ( keywords:: Once , self . token ) ||
829
+ token:: is_keyword ( keywords:: Fn , self . token )
769
830
}
770
831
771
832
pub fn token_is_lifetime ( & self , tok : & token:: Token ) -> bool {
@@ -786,15 +847,15 @@ impl Parser {
786
847
pub fn parse_ty_bare_fn ( & self ) -> ty_ {
787
848
/*
788
849
789
- extern "ABI" [unsafe] fn <'lt> (S) -> T
790
- ^~~~^ ^~~~~~~^ ^~~~^ ^~^ ^
791
- | | | | |
792
- | | | | Return type
793
- | | | Argument types
794
- | | Lifetimes
795
- | |
796
- | Purity
797
- ABI
850
+ [ extern "ABI"] [unsafe] fn <'lt> (S) -> T
851
+ ^~~~^ ^~~~~~~^ ^~~~^ ^~^ ^
852
+ | | | | |
853
+ | | | | Return type
854
+ | | | Argument types
855
+ | | Lifetimes
856
+ | |
857
+ | Purity
858
+ ABI
798
859
799
860
*/
800
861
@@ -828,8 +889,8 @@ impl Parser {
828
889
829
890
// parse a ty_closure type
830
891
pub fn parse_ty_closure ( & self ,
831
- sigil : ast:: Sigil ,
832
- region : Option < ast:: Lifetime > )
892
+ opt_sigil : Option < ast:: Sigil > ,
893
+ mut region : Option < ast:: Lifetime > )
833
894
-> ty_ {
834
895
/*
835
896
@@ -852,10 +913,58 @@ impl Parser {
852
913
853
914
let purity = self . parse_unsafety ( ) ;
854
915
let onceness = parse_onceness ( self ) ;
855
- self . expect_keyword ( keywords:: Fn ) ;
856
- let bounds = self . parse_optional_ty_param_bounds ( ) ;
857
916
858
- let ( decl, lifetimes) = self . parse_ty_fn_decl ( ) ;
917
+ let ( sigil, decl, lifetimes, bounds) = match opt_sigil {
918
+ Some ( sigil) => {
919
+ // Old-style closure syntax (`fn(A)->B`).
920
+ self . expect_keyword ( keywords:: Fn ) ;
921
+ let bounds = self . parse_optional_ty_param_bounds ( ) ;
922
+ let ( decl, lifetimes) = self . parse_ty_fn_decl ( ) ;
923
+ ( sigil, decl, lifetimes, bounds)
924
+ }
925
+ None => {
926
+ // New-style closure syntax (`<'lt>|A|:K -> B`).
927
+ let lifetimes = if self . eat ( & token:: LT ) {
928
+ let lifetimes = self . parse_lifetimes ( ) ;
929
+ self . expect_gt ( ) ;
930
+
931
+ // Re-parse the region here. What a hack.
932
+ if region. is_some ( ) {
933
+ self . span_err ( * self . last_span ,
934
+ "lifetime declarations must precede \
935
+ the lifetime associated with a \
936
+ closure") ;
937
+ }
938
+ region = self . parse_opt_lifetime ( ) ;
939
+
940
+ lifetimes
941
+ } else {
942
+ opt_vec:: Empty
943
+ } ;
944
+
945
+ let inputs = if self . eat ( & token:: OROR ) {
946
+ ~[ ]
947
+ } else {
948
+ self . expect_or ( ) ;
949
+ let inputs = self . parse_seq_to_before_or (
950
+ & token:: COMMA ,
951
+ |p| p. parse_arg_general ( false ) ) ;
952
+ self . expect_or ( ) ;
953
+ inputs
954
+ } ;
955
+
956
+ let bounds = self . parse_optional_ty_param_bounds ( ) ;
957
+
958
+ let ( return_style, output) = self . parse_ret_ty ( ) ;
959
+ let decl = ast:: fn_decl {
960
+ inputs : inputs,
961
+ output : output,
962
+ cf : return_style,
963
+ } ;
964
+
965
+ ( BorrowedSigil , decl, lifetimes, bounds)
966
+ }
967
+ } ;
859
968
860
969
return ty_closure ( @TyClosure {
861
970
sigil : sigil,
@@ -1120,13 +1229,23 @@ impl Parser {
1120
1229
// BORROWED POINTER
1121
1230
self . bump ( ) ;
1122
1231
self . parse_borrowed_pointee ( )
1123
- } else if self . eat_keyword ( keywords:: Extern ) {
1124
- // EXTERN FUNCTION
1232
+ } else if self . is_keyword ( keywords:: Extern ) ||
1233
+ self . token_is_bare_fn_keyword ( ) {
1234
+ // BARE FUNCTION
1125
1235
self . parse_ty_bare_fn ( )
1126
- } else if self . token_is_closure_keyword ( self . token ) {
1236
+ } else if self . token_is_closure_keyword ( ) ||
1237
+ * self . token == token:: BINOP ( token:: OR ) ||
1238
+ * self . token == token:: OROR ||
1239
+ * self . token == token:: LT ||
1240
+ self . token_is_lifetime ( self . token ) {
1127
1241
// CLOSURE
1128
- let result = self . parse_ty_closure ( ast:: BorrowedSigil , None ) ;
1129
- self . obsolete ( * self . last_span , ObsoleteBareFnType ) ;
1242
+ //
1243
+ // XXX(pcwalton): Eventually `token::LT` will not unambiguously
1244
+ // introduce a closure, once procs can have lifetime bounds. We
1245
+ // will need to refactor the grammar a little bit at that point.
1246
+
1247
+ let lifetime = self . parse_opt_lifetime ( ) ;
1248
+ let result = self . parse_ty_closure ( None , lifetime) ;
1130
1249
result
1131
1250
} else if self . eat_keyword ( keywords:: Typeof ) {
1132
1251
// TYPEOF
@@ -1161,12 +1280,12 @@ impl Parser {
1161
1280
match * self . token {
1162
1281
token:: LIFETIME ( * ) => {
1163
1282
let lifetime = self . parse_lifetime ( ) ;
1164
- return self . parse_ty_closure ( sigil, Some ( lifetime) ) ;
1283
+ return self . parse_ty_closure ( Some ( sigil) , Some ( lifetime) ) ;
1165
1284
}
1166
1285
1167
1286
token:: IDENT ( * ) => {
1168
- if self . token_is_closure_keyword ( self . token ) {
1169
- return self . parse_ty_closure ( sigil, None ) ;
1287
+ if self . token_is_old_style_closure_keyword ( ) {
1288
+ return self . parse_ty_closure ( Some ( sigil) , None ) ;
1170
1289
}
1171
1290
}
1172
1291
_ => { }
@@ -1187,8 +1306,8 @@ impl Parser {
1187
1306
// look for `&'lt` or `&'foo ` and interpret `foo` as the region name:
1188
1307
let opt_lifetime = self . parse_opt_lifetime ( ) ;
1189
1308
1190
- if self . token_is_closure_keyword ( self . token ) {
1191
- return self . parse_ty_closure ( BorrowedSigil , opt_lifetime) ;
1309
+ if self . token_is_old_style_closure_keyword ( ) {
1310
+ return self . parse_ty_closure ( Some ( BorrowedSigil ) , opt_lifetime) ;
1192
1311
}
1193
1312
1194
1313
let mt = self . parse_mt ( ) ;
@@ -4390,8 +4509,13 @@ impl Parser {
4390
4509
}
4391
4510
}
4392
4511
4393
- // parse a string as an ABI spec on an extern type or module
4512
+ // Parses a string as an ABI spec on an extern type or module. Consumes
4513
+ // the `extern` keyword, if one is found.
4394
4514
fn parse_opt_abis ( & self ) -> Option < AbiSet > {
4515
+ if !self . eat_keyword ( keywords:: Extern ) {
4516
+ return None
4517
+ }
4518
+
4395
4519
match * self . token {
4396
4520
token:: LIT_STR ( s)
4397
4521
| token:: LIT_STR_RAW ( s, _) => {
@@ -4467,7 +4591,7 @@ impl Parser {
4467
4591
} ) ;
4468
4592
}
4469
4593
// either a view item or an item:
4470
- if self . eat_keyword ( keywords:: Extern ) {
4594
+ if self . is_keyword ( keywords:: Extern ) {
4471
4595
let opt_abis = self . parse_opt_abis ( ) ;
4472
4596
4473
4597
if self . eat_keyword ( keywords:: Fn ) {
0 commit comments