@@ -51,6 +51,12 @@ impl ConstCx<'mir, 'tcx> {
5151 pub fn const_kind ( & self ) -> hir:: ConstContext {
5252 self . const_kind . expect ( "`const_kind` must not be called on a non-const fn" )
5353 }
54+
55+ pub fn is_const_stable_const_fn ( & self ) -> bool {
56+ self . const_kind == Some ( hir:: ConstContext :: ConstFn )
57+ && self . tcx . features ( ) . staged_api
58+ && is_const_stable ( self . tcx , self . def_id . to_def_id ( ) )
59+ }
5460}
5561
5662/// Returns `true` if this `DefId` points to one of the official `panic` lang items.
@@ -63,3 +69,35 @@ pub fn allow_internal_unstable(tcx: TyCtxt<'tcx>, def_id: DefId, feature_gate: S
6369 attr:: allow_internal_unstable ( & tcx. sess , attrs)
6470 . map_or ( false , |mut features| features. any ( |name| name == feature_gate) )
6571}
72+
73+ // Returns `true` if the given `const fn` is "const-stable".
74+ //
75+ // Const-stability is only relevant for `const fn` within a `staged_api` crate. Only "const-stable"
76+ // functions can be called in a const-context by users of the stable compiler. "const-stable"
77+ // functions are subject to more stringent restrictions than "const-unstable" functions: They
78+ // cannot use unstable features and can only call other "const-stable" functions.
79+ pub fn is_const_stable ( tcx : TyCtxt < ' tcx > , def_id : DefId ) -> bool {
80+ use attr:: { ConstStability , Stability , StabilityLevel } ;
81+
82+ // Const-stability is only relevant for `const fn`.
83+ assert ! ( tcx. is_const_fn_raw( def_id) ) ;
84+
85+ // Functions with `#[rustc_const_unstable]` are const-unstable.
86+ match tcx. lookup_const_stability ( def_id) {
87+ Some ( ConstStability { level : StabilityLevel :: Unstable { .. } , .. } ) => return false ,
88+ Some ( ConstStability { level : StabilityLevel :: Stable { .. } , .. } ) => return true ,
89+ None => { }
90+ }
91+
92+ // Functions with `#[unstable]` are const-unstable.
93+ //
94+ // FIXME(ecstaticmorse): We should keep const-stability attributes wholly separate from normal stability
95+ // attributes. `#[unstable]` should be irrelevant.
96+ if let Some ( Stability { level : StabilityLevel :: Unstable { .. } , .. } ) =
97+ tcx. lookup_stability ( def_id)
98+ {
99+ return false ;
100+ }
101+
102+ true
103+ }
0 commit comments