This project aims to provide a lightweight runtime to semi-automatically optimize data flow pipelines for locality. Pipelines are specified as graphs of operators processing data between multi-dimensional buffers. Slinky then allows the user to describe transformations to the pipeline that improve memory locality and reduce memory usage by executing small crops of operator outputs.
Slinky is heavily inspired and motivated by Halide. It can be described by starting with Halide, and making the following changes:
- Slinky is a runtime interpreter, not a compiler.
- All operations that read and write buffers are user defined callbacks (except copies).
- Bounds for each operation are manually provided instead of inferred as in Halide.
Because we are not responsible for generating the inner loop code like Halide, scheduling is a dramatically simpler problem. Without needing to worry about instruction selection, register pressure, and so on, the cost function for scheduling is a much more straightforward function of high level memory access patterns.
The ultimate goal of Slinky is to make automatic scheduling of pipelines reliable and fast enough to implement a just-in-time optimization engine at runtime.
A pipeline is a directed acyclic graph, where the nodes are buffer_exprs, and the edges are funcs.
A func is said to "consume" its input buffer(s), and "produce" its output buffers, of which there can be more than one.
A node that is not produced by any func is a pipeline input, and a node that is not consumed by a func is a pipeline output.
A func encapsulates:
- A symbolic description of data dependecies.
Data dependencies are expressed as a
[min, max]interval for each input dimension as a function of the output dimensions. - A specification of the compute schedule (loop tiling).
- A callback representing the actual implementation.
Here is an example of a simple pipeline of two 1D elementwise funcs:
node_context ctx;
// input buffer expression
auto in = buffer_expr::make(ctx, "in", 1, sizeof(int));
// output buffer expression
auto out = buffer_expr::make(ctx, "out", 1, sizeof(int));
// temporary buffer expression
auto tmp = buffer_expr::make(ctx, "tmp", 1, sizeof(int));
var i(ctx, "i");
// `mul` is automatically registered as the producer of `tmp`.
func mul = func::make(multiply_2<int>, {{in, {point(i)}}}, {{tmp, {i}}});
// one could reuse 'i', but for clarity let's use a new variable.
var j(ctx, "j");
// `add` is automatically registered as the producer of `out`.
func add = func::make(add_1<int>, {{tmp, {point(j)}}}, {{out, {j}}});
pipeline p = build_pipeline(ctx, {in}, {out});Without any further specifications this pipeline would use the following strategy:
- Materialize the whole logical tmp buffer of the same size as out, execute all of
mul. - Execute all of
add.
This approach has multiple downsides:
- Large memory allocation for
tmp. tmpmemory is first fully written and then fully read, resulting in expensive memory traffic and inefficient use of CPU caches.
check(in)
check((buffer_rank(in) == 1))
check(out)
check((buffer_rank(out) == 1))
check((buffer_elem_size(in) == 4))
check((buffer_elem_size(out) == 4))
check(or_else((buffer_fold_factor(out, 0) == 9223372036854775807), (buffer_max(out, 0) < (buffer_fold_factor(out, 0) + buffer_min(out, 0)))))
check((buffer_min(in, 0) <= buffer_min(out, 0)))
check((buffer_max(out, 0) <= buffer_max(in, 0)))
tmp = allocate(automatic, 4, {
{[buffer_min(out, 0), buffer_max(out, 0)], <>, <>}
}) {
call(<mul>, {in}, {tmp})
call(<add>, {tmp}, {out})
}The opposite extreme would be the following:
- Allocate a buffer of size 1 for
tmp. - For each element compute
mulfollowed byaddand store the result toout.
...
const int chunk_size = 1;
add.loops({{x, chunk_size}});
pipeline p = build_pipeline(ctx, {in}, {out});tmp = allocate(automatic, 4, {
{[buffer_min(out, 0), buffer_max(out, 0)], <>, 1}
}) {
out.j = loop(serial, [buffer_min(out, 0), buffer_max(out, 0)], 1) {
tmp = crop_dim(tmp, 0, [out.j, out.j]) {
call(<mull>, {in}, {tmp})
}
out.out.j = crop_dim(out, 0, [out.j, out.j]) {
call(<add>, {tmp}, {out.out.j})
}
}
}This approach seemingly avoids the aforementioned issues but introduces a different set of deficiencies:
- Inefficient use of
mulandaddvectorized implementations. - Unamortized function calls overhead.
The ideal strategy is to split out into chunks, and compute the two operations at each chunk.
Here is an example of a pipeline that has a stage that is a stencil, such as a convolution:
node_context ctx;
auto in = buffer_expr::make(ctx, "in", sizeof(short), 2);
auto out = buffer_expr::make(ctx, "out", sizeof(short), 2);
auto intm = buffer_expr::make(ctx, "intm", sizeof(short), 2);
var x(ctx, "x");
var y(ctx, "y");
func add = func::make(add_1, {in, {point(x), point(y)}}, {intm, {x, y}});
func stencil =
func::make(sum3x3, {intm, {{x - 1, x + 1}, {y - 1, y + 1}}}, {out, {x, y}});
pipeline p = build_pipeline(ctx, {in}, {out});inandoutare the input and output buffers.intmis the intermediate buffer between the two operations.- We need two variables
xandyto describe the buffers in this pipeline. - The first stage
addis an elementwise operation that adds one to each element. - The second stage is a stencil
sum3x3, which computes the sum of the 3x3 neighborhood aroundx, y. - The output of both stages is indexed by
x, y. The first stage is similar to the previous elementwise example, but the stencil has bounds[x - 1, x + 1], [y - 1, y + 1].
The typical way that many systems would execute such a pipeline is to run all of add_1, followed by all of sum3x3, storing the intermediate result in a buffer equal to the size of the input.
Here is a visualization of this strategy.
An interesting way to implement this pipeline is to compute rows of out at a time, keeping the window of rows required from add in memory.
This can be expressed with the following schedule:
stencil.loops({y});
add.compute_at({&stencil, y});This means:
- We want a loop over
y, instead of just passing the whole 2D buffer tosum3x3. - We want to compute add at that same loop over y to compute
stencil.
This generates a program like so:
check(in)
check((buffer_rank(in) == 2))
check(out)
check((buffer_rank(out) == 2))
check((buffer_elem_size(in) == 2))
check((buffer_elem_size(out) == 2))
check(or_else((buffer_fold_factor(out, 0) == 9223372036854775807), (buffer_max(out, 0) < (buffer_fold_factor(out, 0) + buffer_min(out, 0)))))
check(or_else((buffer_fold_factor(out, 1) == 9223372036854775807), (buffer_max(out, 1) < (buffer_fold_factor(out, 1) + buffer_min(out, 1)))))
check((buffer_min(in, 0) < buffer_min(out, 0)))
check((buffer_max(out, 0) < buffer_max(in, 0)))
check((buffer_min(in, 1) < buffer_min(out, 1)))
check((buffer_max(out, 1) < buffer_max(in, 1)))
intm = allocate(automatic, 2, {
{[(buffer_min(out, 0) + -1), (buffer_max(out, 0) + 1)], <>, <>},
{[(buffer_min(out, 1) + -1), (buffer_max(out, 1) + 1)], <>, 3}
}) {
out.y = loop(serial, [(buffer_min(out, 1) + -2), buffer_max(out, 1)], 1) {
intm = crop_dim(intm, 1, [(out.y + 1), (out.y + 1)]) {
call(<anonymous target>, {in}, {intm})
}
out.out.y = crop_dim(out, 1, [out.y, out.y]) {
call(<anonymous target>, {intm}, {out.out.y})
}
}
}
This program does the following:
- Allocates a buffer for
intm, with a fold factor of 3, meaning that the coordinates of the second dimension are modulo 3 when computing addresses. - Runs a loop over
ystarting from 2 rows before the first output row, callingaddandsum3x3at eachy. - The calls are cropped to the line to be produced on the current iteration
y.sum3x3reads rowsy-1,y, andy+1ofintm, so we need to producey+1ofintmbefore producingyofout. - The
intmbuffer persists between loop iterations, so we only need to compute the newly required liney+1ofintmon each iteration, linesy-1andywere already produced on previous iterations. - Because we started the loop two iterations of
yearly, linesy-1andyhave already been produced for the first value ofyofout. For these two "warmup" iterations, thesum3x3call's crop ofoutwill be an empty buffer (because crops clamp to the original bounds). - Because we only need lines
[y-1,y+1], we can "fold" the storage ofintm, by rewriting all accessesyto bey%3.
Here is a visualization of this strategy.
Here is a more involved example, which computes the matrix product d = (a x b) x c:
node_context ctx;
auto a = buffer_expr::make(ctx, "a", sizeof(float), 2);
auto b = buffer_expr::make(ctx, "b", sizeof(float), 2);
auto c = buffer_expr::make(ctx, "c", sizeof(float), 2);
auto abc = buffer_expr::make(ctx, "abc", sizeof(float), 2);
auto ab = buffer_expr::make(ctx, "ab", sizeof(float), 2);
var i(ctx, "i");
var j(ctx, "j");
// The bounds required of the dimensions consumed by the reduction depend on the size of the
// buffers passed in. Note that we haven't used any constants yet.
auto K_ab = a->dim(1).bounds;
auto K_abc = c->dim(0).bounds;
// We use float for this pipeline so we can test for correctness exactly.
func matmul_ab =
func::make<const float, const float, float>(matmul<float>, {a, {point(i), K_ab}}, {b, {K_ab, point(j)}}, {ab, {i, j}});
func matmul_abc = func::make<const float, const float, float>(
matmul<float>, {ab, {point(i), K_abc}}, {c, {K_abc, point(j)}}, {abc, {i, j}});
pipeline p = build_pipeline(ctx, {a, b, c}, {abc});a,b,c,abcare input and output buffers.abis the intermediate producta x b.- We need 2 variables
iandjto describe this pipeline. - Both
funcobjects have the same signature:- Consume two operands, produce one operand.
- The first
funcproducesab, the secondfuncconsumes it. - The bounds required by output element
i,jof the first operand is theith row and all the columns of the first operand. We usedim(1).boundsof the first operand, butdim(0).boundsof the second operand should be equal to this. - The bounds required of the second operand is similar, we just need all the rows and one column instead. We use
dim(0).boundsof the second operand to avoid relying on the intermediate buffer, which will have its bounds inferred (maybe this would still work...). matmulis the callback function implementing the matrix multiply operation.
Much like the elementwise example, we can compute this in a variety of ways between two extremes:
- Allocating
abto have the full extent of the producta x b, and executing all of the first multiply followed by all of the second multiply. - Each row of the second product depends only on the same row of the first product. Therefore, we can allocate
abto hold only one row of the producta x b, and compute both products in a loop over rows of the final result.
In practice, matrix multiplication kernels like to produce multiple rows at once for maximum efficiency.
We expect this approach to fill a gap between two extremes that seem prevalent today (TODO: is this really true? I think so...):
- Pipeline interpreters that execute entire operations one at a time.
- Pipeline compilers that generate code specific to a pipeline.
We expect Slinky to execute suitable pipelines using less memory than (1), but at a lower performance than what is possible with (2). We emphasize possible because actually building a compiler that does this well on novel code is very difficult. We think Slinky's approach is a more easily solved problem, and will degrade more gracefully in failure cases.
This performance app attempts to measure the overhead of interpreting pipelines at runtime.
The test performs a copy between two 2D buffers of "total size" bytes twice: first to an intermediate buffer, and then to the output.
The inner dimension of size "copy size" is copied with memcpy, the outer dimension is a loop implemented in one of two ways:
- An "explicit loop" version, which has a loop in the pipeline for the outer dimension (interpreted by Slinky).
- An "implicit loop" version, which loops over the outer dimension in the callback.
Two factors affect the performance of this pipeline:
- Interpreter and dispatching overhead of slinky.
- Locality of the copy operations.
This is an extreme example, where memcpy is the fastest operation (per memory accessed) that could be performed in a pipeline.
In other words, this is an upper bound on the overhead that could be expected for an operation on the same amount of memory.
On my machine, here are some data points from this pipeline:
| copy size (KB) | loop (GB/s) | no loop (GB/s) | ratio |
|---|---|---|---|
| 1 | 13.0713 | 19.7616 | 0.661 |
| 2 | 19.485 | 23.728 | 0.821 |
| 4 | 24.254 | 25.221 | 0.962 |
| 8 | 27.701 | 26.013 | 1.065 |
| 16 | 26.428 | 25.919 | 1.020 |
| 32 | 25.891 | 26.494 | 0.977 |
| copy size (KB) | loop (GB/s) | no loop (GB/s) | ratio |
|---|---|---|---|
| 1 | 12.947 | 21.410 | 0.605 |
| 2 | 20.459 | 25.705 | 0.796 |
| 4 | 25.456 | 27.320 | 0.932 |
| 8 | 30.462 | 27.514 | 1.107 |
| 16 | 28.804 | 27.578 | 1.044 |
| 32 | 28.480 | 28.026 | 1.016 |
| copy size (KB) | loop (GB/s) | no loop (GB/s) | ratio |
|---|---|---|---|
| 1 | 12.416 | 20.683 | 0.600 |
| 2 | 19.230 | 24.026 | 0.800 |
| 4 | 23.793 | 24.163 | 0.985 |
| 8 | 27.807 | 24.075 | 1.155 |
| 16 | 27.173 | 24.201 | 1.123 |
| 32 | 26.199 | 24.155 | 1.085 |
| copy size (KB) | loop (GB/s) | no loop (GB/s) | ratio |
|---|---|---|---|
| 1 | 12.229 | 20.616 | 0.593 |
| 2 | 19.833 | 24.447 | 0.811 |
| 4 | 24.303 | 24.761 | 0.982 |
| 8 | 28.563 | 24.262 | 1.177 |
| 16 | 27.951 | 24.104 | 1.160 |
| 32 | 26.826 | 24.217 | 1.108 |
| copy size (KB) | loop (GB/s) | no loop (GB/s) | ratio |
|---|---|---|---|
| 1 | 11.978 | 12.023 | 0.996 |
| 2 | 19.676 | 16.441 | 1.197 |
| 4 | 21.588 | 14.013 | 1.541 |
| 8 | 23.544 | 14.536 | 1.620 |
| 16 | 23.892 | 13.440 | 1.778 |
| 32 | 23.965 | 13.942 | 1.719 |
As we should expect, the observations vary depending on the total size of the copy:
- When the total size is small enough to fit in L1 or L2 cache, the cost of the
memcpywill be small, and the overhead will be relatively more expensive. This cost is as much as 40% when copying 1 KB at a time, according to the data above. - Even when the entire copy fits in the L1 cache, the overhead of dispatching 8KB at a time is negligible.
- When the copies are very large, dispatch overhead is insignificant relative to locality improvements.