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"
24 using namespace circt;
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)
110 SCModuleOp::PortDirectionRange
112 std::function<bool(
const BlockArgument &)> predicateFn =
113 [&](
const BlockArgument &arg) ->
bool {
116 return llvm::make_filter_range(getArguments(), predicateFn);
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);
131 mlir::Region *SCModuleOp::getCallableRegion() {
return &getBody(); }
133 StringRef SCModuleOp::getModuleName() {
135 ->getAttrOfType<StringAttr>(SymbolTable::getSymbolAttrName())
139 ParseResult 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",
171 result.addAttribute(SCModuleOp::getFunctionTypeAttrName(result.name),
174 mlir::function_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());
188 void 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);
213 ArrayRef<Type> SCModuleOp::getArgumentTypes() {
214 return getFunctionType().getInputs();
218 ArrayRef<Type> SCModuleOp::getResultTypes() {
219 return getFunctionType().getResults();
223 if (
auto inoutTy = dyn_cast<hw::InOutType>(type))
224 type = inoutTy.getElementType();
234 llvm_unreachable(
"Impossible port direction");
237 void 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),
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);
256 void 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) {
269 void SCModuleOp::build(OpBuilder &odsBuilder, OperationState &odsState,
270 StringAttr name,
const hw::ModulePortInfo &ports,
271 ArrayRef<NamedAttribute> attributes) {
272 MLIRContext *
ctxt = odsBuilder.getContext();
273 SmallVector<Attribute> portNames;
274 SmallVector<Type> portTypes;
275 for (
auto port : ports) {
282 void 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);
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");
316 LogicalResult SCModuleOp::verifyRegions() {
317 auto attachNote = [&](mlir::InFlightDiagnostic &diag) {
318 diag.attachNote(getLoc()) <<
"in module '@" << getModuleName() <<
"'";
323 CtorOp SCModuleOp::getOrCreateCtor() {
325 getBody().walk([&](Operation *op) {
326 if ((ctor = dyn_cast<CtorOp>(op)))
327 return WalkResult::interrupt();
329 return WalkResult::skip();
335 return OpBuilder(getBody()).create<CtorOp>(getLoc());
338 DestructorOp SCModuleOp::getOrCreateDestructor() {
339 DestructorOp destructor;
340 getBody().walk([&](Operation *op) {
341 if ((destructor = dyn_cast<DestructorOp>(op)))
342 return WalkResult::interrupt();
344 return WalkResult::skip();
350 return OpBuilder::atBlockEnd(
getBodyBlock()).create<DestructorOp>(getLoc());
358 setNameFn(getSignal(),
getName());
365 OpFoldResult ConvertOp::fold(FoldAdaptor) {
366 if (getInput().getType() == getResult().getType())
369 if (
auto other = getInput().getDefiningOp<ConvertOp>()) {
370 Type inputType = other.getInput().getType();
371 Type intermediateType = getInput().getType();
373 if (inputType != getResult().getType())
378 bool inputSigned = isa<SignedType, IntBaseType>(inputType);
379 bool intermediateSigned = isa<SignedType, IntBaseType>(intermediateType);
380 if (inputSigned ^ intermediateSigned)
384 if (isa<LogicVectorBaseType, LogicType>(inputType) &&
385 !isa<LogicVectorBaseType, LogicType>(intermediateType))
389 auto intermediateBw =
getBitWidth(intermediateType);
391 if (!inputBw && intermediateBw) {
392 if (isa<IntBaseType, UIntBaseType>(inputType) && *intermediateBw >= 64)
393 return other.getInput();
398 if (!intermediateBw) {
399 if (isa<BitVectorBaseType, LogicVectorBaseType>(intermediateType))
400 return other.getInput();
402 if (!inputBw && isa<IntBaseType, UIntBaseType>(inputType) &&
403 isa<SignedType, UnsignedType>(intermediateType))
404 return other.getInput();
406 if (inputBw && *inputBw <= 64 &&
407 isa<IntBaseType, UIntBaseType, SignedType, UnsignedType>(
409 return other.getInput();
416 if (inputBw && intermediateBw && *inputBw <= *intermediateBw)
417 return other.getInput();
428 if (getBody().getNumArguments() != 0)
429 return emitOpError(
"must not have any arguments");
439 setNameFn(getHandle(),
getName());
443 if (getBody().getNumArguments() != 0)
444 return emitOpError(
"must not have any arguments");
454 setNameFn(getInstanceHandle(),
getName());
458 StringAttr InstanceDeclOp::getInstanceNameAttr() {
return getNameAttr(); }
461 InstanceDeclOp::getReferencedModuleCached(
const hw::HWSymbolCache *cache) {
463 if (
auto *result = cache->getDefinition(getModuleNameAttr()))
466 auto topLevelModuleOp = (*this)->getParentOfType<ModuleOp>();
467 return topLevelModuleOp.lookupSymbol(getModuleName());
471 InstanceDeclOp::verifySymbolUses(SymbolTableCollection &symbolTable) {
473 symbolTable.lookupNearestSymbolFrom(*
this, getModuleNameAttr());
474 if (module ==
nullptr)
475 return emitError(
"cannot find module definition '")
476 << getModuleName() <<
"'";
478 auto emitError = [&](
const std::function<void(InFlightDiagnostic & diag)> &fn)
480 auto diag = emitOpError();
482 diag.attachNote(module->getLoc()) <<
"module declared here";
487 if (!isa<SCModuleOp>(module))
488 return emitError([&](
auto &diag) {
489 diag <<
"symbol reference '" << getModuleName()
490 <<
"' isn't a systemc module";
493 auto scModule = cast<SCModuleOp>(module);
496 if (scModule.getModuleName() != getInstanceType().getModuleName())
497 return emitError([&](
auto &diag) {
498 diag <<
"module names must match; expected '" << scModule.getModuleName()
499 <<
"' but got '" << getInstanceType().getModuleName().getValue()
504 ArrayRef<ModuleType::PortInfo> ports = getInstanceType().getPorts();
505 ArrayAttr modArgNames = scModule.getPortNames();
506 auto numPorts = ports.size();
507 auto expectedPortTypes = scModule.getArgumentTypes();
509 if (expectedPortTypes.size() != numPorts)
510 return emitError([&](
auto &diag) {
511 diag <<
"has a wrong number of ports; expected "
512 << expectedPortTypes.size() <<
" but got " << numPorts;
515 for (
size_t i = 0; i != numPorts; ++i) {
516 if (ports[i].type != expectedPortTypes[i]) {
517 return emitError([&](
auto &diag) {
518 diag <<
"port type #" << i <<
" must be " << expectedPortTypes[i]
519 <<
", but got " << ports[i].type;
523 if (ports[i].name != modArgNames[i])
524 return emitError([&](
auto &diag) {
525 diag <<
"port name #" << i <<
" must be " << modArgNames[i]
526 <<
", but got " << ports[i].name;
534 return cast<hw::PortList>(SymbolTable::lookupNearestSymbolFrom(
535 getOperation(), getReferencedModuleNameAttr()))
544 if (getBody().getNumArguments() != 0)
545 return emitOpError(
"must not have any arguments");
554 ParseResult BindPortOp::parse(OpAsmParser &parser, OperationState &result) {
555 OpAsmParser::UnresolvedOperand instance, channel;
556 std::string portName;
557 if (parser.parseOperand(instance) || parser.parseLSquare() ||
558 parser.parseString(&portName))
561 auto portNameLoc = parser.getCurrentLocation();
563 if (parser.parseRSquare() || parser.parseKeyword(
"to") ||
564 parser.parseOperand(channel))
567 if (parser.parseOptionalAttrDict(result.attributes))
570 auto typeListLoc = parser.getCurrentLocation();
571 SmallVector<Type> types;
572 if (parser.parseColonTypeList(types))
575 if (types.size() != 2)
576 return parser.emitError(typeListLoc,
577 "expected a list of exactly 2 types, but got ")
580 if (parser.resolveOperand(instance, types[0], result.operands))
582 if (parser.resolveOperand(channel, types[1], result.operands))
585 if (
auto moduleType = dyn_cast<ModuleType>(types[0])) {
586 auto ports = moduleType.getPorts();
588 for (
auto port : ports) {
589 if (port.name == portName)
593 if (index >= ports.size())
594 return parser.emitError(portNameLoc,
"port name \"")
595 << portName <<
"\" not found in module";
597 result.addAttribute(
"portId", parser.getBuilder().getIndexAttr(index));
605 void BindPortOp::print(OpAsmPrinter &p) {
606 p <<
" " << getInstance() <<
"["
607 << cast<ModuleType>(getInstance().getType())
608 .getPorts()[getPortId().getZExtValue()]
610 <<
"] to " << getChannel();
611 p.printOptionalAttrDict((*this)->getAttrs(), {
"portId"});
612 p <<
" : " << getInstance().getType() <<
", " << getChannel().getType();
616 auto ports = cast<ModuleType>(getInstance().getType()).getPorts();
617 if (getPortId().getZExtValue() >= ports.size())
618 return emitOpError(
"port #")
619 << getPortId().getZExtValue() <<
" does not exist, there are only "
620 << ports.size() <<
" ports";
623 Type portType = ports[getPortId().getZExtValue()].type;
624 Type channelType = getChannel().getType();
626 return emitOpError() << portType <<
" port cannot be bound to "
627 << channelType <<
" channel due to base type mismatch";
630 if ((isa<InputType>(portType) && isa<OutputType>(channelType)) ||
631 (isa<OutputType>(portType) && isa<InputType>(channelType)))
632 return emitOpError() << portType <<
" port cannot be bound to "
634 <<
" channel due to port direction mismatch";
639 StringRef BindPortOp::getPortName() {
640 return cast<ModuleType>(getInstance().getType())
641 .getPorts()[getPortId().getZExtValue()]
650 PatternRewriter &rewriter) {
651 if (op.getSensitivities().empty()) {
652 rewriter.eraseOp(op);
664 setNameFn(getVariable(),
getName());
667 ParseResult VariableOp::parse(OpAsmParser &parser, OperationState &result) {
671 result.addAttribute(
"name", nameAttr);
673 OpAsmParser::UnresolvedOperand init;
674 auto initResult = parser.parseOptionalOperand(init);
676 if (parser.parseOptionalAttrDict(result.attributes))
680 if (parser.parseColonType(variableType))
683 if (initResult.has_value()) {
684 if (parser.resolveOperand(init, variableType, result.operands))
687 result.addTypes({variableType});
692 void VariableOp::print(::mlir::OpAsmPrinter &p) {
696 p << getInit() <<
" ";
698 p.printOptionalAttrDict(getOperation()->getAttrs(), {
"name"});
699 p <<
": " << getVariable().getType();
703 if (getInit() && getInit().getType() != getVariable().getType())
705 "'init' and 'variable' must have the same type, but got ")
706 << getInit().getType() <<
" and " << getVariable().getType();
716 void InteropVerilatedOp::build(OpBuilder &odsBuilder, OperationState &odsState,
717 Operation *module, StringAttr name,
718 ArrayRef<Value> inputs) {
719 auto mod = cast<hw::HWModuleLike>(module);
720 auto argNames = odsBuilder.getArrayAttr(mod.getInputNames());
721 auto resultNames = odsBuilder.getArrayAttr(mod.getOutputNames());
722 build(odsBuilder, odsState, mod.getHWModuleType().getOutputTypes(), name,
724 resultNames, inputs);
728 InteropVerilatedOp::verifySymbolUses(SymbolTableCollection &symbolTable) {
730 *
this, getModuleNameAttr(), getInputs(), getResultTypes(),
731 getInputNames(), getResultNames(), ArrayAttr(), symbolTable);
738 getResultNames(), getResults());
750 LogicalResult CallOp::verifySymbolUses(SymbolTableCollection &symbolTable) {
752 auto fnAttr = (*this)->getAttrOfType<FlatSymbolRefAttr>(
"callee");
754 return emitOpError(
"requires a 'callee' symbol reference attribute");
755 FuncOp fn = symbolTable.lookupNearestSymbolFrom<FuncOp>(*
this, fnAttr);
757 return emitOpError() <<
"'" << fnAttr.getValue()
758 <<
"' does not reference a valid function";
761 auto fnType = fn.getFunctionType();
762 if (fnType.getNumInputs() != getNumOperands())
763 return emitOpError(
"incorrect number of operands for callee");
765 for (
unsigned i = 0, e = fnType.getNumInputs(); i != e; ++i)
766 if (getOperand(i).getType() != fnType.getInput(i))
767 return emitOpError(
"operand type mismatch: expected operand type ")
768 << fnType.getInput(i) <<
", but provided "
769 << getOperand(i).getType() <<
" for operand number " << i;
771 if (fnType.getNumResults() != getNumResults())
772 return emitOpError(
"incorrect number of results for callee");
774 for (
unsigned i = 0, e = fnType.getNumResults(); i != e; ++i)
775 if (getResult(i).getType() != fnType.getResult(i)) {
776 auto diag = emitOpError(
"result type mismatch at index ") << i;
777 diag.attachNote() <<
" op result types: " << getResultTypes();
778 diag.attachNote() <<
"function result types: " << fnType.getResults();
785 FunctionType CallOp::getCalleeType() {
791 if (getNumResults() > 1)
793 "incorrect number of function results (always has to be 0 or 1)");
804 if (getNumResults() > 1)
806 "incorrect number of function results (always has to be 0 or 1)");
821 FuncOp FuncOp::create(Location location, StringRef name, ArrayAttr argNames,
822 FunctionType type, ArrayRef<NamedAttribute> attrs) {
823 OpBuilder builder(location->getContext());
824 OperationState state(location, getOperationName());
825 FuncOp::build(builder, state, name, argNames, type, attrs);
826 return cast<FuncOp>(Operation::create(state));
829 FuncOp FuncOp::create(Location location, StringRef name, ArrayAttr argNames,
830 FunctionType type, Operation::dialect_attr_range attrs) {
831 SmallVector<NamedAttribute, 8> attrRef(attrs);
832 return create(location, name, argNames, type, ArrayRef(attrRef));
835 FuncOp FuncOp::create(Location location, StringRef name, ArrayAttr argNames,
836 FunctionType type, ArrayRef<NamedAttribute> attrs,
837 ArrayRef<DictionaryAttr> argAttrs) {
838 FuncOp func = create(location, name, argNames, type, attrs);
839 func.setAllArgAttrs(argAttrs);
843 void FuncOp::build(OpBuilder &odsBuilder, OperationState &odsState,
844 StringRef name, ArrayAttr argNames, FunctionType type,
845 ArrayRef<NamedAttribute> attrs,
846 ArrayRef<DictionaryAttr> argAttrs) {
847 odsState.addAttribute(getArgNamesAttrName(odsState.name), argNames);
848 odsState.addAttribute(SymbolTable::getSymbolAttrName(),
849 odsBuilder.getStringAttr(name));
850 odsState.addAttribute(FuncOp::getFunctionTypeAttrName(odsState.name),
852 odsState.attributes.append(attrs.begin(), attrs.end());
853 odsState.addRegion();
855 if (argAttrs.empty())
857 assert(type.getNumInputs() == argAttrs.size());
858 mlir::function_interface_impl::addArgAndResultAttrs(
859 odsBuilder, odsState, argAttrs,
860 std::nullopt, FuncOp::getArgAttrsAttrName(odsState.name),
861 FuncOp::getResAttrsAttrName(odsState.name));
864 ParseResult FuncOp::parse(OpAsmParser &parser, OperationState &result) {
866 [](Builder &builder, ArrayRef<Type> argTypes, ArrayRef<Type> results,
867 mlir::function_interface_impl::VariadicFlag,
868 std::string &) {
return builder.getFunctionType(argTypes, results); };
872 if (succeeded(parser.parseOptionalKeyword(
"externC")))
873 result.addAttribute(getExternCAttrName(result.name),
879 SmallVector<OpAsmParser::Argument> entryArgs;
880 SmallVector<DictionaryAttr> resultAttrs;
881 SmallVector<Type> resultTypes;
882 auto &builder = parser.getBuilder();
885 (void)mlir::impl::parseOptionalVisibilityKeyword(parser, result.attributes);
889 if (parser.parseSymbolName(nameAttr, SymbolTable::getSymbolAttrName(),
894 mlir::SMLoc signatureLocation = parser.getCurrentLocation();
895 bool isVariadic =
false;
896 if (mlir::function_interface_impl::parseFunctionSignature(
897 parser,
false, entryArgs, isVariadic, resultTypes, resultAttrs))
900 std::string errorMessage;
901 SmallVector<Type> argTypes;
902 argTypes.reserve(entryArgs.size());
903 for (
auto &arg : entryArgs)
904 argTypes.push_back(arg.type);
906 Type type = buildFuncType(
907 builder, argTypes, resultTypes,
908 mlir::function_interface_impl::VariadicFlag(isVariadic), errorMessage);
910 return parser.emitError(signatureLocation)
911 <<
"failed to construct function type"
912 << (errorMessage.empty() ?
"" :
": ") << errorMessage;
914 result.addAttribute(FuncOp::getFunctionTypeAttrName(result.name),
918 NamedAttrList parsedAttributes;
919 mlir::SMLoc attributeDictLocation = parser.getCurrentLocation();
920 if (parser.parseOptionalAttrDictWithKeyword(parsedAttributes))
925 for (StringRef disallowed :
926 {SymbolTable::getVisibilityAttrName(), SymbolTable::getSymbolAttrName(),
927 FuncOp::getFunctionTypeAttrName(result.name).getValue()}) {
928 if (parsedAttributes.get(disallowed))
929 return parser.emitError(attributeDictLocation,
"'")
931 <<
"' is an inferred attribute and should not be specified in the "
932 "explicit attribute dictionary";
934 result.attributes.append(parsedAttributes);
937 assert(resultAttrs.size() == resultTypes.size());
938 mlir::function_interface_impl::addArgAndResultAttrs(
939 builder, result, entryArgs, resultAttrs,
940 FuncOp::getArgAttrsAttrName(result.name),
941 FuncOp::getResAttrsAttrName(result.name));
945 auto *body = result.addRegion();
946 mlir::SMLoc loc = parser.getCurrentLocation();
947 mlir::OptionalParseResult parseResult =
948 parser.parseOptionalRegion(*body, entryArgs,
950 if (parseResult.has_value()) {
951 if (failed(*parseResult))
955 return parser.emitError(loc,
"expected non-empty function body");
960 SmallVector<Attribute> argNames;
961 if (!entryArgs.empty() && !entryArgs.front().ssaName.name.empty()) {
962 for (
auto &arg : entryArgs)
967 result.addAttribute(getArgNamesAttrName(result.name),
973 void FuncOp::print(OpAsmPrinter &p) {
977 mlir::FunctionOpInterface op = *
this;
984 op->getAttrOfType<StringAttr>(SymbolTable::getSymbolAttrName())
988 StringRef visibilityAttrName = SymbolTable::getVisibilityAttrName();
989 if (
auto visibility = op->getAttrOfType<StringAttr>(visibilityAttrName))
990 p << visibility.getValue() <<
' ';
991 p.printSymbolName(funcName);
993 ArrayRef<Type> argTypes = op.getArgumentTypes();
994 ArrayRef<Type> resultTypes = op.getResultTypes();
995 mlir::function_interface_impl::printFunctionSignature(p, op, argTypes,
false,
997 mlir::function_interface_impl::printFunctionAttributes(
999 {visibilityAttrName,
"externC",
"argNames", getFunctionTypeAttrName(),
1000 getArgAttrsAttrName(), getResAttrsAttrName()});
1002 Region &body = op->getRegion(0);
1003 if (!body.empty()) {
1005 p.printRegion(body,
false,
1014 void FuncOp::cloneInto(FuncOp dest, IRMapping &mapper) {
1016 llvm::MapVector<StringAttr, Attribute> newAttrMap;
1017 for (
const auto &attr : dest->getAttrs())
1018 newAttrMap.insert({attr.getName(), attr.getValue()});
1019 for (
const auto &attr : (*this)->getAttrs())
1020 newAttrMap.insert({attr.getName(), attr.getValue()});
1022 auto newAttrs = llvm::to_vector(llvm::map_range(
1023 newAttrMap, [](std::pair<StringAttr, Attribute> attrPair) {
1024 return NamedAttribute(attrPair.first, attrPair.second);
1029 getBody().cloneInto(&dest.getBody(), mapper);
1037 FuncOp FuncOp::clone(IRMapping &mapper) {
1039 FuncOp newFunc = cast<FuncOp>(getOperation()->cloneWithoutRegions());
1044 if (!isExternal()) {
1045 FunctionType oldType = getFunctionType();
1047 unsigned oldNumArgs = oldType.getNumInputs();
1048 SmallVector<Type, 4> newInputs;
1049 newInputs.reserve(oldNumArgs);
1050 for (
unsigned i = 0; i != oldNumArgs; ++i)
1051 if (!mapper.contains(getArgument(i)))
1052 newInputs.push_back(oldType.getInput(i));
1056 if (newInputs.size() != oldNumArgs) {
1058 oldType.getResults()));
1060 if (ArrayAttr argAttrs = getAllArgAttrs()) {
1061 SmallVector<Attribute> newArgAttrs;
1062 newArgAttrs.reserve(newInputs.size());
1063 for (
unsigned i = 0; i != oldNumArgs; ++i)
1064 if (!mapper.contains(getArgument(i)))
1065 newArgAttrs.push_back(argAttrs[i]);
1066 newFunc.setAllArgAttrs(newArgAttrs);
1072 cloneInto(newFunc, mapper);
1076 FuncOp FuncOp::clone() {
1078 return clone(mapper);
1083 void FuncOp::getAsmBlockArgumentNames(mlir::Region ®ion,
1088 for (
auto [arg, name] : llvm::zip(getArguments(), getArgNames()))
1089 setNameFn(arg, cast<StringAttr>(name).getValue());
1093 if (getFunctionType().getNumResults() > 1)
1095 "incorrect number of function results (always has to be 0 or 1)");
1097 if (getBody().
empty())
1100 if (getArgNames().size() != getFunctionType().getNumInputs())
1101 return emitOpError(
"incorrect number of argument names");
1103 for (
auto portName : getArgNames()) {
1104 if (cast<StringAttr>(portName).getValue().
empty())
1105 return emitOpError(
"arg name must not be empty");
1111 LogicalResult FuncOp::verifyRegions() {
1112 auto attachNote = [&](mlir::InFlightDiagnostic &diag) {
1113 diag.attachNote(getLoc()) <<
"in function '@" <<
getName() <<
"'";
1127 auto function = cast<FuncOp>((*this)->getParentOp());
1130 const auto &results =
function.getFunctionType().getResults();
1131 if (getNumOperands() != results.size())
1132 return emitOpError(
"has ")
1133 << getNumOperands() <<
" operands, but enclosing function (@"
1134 <<
function.getName() <<
") returns " << results.size();
1136 for (
unsigned i = 0, e = results.size(); i != e; ++i)
1137 if (getOperand(i).getType() != results[i])
1138 return emitError() <<
"type of return operand " << i <<
" ("
1139 << getOperand(i).getType()
1140 <<
") doesn't match function result type ("
1141 << results[i] <<
")"
1142 <<
" in function @" <<
function.getName();
1152 #define GET_OP_CLASSES
1153 #include "circt/Dialect/SystemC/SystemC.cpp.inc"
assert(baseType &&"element must be base type")
static SmallVector< PortInfo > getPortList(ModuleTy &mod)
static InstancePath empty
static Block * getBodyBlock(FModuleLike mod)
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)
static LogicalResult canonicalize(Op op, PatternRewriter &rewriter)
static LogicalResult verify(Value clock, bool eventExists, mlir::Location loc)
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.
circt::hw::InOutType InOutType
Type getSignalBaseType(Type type)
Get the type wrapped by a signal or port (in, inout, out) type.
std::optional< size_t > getBitWidth(Type type)
Return the bitwidth of a 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