mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-26 23:32:58 +00:00
37be0d7c43
This was introduced in a faulty refactoring (r225640, mea culpa): the tests weren't testing the return values, so, for both __strcpy_chk and __stpcpy_chk, we would return the end of the buffer (matching stpcpy) instead of the beginning (for strcpy). The root cause was the prefix "__" being ignored when comparing, which made us always pick LibFunc::stpcpy_chk. Pass the LibFunc::Func directly to avoid this kind of error. Also, make the testcases as explicit as possible to prevent this. The now-useful testcases expose another, entangled, stpcpy problem, with the further simplification. This was introduced in a refactoring (r225640) to match the original behavior. However, this leads to problems when successive simplifications generate several similar instructions, none of which are removed by the custom replaceAllUsesWith. For instance, InstCombine (the main user) doesn't erase the instruction in its custom RAUW. When trying to simplify say __stpcpy_chk: - first, an stpcpy is created (fortified simplifier), - second, a memcpy is created (normal simplifier), but the stpcpy call isn't removed. - third, InstCombine later revisits the instructions, and simplifies the first stpcpy to a memcpy. We now have two memcpys. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@227250 91177308-0d34-0410-b5e6-96231b3b80d8
104 lines
4.5 KiB
LLVM
104 lines
4.5 KiB
LLVM
; Test lib call simplification of __strcpy_chk calls with various values
|
|
; for src, dst, and slen.
|
|
;
|
|
; RUN: opt < %s -instcombine -S | FileCheck %s
|
|
|
|
target datalayout = "e-p:32:32:32-i1:8:8-i8:8:8-i16:16:16-i32:32:32-i64:32:64-f32:32:32-f64:32:64-v64:64:64-v128:128:128-a0:0:64-f80:128:128"
|
|
|
|
@a = common global [60 x i8] zeroinitializer, align 1
|
|
@b = common global [60 x i8] zeroinitializer, align 1
|
|
@.str = private constant [12 x i8] c"abcdefghijk\00"
|
|
|
|
; Check cases where slen >= strlen (src).
|
|
|
|
define i8* @test_simplify1() {
|
|
; CHECK-LABEL: @test_simplify1(
|
|
%dst = getelementptr inbounds [60 x i8]* @a, i32 0, i32 0
|
|
%src = getelementptr inbounds [12 x i8]* @.str, i32 0, i32 0
|
|
|
|
; CHECK-NEXT: call void @llvm.memcpy.p0i8.p0i8.i32(i8* getelementptr inbounds ([60 x i8]* @a, i32 0, i32 0), i8* getelementptr inbounds ([12 x i8]* @.str, i32 0, i32 0), i32 12, i32 1, i1 false)
|
|
; CHECK-NEXT: ret i8* getelementptr inbounds ([60 x i8]* @a, i32 0, i32 0)
|
|
%ret = call i8* @__strcpy_chk(i8* %dst, i8* %src, i32 60)
|
|
ret i8* %ret
|
|
}
|
|
|
|
define i8* @test_simplify2() {
|
|
; CHECK-LABEL: @test_simplify2(
|
|
%dst = getelementptr inbounds [60 x i8]* @a, i32 0, i32 0
|
|
%src = getelementptr inbounds [12 x i8]* @.str, i32 0, i32 0
|
|
|
|
; CHECK-NEXT: call void @llvm.memcpy.p0i8.p0i8.i32(i8* getelementptr inbounds ([60 x i8]* @a, i32 0, i32 0), i8* getelementptr inbounds ([12 x i8]* @.str, i32 0, i32 0), i32 12, i32 1, i1 false)
|
|
; CHECK-NEXT: ret i8* getelementptr inbounds ([60 x i8]* @a, i32 0, i32 0)
|
|
%ret = call i8* @__strcpy_chk(i8* %dst, i8* %src, i32 12)
|
|
ret i8* %ret
|
|
}
|
|
|
|
define i8* @test_simplify3() {
|
|
; CHECK-LABEL: @test_simplify3(
|
|
%dst = getelementptr inbounds [60 x i8]* @a, i32 0, i32 0
|
|
%src = getelementptr inbounds [12 x i8]* @.str, i32 0, i32 0
|
|
|
|
; CHECK-NEXT: call void @llvm.memcpy.p0i8.p0i8.i32(i8* getelementptr inbounds ([60 x i8]* @a, i32 0, i32 0), i8* getelementptr inbounds ([12 x i8]* @.str, i32 0, i32 0), i32 12, i32 1, i1 false)
|
|
; CHECK-NEXT: ret i8* getelementptr inbounds ([60 x i8]* @a, i32 0, i32 0)
|
|
%ret = call i8* @__strcpy_chk(i8* %dst, i8* %src, i32 -1)
|
|
ret i8* %ret
|
|
}
|
|
|
|
; Check cases where there are no string constants.
|
|
|
|
define i8* @test_simplify4() {
|
|
; CHECK-LABEL: @test_simplify4(
|
|
%dst = getelementptr inbounds [60 x i8]* @a, i32 0, i32 0
|
|
%src = getelementptr inbounds [60 x i8]* @b, i32 0, i32 0
|
|
|
|
; CHECK-NEXT: %strcpy = call i8* @strcpy(i8* getelementptr inbounds ([60 x i8]* @a, i32 0, i32 0), i8* getelementptr inbounds ([60 x i8]* @b, i32 0, i32 0))
|
|
; CHECK-NEXT: ret i8* %strcpy
|
|
%ret = call i8* @__strcpy_chk(i8* %dst, i8* %src, i32 -1)
|
|
ret i8* %ret
|
|
}
|
|
|
|
; Check case where the string length is not constant.
|
|
|
|
define i8* @test_simplify5() {
|
|
; CHECK-LABEL: @test_simplify5(
|
|
%dst = getelementptr inbounds [60 x i8]* @a, i32 0, i32 0
|
|
%src = getelementptr inbounds [12 x i8]* @.str, i32 0, i32 0
|
|
|
|
; CHECK-NEXT: %len = call i32 @llvm.objectsize.i32.p0i8(i8* getelementptr inbounds ([60 x i8]* @a, i32 0, i32 0), i1 false)
|
|
; CHECK-NEXT: %1 = call i8* @__memcpy_chk(i8* getelementptr inbounds ([60 x i8]* @a, i32 0, i32 0), i8* getelementptr inbounds ([12 x i8]* @.str, i32 0, i32 0), i32 12, i32 %len)
|
|
; CHECK-NEXT: ret i8* %1
|
|
%len = call i32 @llvm.objectsize.i32.p0i8(i8* %dst, i1 false)
|
|
%ret = call i8* @__strcpy_chk(i8* %dst, i8* %src, i32 %len)
|
|
ret i8* %ret
|
|
}
|
|
|
|
; Check case where the source and destination are the same.
|
|
|
|
define i8* @test_simplify6() {
|
|
; CHECK-LABEL: @test_simplify6(
|
|
%dst = getelementptr inbounds [60 x i8]* @a, i32 0, i32 0
|
|
|
|
; CHECK-NEXT: %len = call i32 @llvm.objectsize.i32.p0i8(i8* getelementptr inbounds ([60 x i8]* @a, i32 0, i32 0), i1 false)
|
|
; CHECK-NEXT: %ret = call i8* @__strcpy_chk(i8* getelementptr inbounds ([60 x i8]* @a, i32 0, i32 0), i8* getelementptr inbounds ([60 x i8]* @a, i32 0, i32 0), i32 %len)
|
|
; CHECK-NEXT: ret i8* %ret
|
|
%len = call i32 @llvm.objectsize.i32.p0i8(i8* %dst, i1 false)
|
|
%ret = call i8* @__strcpy_chk(i8* %dst, i8* %dst, i32 %len)
|
|
ret i8* %ret
|
|
}
|
|
|
|
; Check case where slen < strlen (src).
|
|
|
|
define i8* @test_no_simplify1() {
|
|
; CHECK-LABEL: @test_no_simplify1(
|
|
%dst = getelementptr inbounds [60 x i8]* @a, i32 0, i32 0
|
|
%src = getelementptr inbounds [60 x i8]* @b, i32 0, i32 0
|
|
|
|
; CHECK-NEXT: %ret = call i8* @__strcpy_chk(i8* getelementptr inbounds ([60 x i8]* @a, i32 0, i32 0), i8* getelementptr inbounds ([60 x i8]* @b, i32 0, i32 0), i32 8)
|
|
; CHECK-NEXT: ret i8* %ret
|
|
%ret = call i8* @__strcpy_chk(i8* %dst, i8* %src, i32 8)
|
|
ret i8* %ret
|
|
}
|
|
|
|
declare i8* @__strcpy_chk(i8*, i8*, i32) nounwind
|
|
declare i32 @llvm.objectsize.i32.p0i8(i8*, i1) nounwind readonly
|