18#include "mlir/IR/IRMapping.h"
19#include "mlir/IR/PatternMatch.h"
20#include "mlir/Interfaces/FunctionImplementation.h"
21#include "llvm/ADT/MapVector.h"
22#include "llvm/ADT/TypeSwitch.h"
32 Operation *operation, ArrayAttr argNames,
33 std::function<
void(mlir::InFlightDiagnostic &)> attachNote) {
34 DenseMap<StringRef, BlockArgument> portNames;
35 DenseMap<StringRef, Operation *> memberNames;
36 DenseMap<StringRef, Operation *> localNames;
38 if (operation->getNumRegions() != 1)
39 return operation->emitError(
"required to have exactly one region");
41 bool portsVerified =
true;
43 for (
auto arg : llvm::zip(argNames, operation->getRegion(0).getArguments())) {
44 StringRef argName = cast<StringAttr>(std::get<0>(arg)).getValue();
45 BlockArgument argValue = std::get<1>(arg);
47 if (portNames.count(argName)) {
48 auto diag = mlir::emitError(argValue.getLoc(),
"redefines name '")
50 diag.attachNote(portNames[argName].
getLoc())
51 <<
"'" << argName <<
"' first defined here";
53 portsVerified =
false;
57 portNames.insert({argName, argValue});
61 operation->walk<mlir::WalkOrder::PreOrder>([&](Operation *op) {
62 if (isa<SCModuleOp>(op->getParentOp()))
65 if (
auto nameDeclOp = dyn_cast<SystemCNameDeclOpInterface>(op)) {
66 StringRef name = nameDeclOp.getName();
68 auto reportNameRedefinition = [&](Location firstLoc) -> WalkResult {
69 auto diag = mlir::emitError(op->getLoc(),
"redefines name '")
71 diag.attachNote(firstLoc) <<
"'" << name <<
"' first defined here";
73 return WalkResult::interrupt();
76 if (portNames.count(name))
77 return reportNameRedefinition(portNames[name].
getLoc());
78 if (memberNames.count(name))
79 return reportNameRedefinition(memberNames[name]->
getLoc());
80 if (localNames.count(name))
81 return reportNameRedefinition(localNames[name]->
getLoc());
83 if (isa<SCModuleOp>(op->getParentOp()))
84 memberNames.insert({name, op});
86 localNames.insert({name, op});
89 return WalkResult::advance();
92 if (result.wasInterrupted() || !portsVerified)
103 return TypeSwitch<Type, hw::ModulePort::Direction>(type)
104 .Case<InOutType>([](
auto ty) {
return hw::ModulePort::Direction::InOut; })
105 .Case<InputType>([](
auto ty) {
return hw::ModulePort::Direction::Input; })
107 [](
auto ty) {
return hw::ModulePort::Direction::Output; });
110SCModuleOp::PortDirectionRange
112 std::function<bool(
const BlockArgument &)> predicateFn =
113 [&](
const BlockArgument &arg) ->
bool {
116 return llvm::make_filter_range(getArguments(), predicateFn);
119SmallVector<::circt::hw::PortInfo> SCModuleOp::getPortList() {
120 SmallVector<hw::PortInfo> ports;
121 for (
int i = 0, e = getNumArguments(); i < e; ++i) {
123 info.name = cast<StringAttr>(getPortNames()[i]);
126 ports.push_back(info);
131mlir::Region *SCModuleOp::getCallableRegion() {
return &getBody(); }
133StringRef SCModuleOp::getModuleName() {
135 ->getAttrOfType<StringAttr>(SymbolTable::getSymbolAttrName())
139ParseResult SCModuleOp::parse(OpAsmParser &parser, OperationState &result) {
142 (void)mlir::impl::parseOptionalVisibilityKeyword(parser, result.attributes);
145 StringAttr moduleName;
146 if (parser.parseSymbolName(moduleName, SymbolTable::getSymbolAttrName(),
151 bool isVariadic =
false;
152 SmallVector<OpAsmParser::Argument, 4> entryArgs;
153 SmallVector<Attribute> argNames;
154 SmallVector<Attribute> argLocs;
155 SmallVector<Attribute> resultNames;
156 SmallVector<DictionaryAttr> resultAttrs;
157 SmallVector<Attribute> resultLocs;
158 TypeAttr functionType;
160 parser, isVariadic, entryArgs, argNames, argLocs, resultNames,
161 resultAttrs, resultLocs, functionType)))
165 if (parser.parseOptionalAttrDictWithKeyword(result.attributes))
168 result.addAttribute(
"portNames",
169 ArrayAttr::get(parser.getContext(), argNames));
171 result.addAttribute(SCModuleOp::getFunctionTypeAttrName(result.name),
174 mlir::call_interface_impl::addArgAndResultAttrs(
175 parser.getBuilder(), result, entryArgs, resultAttrs,
176 SCModuleOp::getArgAttrsAttrName(result.name),
177 SCModuleOp::getResAttrsAttrName(result.name));
179 auto &body = *result.addRegion();
180 if (parser.parseRegion(body, entryArgs))
183 body.push_back(std::make_unique<Block>().release());
188void SCModuleOp::print(OpAsmPrinter &p) {
192 StringRef visibilityAttrName = SymbolTable::getVisibilityAttrName();
193 if (
auto visibility =
194 getOperation()->getAttrOfType<StringAttr>(visibilityAttrName))
195 p << visibility.getValue() <<
' ';
197 p.printSymbolName(SymbolTable::getSymbolName(*this).getValue());
200 bool needArgNamesAttr =
false;
202 p, *
this, getFunctionType().getInputs(),
false, {}, needArgNamesAttr);
203 mlir::function_interface_impl::printFunctionAttributes(
205 {
"portNames", getFunctionTypeAttrName(), getArgAttrsAttrName(),
206 getResAttrsAttrName()});
209 p.printRegion(getBody(),
false,
false);
213ArrayRef<Type> SCModuleOp::getArgumentTypes() {
214 return getFunctionType().getInputs();
218ArrayRef<Type> SCModuleOp::getResultTypes() {
219 return getFunctionType().getResults();
223 if (
auto inoutTy = dyn_cast<hw::InOutType>(type))
224 type = inoutTy.getElementType();
227 case hw::ModulePort::Direction::InOut:
228 return InOutType::get(type);
229 case hw::ModulePort::Direction::Input:
230 return InputType::get(type);
231 case hw::ModulePort::Direction::Output:
232 return OutputType::get(type);
234 llvm_unreachable(
"Impossible port direction");
237void SCModuleOp::build(OpBuilder &odsBuilder, OperationState &odsState,
238 StringAttr name, ArrayAttr portNames,
239 ArrayRef<Type> portTypes,
240 ArrayRef<NamedAttribute> attributes) {
241 odsState.addAttribute(getPortNamesAttrName(odsState.name), portNames);
242 Region *region = odsState.addRegion();
244 auto moduleType = odsBuilder.getFunctionType(portTypes, {});
245 odsState.addAttribute(getFunctionTypeAttrName(odsState.name),
246 TypeAttr::get(moduleType));
248 odsState.addAttribute(SymbolTable::getSymbolAttrName(), name);
249 region->push_back(
new Block);
250 region->addArguments(
252 SmallVector<Location>(portTypes.size(), odsBuilder.getUnknownLoc()));
253 odsState.addAttributes(attributes);
256void SCModuleOp::build(OpBuilder &odsBuilder, OperationState &odsState,
257 StringAttr name, ArrayRef<hw::PortInfo> ports,
258 ArrayRef<NamedAttribute> attributes) {
259 MLIRContext *ctxt = odsBuilder.getContext();
260 SmallVector<Attribute> portNames;
261 SmallVector<Type> portTypes;
262 for (
auto port : ports) {
263 portNames.push_back(StringAttr::get(ctxt, port.getName()));
266 build(odsBuilder, odsState, name, ArrayAttr::get(ctxt, portNames), portTypes);
269void SCModuleOp::build(OpBuilder &odsBuilder, OperationState &odsState,
271 ArrayRef<NamedAttribute> attributes) {
272 MLIRContext *ctxt = odsBuilder.getContext();
273 SmallVector<Attribute> portNames;
274 SmallVector<Type> portTypes;
275 for (
auto port : ports) {
276 portNames.push_back(StringAttr::get(ctxt, port.getName()));
279 build(odsBuilder, odsState, name, ArrayAttr::get(ctxt, portNames), portTypes);
282void SCModuleOp::getAsmBlockArgumentNames(mlir::Region ®ion,
287 ArrayAttr portNames = getPortNames();
288 for (
size_t i = 0, e = getNumArguments(); i != e; ++i) {
289 auto str = cast<StringAttr>(portNames[i]).getValue();
290 setNameFn(getArgument(i), str);
294LogicalResult SCModuleOp::verify() {
295 if (getFunctionType().getNumResults() != 0)
297 "incorrect number of function results (always has to be 0)");
298 if (getPortNames().size() != getFunctionType().getNumInputs())
299 return emitOpError(
"incorrect number of port names");
301 for (
auto arg : getArguments()) {
302 if (!hw::type_isa<InputType, OutputType, InOutType>(arg.getType()))
303 return mlir::emitError(
305 "module port must be of type 'sc_in', 'sc_out', or 'sc_inout'");
308 for (
auto portName : getPortNames()) {
309 if (cast<StringAttr>(portName).getValue().
empty())
310 return emitOpError(
"port name must not be empty");
316LogicalResult SCModuleOp::verifyRegions() {
317 auto attachNote = [&](mlir::InFlightDiagnostic &diag) {
318 diag.attachNote(
getLoc()) <<
"in module '@" << getModuleName() <<
"'";
323CtorOp SCModuleOp::getOrCreateCtor(OpBuilder &builder) {
325 getBody().walk([&](Operation *op) {
326 if ((ctor = dyn_cast<CtorOp>(op)))
327 return WalkResult::interrupt();
329 return WalkResult::skip();
335 OpBuilder::InsertionGuard guard(builder);
337 return CtorOp::create(builder,
getLoc());
340DestructorOp SCModuleOp::getOrCreateDestructor() {
341 DestructorOp destructor;
342 getBody().walk([&](Operation *op) {
343 if ((destructor = dyn_cast<DestructorOp>(op)))
344 return WalkResult::interrupt();
346 return WalkResult::skip();
353 return DestructorOp::create(builder,
getLoc());
360void SignalOp::getAsmResultNames(OpAsmSetValueNameFn setNameFn) {
361 setNameFn(getSignal(),
getName());
368OpFoldResult ConvertOp::fold(FoldAdaptor) {
369 if (getInput().getType() == getResult().getType())
372 if (
auto other = getInput().getDefiningOp<ConvertOp>()) {
373 Type inputType = other.getInput().getType();
374 Type intermediateType = getInput().getType();
376 if (inputType != getResult().getType())
381 bool inputSigned = isa<SignedType, IntBaseType>(inputType);
382 bool intermediateSigned = isa<SignedType, IntBaseType>(intermediateType);
383 if (inputSigned ^ intermediateSigned)
387 if (isa<LogicVectorBaseType, LogicType>(inputType) &&
388 !isa<LogicVectorBaseType, LogicType>(intermediateType))
392 auto intermediateBw =
getBitWidth(intermediateType);
394 if (!inputBw && intermediateBw) {
395 if (isa<IntBaseType, UIntBaseType>(inputType) && *intermediateBw >= 64)
396 return other.getInput();
401 if (!intermediateBw) {
402 if (isa<BitVectorBaseType, LogicVectorBaseType>(intermediateType))
403 return other.getInput();
405 if (!inputBw && isa<IntBaseType, UIntBaseType>(inputType) &&
406 isa<SignedType, UnsignedType>(intermediateType))
407 return other.getInput();
409 if (inputBw && *inputBw <= 64 &&
410 isa<IntBaseType, UIntBaseType, SignedType, UnsignedType>(
412 return other.getInput();
419 if (inputBw && intermediateBw && *inputBw <= *intermediateBw)
420 return other.getInput();
430LogicalResult CtorOp::verify() {
431 if (getBody().getNumArguments() != 0)
432 return emitOpError(
"must not have any arguments");
441void SCFuncOp::getAsmResultNames(OpAsmSetValueNameFn setNameFn) {
442 setNameFn(getHandle(),
getName());
445LogicalResult SCFuncOp::verify() {
446 if (getBody().getNumArguments() != 0)
447 return emitOpError(
"must not have any arguments");
456void InstanceDeclOp::getAsmResultNames(OpAsmSetValueNameFn setNameFn) {
457 setNameFn(getInstanceHandle(),
getName());
460StringRef InstanceDeclOp::getInstanceName() {
return getName(); }
461StringAttr InstanceDeclOp::getInstanceNameAttr() {
return getNameAttr(); }
466 if (
auto *result = cache->
getDefinition(getModuleNameAttr()))
469 auto topLevelModuleOp = (*this)->getParentOfType<ModuleOp>();
470 return topLevelModuleOp.lookupSymbol(getModuleName());
474InstanceDeclOp::verifySymbolUses(SymbolTableCollection &symbolTable) {
476 symbolTable.lookupNearestSymbolFrom(*this, getModuleNameAttr());
477 if (module ==
nullptr)
478 return emitError(
"cannot find module definition '")
479 << getModuleName() <<
"'";
481 auto emitError = [&](
const std::function<void(InFlightDiagnostic & diag)> &fn)
483 auto diag = emitOpError();
485 diag.attachNote(module->getLoc()) <<
"module declared here";
490 if (!isa<SCModuleOp>(module))
491 return emitError([&](
auto &diag) {
492 diag <<
"symbol reference '" << getModuleName()
493 <<
"' isn't a systemc module";
496 auto scModule = cast<SCModuleOp>(module);
499 if (scModule.getModuleName() != getInstanceType().getModuleName())
500 return emitError([&](
auto &diag) {
501 diag <<
"module names must match; expected '" << scModule.getModuleName()
502 <<
"' but got '" << getInstanceType().getModuleName().getValue()
507 ArrayRef<ModuleType::PortInfo> ports = getInstanceType().getPorts();
508 ArrayAttr modArgNames = scModule.getPortNames();
509 auto numPorts = ports.
size();
510 auto expectedPortTypes = scModule.getArgumentTypes();
512 if (expectedPortTypes.size() != numPorts)
513 return emitError([&](
auto &diag) {
514 diag <<
"has a wrong number of ports; expected "
515 << expectedPortTypes.size() <<
" but got " << numPorts;
518 for (
size_t i = 0; i != numPorts; ++i) {
519 if (ports[i].type != expectedPortTypes[i]) {
520 return emitError([&](
auto &diag) {
521 diag <<
"port type #" << i <<
" must be " << expectedPortTypes[i]
522 <<
", but got " << ports[i].type;
526 if (ports[i].name != modArgNames[i])
527 return emitError([&](
auto &diag) {
528 diag <<
"port name #" << i <<
" must be " << modArgNames[i]
529 <<
", but got " << ports[i].name;
536SmallVector<hw::PortInfo> InstanceDeclOp::getPortList() {
537 return cast<hw::PortList>(SymbolTable::lookupNearestSymbolFrom(
538 getOperation(), getReferencedModuleNameAttr()))
546LogicalResult DestructorOp::verify() {
547 if (getBody().getNumArguments() != 0)
548 return emitOpError(
"must not have any arguments");
557ParseResult BindPortOp::parse(OpAsmParser &parser, OperationState &result) {
558 OpAsmParser::UnresolvedOperand instance, channel;
559 std::string portName;
560 if (parser.parseOperand(instance) || parser.parseLSquare() ||
561 parser.parseString(&portName))
564 auto portNameLoc = parser.getCurrentLocation();
566 if (parser.parseRSquare() || parser.parseKeyword(
"to") ||
567 parser.parseOperand(channel))
570 if (parser.parseOptionalAttrDict(result.attributes))
573 auto typeListLoc = parser.getCurrentLocation();
574 SmallVector<Type> types;
575 if (parser.parseColonTypeList(types))
578 if (types.size() != 2)
579 return parser.emitError(typeListLoc,
580 "expected a list of exactly 2 types, but got ")
583 if (parser.resolveOperand(instance, types[0], result.operands))
585 if (parser.resolveOperand(channel, types[1], result.operands))
588 if (
auto moduleType = dyn_cast<ModuleType>(types[0])) {
589 auto ports = moduleType.getPorts();
591 for (
auto port : ports) {
592 if (port.name == portName)
596 if (index >= ports.size())
597 return parser.emitError(portNameLoc,
"port name \"")
598 << portName <<
"\" not found in module";
600 result.addAttribute(
"portId", parser.getBuilder().getIndexAttr(index));
608void BindPortOp::print(OpAsmPrinter &p) {
609 p <<
" " << getInstance() <<
"["
610 << cast<ModuleType>(getInstance().getType())
611 .getPorts()[getPortId().getZExtValue()]
613 <<
"] to " << getChannel();
614 p.printOptionalAttrDict((*this)->getAttrs(), {
"portId"});
615 p <<
" : " << getInstance().getType() <<
", " << getChannel().getType();
618LogicalResult BindPortOp::verify() {
619 auto ports = cast<ModuleType>(getInstance().getType()).getPorts();
620 if (getPortId().getZExtValue() >= ports.size())
621 return emitOpError(
"port #")
622 << getPortId().getZExtValue() <<
" does not exist, there are only "
623 << ports.size() <<
" ports";
626 Type portType = ports[getPortId().getZExtValue()].type;
627 Type channelType = getChannel().getType();
629 return emitOpError() << portType <<
" port cannot be bound to "
630 << channelType <<
" channel due to base type mismatch";
633 if ((isa<InputType>(portType) && isa<OutputType>(channelType)) ||
634 (isa<OutputType>(portType) && isa<InputType>(channelType)))
635 return emitOpError() << portType <<
" port cannot be bound to "
637 <<
" channel due to port direction mismatch";
642StringRef BindPortOp::getPortName() {
643 return cast<ModuleType>(getInstance().getType())
644 .getPorts()[getPortId().getZExtValue()]
652LogicalResult SensitiveOp::canonicalize(SensitiveOp op,
653 PatternRewriter &rewriter) {
654 if (op.getSensitivities().empty()) {
655 rewriter.eraseOp(op);
666void VariableOp::getAsmResultNames(OpAsmSetValueNameFn setNameFn) {
667 setNameFn(getVariable(),
getName());
670ParseResult VariableOp::parse(OpAsmParser &parser, OperationState &result) {
674 result.addAttribute(
"name", nameAttr);
676 OpAsmParser::UnresolvedOperand init;
677 auto initResult = parser.parseOptionalOperand(init);
679 if (parser.parseOptionalAttrDict(result.attributes))
683 if (parser.parseColonType(variableType))
686 if (initResult.has_value()) {
687 if (parser.resolveOperand(init, variableType, result.operands))
690 result.addTypes({variableType});
695void VariableOp::print(::mlir::OpAsmPrinter &p) {
699 p << getInit() <<
" ";
701 p.printOptionalAttrDict(getOperation()->getAttrs(), {
"name"});
702 p <<
": " << getVariable().getType();
705LogicalResult VariableOp::verify() {
706 if (getInit() && getInit().getType() != getVariable().getType())
708 "'init' and 'variable' must have the same type, but got ")
709 << getInit().getType() <<
" and " << getVariable().getType();
719void InteropVerilatedOp::build(OpBuilder &odsBuilder, OperationState &odsState,
720 Operation *module, StringAttr name,
721 ArrayRef<Value> inputs) {
722 auto mod = cast<hw::HWModuleLike>(module);
723 auto argNames = odsBuilder.getArrayAttr(mod.getInputNames());
724 auto resultNames = odsBuilder.getArrayAttr(mod.getOutputNames());
725 build(odsBuilder, odsState, mod.getHWModuleType().getOutputTypes(), name,
726 FlatSymbolRefAttr::get(SymbolTable::getSymbolName(module)), argNames,
727 resultNames, inputs);
731InteropVerilatedOp::verifySymbolUses(SymbolTableCollection &symbolTable) {
733 *
this, getModuleNameAttr(), getInputs(), getResultTypes(),
734 getInputNames(), getResultNames(), ArrayAttr(), symbolTable);
739void InteropVerilatedOp::getAsmResultNames(OpAsmSetValueNameFn setNameFn) {
741 getResultNames(), getResults());
753LogicalResult CallOp::verifySymbolUses(SymbolTableCollection &symbolTable) {
755 auto fnAttr = (*this)->getAttrOfType<FlatSymbolRefAttr>(
"callee");
757 return emitOpError(
"requires a 'callee' symbol reference attribute");
758 FuncOp fn = symbolTable.lookupNearestSymbolFrom<FuncOp>(*
this, fnAttr);
760 return emitOpError() <<
"'" << fnAttr.getValue()
761 <<
"' does not reference a valid function";
764 auto fnType = fn.getFunctionType();
765 if (fnType.getNumInputs() != getNumOperands())
766 return emitOpError(
"incorrect number of operands for callee");
768 for (
unsigned i = 0, e = fnType.getNumInputs(); i != e; ++i)
769 if (getOperand(i).getType() != fnType.getInput(i))
770 return emitOpError(
"operand type mismatch: expected operand type ")
771 << fnType.getInput(i) <<
", but provided "
772 << getOperand(i).getType() <<
" for operand number " << i;
774 if (fnType.getNumResults() != getNumResults())
775 return emitOpError(
"incorrect number of results for callee");
777 for (
unsigned i = 0, e = fnType.getNumResults(); i != e; ++i)
778 if (getResult(i).getType() != fnType.getResult(i)) {
779 auto diag = emitOpError(
"result type mismatch at index ") << i;
780 diag.attachNote() <<
" op result types: " << getResultTypes();
781 diag.attachNote() <<
"function result types: " << fnType.getResults();
788FunctionType CallOp::getCalleeType() {
789 return FunctionType::get(getContext(), getOperandTypes(), getResultTypes());
793LogicalResult CallOp::verify() {
794 if (getNumResults() > 1)
796 "incorrect number of function results (always has to be 0 or 1)");
806LogicalResult CallIndirectOp::verify() {
807 if (getNumResults() > 1)
809 "incorrect number of function results (always has to be 0 or 1)");
824FuncOp FuncOp::create(Location location, StringRef name, ArrayAttr argNames,
825 FunctionType type, ArrayRef<NamedAttribute> attrs) {
826 OpBuilder builder(location->getContext());
827 OperationState state(location, getOperationName());
828 FuncOp::build(builder, state, name, argNames, type, attrs);
829 return cast<FuncOp>(Operation::create(state));
832FuncOp FuncOp::create(Location location, StringRef name, ArrayAttr argNames,
833 FunctionType type, Operation::dialect_attr_range attrs) {
834 SmallVector<NamedAttribute, 8> attrRef(attrs);
835 return create(location, name, argNames, type, ArrayRef(attrRef));
838FuncOp FuncOp::create(Location location, StringRef name, ArrayAttr argNames,
839 FunctionType type, ArrayRef<NamedAttribute> attrs,
840 ArrayRef<DictionaryAttr> argAttrs) {
841 FuncOp func = create(location, name, argNames, type, attrs);
842 func.setAllArgAttrs(argAttrs);
846void FuncOp::build(OpBuilder &odsBuilder, OperationState &odsState,
847 StringRef name, ArrayAttr argNames, FunctionType type,
848 ArrayRef<NamedAttribute> attrs,
849 ArrayRef<DictionaryAttr> argAttrs) {
850 odsState.addAttribute(getArgNamesAttrName(odsState.name), argNames);
851 odsState.addAttribute(SymbolTable::getSymbolAttrName(),
852 odsBuilder.getStringAttr(name));
853 odsState.addAttribute(FuncOp::getFunctionTypeAttrName(odsState.name),
854 TypeAttr::get(type));
855 odsState.attributes.append(attrs.begin(), attrs.end());
856 odsState.addRegion();
858 if (argAttrs.empty())
860 assert(type.getNumInputs() == argAttrs.size());
861 mlir::call_interface_impl::addArgAndResultAttrs(
862 odsBuilder, odsState, argAttrs,
863 {}, FuncOp::getArgAttrsAttrName(odsState.name),
864 FuncOp::getResAttrsAttrName(odsState.name));
867ParseResult FuncOp::parse(OpAsmParser &parser, OperationState &result) {
869 [](Builder &builder, ArrayRef<Type> argTypes, ArrayRef<Type> results,
870 mlir::function_interface_impl::VariadicFlag,
871 std::string &) {
return builder.getFunctionType(argTypes, results); };
875 if (succeeded(parser.parseOptionalKeyword(
"externC")))
876 result.addAttribute(getExternCAttrName(result.name),
877 UnitAttr::get(result.getContext()));
882 SmallVector<OpAsmParser::Argument> entryArgs;
883 SmallVector<DictionaryAttr> resultAttrs;
884 SmallVector<Type> resultTypes;
885 auto &builder = parser.getBuilder();
888 (void)mlir::impl::parseOptionalVisibilityKeyword(parser, result.attributes);
892 if (parser.parseSymbolName(nameAttr, SymbolTable::getSymbolAttrName(),
897 mlir::SMLoc signatureLocation = parser.getCurrentLocation();
898 bool isVariadic =
false;
899 if (mlir::function_interface_impl::parseFunctionSignatureWithArguments(
900 parser,
false, entryArgs, isVariadic, resultTypes, resultAttrs))
903 std::string errorMessage;
904 SmallVector<Type> argTypes;
905 argTypes.reserve(entryArgs.size());
906 for (
auto &arg : entryArgs)
907 argTypes.push_back(arg.type);
909 Type type = buildFuncType(
910 builder, argTypes, resultTypes,
911 mlir::function_interface_impl::VariadicFlag(isVariadic), errorMessage);
913 return parser.emitError(signatureLocation)
914 <<
"failed to construct function type"
915 << (errorMessage.empty() ?
"" :
": ") << errorMessage;
917 result.addAttribute(FuncOp::getFunctionTypeAttrName(result.name),
918 TypeAttr::get(type));
921 NamedAttrList parsedAttributes;
922 mlir::SMLoc attributeDictLocation = parser.getCurrentLocation();
923 if (parser.parseOptionalAttrDictWithKeyword(parsedAttributes))
928 for (StringRef disallowed :
929 {SymbolTable::getVisibilityAttrName(), SymbolTable::getSymbolAttrName(),
930 FuncOp::getFunctionTypeAttrName(result.name).getValue()}) {
931 if (parsedAttributes.get(disallowed))
932 return parser.emitError(attributeDictLocation,
"'")
934 <<
"' is an inferred attribute and should not be specified in the "
935 "explicit attribute dictionary";
937 result.attributes.append(parsedAttributes);
940 assert(resultAttrs.size() == resultTypes.size());
941 mlir::call_interface_impl::addArgAndResultAttrs(
942 builder, result, entryArgs, resultAttrs,
943 FuncOp::getArgAttrsAttrName(result.name),
944 FuncOp::getResAttrsAttrName(result.name));
948 auto *body = result.addRegion();
949 mlir::SMLoc loc = parser.getCurrentLocation();
950 mlir::OptionalParseResult parseResult =
951 parser.parseOptionalRegion(*body, entryArgs,
953 if (parseResult.has_value()) {
954 if (failed(*parseResult))
958 return parser.emitError(loc,
"expected non-empty function body");
963 SmallVector<Attribute> argNames;
964 if (!entryArgs.empty() && !entryArgs.front().ssaName.name.empty()) {
965 for (
auto &arg : entryArgs)
967 StringAttr::
get(parser.getContext(), arg.ssaName.name.drop_front()));
970 result.addAttribute(getArgNamesAttrName(result.name),
971 ArrayAttr::get(parser.getContext(), argNames));
976void FuncOp::print(OpAsmPrinter &p) {
980 mlir::FunctionOpInterface op = *
this;
987 op->getAttrOfType<StringAttr>(SymbolTable::getSymbolAttrName())
991 StringRef visibilityAttrName = SymbolTable::getVisibilityAttrName();
992 if (
auto visibility = op->getAttrOfType<StringAttr>(visibilityAttrName))
993 p << visibility.getValue() <<
' ';
994 p.printSymbolName(funcName);
996 ArrayRef<Type> argTypes = op.getArgumentTypes();
997 ArrayRef<Type> resultTypes = op.getResultTypes();
998 mlir::function_interface_impl::printFunctionSignature(p, op, argTypes,
false,
1000 mlir::function_interface_impl::printFunctionAttributes(
1002 {visibilityAttrName,
"externC",
"argNames", getFunctionTypeAttrName(),
1003 getArgAttrsAttrName(), getResAttrsAttrName()});
1005 Region &body = op->getRegion(0);
1006 if (!body.empty()) {
1008 p.printRegion(body,
false,
1017void FuncOp::cloneInto(FuncOp dest, IRMapping &mapper) {
1019 llvm::MapVector<StringAttr, Attribute> newAttrMap;
1020 for (
const auto &attr : dest->getAttrs())
1021 newAttrMap.insert({attr.getName(), attr.getValue()});
1022 for (
const auto &attr : (*this)->getAttrs())
1023 newAttrMap.insert({attr.getName(), attr.getValue()});
1025 auto newAttrs = llvm::to_vector(llvm::map_range(
1026 newAttrMap, [](std::pair<StringAttr, Attribute> attrPair) {
1027 return NamedAttribute(attrPair.first, attrPair.second);
1029 dest->setAttrs(DictionaryAttr::get(getContext(), newAttrs));
1032 getBody().cloneInto(&dest.getBody(), mapper);
1040FuncOp FuncOp::clone(IRMapping &mapper) {
1042 FuncOp newFunc = cast<FuncOp>(getOperation()->cloneWithoutRegions());
1047 if (!isExternal()) {
1048 FunctionType oldType = getFunctionType();
1050 unsigned oldNumArgs = oldType.getNumInputs();
1051 SmallVector<Type, 4> newInputs;
1052 newInputs.reserve(oldNumArgs);
1053 for (
unsigned i = 0; i != oldNumArgs; ++i)
1054 if (!mapper.contains(getArgument(i)))
1055 newInputs.push_back(oldType.getInput(i));
1059 if (newInputs.size() != oldNumArgs) {
1060 newFunc.setType(FunctionType::get(oldType.getContext(), newInputs,
1061 oldType.getResults()));
1063 if (ArrayAttr argAttrs = getAllArgAttrs()) {
1064 SmallVector<Attribute> newArgAttrs;
1065 newArgAttrs.reserve(newInputs.size());
1066 for (
unsigned i = 0; i != oldNumArgs; ++i)
1067 if (!mapper.contains(getArgument(i)))
1068 newArgAttrs.push_back(argAttrs[i]);
1069 newFunc.setAllArgAttrs(newArgAttrs);
1075 cloneInto(newFunc, mapper);
1079FuncOp FuncOp::clone() {
1081 return clone(mapper);
1086void FuncOp::getAsmBlockArgumentNames(mlir::Region ®ion,
1091 for (
auto [arg, name] :
llvm::zip(getArguments(), getArgNames()))
1092 setNameFn(arg, cast<StringAttr>(name).getValue());
1095LogicalResult FuncOp::verify() {
1096 if (getFunctionType().getNumResults() > 1)
1098 "incorrect number of function results (always has to be 0 or 1)");
1100 if (getBody().
empty())
1103 if (getArgNames().size() != getFunctionType().getNumInputs())
1104 return emitOpError(
"incorrect number of argument names");
1106 for (
auto portName : getArgNames()) {
1107 if (cast<StringAttr>(portName).getValue().
empty())
1108 return emitOpError(
"arg name must not be empty");
1114LogicalResult FuncOp::verifyRegions() {
1115 auto attachNote = [&](mlir::InFlightDiagnostic &diag) {
1116 diag.attachNote(
getLoc()) <<
"in function '@" <<
getName() <<
"'";
1129LogicalResult ReturnOp::verify() {
1130 auto function = cast<FuncOp>((*this)->getParentOp());
1133 const auto &results = function.getFunctionType().getResults();
1134 if (getNumOperands() != results.size())
1135 return emitOpError(
"has ")
1136 << getNumOperands() <<
" operands, but enclosing function (@"
1137 << function.getName() <<
") returns " << results.size();
1139 for (
unsigned i = 0, e = results.size(); i != e; ++i)
1140 if (getOperand(i).getType() != results[i])
1141 return emitError() <<
"type of return operand " << i <<
" ("
1142 << getOperand(i).getType()
1143 <<
") doesn't match function result type ("
1144 << results[i] <<
")"
1145 <<
" in function @" << function.getName();
1155#define GET_OP_CLASSES
1156#include "circt/Dialect/SystemC/SystemC.cpp.inc"
assert(baseType &&"element must be base type")
static Location getLoc(DefSlot slot)
static Block * getBodyBlock(FModuleLike mod)
static InstancePath empty
static hw::ModulePort::Direction getDirection(Type type)
static Type wrapPortType(Type type, hw::ModulePort::Direction direction)
static LogicalResult verifyUniqueNamesInRegion(Operation *operation, ArrayAttr argNames, std::function< void(mlir::InFlightDiagnostic &)> attachNote)
This stores lookup tables to make manipulating and working with the IR more efficient.
mlir::Operation * getDefinition(mlir::Attribute attr) const override
Lookup a definition for 'symbol' in the cache.
Direction get(bool isOutput)
Returns an output direction if isOutput is true, otherwise returns an input direction.
std::string getInstanceName(mlir::func::CallOp callOp)
A helper function to get the instance name.
LogicalResult verifyInstanceOfHWModule(Operation *instance, FlatSymbolRefAttr moduleRef, OperandRange inputs, TypeRange results, ArrayAttr argNames, ArrayAttr resultNames, ArrayAttr parameters, SymbolTableCollection &symbolTable)
Combines verifyReferencedModule, verifyInputs, verifyOutputs, and verifyParameters.
StringAttr getName(ArrayAttr names, size_t idx)
Return the name at the specified index of the ArrayAttr or null if it cannot be determined.
void getAsmResultNames(OpAsmSetValueNameFn setNameFn, StringRef instanceName, ArrayAttr resultNames, ValueRange results)
Suggest a name for each result value based on the saved result names attribute.
void printModuleSignature(OpAsmPrinter &p, Operation *op, ArrayRef< Type > argTypes, bool isVariadic, ArrayRef< Type > resultTypes, bool &needArgNamesAttr)
Print a module signature with named results.
ParseResult parseModuleFunctionSignature(OpAsmParser &parser, bool &isVariadic, SmallVectorImpl< OpAsmParser::Argument > &args, SmallVectorImpl< Attribute > &argNames, SmallVectorImpl< Attribute > &argLocs, SmallVectorImpl< Attribute > &resultNames, SmallVectorImpl< DictionaryAttr > &resultAttrs, SmallVectorImpl< Attribute > &resultLocs, TypeAttr &type)
This is a variant of mlir::parseFunctionSignature that allows names on result arguments.
int64_t getBitWidth(mlir::Type type)
Return the hardware bit width of a type.
Type getSignalBaseType(Type type)
Get the type wrapped by a signal or port (in, inout, out) type.
The InstanceGraph op interface, see InstanceGraphInterface.td for more details.
ParseResult parseImplicitSSAName(OpAsmParser &parser, StringAttr &attr)
Parse an implicit SSA name string attribute.
function_ref< void(Value, StringRef)> OpAsmSetValueNameFn
This holds a decoded list of input/inout and output ports for a module or instance.
This holds the name, type, direction of a module's ports.