blob: 21db18dfd47ed28031eddd225bacb77400d7c516 [file] [log] [blame]
//===- Linalg.cpp - C Interface for Linalg dialect ------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#include "mlir-c/Dialect/Linalg.h"
#include "mlir/CAPI/Registration.h"
#include "mlir/Dialect/Linalg/IR/Linalg.h"
using namespace mlir;
using namespace mlir::linalg;
/// Apply the special region builder for the builtin named Linalg op.
/// Assert that `op` is a builtin named Linalg op.
void mlirLinalgFillBuiltinNamedOpRegion(MlirOperation mlirOp) {
Operation *op = unwrap(mlirOp);
auto linalgOp = cast<LinalgOp>(op);
auto *dialect = static_cast<LinalgDialect *>(linalgOp->getDialect());
LinalgDialect::RegionBuilderFunType fun =
dialect->getRegionBuilder(op->getName().getStringRef());
assert(fun && "Expected a builtin named Linalg op.");
assert(op->getNumRegions() == 1 && "Expected Linalg op with 1 region");
assert(op->getRegion(0).getBlocks().empty() &&
"Expected Linalg op with 0 blocks");
SmallVector<Type, 8> argTypes;
SmallVector<Location, 8> argLocs;
for (OpOperand &opOperand : linalgOp->getOpOperands()) {
argTypes.push_back(getElementTypeOrSelf(opOperand.get().getType()));
argLocs.push_back(opOperand.get().getLoc());
}
ImplicitLocOpBuilder b(op->getLoc(), op->getContext());
Region &region = op->getRegion(0);
Block *body = b.createBlock(&region, /*insertPt=*/{}, argTypes, argLocs);
b.setInsertionPointToStart(body);
fun(b, *body, op->getAttrs(), /*emitError=*/{});
}
MLIR_CAPI_EXPORTED bool mlirLinalgIsAContractionOp(MlirOperation op) {
auto linalgOp = llvm::dyn_cast<mlir::linalg::LinalgOp>(unwrap(op));
// isaContractionOpInterface handles null linalgOp internally.
return linalg::isaContractionOpInterface(linalgOp);
}
MLIR_CAPI_EXPORTED MlirLinalgContractionDimensions
mlirLinalgInferContractionDimensions(MlirOperation op) {
MlirLinalgContractionDimensions result{};
auto linalgOp = dyn_cast<linalg::LinalgOp>(unwrap(op));
if (!linalgOp)
return result;
FailureOr<linalg::ContractionDimensions> maybeDims =
linalg::inferContractionDims(linalgOp);
if (failed(maybeDims))
return result;
linalg::ContractionDimensions contractionDims = *maybeDims;
MLIRContext *ctx = linalgOp.getContext();
auto toAttr = [&ctx](const SmallVector<unsigned, 2> &vals) -> MlirAttribute {
return wrap(
DenseI32ArrayAttr::get(ctx, llvm::to_vector_of<int32_t, 2>(vals)));
};
result.batch = toAttr(contractionDims.batch);
result.m = toAttr(contractionDims.m);
result.n = toAttr(contractionDims.n);
result.k = toAttr(contractionDims.k);
return result;
}
MLIR_CAPI_EXPORTED bool mlirLinalgIsAConvolutionOp(MlirOperation op) {
auto linalgOp = llvm::dyn_cast<mlir::linalg::LinalgOp>(unwrap(op));
if (!linalgOp)
return false;
return linalg::isaConvolutionOpInterface(linalgOp);
}
MLIR_CAPI_EXPORTED MlirLinalgConvolutionDimensions
mlirLinalgInferConvolutionDimensions(MlirOperation op) {
MlirLinalgConvolutionDimensions result{};
auto linalgOp = llvm::dyn_cast<mlir::linalg::LinalgOp>(unwrap(op));
if (!linalgOp)
return result;
FailureOr<linalg::ConvolutionDimensions> maybeDims =
linalg::inferConvolutionDims(linalgOp);
if (failed(maybeDims))
return result;
linalg::ConvolutionDimensions dims = *maybeDims;
MLIRContext *ctx = linalgOp.getContext();
auto toI32Attr =
[&ctx](const SmallVector<unsigned, 2> &vals) -> MlirAttribute {
return wrap(DenseI32ArrayAttr::get(ctx, llvm::to_vector_of<int32_t>(vals)));
};
auto toI64Attr =
[&ctx](const SmallVector<int64_t, 2> &vals) -> MlirAttribute {
return wrap(DenseI64ArrayAttr::get(ctx, vals));
};
result.batch = toI32Attr(dims.batch);
result.outputImage = toI32Attr(dims.outputImage);
result.outputChannel = toI32Attr(dims.outputChannel);
result.filterLoop = toI32Attr(dims.filterLoop);
result.inputChannel = toI32Attr(dims.inputChannel);
result.depth = toI32Attr(dims.depth);
result.strides = toI64Attr(dims.strides);
result.dilations = toI64Attr(dims.dilations);
return result;
}
MLIR_CAPI_EXPORTED MlirAttribute
mlirLinalgGetIndexingMapsAttribute(MlirOperation op) {
auto linalgOp = llvm::dyn_cast<mlir::linalg::LinalgOp>(unwrap(op));
if (!linalgOp)
return MlirAttribute{nullptr};
ArrayAttr attr = linalgOp.getIndexingMaps();
return wrap(attr);
}
MLIR_DEFINE_CAPI_DIALECT_REGISTRATION(Linalg, linalg, LinalgDialect)