Skip to content

Commit 4e20754

Browse files
committed
Auto merge of #85919 - workingjubilee:simd-ptrs-are-valid, r=petrochenkov
Allow raw pointers in SIMD types Closes #85915 by loosening the strictness in typechecking and adding a test to guarantee it passes. This still might be too strict, as references currently do pass monomorphization, but my understanding is that they are not guaranteed to be "scalar" in the same way.
2 parents 2c10688 + d9a5df6 commit 4e20754

8 files changed

+134
-8
lines changed

compiler/rustc_middle/src/ty/sty.rs

-5
Original file line numberDiff line numberDiff line change
@@ -1890,11 +1890,6 @@ impl<'tcx> TyS<'tcx> {
18901890
matches!(self.kind(), Int(ty::IntTy::Isize) | Uint(ty::UintTy::Usize))
18911891
}
18921892

1893-
#[inline]
1894-
pub fn is_machine(&self) -> bool {
1895-
matches!(self.kind(), Int(..) | Uint(..) | Float(..))
1896-
}
1897-
18981893
#[inline]
18991894
pub fn has_concrete_skeleton(&self) -> bool {
19001895
!matches!(self.kind(), Param(_) | Infer(_) | Error(_))

compiler/rustc_typeck/src/check/check.rs

+12-3
Original file line numberDiff line numberDiff line change
@@ -1214,10 +1214,19 @@ pub fn check_simd(tcx: TyCtxt<'_>, sp: Span, def_id: LocalDefId) {
12141214
}
12151215
}
12161216

1217+
// Check that we use types valid for use in the lanes of a SIMD "vector register"
1218+
// These are scalar types which directly match a "machine" type
1219+
// Yes: Integers, floats, "thin" pointers
1220+
// No: char, "fat" pointers, compound types
12171221
match e.kind() {
1218-
ty::Param(_) => { /* struct<T>(T, T, T, T) is ok */ }
1219-
_ if e.is_machine() => { /* struct(u8, u8, u8, u8) is ok */ }
1220-
ty::Array(ty, _c) if ty.is_machine() => { /* struct([f32; 4]) */ }
1222+
ty::Param(_) => (), // pass struct<T>(T, T, T, T) through, let monomorphization catch errors
1223+
ty::Int(_) | ty::Uint(_) | ty::Float(_) | ty::RawPtr(_) => (), // struct(u8, u8, u8, u8) is ok
1224+
ty::Array(t, _clen)
1225+
if matches!(
1226+
t.kind(),
1227+
ty::Int(_) | ty::Uint(_) | ty::Float(_) | ty::RawPtr(_)
1228+
) =>
1229+
{ /* struct([f32; 4]) is ok */ }
12211230
_ => {
12221231
struct_span_err!(
12231232
tcx.sess,
+67
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,67 @@
1+
// run-pass
2+
// ignore-emscripten
3+
4+
// Short form of the generic gather/scatter tests,
5+
// verifying simd([*const T; N]) and simd([*mut T; N]) pass typeck and work.
6+
#![feature(repr_simd, platform_intrinsics)]
7+
#![allow(non_camel_case_types)]
8+
9+
#[repr(simd)]
10+
#[derive(Copy, Clone, PartialEq, Debug)]
11+
struct cptrx4<T>([*const T; 4]);
12+
13+
#[repr(simd)]
14+
#[derive(Copy, Clone, PartialEq, Debug)]
15+
struct mptrx4<T>([*mut T; 4]);
16+
17+
#[repr(simd)]
18+
#[derive(Copy, Clone, PartialEq, Debug)]
19+
struct f32x4([f32; 4]);
20+
21+
#[repr(simd)]
22+
#[derive(Copy, Clone, PartialEq, Debug)]
23+
struct i32x4([i32; 4]);
24+
25+
extern "platform-intrinsic" {
26+
fn simd_gather<T, U, V>(x: T, y: U, z: V) -> T;
27+
fn simd_scatter<T, U, V>(x: T, y: U, z: V) -> ();
28+
}
29+
30+
fn main() {
31+
let mut x = [0_f32, 1., 2., 3., 4., 5., 6., 7.];
32+
33+
let default = f32x4([-3_f32, -3., -3., -3.]);
34+
let s_strided = f32x4([0_f32, 2., -3., 6.]);
35+
let mask = i32x4([-1_i32, -1, 0, -1]);
36+
37+
// reading from *const
38+
unsafe {
39+
let pointer = &x as *const f32;
40+
let pointers = cptrx4([
41+
pointer.offset(0) as *const f32,
42+
pointer.offset(2),
43+
pointer.offset(4),
44+
pointer.offset(6)
45+
]);
46+
47+
let r_strided = simd_gather(default, pointers, mask);
48+
49+
assert_eq!(r_strided, s_strided);
50+
}
51+
52+
// writing to *mut
53+
unsafe {
54+
let pointer = &mut x as *mut f32;
55+
let pointers = mptrx4([
56+
pointer.offset(0) as *mut f32,
57+
pointer.offset(2),
58+
pointer.offset(4),
59+
pointer.offset(6)
60+
]);
61+
62+
let values = f32x4([42_f32, 43_f32, 44_f32, 45_f32]);
63+
simd_scatter(values, pointers, mask);
64+
65+
assert_eq!(x, [42., 1., 43., 3., 4., 5., 45., 7.]);
66+
}
67+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
// run-pass
2+
// ignore-emscripten
3+
4+
#![feature(extern_types)]
5+
#![feature(repr_simd)]
6+
7+
use std::ptr::NonNull;
8+
9+
extern {
10+
type Extern;
11+
}
12+
13+
#[repr(simd)]
14+
struct S<T>(T);
15+
16+
#[inline(never)]
17+
fn identity<T>(v: T) -> T {
18+
v
19+
}
20+
21+
fn main() {
22+
let _v: S<[Option<NonNull<Extern>>; 4]> = identity(S([None; 4]));
23+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
// build-fail
2+
3+
#![feature(repr_simd)]
4+
5+
// error-pattern:monomorphising SIMD type `S<[*mut [u8]; 4]>` with a non-primitive-scalar (integer/float/pointer) element type `*mut [u8]`
6+
7+
#[repr(simd)]
8+
struct S<T>(T);
9+
10+
fn main() {
11+
let _v: Option<S<[*mut [u8]; 4]>> = None;
12+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
error: monomorphising SIMD type `S<[*mut [u8]; 4]>` with a non-primitive-scalar (integer/float/pointer) element type `*mut [u8]`
2+
3+
error: aborting due to previous error
4+
+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
// build-fail
2+
3+
#![feature(repr_simd)]
4+
5+
// error-pattern:monomorphising SIMD type `S` with a non-primitive-scalar (integer/float/pointer) element type `*mut [u8]`
6+
7+
#[repr(simd)]
8+
struct S([*mut [u8]; 4]);
9+
10+
fn main() {
11+
let _v: Option<S> = None;
12+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
error: monomorphising SIMD type `S` with a non-primitive-scalar (integer/float/pointer) element type `*mut [u8]`
2+
3+
error: aborting due to previous error
4+

0 commit comments

Comments
 (0)