Skip to content

Commit 9f8ae06

Browse files
committed
Auto merge of #50248 - pietroalbini:beta-backports, r=eddyb
[beta] Backport #50072 * #50072: Allow variant discriminant initializers to refer to other initializers of the same enum
2 parents 1861fb5 + fc2ac26 commit 9f8ae06

File tree

7 files changed

+148
-54
lines changed

7 files changed

+148
-54
lines changed

src/librustc/ty/mod.rs

+23-17
Original file line numberDiff line numberDiff line change
@@ -1942,32 +1942,38 @@ impl<'a, 'gcx, 'tcx> AdtDef {
19421942
tcx: TyCtxt<'a, 'gcx, 'tcx>,
19431943
variant_index: usize)
19441944
-> Discr<'tcx> {
1945-
let repr_type = self.repr.discr_type();
1946-
let mut explicit_value = repr_type.initial_discriminant(tcx.global_tcx());
1945+
let (val, offset) = self.discriminant_def_for_variant(variant_index);
1946+
let explicit_value = val
1947+
.and_then(|expr_did| self.eval_explicit_discr(tcx, expr_did))
1948+
.unwrap_or_else(|| self.repr.discr_type().initial_discriminant(tcx.global_tcx()));
1949+
explicit_value.checked_add(tcx, offset as u128).0
1950+
}
1951+
1952+
/// Yields a DefId for the discriminant and an offset to add to it
1953+
/// Alternatively, if there is no explicit discriminant, returns the
1954+
/// inferred discriminant directly
1955+
pub fn discriminant_def_for_variant(
1956+
&self,
1957+
variant_index: usize,
1958+
) -> (Option<DefId>, usize) {
19471959
let mut explicit_index = variant_index;
1960+
let expr_did;
19481961
loop {
19491962
match self.variants[explicit_index].discr {
1950-
ty::VariantDiscr::Relative(0) => break,
1963+
ty::VariantDiscr::Relative(0) => {
1964+
expr_did = None;
1965+
break;
1966+
},
19511967
ty::VariantDiscr::Relative(distance) => {
19521968
explicit_index -= distance;
19531969
}
1954-
ty::VariantDiscr::Explicit(expr_did) => {
1955-
match self.eval_explicit_discr(tcx, expr_did) {
1956-
Some(discr) => {
1957-
explicit_value = discr;
1958-
break;
1959-
},
1960-
None => {
1961-
if explicit_index == 0 {
1962-
break;
1963-
}
1964-
explicit_index -= 1;
1965-
}
1966-
}
1970+
ty::VariantDiscr::Explicit(did) => {
1971+
expr_did = Some(did);
1972+
break;
19671973
}
19681974
}
19691975
}
1970-
explicit_value.checked_add(tcx, (variant_index - explicit_index) as u128).0
1976+
(expr_did, variant_index - explicit_index)
19711977
}
19721978

19731979
pub fn destructor(&self, tcx: TyCtxt<'a, 'gcx, 'tcx>) -> Option<Destructor> {

src/librustc_mir/hair/cx/expr.rs

+75-3
Original file line numberDiff line numberDiff line change
@@ -605,12 +605,84 @@ fn make_mirror_unadjusted<'a, 'gcx, 'tcx>(cx: &mut Cx<'a, 'gcx, 'tcx>,
605605
// Check to see if this cast is a "coercion cast", where the cast is actually done
606606
// using a coercion (or is a no-op).
607607
if let Some(&TyCastKind::CoercionCast) = cx.tables()
608-
.cast_kinds()
609-
.get(source.hir_id) {
608+
.cast_kinds()
609+
.get(source.hir_id) {
610610
// Convert the lexpr to a vexpr.
611611
ExprKind::Use { source: source.to_ref() }
612612
} else {
613-
ExprKind::Cast { source: source.to_ref() }
613+
// check whether this is casting an enum variant discriminant
614+
// to prevent cycles, we refer to the discriminant initializer
615+
// which is always an integer and thus doesn't need to know the
616+
// enum's layout (or its tag type) to compute it during const eval
617+
// Example:
618+
// enum Foo {
619+
// A,
620+
// B = A as isize + 4,
621+
// }
622+
// The correct solution would be to add symbolic computations to miri,
623+
// so we wouldn't have to compute and store the actual value
624+
let var = if let hir::ExprPath(ref qpath) = source.node {
625+
let def = cx.tables().qpath_def(qpath, source.hir_id);
626+
cx
627+
.tables()
628+
.node_id_to_type(source.hir_id)
629+
.ty_adt_def()
630+
.and_then(|adt_def| {
631+
match def {
632+
Def::VariantCtor(variant_id, CtorKind::Const) => {
633+
let idx = adt_def.variant_index_with_id(variant_id);
634+
let (d, o) = adt_def.discriminant_def_for_variant(idx);
635+
use rustc::ty::util::IntTypeExt;
636+
let ty = adt_def.repr.discr_type().to_ty(cx.tcx());
637+
Some((d, o, ty))
638+
}
639+
_ => None,
640+
}
641+
})
642+
} else {
643+
None
644+
};
645+
let source = if let Some((did, offset, ty)) = var {
646+
let mk_const = |val| Expr {
647+
temp_lifetime,
648+
ty,
649+
span: expr.span,
650+
kind: ExprKind::Literal {
651+
literal: Literal::Value {
652+
value: cx.tcx().mk_const(ty::Const {
653+
val,
654+
ty,
655+
}),
656+
},
657+
},
658+
}.to_ref();
659+
let offset = mk_const(
660+
ConstVal::Value(Value::ByVal(PrimVal::Bytes(offset as u128))),
661+
);
662+
match did {
663+
Some(did) => {
664+
// in case we are offsetting from a computed discriminant
665+
// and not the beginning of discriminants (which is always `0`)
666+
let substs = Substs::identity_for_item(cx.tcx(), did);
667+
let lhs = mk_const(ConstVal::Unevaluated(did, substs));
668+
let bin = ExprKind::Binary {
669+
op: BinOp::Add,
670+
lhs,
671+
rhs: offset,
672+
};
673+
Expr {
674+
temp_lifetime,
675+
ty,
676+
span: expr.span,
677+
kind: bin,
678+
}.to_ref()
679+
},
680+
None => offset,
681+
}
682+
} else {
683+
source.to_ref()
684+
};
685+
ExprKind::Cast { source }
614686
}
615687
}
616688
hir::ExprType(ref source, _) => return source.make_mirror(cx),

src/test/ui/const-eval/enum_discr.rs

+36
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
// Copyright 2018 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
// compile-pass
12+
// run-pass
13+
14+
enum Foo {
15+
X = 42,
16+
Y = Foo::X as isize - 3,
17+
}
18+
19+
enum Bar {
20+
X,
21+
Y = Bar::X as isize + 2,
22+
}
23+
24+
enum Boo {
25+
X = Boo::Y as isize * 2,
26+
Y = 9,
27+
}
28+
29+
fn main() {
30+
assert_eq!(Foo::X as isize, 42);
31+
assert_eq!(Foo::Y as isize, 39);
32+
assert_eq!(Bar::X as isize, 0);
33+
assert_eq!(Bar::Y as isize, 2);
34+
assert_eq!(Boo::X as isize, 18);
35+
assert_eq!(Boo::Y as isize, 9);
36+
}

src/test/ui/issue-23302-1.stderr

+2-7
Original file line numberDiff line numberDiff line change
@@ -4,17 +4,12 @@ error[E0391]: cyclic dependency detected
44
LL | A = X::A as isize, //~ ERROR E0391
55
| ^^^^^^^^^^^^^ cyclic reference
66
|
7-
note: the cycle begins when const-evaluating `X::A::{{initializer}}`...
7+
note: the cycle begins when processing `X::A::{{initializer}}`...
88
--> $DIR/issue-23302-1.rs:14:9
99
|
1010
LL | A = X::A as isize, //~ ERROR E0391
1111
| ^^^^^^^^^^^^^
12-
note: ...which then requires computing layout of `X`...
13-
--> $DIR/issue-23302-1.rs:14:9
14-
|
15-
LL | A = X::A as isize, //~ ERROR E0391
16-
| ^^^^
17-
= note: ...which then again requires const-evaluating `X::A::{{initializer}}`, completing the cycle.
12+
= note: ...which then again requires processing `X::A::{{initializer}}`, completing the cycle.
1813

1914
error: aborting due to previous error
2015

src/test/ui/issue-23302-2.stderr

+2-7
Original file line numberDiff line numberDiff line change
@@ -4,17 +4,12 @@ error[E0391]: cyclic dependency detected
44
LL | A = Y::B as isize, //~ ERROR E0391
55
| ^^^^^^^^^^^^^ cyclic reference
66
|
7-
note: the cycle begins when const-evaluating `Y::A::{{initializer}}`...
7+
note: the cycle begins when processing `Y::A::{{initializer}}`...
88
--> $DIR/issue-23302-2.rs:14:9
99
|
1010
LL | A = Y::B as isize, //~ ERROR E0391
1111
| ^^^^^^^^^^^^^
12-
note: ...which then requires computing layout of `Y`...
13-
--> $DIR/issue-23302-2.rs:14:9
14-
|
15-
LL | A = Y::B as isize, //~ ERROR E0391
16-
| ^^^^
17-
= note: ...which then again requires const-evaluating `Y::A::{{initializer}}`, completing the cycle.
12+
= note: ...which then again requires processing `Y::A::{{initializer}}`, completing the cycle.
1813

1914
error: aborting due to previous error
2015

src/test/ui/issue-36163.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -8,10 +8,10 @@
88
// option. This file may not be copied, modified, or distributed
99
// except according to those terms.
1010

11-
const A: isize = Foo::B as isize;
11+
const A: isize = Foo::B as isize; //~ ERROR E0391
1212

1313
enum Foo {
14-
B = A, //~ ERROR E0391
14+
B = A,
1515
}
1616

1717
fn main() {}

src/test/ui/issue-36163.stderr

+8-18
Original file line numberDiff line numberDiff line change
@@ -1,30 +1,20 @@
11
error[E0391]: cyclic dependency detected
2-
--> $DIR/issue-36163.rs:14:9
2+
--> $DIR/issue-36163.rs:11:18
33
|
4-
LL | B = A, //~ ERROR E0391
5-
| ^ cyclic reference
4+
LL | const A: isize = Foo::B as isize; //~ ERROR E0391
5+
| ^^^^^^^^^^^^^^^ cyclic reference
66
|
7-
note: the cycle begins when const-evaluating `Foo::B::{{initializer}}`...
7+
note: the cycle begins when processing `Foo::B::{{initializer}}`...
88
--> $DIR/issue-36163.rs:14:9
99
|
10-
LL | B = A, //~ ERROR E0391
10+
LL | B = A,
1111
| ^
12-
note: ...which then requires processing `Foo::B::{{initializer}}`...
12+
note: ...which then requires processing `A`...
1313
--> $DIR/issue-36163.rs:14:9
1414
|
15-
LL | B = A, //~ ERROR E0391
15+
LL | B = A,
1616
| ^
17-
note: ...which then requires const-evaluating `A`...
18-
--> $DIR/issue-36163.rs:11:1
19-
|
20-
LL | const A: isize = Foo::B as isize;
21-
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
22-
note: ...which then requires computing layout of `Foo`...
23-
--> $DIR/issue-36163.rs:11:18
24-
|
25-
LL | const A: isize = Foo::B as isize;
26-
| ^^^^^^
27-
= note: ...which then again requires const-evaluating `Foo::B::{{initializer}}`, completing the cycle.
17+
= note: ...which then again requires processing `Foo::B::{{initializer}}`, completing the cycle.
2818

2919
error: aborting due to previous error
3020

0 commit comments

Comments
 (0)