Simple CPU runner
This implements a simple CPU runner based on LLVM Orc JIT. The base
functionality is provided by the ExecutionEngine class that compiles and links
the module, and provides an interface for obtaining function pointers to the
JIT-compiled MLIR functions and for invoking those functions directly. Since
function pointers need to be casted to the correct pointer type, the
ExecutionEngine wraps LLVM IR functions obtained from MLIR into a helper
function with the common signature `void (void **)` where the single argument
is interpreted as a list of pointers to the actual arguments passed to the
function, eventually followed by a pointer to the result of the function.
Additionally, the ExecutionEngine is set up to resolve library functions to
those available in the current process, enabling support for, e.g., simple C
library calls.
For integration purposes, this also provides a simplistic runtime for memref
descriptors as expected by the LLVM IR code produced by MLIR translation. In
particular, memrefs are transformed into LLVM structs (can be mapped to C
structs) with a pointer to the data, followed by dynamic sizes. This
implementation only supports statically-shaped memrefs of type float, but can
be extened if necessary.
Provide a binary for the runner and a test that exercises it.
PiperOrigin-RevId: 230876363
2019-01-25 19:16:06 +08:00
|
|
|
// RUN: mlir-cpu-runner %s | FileCheck %s
|
|
|
|
// RUN: mlir-cpu-runner -e foo -init-value 1000 %s | FileCheck -check-prefix=NOMAIN %s
|
2019-03-29 00:20:13 +08:00
|
|
|
// RUN: mlir-cpu-runner %s -O3 | FileCheck %s
|
2019-09-02 02:32:52 +08:00
|
|
|
// RUN: mlir-cpu-runner -e affine -init-value 2.0 %s | FileCheck -check-prefix=AFFINE %s
|
Simple CPU runner
This implements a simple CPU runner based on LLVM Orc JIT. The base
functionality is provided by the ExecutionEngine class that compiles and links
the module, and provides an interface for obtaining function pointers to the
JIT-compiled MLIR functions and for invoking those functions directly. Since
function pointers need to be casted to the correct pointer type, the
ExecutionEngine wraps LLVM IR functions obtained from MLIR into a helper
function with the common signature `void (void **)` where the single argument
is interpreted as a list of pointers to the actual arguments passed to the
function, eventually followed by a pointer to the result of the function.
Additionally, the ExecutionEngine is set up to resolve library functions to
those available in the current process, enabling support for, e.g., simple C
library calls.
For integration purposes, this also provides a simplistic runtime for memref
descriptors as expected by the LLVM IR code produced by MLIR translation. In
particular, memrefs are transformed into LLVM structs (can be mapped to C
structs) with a pointer to the data, followed by dynamic sizes. This
implementation only supports statically-shaped memrefs of type float, but can
be extened if necessary.
Provide a binary for the runner and a test that exercises it.
PiperOrigin-RevId: 230876363
2019-01-25 19:16:06 +08:00
|
|
|
|
2019-08-31 04:01:34 +08:00
|
|
|
// RUN: cp %s %t
|
|
|
|
// RUN: mlir-cpu-runner %t -dump-object-file | FileCheck %t
|
|
|
|
// RUN: ls %t.o
|
|
|
|
// RUN: rm %t.o
|
|
|
|
|
|
|
|
// RUN: mlir-cpu-runner %s -dump-object-file -object-filename=%T/test.o | FileCheck %s
|
|
|
|
// RUN: ls %T/test.o
|
|
|
|
// RUN: rm %T/test.o
|
|
|
|
|
Simple CPU runner
This implements a simple CPU runner based on LLVM Orc JIT. The base
functionality is provided by the ExecutionEngine class that compiles and links
the module, and provides an interface for obtaining function pointers to the
JIT-compiled MLIR functions and for invoking those functions directly. Since
function pointers need to be casted to the correct pointer type, the
ExecutionEngine wraps LLVM IR functions obtained from MLIR into a helper
function with the common signature `void (void **)` where the single argument
is interpreted as a list of pointers to the actual arguments passed to the
function, eventually followed by a pointer to the result of the function.
Additionally, the ExecutionEngine is set up to resolve library functions to
those available in the current process, enabling support for, e.g., simple C
library calls.
For integration purposes, this also provides a simplistic runtime for memref
descriptors as expected by the LLVM IR code produced by MLIR translation. In
particular, memrefs are transformed into LLVM structs (can be mapped to C
structs) with a pointer to the data, followed by dynamic sizes. This
implementation only supports statically-shaped memrefs of type float, but can
be extened if necessary.
Provide a binary for the runner and a test that exercises it.
PiperOrigin-RevId: 230876363
2019-01-25 19:16:06 +08:00
|
|
|
func @fabsf(f32) -> f32
|
|
|
|
|
|
|
|
func @main(%a : memref<2xf32>, %b : memref<1xf32>) {
|
|
|
|
%c0 = constant 0 : index
|
|
|
|
%c1 = constant 1 : index
|
|
|
|
%0 = constant -420.0 : f32
|
|
|
|
%1 = load %a[%c0] : memref<2xf32>
|
|
|
|
%2 = load %a[%c1] : memref<2xf32>
|
|
|
|
%3 = addf %0, %1 : f32
|
|
|
|
%4 = addf %3, %2 : f32
|
|
|
|
%5 = call @fabsf(%4) : (f32) -> f32
|
|
|
|
store %5, %b[%c0] : memref<1xf32>
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// CHECK: 0.000000e+00 0.000000e+00
|
|
|
|
// CHECK-NEXT: 4.200000e+02
|
|
|
|
|
|
|
|
func @foo(%a : memref<1x1xf32>) -> memref<1x1xf32> {
|
|
|
|
%c0 = constant 0 : index
|
|
|
|
%0 = constant 1234.0 : f32
|
|
|
|
%1 = load %a[%c0, %c0] : memref<1x1xf32>
|
|
|
|
%2 = addf %1, %0 : f32
|
|
|
|
store %2, %a[%c0, %c0] : memref<1x1xf32>
|
|
|
|
return %a : memref<1x1xf32>
|
|
|
|
}
|
|
|
|
// NOMAIN: 2.234000e+03
|
2019-02-16 02:50:28 +08:00
|
|
|
// NOMAIN-NEXT: 2.234000e+03
|
2019-09-02 02:32:52 +08:00
|
|
|
|
|
|
|
func @affine(%a : memref<32xf32>) -> memref<32xf32> {
|
|
|
|
%cf1 = constant 42.0 : f32
|
|
|
|
%N = dim %a, 0 : memref<32xf32>
|
|
|
|
affine.for %i = 0 to %N {
|
|
|
|
affine.store %cf1, %a[%i] : memref<32xf32>
|
|
|
|
}
|
|
|
|
return %a : memref<32xf32>
|
|
|
|
}
|
|
|
|
// AFFINE: 4.2{{0+}}e+01
|