blob: adf3aa12623b9bd9f598f3b16057bde4780b0b11 [file] [log] [blame]
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --version 6
; RUN: opt -S -passes=instcombine < %s | FileCheck %s
; The ptrtoaddr folds are also valid for pointers that have external state.
target datalayout = "pe1:64:64:64:32"
declare void @use.i1(i1)
declare void @use.i32(i32)
declare void @use.i64(i64)
; ptrtoaddr result type is fixed, and can't be combined with integer cast.
define i32 @ptrtoaddr_trunc(ptr %p) {
; CHECK-LABEL: define i32 @ptrtoaddr_trunc(
; CHECK-SAME: ptr [[P:%.*]]) {
; CHECK-NEXT: [[P_ADDR:%.*]] = ptrtoaddr ptr [[P]] to i64
; CHECK-NEXT: [[TRUNC:%.*]] = trunc i64 [[P_ADDR]] to i32
; CHECK-NEXT: ret i32 [[TRUNC]]
;
%p.addr = ptrtoaddr ptr %p to i64
%trunc = trunc i64 %p.addr to i32
ret i32 %trunc
}
define i128 @ptrtoaddr_zext(ptr %p) {
; CHECK-LABEL: define i128 @ptrtoaddr_zext(
; CHECK-SAME: ptr [[P:%.*]]) {
; CHECK-NEXT: [[P_ADDR:%.*]] = ptrtoaddr ptr [[P]] to i64
; CHECK-NEXT: [[EXT:%.*]] = zext i64 [[P_ADDR]] to i128
; CHECK-NEXT: ret i128 [[EXT]]
;
%p.addr = ptrtoaddr ptr %p to i64
%ext = zext i64 %p.addr to i128
ret i128 %ext
}
define i128 @ptrtoaddr_sext(ptr %p) {
; CHECK-LABEL: define i128 @ptrtoaddr_sext(
; CHECK-SAME: ptr [[P:%.*]]) {
; CHECK-NEXT: [[P_ADDR:%.*]] = ptrtoaddr ptr [[P]] to i64
; CHECK-NEXT: [[EXT:%.*]] = sext i64 [[P_ADDR]] to i128
; CHECK-NEXT: ret i128 [[EXT]]
;
%p.addr = ptrtoaddr ptr %p to i64
%ext = sext i64 %p.addr to i128
ret i128 %ext
}
define i64 @sub_ptrtoaddr(ptr %p, i64 %offset) {
; CHECK-LABEL: define i64 @sub_ptrtoaddr(
; CHECK-SAME: ptr [[P:%.*]], i64 [[OFFSET:%.*]]) {
; CHECK-NEXT: ret i64 [[OFFSET]]
;
%p2 = getelementptr i8, ptr %p, i64 %offset
%p.addr = ptrtoaddr ptr %p to i64
%p2.addr = ptrtoaddr ptr %p2 to i64
%sub = sub i64 %p2.addr, %p.addr
ret i64 %sub
}
define i64 @sub_ptrtoint_ptrtoaddr(ptr %p, i64 %offset) {
; CHECK-LABEL: define i64 @sub_ptrtoint_ptrtoaddr(
; CHECK-SAME: ptr [[P:%.*]], i64 [[OFFSET:%.*]]) {
; CHECK-NEXT: ret i64 [[OFFSET]]
;
%p2 = getelementptr i8, ptr %p, i64 %offset
%p.int = ptrtoint ptr %p to i64
%p2.addr = ptrtoaddr ptr %p2 to i64
%sub = sub i64 %p2.addr, %p.int
ret i64 %sub
}
define i32 @sub_ptrtoaddr_addrsize(ptr addrspace(1) %p, i32 %offset) {
; CHECK-LABEL: define i32 @sub_ptrtoaddr_addrsize(
; CHECK-SAME: ptr addrspace(1) [[P:%.*]], i32 [[OFFSET:%.*]]) {
; CHECK-NEXT: ret i32 [[OFFSET]]
;
%p2 = getelementptr i8, ptr addrspace(1) %p, i32 %offset
%p.addr = ptrtoaddr ptr addrspace(1) %p to i32
%p2.addr = ptrtoaddr ptr addrspace(1) %p2 to i32
%sub = sub i32 %p2.addr, %p.addr
ret i32 %sub
}
define i32 @sub_trunc_ptrtoaddr(ptr %p, i64 %offset) {
; CHECK-LABEL: define i32 @sub_trunc_ptrtoaddr(
; CHECK-SAME: ptr [[P:%.*]], i64 [[OFFSET:%.*]]) {
; CHECK-NEXT: [[SUB:%.*]] = trunc i64 [[OFFSET]] to i32
; CHECK-NEXT: ret i32 [[SUB]]
;
%p2 = getelementptr i8, ptr %p, i64 %offset
%p.addr = ptrtoaddr ptr %p to i64
%p2.addr = ptrtoaddr ptr %p2 to i64
%p.addr.trunc = trunc i64 %p.addr to i32
%p2.addr.trunc = trunc i64 %p2.addr to i32
%sub = sub i32 %p2.addr.trunc, %p.addr.trunc
ret i32 %sub
}
define i16 @sub_trunc_ptrtoaddr_addrsize(ptr addrspace(1) %p, i32 %offset) {
; CHECK-LABEL: define i16 @sub_trunc_ptrtoaddr_addrsize(
; CHECK-SAME: ptr addrspace(1) [[P:%.*]], i32 [[OFFSET:%.*]]) {
; CHECK-NEXT: [[SUB:%.*]] = trunc i32 [[OFFSET]] to i16
; CHECK-NEXT: ret i16 [[SUB]]
;
%p2 = getelementptr i8, ptr addrspace(1) %p, i32 %offset
%p.addr = ptrtoaddr ptr addrspace(1) %p to i32
%p2.addr = ptrtoaddr ptr addrspace(1) %p2 to i32
%p.addr.trunc = trunc i32 %p.addr to i16
%p2.addr.trunc = trunc i32 %p2.addr to i16
%sub = sub i16 %p2.addr.trunc, %p.addr.trunc
ret i16 %sub
}
define i16 @sub_trunc_ptrtoint_ptrtoaddr_addrsize(ptr addrspace(1) %p, i32 %offset) {
; CHECK-LABEL: define i16 @sub_trunc_ptrtoint_ptrtoaddr_addrsize(
; CHECK-SAME: ptr addrspace(1) [[P:%.*]], i32 [[OFFSET:%.*]]) {
; CHECK-NEXT: [[SUB:%.*]] = trunc i32 [[OFFSET]] to i16
; CHECK-NEXT: ret i16 [[SUB]]
;
%p2 = getelementptr i8, ptr addrspace(1) %p, i32 %offset
%p.int = ptrtoint ptr addrspace(1) %p to i64
%p2.addr = ptrtoaddr ptr addrspace(1) %p2 to i32
%p.int.trunc = trunc i64 %p.int to i16
%p2.addr.trunc = trunc i32 %p2.addr to i16
%sub = sub i16 %p2.addr.trunc, %p.int.trunc
ret i16 %sub
}
define i128 @sub_zext_ptrtoaddr(ptr %p, i64 %offset) {
; CHECK-LABEL: define i128 @sub_zext_ptrtoaddr(
; CHECK-SAME: ptr [[P:%.*]], i64 [[OFFSET:%.*]]) {
; CHECK-NEXT: [[SUB:%.*]] = zext i64 [[OFFSET]] to i128
; CHECK-NEXT: ret i128 [[SUB]]
;
%p2 = getelementptr nuw i8, ptr %p, i64 %offset
%p.addr = ptrtoaddr ptr %p to i64
%p2.addr = ptrtoaddr ptr %p2 to i64
%p.addr.ext = zext i64 %p.addr to i128
%p2.addr.ext = zext i64 %p2.addr to i128
%sub = sub i128 %p2.addr.ext, %p.addr.ext
ret i128 %sub
}
define i64 @sub_zext_ptrtoaddr_addrsize(ptr addrspace(1) %p, i32 %offset) {
; CHECK-LABEL: define i64 @sub_zext_ptrtoaddr_addrsize(
; CHECK-SAME: ptr addrspace(1) [[P:%.*]], i32 [[OFFSET:%.*]]) {
; CHECK-NEXT: [[SUB:%.*]] = zext i32 [[OFFSET]] to i64
; CHECK-NEXT: ret i64 [[SUB]]
;
%p2 = getelementptr nuw i8, ptr addrspace(1) %p, i32 %offset
%p.addr = ptrtoaddr ptr addrspace(1) %p to i32
%p2.addr = ptrtoaddr ptr addrspace(1) %p2 to i32
%p.addr.ext = zext i32 %p.addr to i64
%p2.addr.ext = zext i32 %p2.addr to i64
%sub = sub i64 %p2.addr.ext, %p.addr.ext
ret i64 %sub
}
define i128 @sub_zext_ptrtoint_ptrtoaddr_addrsize(ptr addrspace(1) %p, i32 %offset) {
; CHECK-LABEL: define i128 @sub_zext_ptrtoint_ptrtoaddr_addrsize(
; CHECK-SAME: ptr addrspace(1) [[P:%.*]], i32 [[OFFSET:%.*]]) {
; CHECK-NEXT: [[P2:%.*]] = getelementptr nuw i8, ptr addrspace(1) [[P]], i32 [[OFFSET]]
; CHECK-NEXT: [[P_INT:%.*]] = ptrtoint ptr addrspace(1) [[P]] to i64
; CHECK-NEXT: [[P2_ADDR:%.*]] = ptrtoaddr ptr addrspace(1) [[P2]] to i32
; CHECK-NEXT: [[P_INT_EXT:%.*]] = zext i64 [[P_INT]] to i128
; CHECK-NEXT: [[P2_ADDR_EXT:%.*]] = zext i32 [[P2_ADDR]] to i128
; CHECK-NEXT: [[SUB:%.*]] = sub nsw i128 [[P2_ADDR_EXT]], [[P_INT_EXT]]
; CHECK-NEXT: ret i128 [[SUB]]
;
%p2 = getelementptr nuw i8, ptr addrspace(1) %p, i32 %offset
%p.int = ptrtoint ptr addrspace(1) %p to i64
%p2.addr = ptrtoaddr ptr addrspace(1) %p2 to i32
%p.int.ext = zext i64 %p.int to i128
%p2.addr.ext = zext i32 %p2.addr to i128
%sub = sub i128 %p2.addr.ext, %p.int.ext
ret i128 %sub
}
; The uses in icmp, ptrtoint, ptrtoaddr should be replaced. The one in the
; return value should not, as the provenance differs.
define ptr @gep_sub_ptrtoaddr_different_obj(ptr %p, ptr %p2, ptr %p3) {
; CHECK-LABEL: define ptr @gep_sub_ptrtoaddr_different_obj(
; CHECK-SAME: ptr [[P:%.*]], ptr [[P2:%.*]], ptr [[P3:%.*]]) {
; CHECK-NEXT: [[P_ADDR:%.*]] = ptrtoaddr ptr [[P]] to i64
; CHECK-NEXT: [[P2_ADDR:%.*]] = ptrtoaddr ptr [[P2]] to i64
; CHECK-NEXT: [[SUB:%.*]] = sub i64 [[P2_ADDR]], [[P_ADDR]]
; CHECK-NEXT: [[GEP:%.*]] = getelementptr i8, ptr [[P]], i64 [[SUB]]
; CHECK-NEXT: [[CMP:%.*]] = icmp eq ptr [[P2]], [[P3]]
; CHECK-NEXT: call void @use.i1(i1 [[CMP]])
; CHECK-NEXT: [[INT:%.*]] = ptrtoint ptr [[P2]] to i64
; CHECK-NEXT: call void @use.i64(i64 [[INT]])
; CHECK-NEXT: [[ADDR:%.*]] = ptrtoaddr ptr [[P2]] to i64
; CHECK-NEXT: call void @use.i64(i64 [[ADDR]])
; CHECK-NEXT: ret ptr [[GEP]]
;
%p.addr = ptrtoaddr ptr %p to i64
%p2.addr = ptrtoaddr ptr %p2 to i64
%sub = sub i64 %p2.addr, %p.addr
%gep = getelementptr i8, ptr %p, i64 %sub
%cmp = icmp eq ptr %gep, %p3
call void @use.i1(i1 %cmp)
%int = ptrtoint ptr %gep to i64
call void @use.i64(i64 %int)
%addr = ptrtoaddr ptr %gep to i64
call void @use.i64(i64 %addr)
ret ptr %gep
}
; The use in ptrtoaddr should be replaced. The uses in ptrtoint and icmp should
; not be replaced, as the non-address bits differ. The use in the return value
; should not be replaced as the provenace differs.
define ptr addrspace(1) @gep_sub_ptrtoaddr_different_obj_addrsize(ptr addrspace(1) %p, ptr addrspace(1) %p2, ptr addrspace(1) %p3) {
; CHECK-LABEL: define ptr addrspace(1) @gep_sub_ptrtoaddr_different_obj_addrsize(
; CHECK-SAME: ptr addrspace(1) [[P:%.*]], ptr addrspace(1) [[P2:%.*]], ptr addrspace(1) [[P3:%.*]]) {
; CHECK-NEXT: [[P_ADDR:%.*]] = ptrtoaddr ptr addrspace(1) [[P]] to i32
; CHECK-NEXT: [[P2_ADDR:%.*]] = ptrtoaddr ptr addrspace(1) [[P2]] to i32
; CHECK-NEXT: [[SUB:%.*]] = sub i32 [[P2_ADDR]], [[P_ADDR]]
; CHECK-NEXT: [[GEP:%.*]] = getelementptr i8, ptr addrspace(1) [[P]], i32 [[SUB]]
; CHECK-NEXT: [[CMP:%.*]] = icmp eq ptr addrspace(1) [[GEP]], [[P3]]
; CHECK-NEXT: call void @use.i1(i1 [[CMP]])
; CHECK-NEXT: [[TMP1:%.*]] = ptrtoint ptr addrspace(1) [[GEP]] to i64
; CHECK-NEXT: [[INT:%.*]] = trunc i64 [[TMP1]] to i32
; CHECK-NEXT: call void @use.i32(i32 [[INT]])
; CHECK-NEXT: [[ADDR:%.*]] = ptrtoaddr ptr addrspace(1) [[P2]] to i32
; CHECK-NEXT: call void @use.i32(i32 [[ADDR]])
; CHECK-NEXT: ret ptr addrspace(1) [[GEP]]
;
%p.addr = ptrtoaddr ptr addrspace(1) %p to i32
%p2.addr = ptrtoaddr ptr addrspace(1) %p2 to i32
%sub = sub i32 %p2.addr, %p.addr
%gep = getelementptr i8, ptr addrspace(1) %p, i32 %sub
%cmp = icmp eq ptr addrspace(1) %gep, %p3
call void @use.i1(i1 %cmp)
%int = ptrtoint ptr addrspace(1) %gep to i32
call void @use.i32(i32 %int)
%addr = ptrtoaddr ptr addrspace(1) %gep to i32
call void @use.i32(i32 %addr)
ret ptr addrspace(1) %gep
}
define i64 @ptrtoaddr_of_ptrmask(ptr %p, i64 %mask) {
; CHECK-LABEL: define i64 @ptrtoaddr_of_ptrmask(
; CHECK-SAME: ptr [[P:%.*]], i64 [[MASK:%.*]]) {
; CHECK-NEXT: [[TMP1:%.*]] = ptrtoaddr ptr [[P]] to i64
; CHECK-NEXT: [[ADDR:%.*]] = and i64 [[MASK]], [[TMP1]]
; CHECK-NEXT: ret i64 [[ADDR]]
;
%masked = call ptr @llvm.ptrmask(ptr %p, i64 %mask)
%addr = ptrtoaddr ptr %masked to i64
ret i64 %addr
}
define i32 @ptrtoaddr_of_ptrmask_addrsize(ptr addrspace(1) %p, i32 %mask) {
; CHECK-LABEL: define i32 @ptrtoaddr_of_ptrmask_addrsize(
; CHECK-SAME: ptr addrspace(1) [[P:%.*]], i32 [[MASK:%.*]]) {
; CHECK-NEXT: [[TMP1:%.*]] = ptrtoaddr ptr addrspace(1) [[P]] to i32
; CHECK-NEXT: [[ADDR:%.*]] = and i32 [[MASK]], [[TMP1]]
; CHECK-NEXT: ret i32 [[ADDR]]
;
%masked = call ptr addrspace(1) @llvm.ptrmask(ptr addrspace(1) %p, i32 %mask)
%addr = ptrtoaddr ptr addrspace(1) %masked to i32
ret i32 %addr
}
define i64 @ptrtoaddr_of_gep_of_inttoptr(i64 %int, i64 %offset) {
; CHECK-LABEL: define i64 @ptrtoaddr_of_gep_of_inttoptr(
; CHECK-SAME: i64 [[INT:%.*]], i64 [[OFFSET:%.*]]) {
; CHECK-NEXT: [[ADDR:%.*]] = add i64 [[INT]], [[OFFSET]]
; CHECK-NEXT: ret i64 [[ADDR]]
;
%ptr = inttoptr i64 %int to ptr
%gep = getelementptr i8, ptr %ptr, i64 %offset
%addr = ptrtoaddr ptr %gep to i64
ret i64 %addr
}
; FIXME: This could be supported by truncating %int before performing the
; arithmetic.
define i32 @ptrtoaddr_of_gep_of_inttoptr_addrsize(i64 %int, i32 %offset) {
; CHECK-LABEL: define i32 @ptrtoaddr_of_gep_of_inttoptr_addrsize(
; CHECK-SAME: i64 [[INT:%.*]], i32 [[OFFSET:%.*]]) {
; CHECK-NEXT: [[PTR:%.*]] = inttoptr i64 [[INT]] to ptr addrspace(1)
; CHECK-NEXT: [[GEP:%.*]] = getelementptr i8, ptr addrspace(1) [[PTR]], i32 [[OFFSET]]
; CHECK-NEXT: [[ADDR:%.*]] = ptrtoaddr ptr addrspace(1) [[GEP]] to i32
; CHECK-NEXT: ret i32 [[ADDR]]
;
%ptr = inttoptr i64 %int to ptr addrspace(1)
%gep = getelementptr i8, ptr addrspace(1) %ptr, i32 %offset
%addr = ptrtoaddr ptr addrspace(1) %gep to i32
ret i32 %addr
}
define i64 @ptrtoaddr_of_gep_of_null(i64 %offset) {
; CHECK-LABEL: define i64 @ptrtoaddr_of_gep_of_null(
; CHECK-SAME: i64 [[OFFSET:%.*]]) {
; CHECK-NEXT: ret i64 [[OFFSET]]
;
%gep = getelementptr i8, ptr null, i64 %offset
%addr = ptrtoaddr ptr %gep to i64
ret i64 %addr
}
define i32 @ptrtoaddr_of_gep_of_null_addrsize(i32 %offset) {
; CHECK-LABEL: define i32 @ptrtoaddr_of_gep_of_null_addrsize(
; CHECK-SAME: i32 [[OFFSET:%.*]]) {
; CHECK-NEXT: ret i32 [[OFFSET]]
;
%gep = getelementptr i8, ptr addrspace(1) null, i32 %offset
%addr = ptrtoaddr ptr addrspace(1) %gep to i32
ret i32 %addr
}