; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
; Verify that equality tests of strnlen calls against zero are folded
; correctly.
;
; RUN: opt < %s -passes=instcombine -S | FileCheck %s
declare i64 @strnlen(ptr, i64)
@ax = external global [0 x i8]
@a5 = external global [5 x i8]
@s5 = constant [6 x i8] c"12345\00"
; Fold strnlen(ax, 0) == 0 to true.
define i1 @fold_strnlen_ax_0_eqz() {
; CHECK-LABEL: @fold_strnlen_ax_0_eqz(
; CHECK-NEXT: ret i1 true
;
%len = tail call i64 @strnlen(ptr @ax, i64 0)
%eqz = icmp eq i64 %len, 0
ret i1 %eqz
}
; Fold strnlen(ax, 0) > 0 to false.
define i1 @fold_strnlen_ax_0_gtz() {
; CHECK-LABEL: @fold_strnlen_ax_0_gtz(
; CHECK-NEXT: ret i1 false
;
%len = tail call i64 @strnlen(ptr @ax, i64 0)
%gtz = icmp ugt i64 %len, 0
ret i1 %gtz
}
; Fold strnlen(ax, 1) == 0 to *ax == 0.
define i1 @fold_strnlen_ax_1_eqz() {
; CHECK-LABEL: @fold_strnlen_ax_1_eqz(
; CHECK-NEXT: [[CHAR0:%.*]] = load i8, ptr @ax, align 1
; CHECK-NEXT: [[EQZ:%.*]] = icmp eq i8 [[CHAR0]], 0
; CHECK-NEXT: ret i1 [[EQZ]]
;
%len = tail call i64 @strnlen(ptr @ax, i64 1)
%eqz = icmp eq i64 %len, 0
ret i1 %eqz
}
; Likewise, fold strnlen(ax, 1) < 1 to *ax == 0.
define i1 @fold_strnlen_ax_1_lt1() {
; CHECK-LABEL: @fold_strnlen_ax_1_lt1(
; CHECK-NEXT: [[STRNLEN_CHAR0:%.*]] = load i8, ptr @ax, align 1
; CHECK-NEXT: [[STRNLEN_CHAR0CMP_NOT:%.*]] = icmp eq i8 [[STRNLEN_CHAR0]], 0
; CHECK-NEXT: ret i1 [[STRNLEN_CHAR0CMP_NOT]]
;
%len = tail call i64 @strnlen(ptr @ax, i64 1)
%nez = icmp ult i64 %len, 1
ret i1 %nez
}
; Fold strnlen(ax, 1) != 0 to *ax != 0.
define i1 @fold_strnlen_ax_1_neqz() {
; CHECK-LABEL: @fold_strnlen_ax_1_neqz(
; CHECK-NEXT: [[CHAR0:%.*]] = load i8, ptr @ax, align 1
; CHECK-NEXT: [[NEZ:%.*]] = icmp ne i8 [[CHAR0]], 0
; CHECK-NEXT: ret i1 [[NEZ]]
;
%len = tail call i64 @strnlen(ptr @ax, i64 1)
%nez = icmp ne i64 %len, 0
ret i1 %nez
}
; Likewise, fold strnlen(ax, 1) > 0 to *ax != 0.
define i1 @fold_strnlen_ax_1_gtz() {
; CHECK-LABEL: @fold_strnlen_ax_1_gtz(
; CHECK-NEXT: [[STRNLEN_CHAR0:%.*]] = load i8, ptr @ax, align 1
; CHECK-NEXT: [[STRNLEN_CHAR0CMP:%.*]] = icmp ne i8 [[STRNLEN_CHAR0]], 0
; CHECK-NEXT: ret i1 [[STRNLEN_CHAR0CMP]]
;
%len = tail call i64 @strnlen(ptr @ax, i64 1)
%nez = icmp ugt i64 %len, 0
ret i1 %nez
}
; Fold strnlen(ax, 9) == 0 to *ax == 0.
define i1 @fold_strnlen_ax_9_eqz() {
; CHECK-LABEL: @fold_strnlen_ax_9_eqz(
; CHECK-NEXT: [[CHAR0:%.*]] = load i8, ptr @ax, align 1
; CHECK-NEXT: [[EQZ:%.*]] = icmp eq i8 [[CHAR0]], 0
; CHECK-NEXT: ret i1 [[EQZ]]
;
%len = tail call i64 @strnlen(ptr @ax, i64 9)
%eqz = icmp eq i64 %len, 0
ret i1 %eqz
}
; Do not fold strnlen(ax, n) == 0 for n that might be zero.
define i1 @call_strnlen_ax_n_eqz(i64 %n) {
; CHECK-LABEL: @call_strnlen_ax_n_eqz(
; CHECK-NEXT: [[LEN:%.*]] = tail call i64 @strnlen(ptr nonnull @ax, i64 [[N:%.*]])
; CHECK-NEXT: [[EQZ:%.*]] = icmp eq i64 [[LEN]], 0
; CHECK-NEXT: ret i1 [[EQZ]]
;
%len = tail call i64 @strnlen(ptr @ax, i64 %n)
%eqz = icmp eq i64 %len, 0
ret i1 %eqz
}
; Fold strnlen(ax, n) == 0 to *ax == 0 for %0 that's not zero.
define i1 @fold_strnlen_ax_nz_eqz(i64 %n) {
; CHECK-LABEL: @fold_strnlen_ax_nz_eqz(
; CHECK-NEXT: [[CHAR0:%.*]] = load i8, ptr @ax, align 1
; CHECK-NEXT: [[EQZ:%.*]] = icmp eq i8 [[CHAR0]], 0
; CHECK-NEXT: ret i1 [[EQZ]]
;
%max = or i64 %n, 1
%len = tail call i64 @strnlen(ptr @ax, i64 %max)
%eqz = icmp eq i64 %len, 0
ret i1 %eqz
}
; Fold strnlen(ax, n) > 0 to *ax != 0 for n that's not zero.
define i1 @fold_strnlen_ax_nz_gtz(i64 %n) {
; CHECK-LABEL: @fold_strnlen_ax_nz_gtz(
; CHECK-NEXT: [[CHAR0:%.*]] = load i8, ptr @ax, align 1
; CHECK-NEXT: [[GTZ:%.*]] = icmp ne i8 [[CHAR0]], 0
; CHECK-NEXT: ret i1 [[GTZ]]
;
%max = or i64 %n, 1
%len = tail call i64 @strnlen(ptr @ax, i64 %max)
%gtz = icmp ugt i64 %len, 0
ret i1 %gtz
}
; Fold strnlen(a5 + i, n) == 0 to a5[i] == 0 for a nonconstant a5
; and a nonzero n.
define i1 @fold_strnlen_a5_pi_nz_eqz(i64 %i, i64 %n) {
; CHECK-LABEL: @fold_strnlen_a5_pi_nz_eqz(
; CHECK-NEXT: [[PTR:%.*]] = getelementptr inbounds [5 x i8], ptr @a5, i64 0, i64 [[I:%.*]]
; CHECK-NEXT: [[CHAR0:%.*]] = load i8, ptr [[PTR]], align 1
; CHECK-NEXT: [[EQZ:%.*]] = icmp eq i8 [[CHAR0]], 0
; CHECK-NEXT: ret i1 [[EQZ]]
;
%nz = or i64 %n, 1
%ptr = getelementptr inbounds [5 x i8], ptr @a5, i64 0, i64 %i
%len = call i64 @strnlen(ptr %ptr, i64 %nz)
%eqz = icmp eq i64 %len, 0
ret i1 %eqz
}
; Fold strnlen(s5 + i, n) == 0 for a constant s5 and nonzero n.
; This is first folded to s5[i] == 0 like the above and then finally
; to %0 == 5.
define i1 @fold_strnlen_s5_pi_nz_eqz(i64 %i, i64 %n) {
; CHECK-LABEL: @fold_strnlen_s5_pi_nz_eqz(
; CHECK-NEXT: [[EQZ:%.*]] = icmp eq i64 [[I:%.*]], 5
; CHECK-NEXT: ret i1 [[EQZ]]
;
%nz = or i64 %n, 1
%ptr = getelementptr inbounds [6 x i8], ptr @s5, i64 0, i64 %i
%len = call i64 @strnlen(ptr %ptr, i64 %nz)
%eqz = icmp eq i64 %len, 0
ret i1 %eqz
}
; Do not fold strnlen(s5 + i, n) for a constant s5 when n might be zero.
define i1 @call_strnlen_s5_pi_n_eqz(i64 %i, i64 %n) {
; CHECK-LABEL: @call_strnlen_s5_pi_n_eqz(
; CHECK-NEXT: [[PTR:%.*]] = getelementptr inbounds [6 x i8], ptr @s5, i64 0, i64 [[I:%.*]]
; CHECK-NEXT: [[LEN:%.*]] = call i64 @strnlen(ptr nonnull [[PTR]], i64 [[N:%.*]])
; CHECK-NEXT: [[EQZ:%.*]] = icmp eq i64 [[LEN]], 0
; CHECK-NEXT: ret i1 [[EQZ]]
;
%ptr = getelementptr inbounds [6 x i8], ptr @s5, i64 0, i64 %i
%len = call i64 @strnlen(ptr %ptr, i64 %n)
%eqz = icmp eq i64 %len, 0
ret i1 %eqz
}