Skip to content

Commit 6ef275e

Browse files
committed
Auto merge of #63770 - oli-obk:allow_internal_unstable, r=Centril
Stabilize `str::len`, `[T]::len` and `str::as_bytes` as const fn r? @Centril cc @RalfJung This also introduces a scheme for making certain feature gates legal in stabilized const fns
2 parents 7fdea7a + 7767e7f commit 6ef275e

File tree

12 files changed

+114
-78
lines changed

12 files changed

+114
-78
lines changed

src/libcore/lib.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -120,9 +120,9 @@
120120
#![feature(rtm_target_feature)]
121121
#![feature(f16c_target_feature)]
122122
#![feature(hexagon_target_feature)]
123-
#![feature(const_slice_len)]
124-
#![feature(const_str_as_bytes)]
125-
#![feature(const_str_len)]
123+
#![cfg_attr(bootstrap, feature(const_slice_len))]
124+
#![cfg_attr(bootstrap, feature(const_str_as_bytes))]
125+
#![cfg_attr(bootstrap, feature(const_str_len))]
126126
#![feature(const_int_conversion)]
127127
#![feature(const_transmute)]
128128
#![feature(non_exhaustive)]

src/libcore/slice/mod.rs

+4-2
Original file line numberDiff line numberDiff line change
@@ -62,7 +62,9 @@ impl<T> [T] {
6262
/// ```
6363
#[stable(feature = "rust1", since = "1.0.0")]
6464
#[inline]
65-
#[rustc_const_unstable(feature = "const_slice_len")]
65+
#[cfg_attr(bootstrap, rustc_const_unstable(feature = "const_slice_len"))]
66+
// SAFETY: const sound because we transmute out the length field as a usize (which it must be)
67+
#[cfg_attr(not(bootstrap), allow_internal_unstable(const_fn_union))]
6668
pub const fn len(&self) -> usize {
6769
unsafe {
6870
crate::ptr::Repr { rust: self }.raw.len
@@ -79,7 +81,7 @@ impl<T> [T] {
7981
/// ```
8082
#[stable(feature = "rust1", since = "1.0.0")]
8183
#[inline]
82-
#[rustc_const_unstable(feature = "const_slice_len")]
84+
#[cfg_attr(bootstrap, rustc_const_unstable(feature = "const_slice_len"))]
8385
pub const fn is_empty(&self) -> bool {
8486
self.len() == 0
8587
}

src/libcore/str/mod.rs

+5-3
Original file line numberDiff line numberDiff line change
@@ -2090,7 +2090,7 @@ impl str {
20902090
/// ```
20912091
#[stable(feature = "rust1", since = "1.0.0")]
20922092
#[inline]
2093-
#[rustc_const_unstable(feature = "const_str_len")]
2093+
#[cfg_attr(bootstrap, rustc_const_unstable(feature = "const_str_len"))]
20942094
pub const fn len(&self) -> usize {
20952095
self.as_bytes().len()
20962096
}
@@ -2110,7 +2110,7 @@ impl str {
21102110
/// ```
21112111
#[inline]
21122112
#[stable(feature = "rust1", since = "1.0.0")]
2113-
#[rustc_const_unstable(feature = "const_str_len")]
2113+
#[cfg_attr(bootstrap, rustc_const_unstable(feature = "const_str_len"))]
21142114
pub const fn is_empty(&self) -> bool {
21152115
self.len() == 0
21162116
}
@@ -2168,7 +2168,9 @@ impl str {
21682168
/// ```
21692169
#[stable(feature = "rust1", since = "1.0.0")]
21702170
#[inline(always)]
2171-
#[rustc_const_unstable(feature="const_str_as_bytes")]
2171+
#[cfg_attr(bootstrap, rustc_const_unstable(feature = "const_str_as_bytes"))]
2172+
// SAFETY: const sound because we transmute two types with the same layout
2173+
#[cfg_attr(not(bootstrap), allow_internal_unstable(const_fn_union))]
21722174
pub const fn as_bytes(&self) -> &[u8] {
21732175
#[repr(C)]
21742176
union Slices<'a> {

src/librustc_mir/transform/check_unsafety.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -305,15 +305,15 @@ impl<'a, 'tcx> Visitor<'tcx> for UnsafetyChecker<'a, 'tcx> {
305305
"assignment to non-`Copy` union field",
306306
"the previous content of the field will be dropped, which \
307307
causes undefined behavior if the field was not properly \
308-
initialized", UnsafetyViolationKind::General)
308+
initialized", UnsafetyViolationKind::GeneralAndConstFn)
309309
} else {
310310
// write to non-move union, safe
311311
}
312312
} else {
313313
self.require_unsafe("access to union field",
314314
"the field may not be properly initialized: using \
315315
uninitialized data will cause undefined behavior",
316-
UnsafetyViolationKind::General)
316+
UnsafetyViolationKind::GeneralAndConstFn)
317317
}
318318
}
319319
}

src/librustc_mir/transform/qualify_min_const_fn.rs

+57-24
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,8 @@ use rustc::ty::{self, Predicate, Ty, TyCtxt, adjustment::{PointerCast}};
55
use rustc_target::spec::abi;
66
use std::borrow::Cow;
77
use syntax_pos::Span;
8+
use syntax::symbol::{sym, Symbol};
9+
use syntax::attr;
810

911
type McfResult = Result<(), (Span, Cow<'static, str>)>;
1012

@@ -67,9 +69,9 @@ pub fn is_min_const_fn(tcx: TyCtxt<'tcx>, def_id: DefId, body: &'a Body<'tcx>) -
6769
)?;
6870

6971
for bb in body.basic_blocks() {
70-
check_terminator(tcx, body, bb.terminator())?;
72+
check_terminator(tcx, body, def_id, bb.terminator())?;
7173
for stmt in &bb.statements {
72-
check_statement(tcx, body, stmt)?;
74+
check_statement(tcx, body, def_id, stmt)?;
7375
}
7476
}
7577
Ok(())
@@ -121,16 +123,17 @@ fn check_ty(tcx: TyCtxt<'tcx>, ty: Ty<'tcx>, span: Span, fn_def_id: DefId) -> Mc
121123

122124
fn check_rvalue(
123125
tcx: TyCtxt<'tcx>,
124-
body: &'a Body<'tcx>,
126+
body: &Body<'tcx>,
127+
def_id: DefId,
125128
rvalue: &Rvalue<'tcx>,
126129
span: Span,
127130
) -> McfResult {
128131
match rvalue {
129132
Rvalue::Repeat(operand, _) | Rvalue::Use(operand) => {
130-
check_operand(operand, span)
133+
check_operand(tcx, operand, span, def_id, body)
131134
}
132135
Rvalue::Len(place) | Rvalue::Discriminant(place) | Rvalue::Ref(_, _, place) => {
133-
check_place(place, span)
136+
check_place(tcx, place, span, def_id, body)
134137
}
135138
Rvalue::Cast(CastKind::Misc, operand, cast_ty) => {
136139
use rustc::ty::cast::CastTy;
@@ -144,11 +147,11 @@ fn check_rvalue(
144147
(CastTy::RPtr(_), CastTy::Float) => bug!(),
145148
(CastTy::RPtr(_), CastTy::Int(_)) => bug!(),
146149
(CastTy::Ptr(_), CastTy::RPtr(_)) => bug!(),
147-
_ => check_operand(operand, span),
150+
_ => check_operand(tcx, operand, span, def_id, body),
148151
}
149152
}
150153
Rvalue::Cast(CastKind::Pointer(PointerCast::MutToConstPointer), operand, _) => {
151-
check_operand(operand, span)
154+
check_operand(tcx, operand, span, def_id, body)
152155
}
153156
Rvalue::Cast(CastKind::Pointer(PointerCast::UnsafeFnPointer), _, _) |
154157
Rvalue::Cast(CastKind::Pointer(PointerCast::ClosureFnPointer(_)), _, _) |
@@ -162,8 +165,8 @@ fn check_rvalue(
162165
)),
163166
// binops are fine on integers
164167
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)?;
167170
let ty = lhs.ty(body, tcx);
168171
if ty.is_integral() || ty.is_bool() || ty.is_char() {
169172
Ok(())
@@ -182,7 +185,7 @@ fn check_rvalue(
182185
Rvalue::UnaryOp(_, operand) => {
183186
let ty = operand.ty(body, tcx);
184187
if ty.is_integral() || ty.is_bool() {
185-
check_operand(operand, span)
188+
check_operand(tcx, operand, span, def_id, body)
186189
} else {
187190
Err((
188191
span,
@@ -192,7 +195,7 @@ fn check_rvalue(
192195
}
193196
Rvalue::Aggregate(_, operands) => {
194197
for operand in operands {
195-
check_operand(operand, span)?;
198+
check_operand(tcx, operand, span, def_id, body)?;
196199
}
197200
Ok(())
198201
}
@@ -201,21 +204,22 @@ fn check_rvalue(
201204

202205
fn check_statement(
203206
tcx: TyCtxt<'tcx>,
204-
body: &'a Body<'tcx>,
207+
body: &Body<'tcx>,
208+
def_id: DefId,
205209
statement: &Statement<'tcx>,
206210
) -> McfResult {
207211
let span = statement.source_info.span;
208212
match &statement.kind {
209213
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)
212216
}
213217

214218
StatementKind::FakeRead(FakeReadCause::ForMatchedPlace, _) => {
215219
Err((span, "loops and conditional expressions are not stable in const fn".into()))
216220
}
217221

218-
StatementKind::FakeRead(_, place) => check_place(place, span),
222+
StatementKind::FakeRead(_, place) => check_place(tcx, place, span, def_id, body),
219223

220224
// just an assignment
221225
StatementKind::SetDiscriminant { .. } => Ok(()),
@@ -234,30 +238,48 @@ fn check_statement(
234238
}
235239

236240
fn check_operand(
241+
tcx: TyCtxt<'tcx>,
237242
operand: &Operand<'tcx>,
238243
span: Span,
244+
def_id: DefId,
245+
body: &Body<'tcx>
239246
) -> McfResult {
240247
match operand {
241248
Operand::Move(place) | Operand::Copy(place) => {
242-
check_place(place, span)
249+
check_place(tcx, place, span, def_id, body)
243250
}
244251
Operand::Constant(_) => Ok(()),
245252
}
246253
}
247254

248255
fn check_place(
256+
tcx: TyCtxt<'tcx>,
249257
place: &Place<'tcx>,
250258
span: Span,
259+
def_id: DefId,
260+
body: &Body<'tcx>
251261
) -> 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;
253265
match elem {
254266
ProjectionElem::Downcast(..) => {
255267
return Err((span, "`match` or `if let` in `const fn` is unstable".into()));
256268
}
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+
}
257280
ProjectionElem::ConstantIndex { .. }
258281
| ProjectionElem::Subslice { .. }
259282
| ProjectionElem::Deref
260-
| ProjectionElem::Field(..)
261283
| ProjectionElem::Index(_) => {}
262284
}
263285
}
@@ -271,9 +293,20 @@ fn check_place(
271293
}
272294
}
273295

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+
274306
fn check_terminator(
275307
tcx: TyCtxt<'tcx>,
276308
body: &'a Body<'tcx>,
309+
def_id: DefId,
277310
terminator: &Terminator<'tcx>,
278311
) -> McfResult {
279312
let span = terminator.source_info.span;
@@ -283,11 +316,11 @@ fn check_terminator(
283316
| TerminatorKind::Resume => Ok(()),
284317

285318
TerminatorKind::Drop { location, .. } => {
286-
check_place(location, span)
319+
check_place(tcx, location, span, def_id, body)
287320
}
288321
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)
291324
},
292325

293326
TerminatorKind::FalseEdges { .. } | TerminatorKind::SwitchInt { .. } => Err((
@@ -339,10 +372,10 @@ fn check_terminator(
339372
)),
340373
}
341374

342-
check_operand(func, span)?;
375+
check_operand(tcx, func, span, def_id, body)?;
343376

344377
for arg in args {
345-
check_operand(arg, span)?;
378+
check_operand(tcx, arg, span, def_id, body)?;
346379
}
347380
Ok(())
348381
} else {
@@ -356,7 +389,7 @@ fn check_terminator(
356389
msg: _,
357390
target: _,
358391
cleanup: _,
359-
} => check_operand(cond, span),
392+
} => check_operand(tcx, cond, span, def_id, body),
360393

361394
TerminatorKind::FalseUnwind { .. } => {
362395
Err((span, "loops are not allowed in const fn".into()))

src/libsyntax/attr/mod.rs

+24
Original file line numberDiff line numberDiff line change
@@ -433,6 +433,30 @@ pub fn find_by_name(attrs: &[Attribute], name: Symbol) -> Option<&Attribute> {
433433
attrs.iter().find(|attr| attr.check_name(name))
434434
}
435435

436+
pub fn allow_internal_unstable<'a>(
437+
attrs: &[Attribute],
438+
span_diagnostic: &'a errors::Handler,
439+
) -> Option<impl Iterator<Item = Symbol> + 'a> {
440+
find_by_name(attrs, sym::allow_internal_unstable).and_then(|attr| {
441+
attr.meta_item_list().or_else(|| {
442+
span_diagnostic.span_err(
443+
attr.span,
444+
"allow_internal_unstable expects list of feature names"
445+
);
446+
None
447+
}).map(|features| features.into_iter().filter_map(move |it| {
448+
let name = it.ident().map(|ident| ident.name);
449+
if name.is_none() {
450+
span_diagnostic.span_err(
451+
it.span(),
452+
"`allow_internal_unstable` expects feature names",
453+
)
454+
}
455+
name
456+
}))
457+
})
458+
}
459+
436460
pub fn filter_by_name(attrs: &[Attribute], name: Symbol)
437461
-> impl Iterator<Item=&Attribute> {
438462
attrs.iter().filter(move |attr| attr.check_name(name))

src/libsyntax/ext/base.rs

+3-27
Original file line numberDiff line numberDiff line change
@@ -762,33 +762,9 @@ impl SyntaxExtension {
762762
name: Name,
763763
attrs: &[ast::Attribute],
764764
) -> SyntaxExtension {
765-
let allow_internal_unstable =
766-
attr::find_by_name(attrs, sym::allow_internal_unstable).map(|attr| {
767-
attr.meta_item_list()
768-
.map(|list| {
769-
list.iter()
770-
.filter_map(|it| {
771-
let name = it.ident().map(|ident| ident.name);
772-
if name.is_none() {
773-
sess.span_diagnostic.span_err(
774-
it.span(), "allow internal unstable expects feature names"
775-
)
776-
}
777-
name
778-
})
779-
.collect::<Vec<Symbol>>()
780-
.into()
781-
})
782-
.unwrap_or_else(|| {
783-
sess.span_diagnostic.span_warn(
784-
attr.span,
785-
"allow_internal_unstable expects list of feature names. In the future \
786-
this will become a hard error. Please use `allow_internal_unstable(\
787-
foo, bar)` to only allow the `foo` and `bar` features",
788-
);
789-
vec![sym::allow_internal_unstable_backcompat_hack].into()
790-
})
791-
});
765+
let allow_internal_unstable = attr::allow_internal_unstable(
766+
&attrs, &sess.span_diagnostic,
767+
).map(|features| features.collect::<Vec<Symbol>>().into());
792768

793769
let mut local_inner_macros = false;
794770
if let Some(macro_export) = attr::find_by_name(attrs, sym::macro_export) {

src/test/ui/consts/const-eval/strlen.rs

+10-2
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,11 @@
11
// run-pass
22

3-
#![feature(const_str_len, const_str_as_bytes)]
4-
53
const S: &str = "foo";
64
pub const B: &[u8] = S.as_bytes();
5+
pub const C: usize = B.len();
6+
pub const D: bool = B.is_empty();
7+
pub const E: bool = S.is_empty();
8+
pub const F: usize = S.len();
79

810
pub fn foo() -> [u8; S.len()] {
911
let mut buf = [0; S.len()];
@@ -20,4 +22,10 @@ fn main() {
2022
assert_eq!(LEN, S.len());
2123
assert_eq!(B, foo());
2224
assert_eq!(B, b"foo");
25+
assert_eq!(C, 3);
26+
assert_eq!(F, 3);
27+
assert!(!D);
28+
assert!(!E);
29+
const EMPTY: bool = "".is_empty();
30+
assert!(EMPTY);
2331
}

0 commit comments

Comments
 (0)