rust/tests/ui/escape_analysis.rs

160 lines
2.8 KiB
Rust
Raw Normal View History

2018-10-06 09:18:06 -07:00
// Copyright 2014-2018 The Rust Project Developers. See the COPYRIGHT
// file at the top-level directory of this distribution.
//
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
// option. This file may not be copied, modified, or distributed
// except according to those terms.
#![feature(box_syntax)]
#![allow(clippy::borrowed_box, clippy::needless_pass_by_value, clippy::unused_unit)]
#![warn(clippy::boxed_local)]
2015-12-04 15:42:53 +05:30
#[derive(Clone)]
struct A;
impl A {
2018-12-09 23:26:16 +01:00
fn foo(&self) {}
2015-12-04 15:42:53 +05:30
}
2016-02-01 20:37:07 +01:00
trait Z {
fn bar(&self);
}
impl Z for A {
fn bar(&self) {
//nothing
}
}
2018-12-09 23:26:16 +01:00
fn main() {}
2015-12-04 15:42:53 +05:30
2016-02-01 20:37:07 +01:00
fn ok_box_trait(boxed_trait: &Box<Z>) {
let boxed_local = boxed_trait;
// done
}
2015-12-04 15:42:53 +05:30
fn warn_call() {
2017-02-08 14:58:07 +01:00
let x = box A;
2016-09-09 20:24:00 +02:00
x.foo();
2015-12-04 15:42:53 +05:30
}
2017-02-08 14:58:07 +01:00
fn warn_arg(x: Box<A>) {
2015-12-28 23:12:57 +09:00
x.foo();
}
fn nowarn_closure_arg() {
let x = Some(box A);
x.map_or((), |x| take_ref(&x));
}
2015-12-04 15:42:53 +05:30
fn warn_rename_call() {
let x = box A;
2017-02-08 14:58:07 +01:00
let y = x;
2015-12-04 15:42:53 +05:30
y.foo(); // via autoderef
}
fn warn_notuse() {
2017-02-08 14:58:07 +01:00
let bz = box A;
2015-12-04 15:42:53 +05:30
}
fn warn_pass() {
2017-02-08 14:58:07 +01:00
let bz = box A;
2015-12-04 15:42:53 +05:30
take_ref(&bz); // via deref coercion
}
fn nowarn_return() -> Box<A> {
box A // moved out, "escapes"
2015-12-04 15:42:53 +05:30
}
fn nowarn_move() {
let bx = box A;
drop(bx) // moved in, "escapes"
}
fn nowarn_call() {
let bx = box A;
bx.clone(); // method only available to Box, not via autoderef
}
fn nowarn_pass() {
let bx = box A;
take_box(&bx); // fn needs &Box
}
fn take_box(x: &Box<A>) {}
fn take_ref(x: &A) {}
fn nowarn_ref_take() {
// false positive, should actually warn
2017-02-08 14:58:07 +01:00
let x = box A;
2015-12-04 15:42:53 +05:30
let y = &x;
take_box(y);
}
fn nowarn_match() {
let x = box A; // moved into a match
match x {
2018-12-09 23:26:16 +01:00
y => drop(y),
2015-12-04 15:42:53 +05:30
}
}
fn warn_match() {
2017-02-08 14:58:07 +01:00
let x = box A;
2018-12-09 23:26:16 +01:00
match &x {
// not moved
ref y => (),
2015-12-04 15:42:53 +05:30
}
2015-12-28 23:12:57 +09:00
}
2016-07-10 18:53:50 +05:30
fn nowarn_large_array() {
// should not warn, is large array
// and should not be on stack
let x = box [1; 10000];
2018-12-09 23:26:16 +01:00
match &x {
// not moved
ref y => (),
2016-07-10 18:53:50 +05:30
}
}
/// ICE regression test
pub trait Foo {
type Item;
}
impl<'a> Foo for &'a () {
type Item = ();
}
pub struct PeekableSeekable<I: Foo> {
_peeked: I::Item,
}
2018-12-09 23:26:16 +01:00
pub fn new(_needs_name: Box<PeekableSeekable<&()>>) -> () {}
/// Regression for #916, #1123
///
/// This shouldn't warn for `boxed_local`as the implementation of a trait
/// can't change much about the trait definition.
trait BoxedAction {
fn do_sth(self: Box<Self>);
}
impl BoxedAction for u64 {
fn do_sth(self: Box<Self>) {
println!("{}", *self)
}
}
/// Regression for #1478
///
/// This shouldn't warn for `boxed_local`as self itself is a box type.
trait MyTrait {
fn do_sth(self);
}
impl<T> MyTrait for Box<T> {
fn do_sth(self) {}
}