Skip to content

Commit 5580012

Browse files
cg_llvm: simplify llvm.masked.gather/scatter naming with opaque pointers
With opaque pointers, there's no longer a need to generate a chain of pointer types in the intrinsic name when arguments are pointers to pointers.
1 parent cf7788d commit 5580012

File tree

3 files changed

+51
-109
lines changed

3 files changed

+51
-109
lines changed

compiler/rustc_codegen_llvm/src/intrinsic.rs

+47-105
Original file line numberDiff line numberDiff line change
@@ -1307,49 +1307,34 @@ fn generic_simd_intrinsic<'ll, 'tcx>(
13071307
// FIXME: use:
13081308
// https://github.com/llvm-mirror/llvm/blob/master/include/llvm/IR/Function.h#L182
13091309
// https://github.com/llvm-mirror/llvm/blob/master/include/llvm/IR/Intrinsics.h#L81
1310-
fn llvm_vector_str(
1311-
elem_ty: Ty<'_>,
1312-
vec_len: u64,
1313-
no_pointers: usize,
1314-
bx: &Builder<'_, '_, '_>,
1315-
) -> String {
1316-
let p0s: String = "p0".repeat(no_pointers);
1310+
fn llvm_vector_str(bx: &Builder<'_, '_, '_>, elem_ty: Ty<'_>, vec_len: u64) -> String {
13171311
match *elem_ty.kind() {
13181312
ty::Int(v) => format!(
1319-
"v{}{}i{}",
1313+
"v{}i{}",
13201314
vec_len,
1321-
p0s,
13221315
// Normalize to prevent crash if v: IntTy::Isize
13231316
v.normalize(bx.target_spec().pointer_width).bit_width().unwrap()
13241317
),
13251318
ty::Uint(v) => format!(
1326-
"v{}{}i{}",
1319+
"v{}i{}",
13271320
vec_len,
1328-
p0s,
13291321
// Normalize to prevent crash if v: UIntTy::Usize
13301322
v.normalize(bx.target_spec().pointer_width).bit_width().unwrap()
13311323
),
1332-
ty::Float(v) => format!("v{}{}f{}", vec_len, p0s, v.bit_width()),
1324+
ty::Float(v) => format!("v{}f{}", vec_len, v.bit_width()),
1325+
ty::RawPtr(_) => format!("v{}p0", vec_len),
13331326
_ => unreachable!(),
13341327
}
13351328
}
13361329

1337-
fn llvm_vector_ty<'ll>(
1338-
cx: &CodegenCx<'ll, '_>,
1339-
elem_ty: Ty<'_>,
1340-
vec_len: u64,
1341-
no_pointers: usize,
1342-
) -> &'ll Type {
1343-
// FIXME: use cx.layout_of(ty).llvm_type() ?
1344-
let mut elem_ty = match *elem_ty.kind() {
1330+
fn llvm_vector_ty<'ll>(cx: &CodegenCx<'ll, '_>, elem_ty: Ty<'_>, vec_len: u64) -> &'ll Type {
1331+
let elem_ty = match *elem_ty.kind() {
13451332
ty::Int(v) => cx.type_int_from_ty(v),
13461333
ty::Uint(v) => cx.type_uint_from_ty(v),
13471334
ty::Float(v) => cx.type_float_from_ty(v),
1335+
ty::RawPtr(_) => cx.type_ptr(),
13481336
_ => unreachable!(),
13491337
};
1350-
if no_pointers > 0 {
1351-
elem_ty = cx.type_ptr();
1352-
}
13531338
cx.type_vector(elem_ty, vec_len)
13541339
}
13551340

@@ -1404,47 +1389,26 @@ fn generic_simd_intrinsic<'ll, 'tcx>(
14041389
InvalidMonomorphization::ExpectedReturnType { span, name, in_ty, ret_ty }
14051390
);
14061391

1407-
// This counts how many pointers
1408-
fn ptr_count(t: Ty<'_>) -> usize {
1409-
match t.kind() {
1410-
ty::RawPtr(p) => 1 + ptr_count(p.ty),
1411-
_ => 0,
1412-
}
1413-
}
1414-
1415-
// Non-ptr type
1416-
fn non_ptr(t: Ty<'_>) -> Ty<'_> {
1417-
match t.kind() {
1418-
ty::RawPtr(p) => non_ptr(p.ty),
1419-
_ => t,
1420-
}
1421-
}
1422-
14231392
// The second argument must be a simd vector with an element type that's a pointer
14241393
// to the element type of the first argument
14251394
let (_, element_ty0) = arg_tys[0].simd_size_and_type(bx.tcx());
14261395
let (_, element_ty1) = arg_tys[1].simd_size_and_type(bx.tcx());
1427-
let (pointer_count, underlying_ty) = match element_ty1.kind() {
1428-
ty::RawPtr(p) if p.ty == in_elem => (ptr_count(element_ty1), non_ptr(element_ty1)),
1429-
_ => {
1430-
require!(
1431-
false,
1432-
InvalidMonomorphization::ExpectedElementType {
1433-
span,
1434-
name,
1435-
expected_element: element_ty1,
1436-
second_arg: arg_tys[1],
1437-
in_elem,
1438-
in_ty,
1439-
mutability: ExpectedPointerMutability::Not,
1440-
}
1441-
);
1442-
unreachable!();
1396+
1397+
require!(
1398+
matches!(
1399+
element_ty1.kind(),
1400+
ty::RawPtr(p) if p.ty == in_elem && p.ty.kind() == element_ty0.kind()
1401+
),
1402+
InvalidMonomorphization::ExpectedElementType {
1403+
span,
1404+
name,
1405+
expected_element: element_ty1,
1406+
second_arg: arg_tys[1],
1407+
in_elem,
1408+
in_ty,
1409+
mutability: ExpectedPointerMutability::Not,
14431410
}
1444-
};
1445-
assert!(pointer_count > 0);
1446-
assert_eq!(pointer_count - 1, ptr_count(element_ty0));
1447-
assert_eq!(underlying_ty, non_ptr(element_ty0));
1411+
);
14481412

14491413
// The element type of the third argument must be a signed integer type of any width:
14501414
let (_, element_ty2) = arg_tys[2].simd_size_and_type(bx.tcx());
@@ -1475,12 +1439,12 @@ fn generic_simd_intrinsic<'ll, 'tcx>(
14751439
};
14761440

14771441
// Type of the vector of pointers:
1478-
let llvm_pointer_vec_ty = llvm_vector_ty(bx, underlying_ty, in_len, pointer_count);
1479-
let llvm_pointer_vec_str = llvm_vector_str(underlying_ty, in_len, pointer_count, bx);
1442+
let llvm_pointer_vec_ty = llvm_vector_ty(bx, element_ty1, in_len);
1443+
let llvm_pointer_vec_str = llvm_vector_str(bx, element_ty1, in_len);
14801444

14811445
// Type of the vector of elements:
1482-
let llvm_elem_vec_ty = llvm_vector_ty(bx, underlying_ty, in_len, pointer_count - 1);
1483-
let llvm_elem_vec_str = llvm_vector_str(underlying_ty, in_len, pointer_count - 1, bx);
1446+
let llvm_elem_vec_ty = llvm_vector_ty(bx, element_ty0, in_len);
1447+
let llvm_elem_vec_str = llvm_vector_str(bx, element_ty0, in_len);
14841448

14851449
let llvm_intrinsic =
14861450
format!("llvm.masked.gather.{}.{}", llvm_elem_vec_str, llvm_pointer_vec_str);
@@ -1544,50 +1508,28 @@ fn generic_simd_intrinsic<'ll, 'tcx>(
15441508
}
15451509
);
15461510

1547-
// This counts how many pointers
1548-
fn ptr_count(t: Ty<'_>) -> usize {
1549-
match t.kind() {
1550-
ty::RawPtr(p) => 1 + ptr_count(p.ty),
1551-
_ => 0,
1552-
}
1553-
}
1554-
1555-
// Non-ptr type
1556-
fn non_ptr(t: Ty<'_>) -> Ty<'_> {
1557-
match t.kind() {
1558-
ty::RawPtr(p) => non_ptr(p.ty),
1559-
_ => t,
1560-
}
1561-
}
1562-
15631511
// The second argument must be a simd vector with an element type that's a pointer
15641512
// to the element type of the first argument
15651513
let (_, element_ty0) = arg_tys[0].simd_size_and_type(bx.tcx());
15661514
let (_, element_ty1) = arg_tys[1].simd_size_and_type(bx.tcx());
15671515
let (_, element_ty2) = arg_tys[2].simd_size_and_type(bx.tcx());
1568-
let (pointer_count, underlying_ty) = match element_ty1.kind() {
1569-
ty::RawPtr(p) if p.ty == in_elem && p.mutbl.is_mut() => {
1570-
(ptr_count(element_ty1), non_ptr(element_ty1))
1571-
}
1572-
_ => {
1573-
require!(
1574-
false,
1575-
InvalidMonomorphization::ExpectedElementType {
1576-
span,
1577-
name,
1578-
expected_element: element_ty1,
1579-
second_arg: arg_tys[1],
1580-
in_elem,
1581-
in_ty,
1582-
mutability: ExpectedPointerMutability::Mut,
1583-
}
1584-
);
1585-
unreachable!();
1516+
1517+
require!(
1518+
matches!(
1519+
element_ty1.kind(),
1520+
ty::RawPtr(p)
1521+
if p.ty == in_elem && p.mutbl.is_mut() && p.ty.kind() == element_ty0.kind()
1522+
),
1523+
InvalidMonomorphization::ExpectedElementType {
1524+
span,
1525+
name,
1526+
expected_element: element_ty1,
1527+
second_arg: arg_tys[1],
1528+
in_elem,
1529+
in_ty,
1530+
mutability: ExpectedPointerMutability::Mut,
15861531
}
1587-
};
1588-
assert!(pointer_count > 0);
1589-
assert_eq!(pointer_count - 1, ptr_count(element_ty0));
1590-
assert_eq!(underlying_ty, non_ptr(element_ty0));
1532+
);
15911533

15921534
// The element type of the third argument must be a signed integer type of any width:
15931535
match element_ty2.kind() {
@@ -1619,12 +1561,12 @@ fn generic_simd_intrinsic<'ll, 'tcx>(
16191561
let ret_t = bx.type_void();
16201562

16211563
// Type of the vector of pointers:
1622-
let llvm_pointer_vec_ty = llvm_vector_ty(bx, underlying_ty, in_len, pointer_count);
1623-
let llvm_pointer_vec_str = llvm_vector_str(underlying_ty, in_len, pointer_count, bx);
1564+
let llvm_pointer_vec_ty = llvm_vector_ty(bx, element_ty1, in_len);
1565+
let llvm_pointer_vec_str = llvm_vector_str(bx, element_ty1, in_len);
16241566

16251567
// Type of the vector of elements:
1626-
let llvm_elem_vec_ty = llvm_vector_ty(bx, underlying_ty, in_len, pointer_count - 1);
1627-
let llvm_elem_vec_str = llvm_vector_str(underlying_ty, in_len, pointer_count - 1, bx);
1568+
let llvm_elem_vec_ty = llvm_vector_ty(bx, element_ty0, in_len);
1569+
let llvm_elem_vec_str = llvm_vector_str(bx, element_ty0, in_len);
16281570

16291571
let llvm_intrinsic =
16301572
format!("llvm.masked.scatter.{}.{}", llvm_elem_vec_str, llvm_pointer_vec_str);

tests/codegen/simd-intrinsic/simd-intrinsic-generic-gather.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -23,14 +23,14 @@ extern "platform-intrinsic" {
2323
#[no_mangle]
2424
pub unsafe fn gather_f32x2(pointers: Vec2<*const f32>, mask: Vec2<i32>,
2525
values: Vec2<f32>) -> Vec2<f32> {
26-
// CHECK: call <2 x float> @llvm.masked.gather.v2f32.{{.+}}(<2 x ptr> {{.*}}, i32 {{.*}}, <2 x i1> {{.*}}, <2 x float> {{.*}})
26+
// CHECK: call <2 x float> @llvm.masked.gather.v2f32.v2p0(<2 x ptr> {{.*}}, i32 {{.*}}, <2 x i1> {{.*}}, <2 x float> {{.*}})
2727
simd_gather(values, pointers, mask)
2828
}
2929

3030
// CHECK-LABEL: @gather_pf32x2
3131
#[no_mangle]
3232
pub unsafe fn gather_pf32x2(pointers: Vec2<*const *const f32>, mask: Vec2<i32>,
3333
values: Vec2<*const f32>) -> Vec2<*const f32> {
34-
// CHECK: call <2 x ptr> @llvm.masked.gather.{{.+}}(<2 x ptr> {{.*}}, i32 {{.*}}, <2 x i1> {{.*}}, <2 x ptr> {{.*}})
34+
// CHECK: call <2 x ptr> @llvm.masked.gather.v2p0.v2p0(<2 x ptr> {{.*}}, i32 {{.*}}, <2 x i1> {{.*}}, <2 x ptr> {{.*}})
3535
simd_gather(values, pointers, mask)
3636
}

tests/codegen/simd-intrinsic/simd-intrinsic-generic-scatter.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,7 @@ extern "platform-intrinsic" {
2323
#[no_mangle]
2424
pub unsafe fn scatter_f32x2(pointers: Vec2<*mut f32>, mask: Vec2<i32>,
2525
values: Vec2<f32>) {
26-
// CHECK: call void @llvm.masked.scatter.v2f32.v2p0{{.*}}(<2 x float> {{.*}}, <2 x ptr> {{.*}}, i32 {{.*}}, <2 x i1> {{.*}})
26+
// CHECK: call void @llvm.masked.scatter.v2f32.v2p0(<2 x float> {{.*}}, <2 x ptr> {{.*}}, i32 {{.*}}, <2 x i1> {{.*}})
2727
simd_scatter(values, pointers, mask)
2828
}
2929

@@ -32,6 +32,6 @@ pub unsafe fn scatter_f32x2(pointers: Vec2<*mut f32>, mask: Vec2<i32>,
3232
#[no_mangle]
3333
pub unsafe fn scatter_pf32x2(pointers: Vec2<*mut *const f32>, mask: Vec2<i32>,
3434
values: Vec2<*const f32>) {
35-
// CHECK: call void @llvm.masked.scatter.v2p0{{.*}}.v2p0{{.*}}(<2 x ptr> {{.*}}, <2 x ptr> {{.*}}, i32 {{.*}}, <2 x i1> {{.*}})
35+
// CHECK: call void @llvm.masked.scatter.v2p0.v2p0(<2 x ptr> {{.*}}, <2 x ptr> {{.*}}, i32 {{.*}}, <2 x i1> {{.*}})
3636
simd_scatter(values, pointers, mask)
3737
}

0 commit comments

Comments
 (0)