19#include "mlir/IR/DialectImplementation.h"
20#include "mlir/IR/OpDefinition.h"
21#include "llvm/ADT/SmallString.h"
22#include "llvm/ADT/TypeSwitch.h"
26using namespace chirrtl;
27using namespace firrtl;
34 NamedAttrList &resultAttrs) {
36 auto result = parser.parseOptionalAttrDict(resultAttrs);
37 if (!resultAttrs.get(
"annotations"))
38 resultAttrs.append(
"annotations", parser.getBuilder().getArrayAttr({}));
42 if (resultAttrs.get(
"name"))
45 auto resultName = parser.getResultName(0).first;
46 if (!resultName.empty() &&
isdigit(resultName[0]))
48 auto nameAttr = parser.getBuilder().getStringAttr(resultName);
49 auto *context = parser.getBuilder().getContext();
50 resultAttrs.push_back({StringAttr::get(context,
"name"), nameAttr});
54static void printCHIRRTLOp(OpAsmPrinter &p, Operation *op, DictionaryAttr attr,
55 ArrayRef<StringRef> extraElides = {}) {
56 SmallVector<StringRef> elides(extraElides.begin(), extraElides.end());
64 SmallString<32> resultNameStr;
65 llvm::raw_svector_ostream tmpStream(resultNameStr);
66 p.printOperand(op->getResult(0), tmpStream);
67 auto actualName = tmpStream.str().drop_front();
68 auto expectedName = op->getAttrOfType<StringAttr>(
"name").getValue();
70 if (actualName == expectedName ||
71 (expectedName.empty() &&
isdigit(actualName[0])))
72 elides.push_back(
"name");
73 elides.push_back(
"nameKind");
76 if (op->getAttrOfType<ArrayAttr>(
"annotations").empty())
77 elides.push_back(
"annotations");
79 p.printOptionalAttrDict(op->getAttrs(), elides);
87 firrtl::NameKindEnumAttr &result) {
90 if (!parser.parseOptionalKeyword(&keyword,
91 {
"interesting_name",
"droppable_name"})) {
92 auto kind = symbolizeNameKindEnum(keyword);
93 result = NameKindEnumAttr::get(parser.getContext(), kind.value());
99 NameKindEnumAttr::get(parser.getContext(), NameKindEnum::DroppableName);
104 firrtl::NameKindEnumAttr attr,
105 ArrayRef<StringRef> extraElides = {}) {
106 if (attr.getValue() != NameKindEnum::DroppableName)
107 p <<
" " << stringifyNameKindEnum(attr.getValue());
114void MemoryPortOp::build(OpBuilder &builder, OperationState &result,
115 Type dataType, Value memory, MemDirAttr direction,
116 StringRef name, ArrayRef<Attribute> annotations) {
117 build(builder, result, CMemoryPortType::get(builder.getContext()), dataType,
118 memory, direction, name, builder.getArrayAttr(annotations));
121LogicalResult MemoryPortOp::inferReturnTypes(
122 MLIRContext *context, std::optional<Location> loc, ValueRange operands,
123 DictionaryAttr attrs, mlir::OpaqueProperties properties,
124 mlir::RegionRange regions, SmallVectorImpl<Type> &results) {
125 auto inType = operands[0].getType();
126 auto memType = type_dyn_cast<CMemoryType>(inType);
129 mlir::emitError(*loc,
"memory port requires memory operand");
132 results.push_back(memType.getElementType());
133 results.push_back(CMemoryPortType::get(context));
137LogicalResult MemoryPortOp::verify() {
141 return emitOpError(
"port should be used by a chirrtl.memoryport.access");
145MemoryPortAccessOp MemoryPortOp::getAccess() {
146 auto uses =
getPort().use_begin();
147 if (uses ==
getPort().use_end())
149 return cast<MemoryPortAccessOp>(uses->getOwner());
152void MemoryPortOp::getAsmResultNames(
153 function_ref<
void(Value, StringRef)> setNameFn) {
157 setNameFn(getData(), (base +
"_data").str());
158 setNameFn(
getPort(), (base +
"_port").str());
162 NamedAttrList &resultAttrs) {
164 auto result = parser.parseOptionalAttrDict(resultAttrs);
165 if (!resultAttrs.get(
"annotations"))
166 resultAttrs.append(
"annotations", parser.getBuilder().getArrayAttr({}));
173 DictionaryAttr attr) {
175 SmallVector<StringRef> elides = {
"direction"};
177 if (op->getAttrOfType<ArrayAttr>(
"annotations").empty())
178 elides.push_back(
"annotations");
179 p.printOptionalAttrDict(op->getAttrs(), elides);
186void MemoryDebugPortOp::build(OpBuilder &builder, OperationState &result,
187 Type dataType, Value memory, StringRef name,
188 ArrayRef<Attribute> annotations) {
189 build(builder, result, dataType, memory, name,
190 builder.getArrayAttr(annotations));
193LogicalResult MemoryDebugPortOp::inferReturnTypes(
194 MLIRContext *context, std::optional<Location> loc, ValueRange operands,
195 DictionaryAttr attrs, mlir::OpaqueProperties properties,
196 mlir::RegionRange regions, SmallVectorImpl<Type> &results) {
197 auto inType = operands[0].getType();
198 auto memType = type_dyn_cast<CMemoryType>(inType);
201 mlir::emitError(*loc,
"memory port requires memory operand");
204 results.push_back(RefType::get(
205 FVectorType::get(memType.getElementType(), memType.getNumElements())));
209void MemoryDebugPortOp::getAsmResultNames(
210 function_ref<
void(Value, StringRef)> setNameFn) {
214 setNameFn(getData(), (base +
"_data").str());
218 NamedAttrList &resultAttrs) {
220 auto result = parser.parseOptionalAttrDict(resultAttrs);
221 if (!resultAttrs.get(
"annotations"))
222 resultAttrs.append(
"annotations", parser.getBuilder().getArrayAttr({}));
229 DictionaryAttr attr) {
230 SmallVector<StringRef, 1> elides;
232 if (op->getAttrOfType<ArrayAttr>(
"annotations").empty())
233 elides.push_back(
"annotations");
234 p.printOptionalAttrDict(op->getAttrs(), elides);
242 NamedAttrList &resultAttrs) {
247 DictionaryAttr attr) {
251void CombMemOp::build(OpBuilder &builder, OperationState &result,
253 StringRef name, NameKindEnum nameKind,
254 ArrayAttr annotations, StringAttr innerSym,
255 MemoryInitAttr init) {
256 build(builder, result,
258 nameKind, annotations,
259 innerSym ? hw::InnerSymAttr::get(innerSym) :
hw::InnerSymAttr(), init,
264 setNameFn(getResult(),
getName());
267std::optional<size_t> CombMemOp::getTargetResultIndex() {
277 NamedAttrList &resultAttrs) {
282static void printSeqMemOp(OpAsmPrinter &p, Operation *op, DictionaryAttr attr) {
286void SeqMemOp::build(OpBuilder &builder, OperationState &result,
288 RUWAttr ruw, StringRef name, NameKindEnum nameKind,
289 ArrayAttr annotations, StringAttr innerSym,
290 MemoryInitAttr init) {
291 build(builder, result,
293 name, nameKind, annotations,
294 innerSym ? hw::InnerSymAttr::get(innerSym) :
hw::InnerSymAttr(), init,
299 setNameFn(getResult(),
getName());
302std::optional<size_t> SeqMemOp::getTargetResultIndex() {
314struct CHIRRTLOpAsmDialectInterface :
public OpAsmDialectInterface {
315 using OpAsmDialectInterface::OpAsmDialectInterface;
319 if (op->getNumResults() == 1)
320 if (
auto nameAttr = op->getAttrOfType<StringAttr>(
"name"))
321 setNameFn(op->getResult(0), nameAttr.getValue());
326#define GET_OP_CLASSES
327#include "circt/Dialect/FIRRTL/CHIRRTL.cpp.inc"
329void CHIRRTLDialect::initialize() {
333#include "circt/Dialect/FIRRTL/CHIRRTL.cpp.inc"
340 addInterfaces<CHIRRTLOpAsmDialectInterface>();
343#include "circt/Dialect/FIRRTL/CHIRRTLDialect.cpp.inc"
static void printSeqMemOp(OpAsmPrinter &p, Operation *op, DictionaryAttr attr)
Always elide "ruw" and elide "annotations" if it exists or if it is empty.
static void printNameKind(OpAsmPrinter &p, Operation *op, firrtl::NameKindEnumAttr attr, ArrayRef< StringRef > extraElides={})
static void printCHIRRTLOp(OpAsmPrinter &p, Operation *op, DictionaryAttr attr, ArrayRef< StringRef > extraElides={})
static ParseResult parseMemoryDebugPortOp(OpAsmParser &parser, NamedAttrList &resultAttrs)
static ParseResult parseNameKind(OpAsmParser &parser, firrtl::NameKindEnumAttr &result)
static void printMemoryPortOp(OpAsmPrinter &p, Operation *op, DictionaryAttr attr)
Always elide "direction" and elide "annotations" if it exists or if it is empty.
static ParseResult parseSeqMemOp(OpAsmParser &parser, NamedAttrList &resultAttrs)
static ParseResult parseCombMemOp(OpAsmParser &parser, NamedAttrList &resultAttrs)
static ParseResult parseCHIRRTLOp(OpAsmParser &parser, NamedAttrList &resultAttrs)
static void printCombMemOp(OpAsmPrinter &p, Operation *op, DictionaryAttr attr)
static void printMemoryDebugPortOp(OpAsmPrinter &p, Operation *op, DictionaryAttr attr)
Always elide "direction" and elide "annotations" if it exists or if it is empty.
static ParseResult parseMemoryPortOp(OpAsmParser &parser, NamedAttrList &resultAttrs)
MlirType uint64_t numElements
static PortInfo getPort(ModuleTy &mod, size_t idx)
static StringRef getInnerSymbolAttrName()
Return the name of the attribute used for inner symbol names.
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.
The InstanceGraph op interface, see InstanceGraphInterface.td for more details.
function_ref< void(Value, StringRef)> OpAsmSetValueNameFn