
This adds support for explicit pointer arithmetic, including unary increment and decrement of pointer values.
78 lines
3.7 KiB
C++
78 lines
3.7 KiB
C++
// RUN: %clang_cc1 -triple x86_64-unknown-linux-gnu -fclangir -emit-cir %s -o %t.cir
|
|
// RUN: FileCheck --input-file=%t.cir %s
|
|
|
|
// Should generate basic pointer arithmetics.
|
|
void foo(int *iptr, char *cptr, unsigned ustride) {
|
|
iptr + 2;
|
|
// CHECK: %[[#STRIDE:]] = cir.const #cir.int<2> : !s32i
|
|
// CHECK: cir.ptr_stride(%{{.+}} : !cir.ptr<!s32i>, %[[#STRIDE]] : !s32i), !cir.ptr<!s32i>
|
|
cptr + 3;
|
|
// CHECK: %[[#STRIDE:]] = cir.const #cir.int<3> : !s32i
|
|
// CHECK: cir.ptr_stride(%{{.+}} : !cir.ptr<!s8i>, %[[#STRIDE]] : !s32i), !cir.ptr<!s8i>
|
|
iptr - 2;
|
|
// CHECK: %[[#STRIDE:]] = cir.const #cir.int<2> : !s32i
|
|
// CHECK: %[[#NEGSTRIDE:]] = cir.unary(minus, %[[#STRIDE]]) : !s32i, !s32i
|
|
// CHECK: cir.ptr_stride(%{{.+}} : !cir.ptr<!s32i>, %[[#NEGSTRIDE]] : !s32i), !cir.ptr<!s32i>
|
|
cptr - 3;
|
|
// CHECK: %[[#STRIDE:]] = cir.const #cir.int<3> : !s32i
|
|
// CHECK: %[[#NEGSTRIDE:]] = cir.unary(minus, %[[#STRIDE]]) : !s32i, !s32i
|
|
// CHECK: cir.ptr_stride(%{{.+}} : !cir.ptr<!s8i>, %[[#NEGSTRIDE]] : !s32i), !cir.ptr<!s8i>
|
|
iptr + ustride;
|
|
// CHECK: %[[#STRIDE:]] = cir.load %{{.+}} : !cir.ptr<!u32i>, !u32i
|
|
// CHECK: cir.ptr_stride(%{{.+}} : !cir.ptr<!s32i>, %[[#STRIDE]] : !u32i), !cir.ptr<!s32i>
|
|
|
|
// Must convert unsigned stride to a signed one.
|
|
iptr - ustride;
|
|
// CHECK: %[[#STRIDE:]] = cir.load %{{.+}} : !cir.ptr<!u32i>, !u32i
|
|
// CHECK: %[[#SIGNSTRIDE:]] = cir.cast(integral, %[[#STRIDE]] : !u32i), !s32i
|
|
// CHECK: %[[#NEGSTRIDE:]] = cir.unary(minus, %[[#SIGNSTRIDE]]) : !s32i, !s32i
|
|
// CHECK: cir.ptr_stride(%{{.+}} : !cir.ptr<!s32i>, %[[#NEGSTRIDE]] : !s32i), !cir.ptr<!s32i>
|
|
|
|
4 + iptr;
|
|
// CHECK: %[[#STRIDE:]] = cir.const #cir.int<4> : !s32i
|
|
// CHECK: cir.ptr_stride(%{{.+}} : !cir.ptr<!s32i>, %[[#STRIDE]] : !s32i), !cir.ptr<!s32i>
|
|
|
|
iptr++;
|
|
// CHECK: %[[#STRIDE:]] = cir.const #cir.int<1> : !s32i
|
|
// CHECK: cir.ptr_stride(%{{.+}} : !cir.ptr<!s32i>, %[[#STRIDE]] : !s32i), !cir.ptr<!s32i>
|
|
|
|
iptr--;
|
|
// CHECK: %[[#STRIDE:]] = cir.const #cir.int<-1> : !s32i
|
|
// CHECK: cir.ptr_stride(%{{.+}} : !cir.ptr<!s32i>, %[[#STRIDE]] : !s32i), !cir.ptr<!s32i>
|
|
}
|
|
|
|
void testPointerSubscriptAccess(int *ptr) {
|
|
// CHECK: testPointerSubscriptAccess
|
|
ptr[1];
|
|
// CHECK: %[[#STRIDE:]] = cir.const #cir.int<1> : !s32i
|
|
// CHECK: %[[#PTR:]] = cir.load %{{.+}} : !cir.ptr<!cir.ptr<!s32i>>, !cir.ptr<!s32i>
|
|
// CHECK: cir.ptr_stride(%[[#PTR]] : !cir.ptr<!s32i>, %[[#STRIDE]] : !s32i), !cir.ptr<!s32i>
|
|
}
|
|
|
|
void testPointerMultiDimSubscriptAccess(int **ptr) {
|
|
// CHECK: testPointerMultiDimSubscriptAccess
|
|
ptr[1][2];
|
|
// CHECK: %[[#STRIDE2:]] = cir.const #cir.int<2> : !s32i
|
|
// CHECK: %[[#STRIDE1:]] = cir.const #cir.int<1> : !s32i
|
|
// CHECK: %[[#PTR1:]] = cir.load %{{.+}} : !cir.ptr<!cir.ptr<!cir.ptr<!s32i>>>, !cir.ptr<!cir.ptr<!s32i>>
|
|
// CHECK: %[[#PTR2:]] = cir.ptr_stride(%[[#PTR1]] : !cir.ptr<!cir.ptr<!s32i>>, %[[#STRIDE1]] : !s32i), !cir.ptr<!cir.ptr<!s32i>>
|
|
// CHECK: %[[#PTR3:]] = cir.load %[[#PTR2]] : !cir.ptr<!cir.ptr<!s32i>>, !cir.ptr<!s32i>
|
|
// CHECK: cir.ptr_stride(%[[#PTR3]] : !cir.ptr<!s32i>, %[[#STRIDE2]] : !s32i), !cir.ptr<!s32i>
|
|
}
|
|
|
|
// This test is meant to verify code that handles the 'p = nullptr + n' idiom
|
|
// used by some versions of glibc and gcc. This is undefined behavior but
|
|
// it is intended there to act like a conversion from a pointer-sized integer
|
|
// to a pointer, and we would like to tolerate that.
|
|
|
|
#define NULLPTRINT ((int*)0)
|
|
|
|
// This should get the inttoptr instruction.
|
|
int *testGnuNullPtrArithmetic(unsigned n) {
|
|
// CHECK: testGnuNullPtrArithmetic
|
|
return NULLPTRINT + n;
|
|
// CHECK: %[[NULLPTR:.*]] = cir.const #cir.ptr<null> : !cir.ptr<!s32i>
|
|
// CHECK: %[[N:.*]] = cir.load %{{.*}} : !cir.ptr<!u32i>, !u32i
|
|
// CHECK: %[[RESULT:.*]] = cir.ptr_stride(%[[NULLPTR]] : !cir.ptr<!s32i>, %[[N]] : !u32i), !cir.ptr<!s32i>
|
|
}
|