@@ -5,6 +5,8 @@ use rustc::ty::{self, Predicate, Ty, TyCtxt, adjustment::{PointerCast}};
5
5
use rustc_target:: spec:: abi;
6
6
use std:: borrow:: Cow ;
7
7
use syntax_pos:: Span ;
8
+ use syntax:: symbol:: { sym, Symbol } ;
9
+ use syntax:: attr;
8
10
9
11
type McfResult = Result < ( ) , ( Span , Cow < ' static , str > ) > ;
10
12
@@ -67,9 +69,9 @@ pub fn is_min_const_fn(tcx: TyCtxt<'tcx>, def_id: DefId, body: &'a Body<'tcx>) -
67
69
) ?;
68
70
69
71
for bb in body. basic_blocks ( ) {
70
- check_terminator ( tcx, body, bb. terminator ( ) ) ?;
72
+ check_terminator ( tcx, body, def_id , bb. terminator ( ) ) ?;
71
73
for stmt in & bb. statements {
72
- check_statement ( tcx, body, stmt) ?;
74
+ check_statement ( tcx, body, def_id , stmt) ?;
73
75
}
74
76
}
75
77
Ok ( ( ) )
@@ -121,16 +123,17 @@ fn check_ty(tcx: TyCtxt<'tcx>, ty: Ty<'tcx>, span: Span, fn_def_id: DefId) -> Mc
121
123
122
124
fn check_rvalue (
123
125
tcx : TyCtxt < ' tcx > ,
124
- body : & ' a Body < ' tcx > ,
126
+ body : & Body < ' tcx > ,
127
+ def_id : DefId ,
125
128
rvalue : & Rvalue < ' tcx > ,
126
129
span : Span ,
127
130
) -> McfResult {
128
131
match rvalue {
129
132
Rvalue :: Repeat ( operand, _) | Rvalue :: Use ( operand) => {
130
- check_operand ( operand, span)
133
+ check_operand ( tcx , operand, span, def_id , body )
131
134
}
132
135
Rvalue :: Len ( place) | Rvalue :: Discriminant ( place) | Rvalue :: Ref ( _, _, place) => {
133
- check_place ( place, span)
136
+ check_place ( tcx , place, span, def_id , body )
134
137
}
135
138
Rvalue :: Cast ( CastKind :: Misc , operand, cast_ty) => {
136
139
use rustc:: ty:: cast:: CastTy ;
@@ -144,11 +147,11 @@ fn check_rvalue(
144
147
( CastTy :: RPtr ( _) , CastTy :: Float ) => bug ! ( ) ,
145
148
( CastTy :: RPtr ( _) , CastTy :: Int ( _) ) => bug ! ( ) ,
146
149
( CastTy :: Ptr ( _) , CastTy :: RPtr ( _) ) => bug ! ( ) ,
147
- _ => check_operand ( operand, span) ,
150
+ _ => check_operand ( tcx , operand, span, def_id , body ) ,
148
151
}
149
152
}
150
153
Rvalue :: Cast ( CastKind :: Pointer ( PointerCast :: MutToConstPointer ) , operand, _) => {
151
- check_operand ( operand, span)
154
+ check_operand ( tcx , operand, span, def_id , body )
152
155
}
153
156
Rvalue :: Cast ( CastKind :: Pointer ( PointerCast :: UnsafeFnPointer ) , _, _) |
154
157
Rvalue :: Cast ( CastKind :: Pointer ( PointerCast :: ClosureFnPointer ( _) ) , _, _) |
@@ -162,8 +165,8 @@ fn check_rvalue(
162
165
) ) ,
163
166
// binops are fine on integers
164
167
Rvalue :: BinaryOp ( _, lhs, rhs) | Rvalue :: CheckedBinaryOp ( _, lhs, rhs) => {
165
- check_operand ( lhs, span) ?;
166
- check_operand ( rhs, span) ?;
168
+ check_operand ( tcx , lhs, span, def_id , body ) ?;
169
+ check_operand ( tcx , rhs, span, def_id , body ) ?;
167
170
let ty = lhs. ty ( body, tcx) ;
168
171
if ty. is_integral ( ) || ty. is_bool ( ) || ty. is_char ( ) {
169
172
Ok ( ( ) )
@@ -182,7 +185,7 @@ fn check_rvalue(
182
185
Rvalue :: UnaryOp ( _, operand) => {
183
186
let ty = operand. ty ( body, tcx) ;
184
187
if ty. is_integral ( ) || ty. is_bool ( ) {
185
- check_operand ( operand, span)
188
+ check_operand ( tcx , operand, span, def_id , body )
186
189
} else {
187
190
Err ( (
188
191
span,
@@ -192,7 +195,7 @@ fn check_rvalue(
192
195
}
193
196
Rvalue :: Aggregate ( _, operands) => {
194
197
for operand in operands {
195
- check_operand ( operand, span) ?;
198
+ check_operand ( tcx , operand, span, def_id , body ) ?;
196
199
}
197
200
Ok ( ( ) )
198
201
}
@@ -201,21 +204,22 @@ fn check_rvalue(
201
204
202
205
fn check_statement (
203
206
tcx : TyCtxt < ' tcx > ,
204
- body : & ' a Body < ' tcx > ,
207
+ body : & Body < ' tcx > ,
208
+ def_id : DefId ,
205
209
statement : & Statement < ' tcx > ,
206
210
) -> McfResult {
207
211
let span = statement. source_info . span ;
208
212
match & statement. kind {
209
213
StatementKind :: Assign ( box( place, rval) ) => {
210
- check_place ( place, span) ?;
211
- check_rvalue ( tcx, body, rval, span)
214
+ check_place ( tcx , place, span, def_id , body ) ?;
215
+ check_rvalue ( tcx, body, def_id , rval, span)
212
216
}
213
217
214
218
StatementKind :: FakeRead ( FakeReadCause :: ForMatchedPlace , _) => {
215
219
Err ( ( span, "loops and conditional expressions are not stable in const fn" . into ( ) ) )
216
220
}
217
221
218
- StatementKind :: FakeRead ( _, place) => check_place ( place, span) ,
222
+ StatementKind :: FakeRead ( _, place) => check_place ( tcx , place, span, def_id , body ) ,
219
223
220
224
// just an assignment
221
225
StatementKind :: SetDiscriminant { .. } => Ok ( ( ) ) ,
@@ -234,30 +238,48 @@ fn check_statement(
234
238
}
235
239
236
240
fn check_operand (
241
+ tcx : TyCtxt < ' tcx > ,
237
242
operand : & Operand < ' tcx > ,
238
243
span : Span ,
244
+ def_id : DefId ,
245
+ body : & Body < ' tcx >
239
246
) -> McfResult {
240
247
match operand {
241
248
Operand :: Move ( place) | Operand :: Copy ( place) => {
242
- check_place ( place, span)
249
+ check_place ( tcx , place, span, def_id , body )
243
250
}
244
251
Operand :: Constant ( _) => Ok ( ( ) ) ,
245
252
}
246
253
}
247
254
248
255
fn check_place (
256
+ tcx : TyCtxt < ' tcx > ,
249
257
place : & Place < ' tcx > ,
250
258
span : Span ,
259
+ def_id : DefId ,
260
+ body : & Body < ' tcx >
251
261
) -> McfResult {
252
- for elem in place. projection . iter ( ) {
262
+ let mut cursor = & * place. projection ;
263
+ while let [ proj_base @ .., elem] = cursor {
264
+ cursor = proj_base;
253
265
match elem {
254
266
ProjectionElem :: Downcast ( ..) => {
255
267
return Err ( ( span, "`match` or `if let` in `const fn` is unstable" . into ( ) ) ) ;
256
268
}
269
+ ProjectionElem :: Field ( ..) => {
270
+ let base_ty = Place :: ty_from ( & place. base , & proj_base, body, tcx) . ty ;
271
+ if let Some ( def) = base_ty. ty_adt_def ( ) {
272
+ // No union field accesses in `const fn`
273
+ if def. is_union ( ) {
274
+ if !feature_allowed ( tcx, def_id, sym:: const_fn_union) {
275
+ return Err ( ( span, "accessing union fields is unstable" . into ( ) ) ) ;
276
+ }
277
+ }
278
+ }
279
+ }
257
280
ProjectionElem :: ConstantIndex { .. }
258
281
| ProjectionElem :: Subslice { .. }
259
282
| ProjectionElem :: Deref
260
- | ProjectionElem :: Field ( ..)
261
283
| ProjectionElem :: Index ( _) => { }
262
284
}
263
285
}
@@ -271,9 +293,20 @@ fn check_place(
271
293
}
272
294
}
273
295
296
+ /// Returns whether `allow_internal_unstable(..., <feature_gate>, ...)` is present.
297
+ fn feature_allowed (
298
+ tcx : TyCtxt < ' tcx > ,
299
+ def_id : DefId ,
300
+ feature_gate : Symbol ,
301
+ ) -> bool {
302
+ attr:: allow_internal_unstable ( & tcx. get_attrs ( def_id) , & tcx. sess . diagnostic ( ) )
303
+ . map_or ( false , |mut features| features. any ( |name| name == feature_gate) )
304
+ }
305
+
274
306
fn check_terminator (
275
307
tcx : TyCtxt < ' tcx > ,
276
308
body : & ' a Body < ' tcx > ,
309
+ def_id : DefId ,
277
310
terminator : & Terminator < ' tcx > ,
278
311
) -> McfResult {
279
312
let span = terminator. source_info . span ;
@@ -283,11 +316,11 @@ fn check_terminator(
283
316
| TerminatorKind :: Resume => Ok ( ( ) ) ,
284
317
285
318
TerminatorKind :: Drop { location, .. } => {
286
- check_place ( location, span)
319
+ check_place ( tcx , location, span, def_id , body )
287
320
}
288
321
TerminatorKind :: DropAndReplace { location, value, .. } => {
289
- check_place ( location, span) ?;
290
- check_operand ( value, span)
322
+ check_place ( tcx , location, span, def_id , body ) ?;
323
+ check_operand ( tcx , value, span, def_id , body )
291
324
} ,
292
325
293
326
TerminatorKind :: FalseEdges { .. } | TerminatorKind :: SwitchInt { .. } => Err ( (
@@ -339,10 +372,10 @@ fn check_terminator(
339
372
) ) ,
340
373
}
341
374
342
- check_operand ( func, span) ?;
375
+ check_operand ( tcx , func, span, def_id , body ) ?;
343
376
344
377
for arg in args {
345
- check_operand ( arg, span) ?;
378
+ check_operand ( tcx , arg, span, def_id , body ) ?;
346
379
}
347
380
Ok ( ( ) )
348
381
} else {
@@ -356,7 +389,7 @@ fn check_terminator(
356
389
msg : _,
357
390
target : _,
358
391
cleanup : _,
359
- } => check_operand ( cond, span) ,
392
+ } => check_operand ( tcx , cond, span, def_id , body ) ,
360
393
361
394
TerminatorKind :: FalseUnwind { .. } => {
362
395
Err ( ( span, "loops are not allowed in const fn" . into ( ) ) )
0 commit comments