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
|
#include "Halide.h"
#include <stdio.h>
using namespace Halide;
int call_counter = 0;
extern "C" HALIDE_EXPORT_SYMBOL float my_func(int x, float y) {
call_counter++;
return x * y;
}
HalideExtern_2(float, my_func, int, float);
int main(int argc, char **argv) {
// set_jit_externs() implicitly adds a user_context arg to the externs, which
// we can't yet support. TODO: this actually should work, but doesn't.
if (get_jit_target_from_environment().arch == Target::WebAssembly) {
printf("[SKIP] WebAssembly JIT does not support passing arbitrary pointers to/from HalideExtern code.\n");
return 0;
}
std::vector<ExternFuncArgument> args;
args.push_back(user_context_value());
Var x, y;
Func monitor;
monitor(x, y) = my_func(x, cast<float>(y));
Func f;
f.define_extern("extern_func", args, Float(32), 2);
Pipeline p(f);
p.set_jit_externs({{"extern_func", JITExtern{monitor}}});
Buffer<float> imf = p.realize({32, 32});
// Check the result was what we expected
for (int i = 0; i < 32; i++) {
for (int j = 0; j < 32; j++) {
float correct = (float)(i * j);
float delta = imf(i, j) - correct;
if (delta < -0.001 || delta > 0.001) {
printf("imf[%d, %d] = %f instead of %f\n", i, j, imf(i, j), correct);
return 1;
}
}
}
if (call_counter != 32 * 32) {
printf("In pipeline_set_jit_externs_func, my_func was called %d times instead of %d\n", call_counter, 32 * 32);
return 1;
}
printf("Success!\n");
return 0;
}
|