@@ -49,6 +49,10 @@ pub enum Error {
49
49
#[ diagnostic( code( "Qasm3.Lex.UnterminatedString" ) ) ]
50
50
UnterminatedString ( #[ label] Span ) ,
51
51
52
+ #[ error( "string literal with an invalid escape sequence" ) ]
53
+ #[ diagnostic( code( "Qasm3.Lex.InvalidEscapeSequence" ) ) ]
54
+ InvalidEscapeSequence ( #[ label] Span ) ,
55
+
52
56
#[ error( "unrecognized character `{0}`" ) ]
53
57
#[ diagnostic( code( "Qasm3.Lex.UnknownChar" ) ) ]
54
58
Unknown ( char , #[ label] Span ) ,
@@ -64,6 +68,7 @@ impl Error {
64
68
Self :: IncompleteEof ( expected, token, span + offset)
65
69
}
66
70
Self :: UnterminatedString ( span) => Self :: UnterminatedString ( span + offset) ,
71
+ Self :: InvalidEscapeSequence ( span) => Self :: InvalidEscapeSequence ( span + offset) ,
67
72
Self :: Unknown ( c, span) => Self :: Unknown ( c, span + offset) ,
68
73
}
69
74
}
@@ -73,6 +78,7 @@ impl Error {
73
78
Error :: Incomplete ( _, _, _, s)
74
79
| Error :: IncompleteEof ( _, _, s)
75
80
| Error :: UnterminatedString ( s)
81
+ | Error :: InvalidEscapeSequence ( s)
76
82
| Error :: Unknown ( _, s) => s,
77
83
}
78
84
}
@@ -82,6 +88,7 @@ impl Error {
82
88
#[ derive( Clone , Copy , Debug , Eq , PartialEq , Sequence ) ]
83
89
pub enum TokenKind {
84
90
Annotation ,
91
+ Pragma ,
85
92
Keyword ( Keyword ) ,
86
93
Type ( Type ) ,
87
94
@@ -118,7 +125,6 @@ pub enum TokenKind {
118
125
PlusPlus ,
119
126
/// `->`
120
127
Arrow ,
121
- At ,
122
128
123
129
// Operators,
124
130
ClosedBinOp ( ClosedBinOp ) ,
@@ -141,6 +147,7 @@ impl Display for TokenKind {
141
147
fn fmt ( & self , f : & mut Formatter ) -> fmt:: Result {
142
148
match self {
143
149
TokenKind :: Annotation => write ! ( f, "annotation" ) ,
150
+ TokenKind :: Pragma => write ! ( f, "pragma" ) ,
144
151
TokenKind :: Keyword ( keyword) => write ! ( f, "keyword `{keyword}`" ) ,
145
152
TokenKind :: Type ( type_) => write ! ( f, "keyword `{type_}`" ) ,
146
153
TokenKind :: GPhase => write ! ( f, "gphase" ) ,
@@ -166,7 +173,6 @@ impl Display for TokenKind {
166
173
TokenKind :: Comma => write ! ( f, "`,`" ) ,
167
174
TokenKind :: PlusPlus => write ! ( f, "`++`" ) ,
168
175
TokenKind :: Arrow => write ! ( f, "`->`" ) ,
169
- TokenKind :: At => write ! ( f, "`@`" ) ,
170
176
TokenKind :: ClosedBinOp ( op) => write ! ( f, "`{op}`" ) ,
171
177
TokenKind :: BinOpEq ( op) => write ! ( f, "`{op}=`" ) ,
172
178
TokenKind :: ComparisonOp ( op) => write ! ( f, "`{op}`" ) ,
@@ -273,7 +279,6 @@ impl FromStr for Type {
273
279
#[ derive( Clone , Copy , Debug , Eq , PartialEq , Sequence ) ]
274
280
pub enum Literal {
275
281
Bitstring ,
276
- Boolean ,
277
282
Float ,
278
283
Imaginary ,
279
284
Integer ( Radix ) ,
@@ -285,7 +290,6 @@ impl Display for Literal {
285
290
fn fmt ( & self , f : & mut Formatter < ' _ > ) -> fmt:: Result {
286
291
f. write_str ( match self {
287
292
Literal :: Bitstring => "bitstring" ,
288
- Literal :: Boolean => "boolean" ,
289
293
Literal :: Float => "float" ,
290
294
Literal :: Imaginary => "imaginary" ,
291
295
Literal :: Integer ( _) => "integer" ,
@@ -459,6 +463,23 @@ impl<'a> Lexer<'a> {
459
463
tokens. next ( ) . map ( |i| i. kind )
460
464
}
461
465
466
+ /// Consumes the characters while they satisfy `f`. Returns the last character eaten, if any.
467
+ fn eat_while ( & mut self , mut f : impl FnMut ( raw:: TokenKind ) -> bool ) -> Option < raw:: TokenKind > {
468
+ let mut last_eaten: Option < raw:: Token > = None ;
469
+ loop {
470
+ let t = self . tokens . next_if ( |t| f ( t. kind ) ) ;
471
+ if t. is_none ( ) {
472
+ return last_eaten. map ( |t| t. kind ) ;
473
+ }
474
+ last_eaten = t;
475
+ }
476
+ }
477
+
478
+ fn eat_to_end_of_line ( & mut self ) {
479
+ self . eat_while ( |t| t != raw:: TokenKind :: Newline ) ;
480
+ self . next_if_eq ( raw:: TokenKind :: Newline ) ;
481
+ }
482
+
462
483
/// Consumes a list of tokens zero or more times.
463
484
fn kleen_star ( & mut self , tokens : & [ raw:: TokenKind ] , complete : TokenKind ) -> Result < ( ) , Error > {
464
485
let mut iter = tokens. iter ( ) ;
@@ -471,6 +492,7 @@ impl<'a> Lexer<'a> {
471
492
Ok ( ( ) )
472
493
}
473
494
495
+ #[ allow( clippy:: too_many_lines) ]
474
496
fn cook ( & mut self , token : & raw:: Token ) -> Result < Option < Token > , Error > {
475
497
let kind = match token. kind {
476
498
raw:: TokenKind :: Bitstring { terminated : true } => {
@@ -487,7 +509,13 @@ impl<'a> Lexer<'a> {
487
509
}
488
510
raw:: TokenKind :: Ident => {
489
511
let ident = & self . input [ ( token. offset as usize ) ..( self . offset ( ) as usize ) ] ;
490
- Ok ( Some ( Self :: ident ( ident) ) )
512
+ let cooked_ident = Self :: ident ( ident) ;
513
+ if matches ! ( cooked_ident, TokenKind :: Keyword ( Keyword :: Pragma ) ) {
514
+ self . eat_to_end_of_line ( ) ;
515
+ Ok ( Some ( TokenKind :: Pragma ) )
516
+ } else {
517
+ Ok ( Some ( cooked_ident) )
518
+ }
491
519
}
492
520
raw:: TokenKind :: HardwareQubit => Ok ( Some ( TokenKind :: HardwareQubit ) ) ,
493
521
raw:: TokenKind :: LiteralFragment ( _) => {
@@ -525,6 +553,26 @@ impl<'a> Lexer<'a> {
525
553
_ => Ok ( Some ( number. into ( ) ) ) ,
526
554
}
527
555
}
556
+ raw:: TokenKind :: Single ( Single :: Sharp ) => {
557
+ let complete = TokenKind :: Pragma ;
558
+ self . expect ( raw:: TokenKind :: Ident , complete) ?;
559
+ let ident = & self . input [ ( token. offset as usize + 1 ) ..( self . offset ( ) as usize ) ] ;
560
+ if matches ! ( Self :: ident( ident) , TokenKind :: Keyword ( Keyword :: Pragma ) ) {
561
+ self . eat_to_end_of_line ( ) ;
562
+ Ok ( Some ( complete) )
563
+ } else {
564
+ let span = Span {
565
+ lo : token. offset ,
566
+ hi : self . offset ( ) ,
567
+ } ;
568
+ Err ( Error :: Incomplete (
569
+ raw:: TokenKind :: Ident ,
570
+ complete,
571
+ raw:: TokenKind :: Ident ,
572
+ span,
573
+ ) )
574
+ }
575
+ }
528
576
raw:: TokenKind :: Single ( single) => self . single ( single) . map ( Some ) ,
529
577
raw:: TokenKind :: String { terminated : true } => {
530
578
Ok ( Some ( TokenKind :: Literal ( Literal :: String ) ) )
@@ -565,7 +613,13 @@ impl<'a> Lexer<'a> {
565
613
Ok ( self . closed_bin_op ( ClosedBinOp :: Amp ) )
566
614
}
567
615
}
568
- Single :: At => Ok ( TokenKind :: At ) ,
616
+ Single :: At => {
617
+ // AnnotationKeyword: '@' Identifier ('.' Identifier)* -> pushMode(EAT_TO_LINE_END);
618
+ let complete = TokenKind :: Annotation ;
619
+ self . expect ( raw:: TokenKind :: Ident , complete) ;
620
+ self . eat_to_end_of_line ( ) ;
621
+ Ok ( complete)
622
+ }
569
623
Single :: Bang => {
570
624
if self . next_if_eq_single ( Single :: Eq ) {
571
625
Ok ( TokenKind :: ComparisonOp ( ComparisonOp :: BangEq ) )
@@ -627,6 +681,7 @@ impl<'a> Lexer<'a> {
627
681
}
628
682
}
629
683
Single :: Semi => Ok ( TokenKind :: Semicolon ) ,
684
+ Single :: Sharp => unreachable ! ( ) ,
630
685
Single :: Slash => Ok ( self . closed_bin_op ( ClosedBinOp :: Slash ) ) ,
631
686
Single :: Star => {
632
687
if self . next_if_eq_single ( Single :: Star ) {
@@ -659,7 +714,6 @@ impl<'a> Lexer<'a> {
659
714
"delay" => TokenKind :: Delay ,
660
715
"reset" => TokenKind :: Reset ,
661
716
"measure" => TokenKind :: Measure ,
662
- "false" | "true" => TokenKind :: Literal ( Literal :: Boolean ) ,
663
717
ident => {
664
718
if let Ok ( keyword) = ident. parse :: < Keyword > ( ) {
665
719
TokenKind :: Keyword ( keyword)
0 commit comments