2023-07-27 15:51:02 +00:00
|
|
|
// FIXME run-pass
|
|
|
|
// known-bug: #110395
|
2021-09-03 09:53:57 +00:00
|
|
|
// revisions: stock precise
|
2021-09-01 11:55:16 +00:00
|
|
|
#![feature(const_trait_impl)]
|
|
|
|
#![feature(const_mut_refs)]
|
2022-01-19 12:59:28 -08:00
|
|
|
#![feature(never_type)]
|
2023-07-27 15:51:02 +00:00
|
|
|
// #![cfg_attr(precise, feature(const_precise_live_drops))]
|
2021-09-01 11:55:16 +00:00
|
|
|
|
2022-03-21 16:52:41 +11:00
|
|
|
use std::marker::Destruct;
|
|
|
|
|
2021-09-01 13:07:49 +00:00
|
|
|
struct S<'a>(&'a mut u8);
|
2021-09-01 11:55:16 +00:00
|
|
|
|
2021-09-01 13:07:49 +00:00
|
|
|
impl<'a> const Drop for S<'a> {
|
2021-09-01 11:55:16 +00:00
|
|
|
fn drop(&mut self) {
|
2021-09-01 13:07:49 +00:00
|
|
|
*self.0 += 1;
|
2021-09-01 11:55:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-21 16:52:41 +11:00
|
|
|
const fn a<T: ~const Destruct>(_: T) {}
|
2023-07-27 15:51:02 +00:00
|
|
|
//FIXME ~^ ERROR destructor of
|
2021-09-01 11:55:16 +00:00
|
|
|
|
2021-09-01 13:07:49 +00:00
|
|
|
const fn b() -> u8 {
|
|
|
|
let mut c = 0;
|
|
|
|
let _ = S(&mut c);
|
2023-07-27 15:51:02 +00:00
|
|
|
//FIXME ~^ ERROR destructor of
|
2021-09-01 13:07:49 +00:00
|
|
|
a(S(&mut c));
|
|
|
|
c
|
|
|
|
}
|
|
|
|
|
|
|
|
const C: u8 = b();
|
2021-09-01 11:55:16 +00:00
|
|
|
|
2021-09-01 16:34:28 +00:00
|
|
|
macro_rules! implements_const_drop {
|
|
|
|
($($exp:expr),*$(,)?) => {
|
|
|
|
$(
|
|
|
|
const _: () = a($exp);
|
|
|
|
)*
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[allow(dead_code)]
|
|
|
|
mod t {
|
|
|
|
pub struct Foo;
|
|
|
|
pub enum Bar { A }
|
|
|
|
pub fn foo() {}
|
|
|
|
pub struct ConstDrop;
|
|
|
|
|
|
|
|
impl const Drop for ConstDrop {
|
|
|
|
fn drop(&mut self) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct HasConstDrop(pub ConstDrop);
|
|
|
|
pub struct TrivialFields(pub u8, pub i8, pub usize, pub isize);
|
2022-01-18 01:42:35 -08:00
|
|
|
|
2022-08-28 06:27:31 +00:00
|
|
|
#[const_trait]
|
2022-01-18 01:42:35 -08:00
|
|
|
pub trait SomeTrait {
|
|
|
|
fn foo();
|
|
|
|
}
|
|
|
|
impl const SomeTrait for () {
|
|
|
|
fn foo() {}
|
|
|
|
}
|
2022-01-19 20:07:04 -08:00
|
|
|
// non-const impl
|
|
|
|
impl SomeTrait for i32 {
|
|
|
|
fn foo() {}
|
|
|
|
}
|
2022-01-18 01:42:35 -08:00
|
|
|
|
2023-11-11 15:01:58 +01:00
|
|
|
// FIXME(effects): This should be a `const` bound instead of a `~const` one.
|
2022-10-21 13:49:11 +00:00
|
|
|
pub struct ConstDropWithBound<T: ~const SomeTrait>(pub core::marker::PhantomData<T>);
|
2022-01-18 01:42:35 -08:00
|
|
|
|
|
|
|
impl<T: ~const SomeTrait> const Drop for ConstDropWithBound<T> {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
T::foo();
|
|
|
|
}
|
|
|
|
}
|
2022-01-19 20:07:04 -08:00
|
|
|
|
|
|
|
pub struct ConstDropWithNonconstBound<T: SomeTrait>(pub core::marker::PhantomData<T>);
|
|
|
|
|
|
|
|
impl<T: SomeTrait> const Drop for ConstDropWithNonconstBound<T> {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
// Note: we DON'T use the `T: SomeTrait` bound
|
|
|
|
}
|
|
|
|
}
|
2021-09-01 16:34:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
use t::*;
|
|
|
|
|
|
|
|
implements_const_drop! {
|
|
|
|
1u8,
|
|
|
|
2,
|
|
|
|
3.0,
|
|
|
|
Foo,
|
|
|
|
Bar::A,
|
|
|
|
foo,
|
|
|
|
ConstDrop,
|
|
|
|
HasConstDrop(ConstDrop),
|
|
|
|
TrivialFields(1, 2, 3, 4),
|
|
|
|
&1,
|
|
|
|
&1 as *const i32,
|
2022-01-18 01:42:35 -08:00
|
|
|
ConstDropWithBound::<()>,
|
2022-01-19 20:07:04 -08:00
|
|
|
ConstDropWithNonconstBound::<i32>,
|
2022-01-19 12:59:28 -08:00
|
|
|
Result::<i32, !>::Ok(1),
|
2021-09-01 16:34:28 +00:00
|
|
|
}
|
|
|
|
|
2021-09-01 13:07:49 +00:00
|
|
|
fn main() {
|
2022-07-25 22:36:03 +02:00
|
|
|
struct HasDropGlue(#[allow(unused_tuple_struct_fields)] Box<u8>);
|
2021-09-01 16:34:28 +00:00
|
|
|
struct HasDropImpl;
|
|
|
|
impl Drop for HasDropImpl {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
println!("not trivial drop");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// These types should pass because ~const in a non-const context should have no effect.
|
|
|
|
a(HasDropGlue(Box::new(0)));
|
|
|
|
a(HasDropImpl);
|
|
|
|
|
2021-09-01 13:07:49 +00:00
|
|
|
assert_eq!(C, 2);
|
|
|
|
}
|