-
-
Notifications
You must be signed in to change notification settings - Fork 5.5k
/
llvmcall2.jl
84 lines (72 loc) · 3.69 KB
/
llvmcall2.jl
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
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
# This file is a part of Julia. License is MIT: https://julialang.org/license
using InteractiveUtils: code_llvm
function declared_floor(x::Float64)
return ccall("llvm.floor.f64", llvmcall, Float64, (Float64,), x)
end
@test declared_floor(4.2) == 4.0
ir = sprint(code_llvm, declared_floor, Tuple{Float64})
@test occursin("call double @llvm.floor.f64", ir) # should be inlined
function doubly_declared_floor(x::Float64)
a = ccall("llvm.floor.f64", llvmcall, Float64, (Float64,), x)
b = ccall("llvm.floor.f64", llvmcall, Float64, (Float64,), x + 1) - 1
return a + b
end
@test doubly_declared_floor(4.2) == 8.0
function doubly_declared2_trunc(x::Float64)
a = ccall("llvm.trunc.f64", llvmcall, Float64, (Float64,), x)
b = ccall("llvm.trunc.f64", llvmcall, Float64, (Float64,), x + 1) - 1
return a + b
end
@test doubly_declared2_trunc(4.2) == 8.0
# Test for single line
function declared_ceil(x::Float64)
return ccall("llvm.ceil.f64", llvmcall, Float64, (Float64,), x)
end
@test declared_ceil(4.2) == 5.0
# Test for multiple lines
function ceilfloor(x::Float64)
a = ccall("llvm.ceil.f64", llvmcall, Float64, (Float64,), x)
b = ccall("llvm.floor.f64", llvmcall, Float64, (Float64,), a)
return b
end
@test ceilfloor(7.4) == 8.0
let err = ErrorException("llvmcall only supports intrinsic calls")
# support for calling external functions
@test_throws err @eval ccall("time", llvmcall, Cvoid, (Ptr{Cvoid},), C_NULL)
g() = ccall("extern time", llvmcall, Cvoid, (Ptr{Cvoid},), C_NULL)
g()
@test_throws err @eval ccall("extern llvm.floor", llvmcall, Float64, (Float64,), 0.0)
# support for mangling
@test (@eval ccall("llvm.floor.f64", llvmcall, Float64, (Float64,), 0.0)) === 0.0
@test (@eval ccall("llvm.floor", llvmcall, Float64, (Float64,), 0.0),
ccall("llvm.floor", llvmcall, Float32, (Float32,), 0.0)) === (0.0, 0.0f0)
@test_throws err @eval ccall("llvm.floor.f64", llvmcall, Float32, (Float64,), 0.0)
@test_throws err @eval ccall("llvm.floor.f64", llvmcall, Float32, (Float32,), 0.0f0)
@test_throws err @eval ccall("llvm.floor.f64", llvmcall, Float64, (Float32,), 0.0f0)
@test_throws err @eval ccall("llvm.floor.f64", llvmcall, Float64, (Int,), 0)
@test_throws err @eval ccall("llvm.floor.f64", llvmcall, Int, (Int,), 0)
@test_throws err @eval ccall("llvm.floor", llvmcall, Float64, (Float32,), 0.0f0)
@test_throws err @eval ccall("llvm.floor", llvmcall, Float64, (Int,), 0)
@test_throws err @eval ccall("llvm.floor", llvmcall, Int, (Int,), 0)
@test_throws err (@eval ccall("llvm.floor.f64", llvmcall, Float64, (Float64, Float64...,), 0.0)) === 0.0
@test_throws err (@eval ccall("llvm.floor", llvmcall, Float64, (Float64, Float64...,), 0.0)) === 0.0
end
@testset "JLJIT API" begin
function JLJITGetJuliaOJIT()
ccall(:JLJITGetJuliaOJIT, Ptr{Cvoid}, ())
end
function JLJITGetTripleString(JIT)
ccall(:JLJITGetTripleString, Cstring, (Ptr{Cvoid},), JIT)
end
jit = JLJITGetJuliaOJIT()
str = JLJITGetTripleString(jit)
jl_str = unsafe_string(str)
@test length(jl_str) > 4
end
# boolean structs
const NT4I = NTuple{4, VecElement{Int}}
const NT4B = NTuple{4, VecElement{Bool}}
f_nt4b(x, y) = ccall("llvm.sadd.with.overflow", llvmcall, Pair{NT4B, NT4B}, (NT4B, NT4B), x, y)
f_nt4i(x, y) = ccall("llvm.sadd.with.overflow", llvmcall, Pair{NT4I, NT4B}, (NT4I, NT4I), x, y)
@test f_nt4b((false, true, false, true), (false, false, true, true)) === (NT4B((false, true, true, false)) => NT4B((false, false, false, true)))
@test f_nt4i((typemin(Int), 0, typemax(Int), typemax(Int)), (-1, typemax(Int),-1, 1)) === (NT4I((typemax(Int), typemax(Int), typemax(Int)-1, typemin(Int))) => NT4B((true, false, false, true)))