2018-10-06 11:18:06 -05: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.
|
|
|
|
|
|
|
|
|
2018-07-28 10:34:52 -05:00
|
|
|
#![feature(tool_lints)]
|
2017-09-18 05:47:33 -05:00
|
|
|
|
2015-11-11 08:28:31 -06:00
|
|
|
|
2016-09-19 10:59:12 -05:00
|
|
|
#![allow(dead_code)]
|
|
|
|
|
2015-11-11 08:28:31 -06:00
|
|
|
extern crate core;
|
|
|
|
|
|
|
|
use std::mem::transmute as my_transmute;
|
|
|
|
use std::vec::Vec as MyVec;
|
|
|
|
|
2016-06-28 07:08:08 -05:00
|
|
|
fn my_int() -> Usize {
|
|
|
|
Usize(42)
|
2016-04-07 10:46:48 -05:00
|
|
|
}
|
|
|
|
|
2015-11-11 08:28:31 -06:00
|
|
|
fn my_vec() -> MyVec<i32> {
|
|
|
|
vec![]
|
|
|
|
}
|
|
|
|
|
2018-07-28 10:34:52 -05:00
|
|
|
#[allow(clippy::needless_lifetimes, clippy::transmute_ptr_to_ptr)]
|
|
|
|
#[warn(clippy::useless_transmute)]
|
2015-11-11 08:28:31 -06:00
|
|
|
unsafe fn _generic<'a, T, U: 'a>(t: &'a T) {
|
|
|
|
let _: &'a T = core::intrinsics::transmute(t);
|
2017-02-08 07:58:07 -06:00
|
|
|
|
2015-11-11 08:28:31 -06:00
|
|
|
let _: &'a U = core::intrinsics::transmute(t);
|
2016-06-27 09:12:48 -05:00
|
|
|
|
|
|
|
let _: *const T = core::intrinsics::transmute(t);
|
2017-02-08 07:58:07 -06:00
|
|
|
|
2016-06-27 09:12:48 -05:00
|
|
|
let _: *mut T = core::intrinsics::transmute(t);
|
2017-02-08 07:58:07 -06:00
|
|
|
|
2016-06-27 09:12:48 -05:00
|
|
|
let _: *const U = core::intrinsics::transmute(t);
|
2015-11-11 08:28:31 -06:00
|
|
|
}
|
|
|
|
|
2018-07-28 10:34:52 -05:00
|
|
|
#[warn(clippy::transmute_ptr_to_ref)]
|
2016-03-25 17:22:17 -05:00
|
|
|
unsafe fn _ptr_to_ref<T, U>(p: *const T, m: *mut T, o: *const U, om: *mut U) {
|
|
|
|
let _: &T = std::mem::transmute(p);
|
|
|
|
let _: &T = &*p;
|
|
|
|
|
|
|
|
let _: &mut T = std::mem::transmute(m);
|
|
|
|
let _: &mut T = &mut *m;
|
|
|
|
|
|
|
|
let _: &T = std::mem::transmute(m);
|
|
|
|
let _: &T = &*m;
|
|
|
|
|
2016-06-28 01:48:44 -05:00
|
|
|
let _: &mut T = std::mem::transmute(p as *mut T);
|
2016-06-29 16:02:15 -05:00
|
|
|
let _ = &mut *(p as *mut T);
|
2016-06-28 01:48:44 -05:00
|
|
|
|
2016-03-25 17:22:17 -05:00
|
|
|
let _: &T = std::mem::transmute(o);
|
|
|
|
let _: &T = &*(o as *const T);
|
|
|
|
|
|
|
|
let _: &mut T = std::mem::transmute(om);
|
|
|
|
let _: &mut T = &mut *(om as *mut T);
|
|
|
|
|
|
|
|
let _: &T = std::mem::transmute(om);
|
|
|
|
let _: &T = &*(om as *const T);
|
|
|
|
}
|
|
|
|
|
2018-07-28 10:34:52 -05:00
|
|
|
#[warn(clippy::transmute_ptr_to_ref)]
|
2016-09-19 10:59:12 -05:00
|
|
|
fn issue1231() {
|
|
|
|
struct Foo<'a, T: 'a> {
|
|
|
|
bar: &'a T,
|
|
|
|
}
|
|
|
|
|
2017-02-20 21:36:59 -06:00
|
|
|
let raw = 42 as *const i32;
|
2016-09-19 10:59:12 -05:00
|
|
|
let _: &Foo<u8> = unsafe { std::mem::transmute::<_, &Foo<_>>(raw) };
|
2017-02-08 07:58:07 -06:00
|
|
|
|
2016-09-19 10:59:12 -05:00
|
|
|
let _: &Foo<&u8> = unsafe { std::mem::transmute::<_, &Foo<&_>>(raw) };
|
2017-02-08 07:58:07 -06:00
|
|
|
|
2016-09-19 10:59:12 -05:00
|
|
|
type Bar<'a> = &'a u8;
|
2017-02-20 21:36:59 -06:00
|
|
|
let raw = 42 as *const i32;
|
2016-09-19 10:59:12 -05:00
|
|
|
unsafe { std::mem::transmute::<_, Bar>(raw) };
|
|
|
|
}
|
|
|
|
|
2018-07-28 10:34:52 -05:00
|
|
|
#[warn(clippy::useless_transmute)]
|
2016-03-24 17:48:38 -05:00
|
|
|
fn useless() {
|
2015-11-11 08:28:31 -06:00
|
|
|
unsafe {
|
|
|
|
let _: Vec<i32> = core::intrinsics::transmute(my_vec());
|
2017-02-08 07:58:07 -06:00
|
|
|
|
2015-11-11 08:28:31 -06:00
|
|
|
let _: Vec<i32> = core::mem::transmute(my_vec());
|
2017-02-08 07:58:07 -06:00
|
|
|
|
2015-11-11 08:28:31 -06:00
|
|
|
let _: Vec<i32> = std::intrinsics::transmute(my_vec());
|
2017-02-08 07:58:07 -06:00
|
|
|
|
2015-11-11 08:28:31 -06:00
|
|
|
let _: Vec<i32> = std::mem::transmute(my_vec());
|
2017-02-08 07:58:07 -06:00
|
|
|
|
2015-11-11 08:28:31 -06:00
|
|
|
let _: Vec<i32> = my_transmute(my_vec());
|
2017-02-08 07:58:07 -06:00
|
|
|
|
2015-11-11 08:28:31 -06:00
|
|
|
let _: Vec<u32> = core::intrinsics::transmute(my_vec());
|
|
|
|
let _: Vec<u32> = core::mem::transmute(my_vec());
|
|
|
|
let _: Vec<u32> = std::intrinsics::transmute(my_vec());
|
|
|
|
let _: Vec<u32> = std::mem::transmute(my_vec());
|
|
|
|
let _: Vec<u32> = my_transmute(my_vec());
|
2016-06-28 07:08:08 -05:00
|
|
|
|
|
|
|
let _: *const usize = std::mem::transmute(5_isize);
|
2017-02-08 07:58:07 -06:00
|
|
|
|
2016-07-03 19:22:57 -05:00
|
|
|
let _ = 5_isize as *const usize;
|
|
|
|
|
|
|
|
let _: *const usize = std::mem::transmute(1+1usize);
|
2017-02-08 07:58:07 -06:00
|
|
|
|
2016-07-03 19:22:57 -05:00
|
|
|
let _ = (1+1_usize) as *const usize;
|
2015-11-11 08:28:31 -06:00
|
|
|
}
|
|
|
|
}
|
2016-03-24 17:48:38 -05:00
|
|
|
|
2016-06-28 07:08:08 -05:00
|
|
|
struct Usize(usize);
|
|
|
|
|
2018-07-28 10:34:52 -05:00
|
|
|
#[warn(clippy::crosspointer_transmute)]
|
2016-03-24 17:48:38 -05:00
|
|
|
fn crosspointer() {
|
2016-06-28 07:08:08 -05:00
|
|
|
let mut int: Usize = Usize(0);
|
|
|
|
let int_const_ptr: *const Usize = &int as *const Usize;
|
|
|
|
let int_mut_ptr: *mut Usize = &mut int as *mut Usize;
|
2016-03-24 17:48:38 -05:00
|
|
|
|
|
|
|
unsafe {
|
2016-06-28 07:08:08 -05:00
|
|
|
let _: Usize = core::intrinsics::transmute(int_const_ptr);
|
2017-02-08 07:58:07 -06:00
|
|
|
|
2016-06-28 07:08:08 -05:00
|
|
|
let _: Usize = core::intrinsics::transmute(int_mut_ptr);
|
2017-02-08 07:58:07 -06:00
|
|
|
|
2016-06-28 07:08:08 -05:00
|
|
|
let _: *const Usize = core::intrinsics::transmute(my_int());
|
2017-02-08 07:58:07 -06:00
|
|
|
|
2016-06-28 07:08:08 -05:00
|
|
|
let _: *mut Usize = core::intrinsics::transmute(my_int());
|
2016-03-24 17:48:38 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-28 10:34:52 -05:00
|
|
|
#[warn(clippy::transmute_int_to_char)]
|
2017-10-02 10:23:24 -05:00
|
|
|
fn int_to_char() {
|
|
|
|
let _: char = unsafe { std::mem::transmute(0_u32) };
|
|
|
|
let _: char = unsafe { std::mem::transmute(0_i32) };
|
|
|
|
}
|
|
|
|
|
2018-07-28 10:34:52 -05:00
|
|
|
#[warn(clippy::transmute_int_to_bool)]
|
2017-10-02 10:23:24 -05:00
|
|
|
fn int_to_bool() {
|
|
|
|
let _: bool = unsafe { std::mem::transmute(0_u8) };
|
|
|
|
}
|
|
|
|
|
2018-07-28 10:34:52 -05:00
|
|
|
#[warn(clippy::transmute_int_to_float)]
|
2017-10-02 10:23:24 -05:00
|
|
|
fn int_to_float() {
|
|
|
|
let _: f32 = unsafe { std::mem::transmute(0_u32) };
|
|
|
|
let _: f32 = unsafe { std::mem::transmute(0_i32) };
|
|
|
|
}
|
|
|
|
|
2017-10-28 20:27:45 -05:00
|
|
|
fn bytes_to_str(b: &[u8], mb: &mut [u8]) {
|
|
|
|
let _: &str = unsafe { std::mem::transmute(b) };
|
|
|
|
let _: &mut str = unsafe { std::mem::transmute(mb) };
|
|
|
|
}
|
|
|
|
|
2018-05-22 19:18:56 -05:00
|
|
|
// Make sure we can modify lifetimes, which is one of the recommended uses
|
|
|
|
// of transmute
|
|
|
|
|
|
|
|
// Make sure we can do static lifetime transmutes
|
2018-07-28 10:34:52 -05:00
|
|
|
#[warn(clippy::transmute_ptr_to_ptr)]
|
2018-05-22 19:18:56 -05:00
|
|
|
unsafe fn transmute_lifetime_to_static<'a, T>(t: &'a T) -> &'static T {
|
|
|
|
std::mem::transmute::<&'a T, &'static T>(t)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure we can do non-static lifetime transmutes
|
2018-07-28 10:34:52 -05:00
|
|
|
#[warn(clippy::transmute_ptr_to_ptr)]
|
2018-05-22 19:18:56 -05:00
|
|
|
unsafe fn transmute_lifetime<'a, 'b, T>(t: &'a T, u: &'b T) -> &'b T {
|
|
|
|
std::mem::transmute::<&'a T, &'b T>(t)
|
|
|
|
}
|
|
|
|
|
2018-05-24 22:02:42 -05:00
|
|
|
struct LifetimeParam<'a> {
|
|
|
|
s: &'a str,
|
|
|
|
}
|
|
|
|
|
|
|
|
struct GenericParam<T> {
|
|
|
|
t: T,
|
|
|
|
}
|
|
|
|
|
2018-07-28 10:34:52 -05:00
|
|
|
#[warn(clippy::transmute_ptr_to_ptr)]
|
2018-04-11 04:17:59 -05:00
|
|
|
fn transmute_ptr_to_ptr() {
|
|
|
|
let ptr = &1u32 as *const u32;
|
|
|
|
let mut_ptr = &mut 1u32 as *mut u32;
|
|
|
|
unsafe {
|
|
|
|
// pointer-to-pointer transmutes; bad
|
|
|
|
let _: *const f32 = std::mem::transmute(ptr);
|
|
|
|
let _: *mut f32 = std::mem::transmute(mut_ptr);
|
|
|
|
// ref-ref transmutes; bad
|
|
|
|
let _: &f32 = std::mem::transmute(&1u32);
|
2018-05-24 22:02:42 -05:00
|
|
|
let _: &f64 = std::mem::transmute(&1f32);
|
|
|
|
// ^ this test is here because both f32 and f64 are the same TypeVariant, but they are not
|
|
|
|
// the same type
|
2018-04-11 04:17:59 -05:00
|
|
|
let _: &mut f32 = std::mem::transmute(&mut 1u32);
|
2018-05-24 22:02:42 -05:00
|
|
|
let _: &GenericParam<f32> = std::mem::transmute(&GenericParam { t: 1u32 });
|
2018-04-11 04:17:59 -05:00
|
|
|
}
|
2018-05-24 22:02:42 -05:00
|
|
|
|
|
|
|
// these are recommendations for solving the above; if these lint we need to update
|
2018-05-14 12:24:31 -05:00
|
|
|
// those suggestions
|
2018-04-11 04:17:59 -05:00
|
|
|
let _ = ptr as *const f32;
|
|
|
|
let _ = mut_ptr as *mut f32;
|
|
|
|
let _ = unsafe { &*(&1u32 as *const u32 as *const f32) };
|
|
|
|
let _ = unsafe { &mut *(&mut 1u32 as *mut u32 as *mut f32) };
|
2018-05-24 22:02:42 -05:00
|
|
|
|
|
|
|
// transmute internal lifetimes, should not lint
|
|
|
|
let s = "hello world".to_owned();
|
|
|
|
let lp = LifetimeParam { s: &s };
|
|
|
|
let _: &LifetimeParam<'static> = unsafe { std::mem::transmute(&lp) };
|
|
|
|
let _: &GenericParam<&LifetimeParam<'static>> = unsafe {
|
|
|
|
std::mem::transmute(&GenericParam { t: &lp})
|
|
|
|
};
|
2018-04-11 04:17:59 -05:00
|
|
|
}
|
|
|
|
|
2016-09-19 10:59:12 -05:00
|
|
|
fn main() { }
|