-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathtvm_test.clj
116 lines (102 loc) · 4.59 KB
/
tvm_test.clj
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
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
(ns tvm-clj.tvm-test
(:require [tvm-clj.ast :as ast]
[tvm-clj.ast.elemwise-op :as ast-op]
[tvm-clj.schedule :as schedule]
[tvm-clj.compiler :as compiler]
[tvm-clj.module :as module]
[tvm-clj.device :as device]
[tech.v3.tensor :as dtt]
[tech.v3.datatype.functional :as dfn]
[clojure.test :refer [deftest is]]))
(defn make-add-fn
[]
;;Default datatype of variable is integer
(let [n (ast/variable "n")
;;Default datatype of placeholder is float32
A (ast/placeholder [n] "A")
B (ast/placeholder [n] "B")
compute-op (ast/compute [n]
;;Attaches metadata to the fn so we know the argument
;;count.
(ast/tvm-fn
[i]
(ast-op/+ (ast/tget A [i])
(ast/tget B [i])))
"C")
C (first (ast/output-tensors compute-op))]
{:schedule (schedule/create-schedule compute-op)
:arguments [A B C]
:compute-op compute-op}))
(deftest cpu-add
(let [{:keys [schedule arguments compute-op]} (make-add-fn)
_ (schedule/stage-cpu-injective schedule compute-op)
module (compiler/compile {"cpu_add" {:schedule schedule
:arguments arguments}})
add-fn (module/find-function module "cpu_add")
tens-a (dtt/->tensor (range 10) :datatype :float32
:container-type :native-heap)
tens-b (dtt/->tensor (range 10 20) :datatype :float32
:container-type :native-heap)
tens-c (dtt/new-tensor [10] :datatype :float32
:container-type :native-heap)]
(add-fn tens-a tens-b tens-c)
(is (dfn/equals tens-c (dfn/+ tens-a tens-b)))))
(defn device-add-test
[device-type]
(let [{:keys [schedule arguments compute-op]} (make-add-fn)
_ (schedule/stage-gpu-injective schedule compute-op)
module (compiler/compile {"device_add" {:schedule schedule
:arguments arguments
:target device-type}})
add-fn (module/find-function module "device_add")
tens-a (dtt/->tensor (range 10) :datatype :float32
:container-type :native-heap)
tens-b (dtt/->tensor (range 10 20) :datatype :float32
:container-type :native-heap)
device-id 0
dev-a (device/cpu->device tens-a device-type device-id)
dev-b (device/cpu->device tens-b device-type device-id)
;;Create a device tensor taking the shape and elemwise datatype
;;from the input.
dev-c (device/device-tensor tens-a device-type device-id)
_ (add-fn dev-a dev-b dev-c)
tens-c (device/device->cpu dev-c)]
(is (dfn/equals tens-c (dfn/+ tens-a tens-b)))))
(deftest ^:cuda cuda-add
(device-add-test :cuda))
(deftest ^:opencl opencl-add
(device-add-test :opencl))
(deftest cpu-reduction
(let [n (ast/variable "n")
A (ast/placeholder [n] "A")
reducer (ast/tvm-fn->commutative-reducer
;;reduce-fn, arguments are divided into accumulators
;;and inputs. Accum args are implicitly defined by the
;;number of identity values passed in..
(ast/tvm-fn
[lhs rhs]
(ast-op/max lhs rhs))
;;reduction identity values, one for each accumulator argument.
[(ast-op/min-value :float32)])
compute-op (ast/compute
[1]
(ast/tvm-fn
[i]
(ast/commutative-reduce
reducer
[{:name "reduce-n" :domain [0 n]}]
[#(ast/tget A [%])]))
"C")
C (first (ast/output-tensors compute-op))
schedule (schedule/create-schedule compute-op)
arguments [A C]
module (compiler/compile {"vec_max" {:schedule schedule
:arguments arguments}})
max-fn (module/find-function module "vec_max")
tens-a (dtt/->tensor (range 10) :datatype :float32
:container-type :native-heap)
tens-c (dtt/new-tensor [1] :datatype :float32
:container-type :native-heap)]
(max-fn tens-a tens-c)
(is (= 9.0
(double (first tens-c))))))