//===- ComplexOps.cpp - MLIR Complex Operations ---------------------------===// // // 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/Dialect/Complex/IR/Complex.h" #include "mlir/IR/Builders.h" #include "mlir/IR/Matchers.h" using namespace mlir; using namespace mlir::complex; //===----------------------------------------------------------------------===// // ConstantOp //===----------------------------------------------------------------------===// OpFoldResult ConstantOp::fold(ArrayRef operands) { assert(operands.empty() && "constant has no operands"); return getValue(); } void ConstantOp::getAsmResultNames( function_ref setNameFn) { setNameFn(getResult(), "cst"); } bool ConstantOp::isBuildableWith(Attribute value, Type type) { if (auto arrAttr = value.dyn_cast()) { auto complexTy = type.dyn_cast(); if (!complexTy) return false; auto complexEltTy = complexTy.getElementType(); return arrAttr.size() == 2 && arrAttr[0].getType() == complexEltTy && arrAttr[1].getType() == complexEltTy; } return false; } LogicalResult ConstantOp::verify() { ArrayAttr arrayAttr = getValue(); if (arrayAttr.size() != 2) { return emitOpError( "requires 'value' to be a complex constant, represented as array of " "two values"); } auto complexEltTy = getType().getElementType(); if (complexEltTy != arrayAttr[0].getType() || complexEltTy != arrayAttr[1].getType()) { return emitOpError() << "requires attribute's element types (" << arrayAttr[0].getType() << ", " << arrayAttr[1].getType() << ") to match the element type of the op's return type (" << complexEltTy << ")"; } return success(); } //===----------------------------------------------------------------------===// // CreateOp //===----------------------------------------------------------------------===// OpFoldResult CreateOp::fold(ArrayRef operands) { assert(operands.size() == 2 && "binary op takes two operands"); // Fold complex.create(complex.re(op), complex.im(op)). if (auto reOp = getOperand(0).getDefiningOp()) { if (auto imOp = getOperand(1).getDefiningOp()) { if (reOp.getOperand() == imOp.getOperand()) { return reOp.getOperand(); } } } return {}; } //===----------------------------------------------------------------------===// // ImOp //===----------------------------------------------------------------------===// OpFoldResult ImOp::fold(ArrayRef operands) { assert(operands.size() == 1 && "unary op takes 1 operand"); ArrayAttr arrayAttr = operands[0].dyn_cast_or_null(); if (arrayAttr && arrayAttr.size() == 2) return arrayAttr[1]; if (auto createOp = getOperand().getDefiningOp()) return createOp.getOperand(1); return {}; } //===----------------------------------------------------------------------===// // ReOp //===----------------------------------------------------------------------===// OpFoldResult ReOp::fold(ArrayRef operands) { assert(operands.size() == 1 && "unary op takes 1 operand"); ArrayAttr arrayAttr = operands[0].dyn_cast_or_null(); if (arrayAttr && arrayAttr.size() == 2) return arrayAttr[0]; if (auto createOp = getOperand().getDefiningOp()) return createOp.getOperand(0); return {}; } //===----------------------------------------------------------------------===// // AddOp //===----------------------------------------------------------------------===// OpFoldResult AddOp::fold(ArrayRef operands) { assert(operands.size() == 2 && "binary op takes 2 operands"); // complex.add(complex.sub(a, b), b) -> a if (auto sub = getLhs().getDefiningOp()) if (getRhs() == sub.getRhs()) return sub.getLhs(); // complex.add(b, complex.sub(a, b)) -> a if (auto sub = getRhs().getDefiningOp()) if (getLhs() == sub.getRhs()) return sub.getLhs(); return {}; } //===----------------------------------------------------------------------===// // NegOp //===----------------------------------------------------------------------===// OpFoldResult NegOp::fold(ArrayRef operands) { assert(operands.size() == 1 && "unary op takes 1 operand"); // complex.neg(complex.neg(a)) -> a if (auto negOp = getOperand().getDefiningOp()) return negOp.getOperand(); return {}; } //===----------------------------------------------------------------------===// // LogOp //===----------------------------------------------------------------------===// OpFoldResult LogOp::fold(ArrayRef operands) { assert(operands.size() == 1 && "unary op takes 1 operand"); // complex.log(complex.exp(a)) -> a if (auto expOp = getOperand().getDefiningOp()) return expOp.getOperand(); return {}; } //===----------------------------------------------------------------------===// // ExpOp //===----------------------------------------------------------------------===// OpFoldResult ExpOp::fold(ArrayRef operands) { assert(operands.size() == 1 && "unary op takes 1 operand"); // complex.exp(complex.log(a)) -> a if (auto logOp = getOperand().getDefiningOp()) return logOp.getOperand(); return {}; } //===----------------------------------------------------------------------===// // TableGen'd op method definitions //===----------------------------------------------------------------------===// #define GET_OP_CLASSES #include "mlir/Dialect/Complex/IR/ComplexOps.cpp.inc"