@@ -27,12 +27,32 @@ struct Compiler<O: OutputStream = BasicOutputStream> {
27
27
source_path : Option < String > ,
28
28
current_source_location : ast:: Location ,
29
29
current_qualified_path : Option < String > ,
30
- in_loop : bool ,
31
- in_function_def : bool ,
32
- in_async_func : bool ,
30
+ ctx : CompileContext ,
33
31
optimize : u8 ,
34
32
}
35
33
34
+ #[ derive( Clone , Copy ) ]
35
+ struct CompileContext {
36
+ in_loop : bool ,
37
+ func : FunctionContext ,
38
+ }
39
+
40
+ #[ derive( Clone , Copy ) ]
41
+ enum FunctionContext {
42
+ NoFunction ,
43
+ Function ,
44
+ AsyncFunction ,
45
+ }
46
+
47
+ impl CompileContext {
48
+ fn in_func ( self ) -> bool {
49
+ match self . func {
50
+ FunctionContext :: NoFunction => false ,
51
+ _ => true ,
52
+ }
53
+ }
54
+ }
55
+
36
56
/// Compile a given sourcecode into a bytecode object.
37
57
pub fn compile (
38
58
source : & str ,
@@ -125,9 +145,10 @@ impl<O: OutputStream> Compiler<O> {
125
145
source_path : None ,
126
146
current_source_location : ast:: Location :: default ( ) ,
127
147
current_qualified_path : None ,
128
- in_loop : false ,
129
- in_function_def : false ,
130
- in_async_func : false ,
148
+ ctx : CompileContext {
149
+ in_loop : false ,
150
+ func : FunctionContext :: NoFunction ,
151
+ } ,
131
152
optimize,
132
153
}
133
154
}
@@ -510,7 +531,7 @@ impl<O: OutputStream> Compiler<O> {
510
531
}
511
532
}
512
533
Break => {
513
- if !self . in_loop {
534
+ if !self . ctx . in_loop {
514
535
return Err ( CompileError {
515
536
error : CompileErrorType :: InvalidBreak ,
516
537
location : statement. location . clone ( ) ,
@@ -519,7 +540,7 @@ impl<O: OutputStream> Compiler<O> {
519
540
self . emit ( Instruction :: Break ) ;
520
541
}
521
542
Continue => {
522
- if !self . in_loop {
543
+ if !self . ctx . in_loop {
523
544
return Err ( CompileError {
524
545
error : CompileErrorType :: InvalidContinue ,
525
546
location : statement. location . clone ( ) ,
@@ -528,7 +549,7 @@ impl<O: OutputStream> Compiler<O> {
528
549
self . emit ( Instruction :: Continue ) ;
529
550
}
530
551
Return { value } => {
531
- if !self . in_function_def {
552
+ if !self . ctx . in_func ( ) {
532
553
return Err ( CompileError {
533
554
error : CompileErrorType :: InvalidReturn ,
534
555
location : statement. location . clone ( ) ,
@@ -814,17 +835,20 @@ impl<O: OutputStream> Compiler<O> {
814
835
is_async : bool ,
815
836
) -> Result < ( ) , CompileError > {
816
837
// Create bytecode for this function:
817
- // remember to restore self.in_loop to the original after the function is compiled
818
- let was_in_loop = self . in_loop ;
819
- let was_in_function_def = self . in_function_def ;
838
+ // remember to restore self.ctx.in_loop to the original after the function is compiled
839
+ let prev_ctx = self . ctx ;
820
840
821
- let was_in_async_func = self . in_async_func ;
822
- self . in_async_func = is_async;
823
- self . in_loop = false ;
824
- self . in_function_def = true ;
841
+ self . ctx = CompileContext {
842
+ in_loop : false ,
843
+ func : if is_async {
844
+ FunctionContext :: AsyncFunction
845
+ } else {
846
+ FunctionContext :: Function
847
+ } ,
848
+ } ;
825
849
826
- let old_qualified_path = self . current_qualified_path . clone ( ) ;
827
850
let qualified_name = self . create_qualified_name ( name, "" ) ;
851
+ let old_qualified_path = self . current_qualified_path . take ( ) ;
828
852
self . current_qualified_path = Some ( self . create_qualified_name ( name, ".<locals>" ) ) ;
829
853
830
854
self . prepare_decorators ( decorator_list) ?;
@@ -911,9 +935,7 @@ impl<O: OutputStream> Compiler<O> {
911
935
self . store_name ( name) ;
912
936
913
937
self . current_qualified_path = old_qualified_path;
914
- self . in_loop = was_in_loop;
915
- self . in_function_def = was_in_function_def;
916
- self . in_async_func = was_in_async_func;
938
+ self . ctx = prev_ctx;
917
939
Ok ( ( ) )
918
940
}
919
941
@@ -925,11 +947,14 @@ impl<O: OutputStream> Compiler<O> {
925
947
keywords : & [ ast:: Keyword ] ,
926
948
decorator_list : & [ ast:: Expression ] ,
927
949
) -> Result < ( ) , CompileError > {
928
- let was_in_loop = self . in_loop ;
929
- self . in_loop = false ;
950
+ let prev_ctx = self . ctx ;
951
+ self . ctx = CompileContext {
952
+ func : FunctionContext :: NoFunction ,
953
+ in_loop : false ,
954
+ } ;
930
955
931
- let old_qualified_path = self . current_qualified_path . clone ( ) ;
932
956
let qualified_name = self . create_qualified_name ( name, "" ) ;
957
+ let old_qualified_path = self . current_qualified_path . take ( ) ;
933
958
self . current_qualified_path = Some ( qualified_name. clone ( ) ) ;
934
959
935
960
self . prepare_decorators ( decorator_list) ?;
@@ -1033,7 +1058,7 @@ impl<O: OutputStream> Compiler<O> {
1033
1058
1034
1059
self . store_name ( name) ;
1035
1060
self . current_qualified_path = old_qualified_path;
1036
- self . in_loop = was_in_loop ;
1061
+ self . ctx = prev_ctx ;
1037
1062
Ok ( ( ) )
1038
1063
}
1039
1064
@@ -1075,10 +1100,10 @@ impl<O: OutputStream> Compiler<O> {
1075
1100
1076
1101
self . compile_jump_if ( test, false , else_label) ?;
1077
1102
1078
- let was_in_loop = self . in_loop ;
1079
- self . in_loop = true ;
1103
+ let was_in_loop = self . ctx . in_loop ;
1104
+ self . ctx . in_loop = true ;
1080
1105
self . compile_statements ( body) ?;
1081
- self . in_loop = was_in_loop;
1106
+ self . ctx . in_loop = was_in_loop;
1082
1107
self . emit ( Instruction :: Jump {
1083
1108
target : start_label,
1084
1109
} ) ;
@@ -1143,11 +1168,11 @@ impl<O: OutputStream> Compiler<O> {
1143
1168
self . emit ( Instruction :: JumpIfTrue { target : else_label } ) ;
1144
1169
self . emit ( Instruction :: Raise { argc : 0 } ) ;
1145
1170
1146
- let was_in_loop = self . in_loop ;
1147
- self . in_loop = true ;
1171
+ let was_in_loop = self . ctx . in_loop ;
1172
+ self . ctx . in_loop = true ;
1148
1173
self . set_label ( body_label) ;
1149
1174
self . compile_statements ( body) ?;
1150
- self . in_loop = was_in_loop;
1175
+ self . ctx . in_loop = was_in_loop;
1151
1176
} else {
1152
1177
// Retrieve Iterator
1153
1178
self . emit ( Instruction :: GetIter ) ;
@@ -1158,10 +1183,10 @@ impl<O: OutputStream> Compiler<O> {
1158
1183
// Start of loop iteration, set targets:
1159
1184
self . compile_store ( target) ?;
1160
1185
1161
- let was_in_loop = self . in_loop ;
1162
- self . in_loop = true ;
1186
+ let was_in_loop = self . ctx . in_loop ;
1187
+ self . ctx . in_loop = true ;
1163
1188
self . compile_statements ( body) ?;
1164
- self . in_loop = was_in_loop;
1189
+ self . ctx . in_loop = was_in_loop;
1165
1190
}
1166
1191
1167
1192
self . emit ( Instruction :: Jump {
@@ -1617,7 +1642,7 @@ impl<O: OutputStream> Compiler<O> {
1617
1642
self . emit ( Instruction :: BuildSlice { size } ) ;
1618
1643
}
1619
1644
Yield { value } => {
1620
- if !self . in_function_def || self . in_async_func {
1645
+ if !self . ctx . in_func ( ) {
1621
1646
return Err ( CompileError {
1622
1647
error : CompileErrorType :: InvalidYield ,
1623
1648
location : self . current_source_location . clone ( ) ,
@@ -1683,12 +1708,11 @@ impl<O: OutputStream> Compiler<O> {
1683
1708
self . load_name ( name) ;
1684
1709
}
1685
1710
Lambda { args, body } => {
1686
- let was_in_loop = self . in_loop ;
1687
- let was_in_function_def = self . in_function_def ;
1688
- let was_in_async_func = self . in_async_func ;
1689
- self . in_async_func = false ;
1690
- self . in_loop = false ;
1691
- self . in_function_def = true ;
1711
+ let prev_ctx = self . ctx ;
1712
+ self . ctx = CompileContext {
1713
+ in_loop : false ,
1714
+ func : FunctionContext :: Function ,
1715
+ } ;
1692
1716
1693
1717
let name = "<lambda>" . to_string ( ) ;
1694
1718
self . enter_function ( & name, args) ?;
@@ -1707,9 +1731,7 @@ impl<O: OutputStream> Compiler<O> {
1707
1731
// Turn code object into function object:
1708
1732
self . emit ( Instruction :: MakeFunction ) ;
1709
1733
1710
- self . in_loop = was_in_loop;
1711
- self . in_function_def = was_in_function_def;
1712
- self . in_async_func = was_in_async_func;
1734
+ self . ctx = prev_ctx;
1713
1735
}
1714
1736
Comprehension { kind, generators } => {
1715
1737
self . compile_comprehension ( kind, generators) ?;
0 commit comments