diff --git a/mlir/include/mlir/Dialect/SparseTensor/IR/SparseTensorOps.td b/mlir/include/mlir/Dialect/SparseTensor/IR/SparseTensorOps.td --- a/mlir/include/mlir/Dialect/SparseTensor/IR/SparseTensorOps.td +++ b/mlir/include/mlir/Dialect/SparseTensor/IR/SparseTensorOps.td @@ -232,6 +232,35 @@ " type($tensor) `,` type($indices) `,` type($value)"; } +def SparseTensor_PushBackOp : SparseTensor_Op<"push_back", []>, + Arguments<(ins StridedMemRefRankOf<[Index], [1]>:$bufferSizes, + StridedMemRefRankOf<[AnyType], [1]>:$inBuffer, + AnyType:$value, IndexAttr:$idx)>, + Results<(outs StridedMemRefRankOf<[AnyType], [1]>:$outBuffer)> { + string summary = "Pushes a value to the back of a given buffer"; + string description = [{ + Push `value` to the end of the given sparse tensor storage buffer + `inBuffer` and update the size of the buffer in `bufferSizes[idx]`. The + capacity of the buffer is record in the memref type of `inBuffer `. If the + current buffer is full, then `inBuffer.realloc` is called before pushing the + data to the buffer. This is similar to std::vector push_back. + + The operation returns an SSA value for the memref. Referencing the memref + through the old SSA value after this operation is undefined behavior. + + Example: + + ```mlir + %r = sparse_tensor.push_back %bufferSizes, %buffer, %val {idx = 0 : index} + : memref, memref, f64 -> memref + ``` + }]; + let assemblyFormat = "$bufferSizes `,` $inBuffer `,` $value" + " attr-dict `:` type($bufferSizes) `,`" + " type($inBuffer) `,` type($value) `to`" + " type($outBuffer)"; +} + def SparseTensor_ExpandOp : SparseTensor_Op<"expand", []>, Arguments<(ins AnySparseTensor:$tensor)>, Results<(outs AnyStridedMemRefOfRank<1>:$values, diff --git a/mlir/test/Dialect/SparseTensor/roundtrip.mlir b/mlir/test/Dialect/SparseTensor/roundtrip.mlir --- a/mlir/test/Dialect/SparseTensor/roundtrip.mlir +++ b/mlir/test/Dialect/SparseTensor/roundtrip.mlir @@ -132,6 +132,19 @@ // ----- +// CHECK-LABEL: func @sparse_push_back( +// CHECK-SAME: %[[A:.*]]: memref, +// CHECK-SAME: %[[B:.*]]: memref, +// CHECK-SAME: %[[C:.*]]: f64) -> memref { +// CHECK: %[[D:.*]] = sparse_tensor.push_back %[[A]], %[[B]], %[[C]] {idx = 2 : index} : memref, memref, f64 to memref +// CHECK: return %[[D]] +func.func @sparse_push_back(%arg0: memref, %arg1: memref, %arg2: f64) -> memref { + %0 = sparse_tensor.push_back %arg0, %arg1, %arg2 {idx = 2 : index} : memref, memref, f64 to memref + return %0 : memref +} + +// ----- + #SparseMatrix = #sparse_tensor.encoding<{dimLevelType = ["compressed", "compressed"]}> // CHECK-LABEL: func @sparse_expansion(