@@ -7,12 +7,14 @@ use rustc_hir::{self as hir, LangItem};
7
7
use rustc_middle:: bug;
8
8
use rustc_middle:: ty:: { self , FloatTy , IntTy , Ty , TyCtxt , TypeVisitableExt , UintTy } ;
9
9
use rustc_session:: lint;
10
- use rustc_span:: Symbol ;
11
10
use rustc_span:: def_id:: LocalDefId ;
11
+ use rustc_span:: { Symbol , sym} ;
12
12
use rustc_target:: asm:: {
13
13
InlineAsmReg , InlineAsmRegClass , InlineAsmRegOrRegClass , InlineAsmType , ModifierInfo ,
14
14
} ;
15
15
16
+ use crate :: errors:: RegisterTypeUnstable ;
17
+
16
18
pub struct InlineAsmCtxt < ' a , ' tcx > {
17
19
tcx : TyCtxt < ' tcx > ,
18
20
typing_env : ty:: TypingEnv < ' tcx > ,
@@ -218,17 +220,29 @@ impl<'a, 'tcx> InlineAsmCtxt<'a, 'tcx> {
218
220
// Check the type against the list of types supported by the selected
219
221
// register class.
220
222
let asm_arch = self . tcx . sess . asm_arch . unwrap ( ) ;
223
+ let allow_experimental_reg = self . tcx . features ( ) . asm_experimental_reg ( ) ;
221
224
let reg_class = reg. reg_class ( ) ;
222
- let supported_tys = reg_class. supported_types ( asm_arch) ;
225
+ let supported_tys = reg_class. supported_types ( asm_arch, allow_experimental_reg ) ;
223
226
let Some ( ( _, feature) ) = supported_tys. iter ( ) . find ( |& & ( t, _) | t == asm_ty) else {
224
- let msg = format ! ( "type `{ty}` cannot be used with this register class" ) ;
225
- let mut err = self . tcx . dcx ( ) . struct_span_err ( expr. span , msg) ;
226
- let supported_tys: Vec < _ > = supported_tys. iter ( ) . map ( |( t, _) | t. to_string ( ) ) . collect ( ) ;
227
- err. note ( format ! (
228
- "register class `{}` supports these types: {}" ,
229
- reg_class. name( ) ,
230
- supported_tys. join( ", " ) ,
231
- ) ) ;
227
+ let mut err = if !allow_experimental_reg
228
+ && reg_class. supported_types ( asm_arch, true ) . iter ( ) . any ( |& ( t, _) | t == asm_ty)
229
+ {
230
+ self . tcx . sess . create_feature_err (
231
+ RegisterTypeUnstable { span : expr. span , ty } ,
232
+ sym:: asm_experimental_reg,
233
+ )
234
+ } else {
235
+ let msg = format ! ( "type `{ty}` cannot be used with this register class" ) ;
236
+ let mut err = self . tcx . dcx ( ) . struct_span_err ( expr. span , msg) ;
237
+ let supported_tys: Vec < _ > =
238
+ supported_tys. iter ( ) . map ( |( t, _) | t. to_string ( ) ) . collect ( ) ;
239
+ err. note ( format ! (
240
+ "register class `{}` supports these types: {}" ,
241
+ reg_class. name( ) ,
242
+ supported_tys. join( ", " ) ,
243
+ ) ) ;
244
+ err
245
+ } ;
232
246
if let Some ( suggest) = reg_class. suggest_class ( asm_arch, asm_ty) {
233
247
err. help ( format ! ( "consider using the `{}` register class instead" , suggest. name( ) ) ) ;
234
248
}
@@ -313,6 +327,7 @@ impl<'a, 'tcx> InlineAsmCtxt<'a, 'tcx> {
313
327
self . tcx . dcx ( ) . delayed_bug ( "target architecture does not support asm" ) ;
314
328
return ;
315
329
} ;
330
+ let allow_experimental_reg = self . tcx . features ( ) . asm_experimental_reg ( ) ;
316
331
for ( idx, ( op, op_sp) ) in asm. operands . iter ( ) . enumerate ( ) {
317
332
// Validate register classes against currently enabled target
318
333
// features. We check that at least one type is available for
@@ -352,7 +367,8 @@ impl<'a, 'tcx> InlineAsmCtxt<'a, 'tcx> {
352
367
if let InlineAsmRegClass :: Err = reg_class {
353
368
continue ;
354
369
}
355
- for & ( _, feature) in reg_class. supported_types ( asm_arch) {
370
+ for & ( _, feature) in reg_class. supported_types ( asm_arch, allow_experimental_reg)
371
+ {
356
372
match feature {
357
373
Some ( feature) => {
358
374
if target_features. contains ( & feature) {
0 commit comments