about summary refs log tree commit diff homepage
path: root/test/Intrinsics/abs-no-overflow.ll
blob: 7f476e2ae2f8b0f88c8bfdc1244cf85dde957dcb (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
; LLVM has an intrinsic for abs.
; This file is based on the following code with poisoning of llvm.abs disabled.
; ```
; #include "klee/klee.h"
;
; #include <assert.h>
; #include <limits.h>
;
; volatile int abs_a;
;
; int main(void)
; {
;   int a;
;   klee_make_symbolic(&a, sizeof(a), "a");
;
;   abs_a = a < 0 ? -a : a;
;   if (abs_a == INT_MIN)
;     assert(abs_a == a);
;   else
;     assert(abs_a >= 0);
;   return abs_a;
; }
; ```
; REQUIRES: geq-llvm-12.0
; RUN: rm -rf %t.klee-out
; RUN: %klee -exit-on-error --output-dir=%t.klee-out --optimize=false %s

@0 = private unnamed_addr constant [2 x i8] c"a\00", align 1
define dso_local i32 @main() local_unnamed_addr {
  %1 = alloca i32, align 4
  %2 = bitcast i32* %1 to i8*
  call void @klee_make_symbolic(i8* nonnull %2, i64 4, i8* getelementptr inbounds ([2 x i8], [2 x i8]* @0, i64 0, i64 0))
  %3 = load i32, i32* %1, align 4
  ; Call with is_int_min_poison == 0 (false), expects INT_MIN to be returned
  %abs_a = call i32 @llvm.abs.i32(i32 %3, i1 false)
  %4 = icmp eq i32 %abs_a, -2147483648
  br i1 %4, label %int_min, label %not_int_min

int_min:
  %5 = icmp eq i32 %abs_a, %3
  br i1 %5, label %exit.block, label %abort.block

not_int_min:
  %6 = icmp sgt i32 %abs_a, -1
  br i1 %6, label %exit.block, label %abort.block

exit.block:
  ret i32 0

abort.block:
  call void @abort()
  unreachable
}

declare dso_local void @klee_make_symbolic(i8*, i64, i8*)
declare i32 @llvm.abs.i32(i32, i1 immarg)
declare void @abort() noreturn nounwind