forked from rust-lang/rust
-
Notifications
You must be signed in to change notification settings - Fork 7
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Auto merge of rust-lang#137669 - DianQK:fn-atts-virtual, r=saethlin
Don't infer attributes of virtual calls based on the function body Fixes (after backport) rust-lang#137646. Since we don't know the exact implementation of the virtual call, it might write to parameters, we can't infer the readonly attribute.
- Loading branch information
Showing
4 changed files
with
123 additions
and
38 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,37 @@ | ||
//! Regression test for https://github.com/rust-lang/rust/issues/137646. | ||
//! Since we don't know the exact implementation of the virtual call, | ||
//! it might write to parameters, we can't infer the readonly attribute. | ||
//@ compile-flags: -C opt-level=3 -C no-prepopulate-passes | ||
|
||
#![crate_type = "lib"] | ||
#![feature(rustc_attrs)] | ||
|
||
pub trait Trait { | ||
#[rustc_nounwind] | ||
fn m(&self, _: (i32, i32, i32)) {} | ||
} | ||
|
||
#[no_mangle] | ||
pub fn foo(trait_: &dyn Trait) { | ||
// CHECK-LABEL: @foo( | ||
// CHECK: call void | ||
// CHECK-NOT: readonly | ||
trait_.m((1, 1, 1)); | ||
} | ||
|
||
#[no_mangle] | ||
#[rustc_nounwind] | ||
pub fn foo_nounwind(trait_: &dyn Trait) { | ||
// CHECK-LABEL: @foo_nounwind( | ||
// FIXME: Here should be invoke. | ||
// COM: CHECK: invoke | ||
trait_.m((1, 1, 1)); | ||
} | ||
|
||
#[no_mangle] | ||
pub extern "C" fn c_nounwind(trait_: &dyn Trait) { | ||
// CHECK-LABEL: @c_nounwind( | ||
// FIXME: Here should be invoke. | ||
// COM: CHECK: invoke | ||
trait_.m((1, 1, 1)); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,45 @@ | ||
//! Regression test for https://github.com/rust-lang/rust/issues/137646. | ||
//! The parameter value at all calls to `check` should be `(1, 1, 1)`. | ||
//@ run-pass | ||
|
||
use std::hint::black_box; | ||
|
||
type T = (i32, i32, i32); | ||
|
||
pub trait Trait { | ||
fn m(&self, _: T, _: T) {} | ||
} | ||
|
||
impl Trait for () { | ||
fn m(&self, mut _v1: T, v2: T) { | ||
_v1 = (0, 0, 0); | ||
check(v2); | ||
} | ||
} | ||
|
||
pub fn run_1(trait_: &dyn Trait) { | ||
let v1 = (1, 1, 1); | ||
let v2 = (1, 1, 1); | ||
trait_.m(v1, v2); | ||
} | ||
|
||
pub fn run_2(trait_: &dyn Trait) { | ||
let v1 = (1, 1, 1); | ||
let v2 = (1, 1, 1); | ||
trait_.m(v1, v2); | ||
check(v1); | ||
check(v2); | ||
} | ||
|
||
#[inline(never)] | ||
fn check(v: T) { | ||
assert_eq!(v, (1, 1, 1)); | ||
} | ||
|
||
fn main() { | ||
black_box(run_1 as fn(&dyn Trait)); | ||
black_box(run_2 as fn(&dyn Trait)); | ||
run_1(&()); | ||
run_2(&()); | ||
} |