Auto merge of #71593 - Dylan-DPC:rollup-ms7j94u, r=Dylan-DPC

Rollup of 5 pull requests

Successful merges:

 - #71421 (Add a function to turn Box<T> into Box<[T]>)
 - #71537 (Remove support for self-opening)
 - #71551 (Minor refactoring around IndexVec usage in generator transformation)
 - #71569 ([miri] Throw UB if target size and data size don't match)
 - #71576 (check that `AsRef` and `AsMut` are inlined)

Failed merges:

 - #71558 (Cleanup and document `-Z tls-model` )

r? @ghost
This commit is contained in:
bors 2020-04-26 20:12:48 +00:00
commit e83f756349
20 changed files with 263 additions and 139 deletions

View File

@ -239,6 +239,16 @@ impl<T> Box<T> {
pub fn pin(x: T) -> Pin<Box<T>> { pub fn pin(x: T) -> Pin<Box<T>> {
(box x).into() (box x).into()
} }
/// Converts a `Box<T>` into a `Box<[T]>`
///
/// This conversion does not allocate on the heap and happens in place.
///
#[unstable(feature = "box_into_boxed_slice", issue = "71582")]
pub fn into_boxed_slice(boxed: Box<T>) -> Box<[T]> {
// *mut T and *mut [T; 1] have the same size and alignment
unsafe { Box::from_raw(Box::into_raw(boxed) as *mut [T; 1] as *mut [T]) }
}
} }
impl<T> Box<[T]> { impl<T> Box<[T]> {

View File

@ -205,7 +205,7 @@ pub fn spawn_thread_pool<F: FnOnce() -> R + Send, R: Send>(
} }
fn load_backend_from_dylib(path: &Path) -> fn() -> Box<dyn CodegenBackend> { fn load_backend_from_dylib(path: &Path) -> fn() -> Box<dyn CodegenBackend> {
let lib = DynamicLibrary::open(Some(path)).unwrap_or_else(|err| { let lib = DynamicLibrary::open(path).unwrap_or_else(|err| {
let err = format!("couldn't load codegen backend {:?}: {:?}", path, err); let err = format!("couldn't load codegen backend {:?}: {:?}", path, err);
early_error(ErrorOutputType::default(), &err); early_error(ErrorOutputType::default(), &err);
}); });

View File

@ -591,7 +591,7 @@ impl<'a> CrateLoader<'a> {
// Make sure the path contains a / or the linker will search for it. // Make sure the path contains a / or the linker will search for it.
let path = env::current_dir().unwrap().join(path); let path = env::current_dir().unwrap().join(path);
let lib = match DynamicLibrary::open(Some(&path)) { let lib = match DynamicLibrary::open(&path) {
Ok(lib) => lib, Ok(lib) => lib,
Err(err) => self.sess.span_fatal(span, &err), Err(err) => self.sess.span_fatal(span, &err),
}; };

View File

@ -16,10 +16,9 @@ impl Drop for DynamicLibrary {
} }
impl DynamicLibrary { impl DynamicLibrary {
/// Lazily open a dynamic library. When passed None it gives a /// Lazily open a dynamic library.
/// handle to the calling process pub fn open(filename: &Path) -> Result<DynamicLibrary, String> {
pub fn open(filename: Option<&Path>) -> Result<DynamicLibrary, String> { let maybe_library = dl::open(filename.as_os_str());
let maybe_library = dl::open(filename.map(|path| path.as_os_str()));
// The dynamic library must not be constructed if there is // The dynamic library must not be constructed if there is
// an error opening the library so the destructor does not // an error opening the library so the destructor does not
@ -57,22 +56,11 @@ mod dl {
use std::ptr; use std::ptr;
use std::str; use std::str;
pub(super) fn open(filename: Option<&OsStr>) -> Result<*mut u8, String> { pub(super) fn open(filename: &OsStr) -> Result<*mut u8, String> {
check_for_errors_in(|| unsafe { check_for_errors_in(|| unsafe {
match filename {
Some(filename) => open_external(filename),
None => open_internal(),
}
})
}
unsafe fn open_external(filename: &OsStr) -> *mut u8 {
let s = CString::new(filename.as_bytes()).unwrap(); let s = CString::new(filename.as_bytes()).unwrap();
libc::dlopen(s.as_ptr(), libc::RTLD_LAZY) as *mut u8 libc::dlopen(s.as_ptr(), libc::RTLD_LAZY) as *mut u8
} })
unsafe fn open_internal() -> *mut u8 {
libc::dlopen(ptr::null(), libc::RTLD_LAZY) as *mut u8
} }
fn check_for_errors_in<T, F>(f: F) -> Result<T, String> fn check_for_errors_in<T, F>(f: F) -> Result<T, String>
@ -124,10 +112,10 @@ mod dl {
use winapi::shared::minwindef::HMODULE; use winapi::shared::minwindef::HMODULE;
use winapi::um::errhandlingapi::SetThreadErrorMode; use winapi::um::errhandlingapi::SetThreadErrorMode;
use winapi::um::libloaderapi::{FreeLibrary, GetModuleHandleExW, GetProcAddress, LoadLibraryW}; use winapi::um::libloaderapi::{FreeLibrary, GetProcAddress, LoadLibraryW};
use winapi::um::winbase::SEM_FAILCRITICALERRORS; use winapi::um::winbase::SEM_FAILCRITICALERRORS;
pub(super) fn open(filename: Option<&OsStr>) -> Result<*mut u8, String> { pub(super) fn open(filename: &OsStr) -> Result<*mut u8, String> {
// disable "dll load failed" error dialog. // disable "dll load failed" error dialog.
let prev_error_mode = unsafe { let prev_error_mode = unsafe {
let new_error_mode = SEM_FAILCRITICALERRORS; let new_error_mode = SEM_FAILCRITICALERRORS;
@ -139,22 +127,9 @@ mod dl {
prev_error_mode prev_error_mode
}; };
let result = match filename {
Some(filename) => {
let filename_str: Vec<_> = filename.encode_wide().chain(Some(0)).collect(); let filename_str: Vec<_> = filename.encode_wide().chain(Some(0)).collect();
let result = unsafe { LoadLibraryW(filename_str.as_ptr()) } as *mut u8; let result = unsafe { LoadLibraryW(filename_str.as_ptr()) } as *mut u8;
ptr_result(result) let result = ptr_result(result);
}
None => {
let mut handle = ptr::null_mut();
let succeeded = unsafe { GetModuleHandleExW(0, ptr::null(), &mut handle) };
if succeeded == 0 {
Err(io::Error::last_os_error().to_string())
} else {
Ok(handle as *mut u8)
}
}
};
unsafe { unsafe {
SetThreadErrorMode(prev_error_mode, ptr::null_mut()); SetThreadErrorMode(prev_error_mode, ptr::null_mut());

View File

@ -1,32 +1,4 @@
use super::*; use super::*;
use std::mem;
#[test]
fn test_loading_atoi() {
if cfg!(windows) {
return;
}
// The C library does not need to be loaded since it is already linked in
let lib = match DynamicLibrary::open(None) {
Err(error) => panic!("Could not load self as module: {}", error),
Ok(lib) => lib,
};
let atoi: extern "C" fn(*const libc::c_char) -> libc::c_int = unsafe {
match lib.symbol("atoi") {
Err(error) => panic!("Could not load function atoi: {}", error),
Ok(atoi) => mem::transmute::<*mut u8, _>(atoi),
}
};
let argument = CString::new("1383428980").unwrap();
let expected_result = 0x52757374;
let result = atoi(argument.as_ptr());
if result != expected_result {
panic!("atoi({:?}) != {} but equaled {} instead", argument, expected_result, result)
}
}
#[test] #[test]
fn test_errors_do_not_crash() { fn test_errors_do_not_crash() {
@ -39,7 +11,7 @@ fn test_errors_do_not_crash() {
// Open /dev/null as a library to get an error, and make sure // Open /dev/null as a library to get an error, and make sure
// that only causes an error, and not a crash. // that only causes an error, and not a crash.
let path = Path::new("/dev/null"); let path = Path::new("/dev/null");
match DynamicLibrary::open(Some(&path)) { match DynamicLibrary::open(&path) {
Err(_) => {} Err(_) => {}
Ok(_) => panic!("Successfully opened the empty library."), Ok(_) => panic!("Successfully opened the empty library."),
} }

View File

@ -360,6 +360,11 @@ pub enum UndefinedBehaviorInfo {
InvalidUndefBytes(Option<Pointer>), InvalidUndefBytes(Option<Pointer>),
/// Working with a local that is not currently live. /// Working with a local that is not currently live.
DeadLocal, DeadLocal,
/// Data size is not equal to target size.
ScalarSizeMismatch {
target_size: u64,
data_size: u64,
},
} }
impl fmt::Debug for UndefinedBehaviorInfo { impl fmt::Debug for UndefinedBehaviorInfo {
@ -421,6 +426,11 @@ impl fmt::Debug for UndefinedBehaviorInfo {
"using uninitialized data, but this operation requires initialized memory" "using uninitialized data, but this operation requires initialized memory"
), ),
DeadLocal => write!(f, "accessing a dead local variable"), DeadLocal => write!(f, "accessing a dead local variable"),
ScalarSizeMismatch { target_size, data_size } => write!(
f,
"scalar size mismatch: expected {} bytes but got {} bytes instead",
target_size, data_size
),
} }
} }
} }

View File

@ -393,7 +393,12 @@ impl<'tcx, Tag> Scalar<Tag> {
assert_ne!(target_size.bytes(), 0, "you should never look at the bits of a ZST"); assert_ne!(target_size.bytes(), 0, "you should never look at the bits of a ZST");
match self { match self {
Scalar::Raw { data, size } => { Scalar::Raw { data, size } => {
assert_eq!(target_size.bytes(), u64::from(size)); if target_size.bytes() != u64::from(size) {
throw_ub!(ScalarSizeMismatch {
target_size: target_size.bytes(),
data_size: u64::from(size),
});
}
Scalar::check_data(data, size); Scalar::check_data(data, size);
Ok(data) Ok(data)
} }

View File

@ -210,8 +210,7 @@ struct TransformVisitor<'tcx> {
remap: FxHashMap<Local, (Ty<'tcx>, VariantIdx, usize)>, remap: FxHashMap<Local, (Ty<'tcx>, VariantIdx, usize)>,
// A map from a suspension point in a block to the locals which have live storage at that point // A map from a suspension point in a block to the locals which have live storage at that point
// FIXME(eddyb) This should use `IndexVec<BasicBlock, Option<_>>`. storage_liveness: IndexVec<BasicBlock, Option<liveness::LiveVarSet>>,
storage_liveness: FxHashMap<BasicBlock, liveness::LiveVarSet>,
// A list of suspension points, generated during the transform // A list of suspension points, generated during the transform
suspension_points: Vec<SuspensionPoint<'tcx>>, suspension_points: Vec<SuspensionPoint<'tcx>>,
@ -338,7 +337,7 @@ impl MutVisitor<'tcx> for TransformVisitor<'tcx> {
resume, resume,
resume_arg, resume_arg,
drop, drop,
storage_liveness: self.storage_liveness.get(&block).unwrap().clone(), storage_liveness: self.storage_liveness[block].clone().unwrap(),
}); });
VariantIdx::new(state) VariantIdx::new(state)
@ -404,8 +403,7 @@ fn replace_local<'tcx>(
is_block_tail: None, is_block_tail: None,
local_info: LocalInfo::Other, local_info: LocalInfo::Other,
}; };
let new_local = Local::new(body.local_decls.len()); let new_local = body.local_decls.push(new_decl);
body.local_decls.push(new_decl);
body.local_decls.swap(local, new_local); body.local_decls.swap(local, new_local);
RenameLocalVisitor { from: local, to: new_local, tcx }.visit_body(body); RenameLocalVisitor { from: local, to: new_local, tcx }.visit_body(body);
@ -431,7 +429,7 @@ struct LivenessInfo {
/// For every suspending block, the locals which are storage-live across /// For every suspending block, the locals which are storage-live across
/// that suspension point. /// that suspension point.
storage_liveness: FxHashMap<BasicBlock, liveness::LiveVarSet>, storage_liveness: IndexVec<BasicBlock, Option<liveness::LiveVarSet>>,
} }
fn locals_live_across_suspend_points( fn locals_live_across_suspend_points(
@ -472,7 +470,7 @@ fn locals_live_across_suspend_points(
let mut liveness = liveness::liveness_of_locals(body); let mut liveness = liveness::liveness_of_locals(body);
liveness::dump_mir(tcx, "generator_liveness", source, body_ref, &liveness); liveness::dump_mir(tcx, "generator_liveness", source, body_ref, &liveness);
let mut storage_liveness_map = FxHashMap::default(); let mut storage_liveness_map = IndexVec::from_elem(None, body.basic_blocks());
let mut live_locals_at_suspension_points = Vec::new(); let mut live_locals_at_suspension_points = Vec::new();
for (block, data) in body.basic_blocks().iter_enumerated() { for (block, data) in body.basic_blocks().iter_enumerated() {
@ -502,7 +500,7 @@ fn locals_live_across_suspend_points(
// Store the storage liveness for later use so we can restore the state // Store the storage liveness for later use so we can restore the state
// after a suspension point // after a suspension point
storage_liveness_map.insert(block, storage_liveness); storage_liveness_map[block] = Some(storage_liveness);
requires_storage_cursor.seek_before(loc); requires_storage_cursor.seek_before(loc);
let storage_required = requires_storage_cursor.get().clone(); let storage_required = requires_storage_cursor.get().clone();
@ -690,7 +688,7 @@ fn compute_layout<'tcx>(
) -> ( ) -> (
FxHashMap<Local, (Ty<'tcx>, VariantIdx, usize)>, FxHashMap<Local, (Ty<'tcx>, VariantIdx, usize)>,
GeneratorLayout<'tcx>, GeneratorLayout<'tcx>,
FxHashMap<BasicBlock, liveness::LiveVarSet>, IndexVec<BasicBlock, Option<liveness::LiveVarSet>>,
) { ) {
// Use a liveness analysis to compute locals which are live across a suspension point // Use a liveness analysis to compute locals which are live across a suspension point
let LivenessInfo { let LivenessInfo {
@ -925,14 +923,12 @@ fn create_generator_drop_shim<'tcx>(
} }
fn insert_term_block<'tcx>(body: &mut Body<'tcx>, kind: TerminatorKind<'tcx>) -> BasicBlock { fn insert_term_block<'tcx>(body: &mut Body<'tcx>, kind: TerminatorKind<'tcx>) -> BasicBlock {
let term_block = BasicBlock::new(body.basic_blocks().len());
let source_info = source_info(body); let source_info = source_info(body);
body.basic_blocks_mut().push(BasicBlockData { body.basic_blocks_mut().push(BasicBlockData {
statements: Vec::new(), statements: Vec::new(),
terminator: Some(Terminator { source_info, kind }), terminator: Some(Terminator { source_info, kind }),
is_cleanup: false, is_cleanup: false,
}); })
term_block
} }
fn insert_panic_block<'tcx>( fn insert_panic_block<'tcx>(
@ -1030,9 +1026,8 @@ fn create_generator_resume_function<'tcx>(
// Poison the generator when it unwinds // Poison the generator when it unwinds
if can_unwind { if can_unwind {
let poison_block = BasicBlock::new(body.basic_blocks().len());
let source_info = source_info(body); let source_info = source_info(body);
body.basic_blocks_mut().push(BasicBlockData { let poison_block = body.basic_blocks_mut().push(BasicBlockData {
statements: vec![transform.set_discr(VariantIdx::new(POISONED), source_info)], statements: vec![transform.set_discr(VariantIdx::new(POISONED), source_info)],
terminator: Some(Terminator { source_info, kind: TerminatorKind::Resume }), terminator: Some(Terminator { source_info, kind: TerminatorKind::Resume }),
is_cleanup: true, is_cleanup: true,
@ -1105,21 +1100,19 @@ fn source_info(body: &Body<'_>) -> SourceInfo {
fn insert_clean_drop(body: &mut Body<'_>) -> BasicBlock { fn insert_clean_drop(body: &mut Body<'_>) -> BasicBlock {
let return_block = insert_term_block(body, TerminatorKind::Return); let return_block = insert_term_block(body, TerminatorKind::Return);
// Create a block to destroy an unresumed generators. This can only destroy upvars.
let drop_clean = BasicBlock::new(body.basic_blocks().len());
let term = TerminatorKind::Drop { let term = TerminatorKind::Drop {
location: Place::from(SELF_ARG), location: Place::from(SELF_ARG),
target: return_block, target: return_block,
unwind: None, unwind: None,
}; };
let source_info = source_info(body); let source_info = source_info(body);
// Create a block to destroy an unresumed generators. This can only destroy upvars.
body.basic_blocks_mut().push(BasicBlockData { body.basic_blocks_mut().push(BasicBlockData {
statements: Vec::new(), statements: Vec::new(),
terminator: Some(Terminator { source_info, kind: term }), terminator: Some(Terminator { source_info, kind: term }),
is_cleanup: false, is_cleanup: false,
}); })
drop_clean
} }
/// An operation that can be performed on a generator. /// An operation that can be performed on a generator.
@ -1151,7 +1144,6 @@ fn create_cases<'tcx>(
.filter_map(|point| { .filter_map(|point| {
// Find the target for this suspension point, if applicable // Find the target for this suspension point, if applicable
operation.target_block(point).map(|target| { operation.target_block(point).map(|target| {
let block = BasicBlock::new(body.basic_blocks().len());
let mut statements = Vec::new(); let mut statements = Vec::new();
// Create StorageLive instructions for locals with live storage // Create StorageLive instructions for locals with live storage
@ -1186,7 +1178,7 @@ fn create_cases<'tcx>(
} }
// Then jump to the real target // Then jump to the real target
body.basic_blocks_mut().push(BasicBlockData { let block = body.basic_blocks_mut().push(BasicBlockData {
statements, statements,
terminator: Some(Terminator { terminator: Some(Terminator {
source_info, source_info,

View File

@ -76,7 +76,7 @@ fn dylink_registrar(
// Make sure the path contains a / or the linker will search for it. // Make sure the path contains a / or the linker will search for it.
let path = env::current_dir().unwrap().join(&path); let path = env::current_dir().unwrap().join(&path);
let lib = match DynamicLibrary::open(Some(&path)) { let lib = match DynamicLibrary::open(&path) {
Ok(lib) => lib, Ok(lib) => lib,
// this is fatal: there are almost certainly macros we need // this is fatal: there are almost certainly macros we need
// inside this crate, so continue would spew "macro undefined" // inside this crate, so continue would spew "macro undefined"

View File

@ -0,0 +1,27 @@
// EMIT_MIR rustc.a.Inline.after.mir
pub fn a<T>(x: &mut [T]) -> &mut [T] {
x.as_mut()
}
// EMIT_MIR rustc.b.Inline.after.mir
pub fn b<T>(x: &mut Box<T>) -> &mut T {
x.as_mut()
}
// EMIT_MIR rustc.c.Inline.after.mir
pub fn c<T>(x: &[T]) -> &[T] {
x.as_ref()
}
// EMIT_MIR rustc.d.Inline.after.mir
pub fn d<T>(x: &Box<T>) -> &T {
x.as_ref()
}
fn main() {
let mut boxed = Box::new(1);
println!("{:?}", a(&mut [1]));
println!("{:?}", b(&mut boxed));
println!("{:?}", c(&[1]));
println!("{:?}", d(&boxed));
}

View File

@ -0,0 +1,30 @@
// MIR for `a` after Inline
fn a(_1: &mut [T]) -> &mut [T] {
debug x => _1; // in scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:2:13: 2:14
let mut _0: &mut [T]; // return place in scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:2:29: 2:37
let mut _2: &mut [T]; // in scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:3:5: 3:15
let mut _3: &mut [T]; // in scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:3:5: 3:15
let mut _4: &mut [T]; // in scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:3:5: 3:6
scope 1 {
debug self => _4; // in scope 1 at $SRC_DIR/libcore/convert/mod.rs:LL:COL
let mut _5: &mut [T]; // in scope 1 at $DIR/issue-58867-inline-as-ref-as-mut.rs:3:5: 3:15
}
bb0: {
StorageLive(_2); // bb0[0]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:3:5: 3:15
StorageLive(_3); // bb0[1]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:3:5: 3:15
StorageLive(_4); // bb0[2]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:3:5: 3:6
_4 = &mut (*_1); // bb0[3]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:3:5: 3:6
StorageLive(_5); // bb0[4]: scope 1 at $SRC_DIR/libcore/convert/mod.rs:LL:COL
_5 = _4; // bb0[5]: scope 1 at $SRC_DIR/libcore/convert/mod.rs:LL:COL
_3 = _5; // bb0[6]: scope 1 at $SRC_DIR/libcore/convert/mod.rs:LL:COL
StorageDead(_5); // bb0[7]: scope 1 at $SRC_DIR/libcore/convert/mod.rs:LL:COL
_2 = &mut (*_3); // bb0[8]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:3:5: 3:15
StorageDead(_4); // bb0[9]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:3:14: 3:15
_0 = &mut (*_2); // bb0[10]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:3:5: 3:15
StorageDead(_3); // bb0[11]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:4:1: 4:2
StorageDead(_2); // bb0[12]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:4:1: 4:2
return; // bb0[13]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:4:2: 4:2
}
}

View File

@ -0,0 +1,34 @@
// MIR for `b` after Inline
fn b(_1: &mut std::boxed::Box<T>) -> &mut T {
debug x => _1; // in scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:7:13: 7:14
let mut _0: &mut T; // return place in scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:7:32: 7:38
let mut _2: &mut T; // in scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:8:5: 8:15
let mut _3: &mut T; // in scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:8:5: 8:15
let mut _4: &mut std::boxed::Box<T>; // in scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:8:5: 8:6
scope 1 {
debug self => _4; // in scope 1 at $SRC_DIR/liballoc/boxed.rs:LL:COL
let mut _5: &mut T; // in scope 1 at $DIR/issue-58867-inline-as-ref-as-mut.rs:8:5: 8:15
let mut _6: &mut T; // in scope 1 at $DIR/issue-58867-inline-as-ref-as-mut.rs:8:5: 8:15
}
bb0: {
StorageLive(_2); // bb0[0]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:8:5: 8:15
StorageLive(_3); // bb0[1]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:8:5: 8:15
StorageLive(_4); // bb0[2]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:8:5: 8:6
_4 = &mut (*_1); // bb0[3]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:8:5: 8:6
StorageLive(_5); // bb0[4]: scope 1 at $SRC_DIR/liballoc/boxed.rs:LL:COL
StorageLive(_6); // bb0[5]: scope 1 at $SRC_DIR/liballoc/boxed.rs:LL:COL
_6 = &mut (*(*_4)); // bb0[6]: scope 1 at $SRC_DIR/liballoc/boxed.rs:LL:COL
_5 = _6; // bb0[7]: scope 1 at $SRC_DIR/liballoc/boxed.rs:LL:COL
_3 = _5; // bb0[8]: scope 1 at $SRC_DIR/liballoc/boxed.rs:LL:COL
StorageDead(_6); // bb0[9]: scope 1 at $SRC_DIR/liballoc/boxed.rs:LL:COL
StorageDead(_5); // bb0[10]: scope 1 at $SRC_DIR/liballoc/boxed.rs:LL:COL
_2 = &mut (*_3); // bb0[11]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:8:5: 8:15
StorageDead(_4); // bb0[12]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:8:14: 8:15
_0 = &mut (*_2); // bb0[13]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:8:5: 8:15
StorageDead(_3); // bb0[14]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:9:1: 9:2
StorageDead(_2); // bb0[15]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:9:1: 9:2
return; // bb0[16]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:9:2: 9:2
}
}

View File

@ -0,0 +1,22 @@
// MIR for `c` after Inline
fn c(_1: &[T]) -> &[T] {
debug x => _1; // in scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:12:13: 12:14
let mut _0: &[T]; // return place in scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:12:25: 12:29
let _2: &[T]; // in scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:13:5: 13:15
let mut _3: &[T]; // in scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:13:5: 13:6
scope 1 {
debug self => _3; // in scope 1 at $SRC_DIR/libcore/convert/mod.rs:LL:COL
}
bb0: {
StorageLive(_2); // bb0[0]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:13:5: 13:15
StorageLive(_3); // bb0[1]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:13:5: 13:6
_3 = &(*_1); // bb0[2]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:13:5: 13:6
_2 = _3; // bb0[3]: scope 1 at $SRC_DIR/libcore/convert/mod.rs:LL:COL
_0 = &(*_2); // bb0[4]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:13:5: 13:15
StorageDead(_3); // bb0[5]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:13:14: 13:15
StorageDead(_2); // bb0[6]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:14:1: 14:2
return; // bb0[7]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:14:2: 14:2
}
}

View File

@ -0,0 +1,26 @@
// MIR for `d` after Inline
fn d(_1: &std::boxed::Box<T>) -> &T {
debug x => _1; // in scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:17:13: 17:14
let mut _0: &T; // return place in scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:17:28: 17:30
let _2: &T; // in scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:18:5: 18:15
let mut _3: &std::boxed::Box<T>; // in scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:18:5: 18:6
scope 1 {
debug self => _3; // in scope 1 at $SRC_DIR/liballoc/boxed.rs:LL:COL
let _4: &T; // in scope 1 at $DIR/issue-58867-inline-as-ref-as-mut.rs:18:5: 18:15
}
bb0: {
StorageLive(_2); // bb0[0]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:18:5: 18:15
StorageLive(_3); // bb0[1]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:18:5: 18:6
_3 = &(*_1); // bb0[2]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:18:5: 18:6
StorageLive(_4); // bb0[3]: scope 1 at $SRC_DIR/liballoc/boxed.rs:LL:COL
_4 = &(*(*_3)); // bb0[4]: scope 1 at $SRC_DIR/liballoc/boxed.rs:LL:COL
_2 = _4; // bb0[5]: scope 1 at $SRC_DIR/liballoc/boxed.rs:LL:COL
StorageDead(_4); // bb0[6]: scope 1 at $SRC_DIR/liballoc/boxed.rs:LL:COL
_0 = &(*_2); // bb0[7]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:18:5: 18:15
StorageDead(_3); // bb0[8]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:18:14: 18:15
StorageDead(_2); // bb0[9]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:19:1: 19:2
return; // bb0[10]: scope 0 at $DIR/issue-58867-inline-as-ref-as-mut.rs:19:2: 19:2
}
}

View File

@ -8,7 +8,7 @@ use std::path::Path;
pub fn main() { pub fn main() {
unsafe { unsafe {
let path = Path::new("libdylib.so"); let path = Path::new("libdylib.so");
let a = DynamicLibrary::open(Some(&path)).unwrap(); let a = DynamicLibrary::open(&path).unwrap();
assert!(a.symbol::<isize>("fun1").is_ok()); assert!(a.symbol::<isize>("fun1").is_ok());
assert!(a.symbol::<isize>("fun2").is_ok()); assert!(a.symbol::<isize>("fun2").is_ok());
assert!(a.symbol::<isize>("fun3").is_ok()); assert!(a.symbol::<isize>("fun3").is_ok());

View File

@ -1,35 +0,0 @@
// ignore-musl - dlsym doesn't see symbols without "-C link-arg=-Wl,--export-dynamic"
#![feature(rustc_private)]
extern crate rustc_metadata;
use rustc_metadata::dynamic_lib::DynamicLibrary;
#[no_mangle]
pub fn foo() {
bar();
}
pub fn foo2<T>() {
fn bar2() {
bar();
}
bar2();
}
#[no_mangle]
fn bar() {}
#[allow(dead_code)]
#[no_mangle]
fn baz() {}
pub fn test() {
let lib = DynamicLibrary::open(None).unwrap();
unsafe {
assert!(lib.symbol::<isize>("foo").is_ok());
assert!(lib.symbol::<isize>("baz").is_ok());
assert!(lib.symbol::<isize>("bar").is_ok());
}
}

View File

@ -1,13 +0,0 @@
// run-pass
// aux-build:linkage-visibility.rs
// ignore-android: FIXME(#10356)
// ignore-windows: std::dynamic_lib does not work on Windows well
// ignore-emscripten no dynamic linking
extern crate linkage_visibility as foo;
pub fn main() {
foo::test();
foo::foo2::<isize>();
foo::foo();
}

View File

@ -0,0 +1,15 @@
// ignore-tidy-linelength
#![feature(box_into_boxed_slice)]
use std::boxed::Box;
use std::fmt::Debug;
fn main() {
let boxed_slice = Box::new([1,2,3]) as Box<[u8]>;
let _ = Box::into_boxed_slice(boxed_slice);
//~^ ERROR the size for values of type `[u8]` cannot be known at compilation time
//~^^ ERROR the size for values of type `[u8]` cannot be known at compilation time
let boxed_trait: Box<dyn Debug> = Box::new(5u8);
let _ = Box::into_boxed_slice(boxed_trait);
//~^ ERROR the size for values of type `dyn std::fmt::Debug` cannot be known at compilation time
//~^^ ERROR the size for values of type `dyn std::fmt::Debug` cannot be known at compilation time
}

View File

@ -0,0 +1,43 @@
error[E0277]: the size for values of type `[u8]` cannot be known at compilation time
--> $DIR/box-into-boxed-slice-fail.rs:8:35
|
LL | let _ = Box::into_boxed_slice(boxed_slice);
| ^^^^^^^^^^^ doesn't have a size known at compile-time
|
= help: the trait `std::marker::Sized` is not implemented for `[u8]`
= note: to learn more, visit <https://doc.rust-lang.org/book/ch19-04-advanced-types.html#dynamically-sized-types-and-the-sized-trait>
= note: required by `std::boxed::Box::<T>::into_boxed_slice`
error[E0277]: the size for values of type `[u8]` cannot be known at compilation time
--> $DIR/box-into-boxed-slice-fail.rs:8:13
|
LL | let _ = Box::into_boxed_slice(boxed_slice);
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ doesn't have a size known at compile-time
|
= help: the trait `std::marker::Sized` is not implemented for `[u8]`
= note: to learn more, visit <https://doc.rust-lang.org/book/ch19-04-advanced-types.html#dynamically-sized-types-and-the-sized-trait>
= note: slice and array elements must have `Sized` type
error[E0277]: the size for values of type `dyn std::fmt::Debug` cannot be known at compilation time
--> $DIR/box-into-boxed-slice-fail.rs:12:35
|
LL | let _ = Box::into_boxed_slice(boxed_trait);
| ^^^^^^^^^^^ doesn't have a size known at compile-time
|
= help: the trait `std::marker::Sized` is not implemented for `dyn std::fmt::Debug`
= note: to learn more, visit <https://doc.rust-lang.org/book/ch19-04-advanced-types.html#dynamically-sized-types-and-the-sized-trait>
= note: required by `std::boxed::Box::<T>::into_boxed_slice`
error[E0277]: the size for values of type `dyn std::fmt::Debug` cannot be known at compilation time
--> $DIR/box-into-boxed-slice-fail.rs:12:13
|
LL | let _ = Box::into_boxed_slice(boxed_trait);
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ doesn't have a size known at compile-time
|
= help: the trait `std::marker::Sized` is not implemented for `dyn std::fmt::Debug`
= note: to learn more, visit <https://doc.rust-lang.org/book/ch19-04-advanced-types.html#dynamically-sized-types-and-the-sized-trait>
= note: slice and array elements must have `Sized` type
error: aborting due to 4 previous errors
For more information about this error, try `rustc --explain E0277`.

View File

@ -0,0 +1,11 @@
// run-pass
#![feature(box_into_boxed_slice)]
use std::boxed::Box;
fn main() {
assert_eq!(Box::into_boxed_slice(Box::new(5u8)), Box::new([5u8]) as Box<[u8]>);
assert_eq!(Box::into_boxed_slice(Box::new([25u8])), Box::new([[25u8]]) as Box<[[u8; 1]]>);
let a: Box<[Box<[u8; 1]>]> = Box::into_boxed_slice(Box::new(Box::new([5u8])));
let b: Box<[Box<[u8; 1]>]> = Box::new([Box::new([5u8])]);
assert_eq!(a, b);
}