]> git.lizzy.rs Git - rust.git/blob - tests/ui/transmute.rs
Add lint to detect transmutes from float to integer
[rust.git] / tests / ui / transmute.rs
1 #![allow(dead_code)]
2
3 extern crate core;
4
5 use std::mem::transmute as my_transmute;
6 use std::vec::Vec as MyVec;
7
8 fn my_int() -> Usize {
9     Usize(42)
10 }
11
12 fn my_vec() -> MyVec<i32> {
13     vec![]
14 }
15
16 #[allow(clippy::needless_lifetimes, clippy::transmute_ptr_to_ptr)]
17 #[warn(clippy::useless_transmute)]
18 unsafe fn _generic<'a, T, U: 'a>(t: &'a T) {
19     let _: &'a T = core::intrinsics::transmute(t);
20
21     let _: &'a U = core::intrinsics::transmute(t);
22
23     let _: *const T = core::intrinsics::transmute(t);
24
25     let _: *mut T = core::intrinsics::transmute(t);
26
27     let _: *const U = core::intrinsics::transmute(t);
28 }
29
30 #[warn(clippy::transmute_ptr_to_ref)]
31 unsafe fn _ptr_to_ref<T, U>(p: *const T, m: *mut T, o: *const U, om: *mut U) {
32     let _: &T = std::mem::transmute(p);
33     let _: &T = &*p;
34
35     let _: &mut T = std::mem::transmute(m);
36     let _: &mut T = &mut *m;
37
38     let _: &T = std::mem::transmute(m);
39     let _: &T = &*m;
40
41     let _: &mut T = std::mem::transmute(p as *mut T);
42     let _ = &mut *(p as *mut T);
43
44     let _: &T = std::mem::transmute(o);
45     let _: &T = &*(o as *const T);
46
47     let _: &mut T = std::mem::transmute(om);
48     let _: &mut T = &mut *(om as *mut T);
49
50     let _: &T = std::mem::transmute(om);
51     let _: &T = &*(om as *const T);
52 }
53
54 #[warn(clippy::transmute_ptr_to_ref)]
55 fn issue1231() {
56     struct Foo<'a, T> {
57         bar: &'a T,
58     }
59
60     let raw = 42 as *const i32;
61     let _: &Foo<u8> = unsafe { std::mem::transmute::<_, &Foo<_>>(raw) };
62
63     let _: &Foo<&u8> = unsafe { std::mem::transmute::<_, &Foo<&_>>(raw) };
64
65     type Bar<'a> = &'a u8;
66     let raw = 42 as *const i32;
67     unsafe { std::mem::transmute::<_, Bar>(raw) };
68 }
69
70 #[warn(clippy::useless_transmute)]
71 fn useless() {
72     unsafe {
73         let _: Vec<i32> = core::intrinsics::transmute(my_vec());
74
75         let _: Vec<i32> = core::mem::transmute(my_vec());
76
77         let _: Vec<i32> = std::intrinsics::transmute(my_vec());
78
79         let _: Vec<i32> = std::mem::transmute(my_vec());
80
81         let _: Vec<i32> = my_transmute(my_vec());
82
83         let _: *const usize = std::mem::transmute(5_isize);
84
85         let _ = 5_isize as *const usize;
86
87         let _: *const usize = std::mem::transmute(1 + 1usize);
88
89         let _ = (1 + 1_usize) as *const usize;
90     }
91 }
92
93 struct Usize(usize);
94
95 #[warn(clippy::crosspointer_transmute)]
96 fn crosspointer() {
97     let mut int: Usize = Usize(0);
98     let int_const_ptr: *const Usize = &int as *const Usize;
99     let int_mut_ptr: *mut Usize = &mut int as *mut Usize;
100
101     unsafe {
102         let _: Usize = core::intrinsics::transmute(int_const_ptr);
103
104         let _: Usize = core::intrinsics::transmute(int_mut_ptr);
105
106         let _: *const Usize = core::intrinsics::transmute(my_int());
107
108         let _: *mut Usize = core::intrinsics::transmute(my_int());
109     }
110 }
111
112 #[warn(clippy::transmute_int_to_char)]
113 fn int_to_char() {
114     let _: char = unsafe { std::mem::transmute(0_u32) };
115     let _: char = unsafe { std::mem::transmute(0_i32) };
116 }
117
118 #[warn(clippy::transmute_int_to_bool)]
119 fn int_to_bool() {
120     let _: bool = unsafe { std::mem::transmute(0_u8) };
121 }
122
123 #[warn(clippy::transmute_int_to_float)]
124 fn int_to_float() {
125     let _: f32 = unsafe { std::mem::transmute(0_u32) };
126     let _: f32 = unsafe { std::mem::transmute(0_i32) };
127 }
128
129 #[warn(clippy::transmute_float_to_int)]
130 fn float_to_int() {
131     let _: u32 = unsafe { std::mem::transmute(1f32) };
132     let _: i32 = unsafe { std::mem::transmute(1f32) };
133     let _: u64 = unsafe { std::mem::transmute(1f64) };
134     let _: i64 = unsafe { std::mem::transmute(1f64) };
135     let _: u64 = unsafe { std::mem::transmute(1.0) };
136     let _: u64 = unsafe { std::mem::transmute(-1.0) };
137 }
138
139 fn bytes_to_str(b: &[u8], mb: &mut [u8]) {
140     let _: &str = unsafe { std::mem::transmute(b) };
141     let _: &mut str = unsafe { std::mem::transmute(mb) };
142 }
143
144 // Make sure we can modify lifetimes, which is one of the recommended uses
145 // of transmute
146
147 // Make sure we can do static lifetime transmutes
148 #[warn(clippy::transmute_ptr_to_ptr)]
149 unsafe fn transmute_lifetime_to_static<'a, T>(t: &'a T) -> &'static T {
150     std::mem::transmute::<&'a T, &'static T>(t)
151 }
152
153 // Make sure we can do non-static lifetime transmutes
154 #[warn(clippy::transmute_ptr_to_ptr)]
155 unsafe fn transmute_lifetime<'a, 'b, T>(t: &'a T, u: &'b T) -> &'b T {
156     std::mem::transmute::<&'a T, &'b T>(t)
157 }
158
159 struct LifetimeParam<'a> {
160     s: &'a str,
161 }
162
163 struct GenericParam<T> {
164     t: T,
165 }
166
167 #[warn(clippy::transmute_ptr_to_ptr)]
168 fn transmute_ptr_to_ptr() {
169     let ptr = &1u32 as *const u32;
170     let mut_ptr = &mut 1u32 as *mut u32;
171     unsafe {
172         // pointer-to-pointer transmutes; bad
173         let _: *const f32 = std::mem::transmute(ptr);
174         let _: *mut f32 = std::mem::transmute(mut_ptr);
175         // ref-ref transmutes; bad
176         let _: &f32 = std::mem::transmute(&1u32);
177         let _: &f64 = std::mem::transmute(&1f32);
178         // ^ this test is here because both f32 and f64 are the same TypeVariant, but they are not
179         // the same type
180         let _: &mut f32 = std::mem::transmute(&mut 1u32);
181         let _: &GenericParam<f32> = std::mem::transmute(&GenericParam { t: 1u32 });
182     }
183
184     // these are recommendations for solving the above; if these lint we need to update
185     // those suggestions
186     let _ = ptr as *const f32;
187     let _ = mut_ptr as *mut f32;
188     let _ = unsafe { &*(&1u32 as *const u32 as *const f32) };
189     let _ = unsafe { &mut *(&mut 1u32 as *mut u32 as *mut f32) };
190
191     // transmute internal lifetimes, should not lint
192     let s = "hello world".to_owned();
193     let lp = LifetimeParam { s: &s };
194     let _: &LifetimeParam<'static> = unsafe { std::mem::transmute(&lp) };
195     let _: &GenericParam<&LifetimeParam<'static>> = unsafe { std::mem::transmute(&GenericParam { t: &lp }) };
196 }
197
198 fn main() {}