CIRCT  20.0.0git
ArcOps.cpp
Go to the documentation of this file.
1 //===- ArcOps.cpp ---------------------------------------------------------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 
11 #include "mlir/Dialect/Func/IR/FuncOps.h"
12 #include "mlir/IR/Builders.h"
13 #include "mlir/IR/OpImplementation.h"
14 #include "mlir/IR/PatternMatch.h"
15 #include "mlir/IR/SymbolTable.h"
16 #include "mlir/Interfaces/FunctionImplementation.h"
17 #include "mlir/Interfaces/SideEffectInterfaces.h"
18 #include "llvm/ADT/SmallPtrSet.h"
19 #include "llvm/ADT/TypeSwitch.h"
20 
21 using namespace circt;
22 using namespace arc;
23 using namespace mlir;
24 
25 //===----------------------------------------------------------------------===//
26 // Helpers
27 //===----------------------------------------------------------------------===//
28 
29 static LogicalResult verifyTypeListEquivalence(Operation *op,
30  TypeRange expectedTypeList,
31  TypeRange actualTypeList,
32  StringRef elementName) {
33  if (expectedTypeList.size() != actualTypeList.size())
34  return op->emitOpError("incorrect number of ")
35  << elementName << "s: expected " << expectedTypeList.size()
36  << ", but got " << actualTypeList.size();
37 
38  for (unsigned i = 0, e = expectedTypeList.size(); i != e; ++i) {
39  if (expectedTypeList[i] != actualTypeList[i]) {
40  auto diag = op->emitOpError(elementName)
41  << " type mismatch: " << elementName << " #" << i;
42  diag.attachNote() << "expected type: " << expectedTypeList[i];
43  diag.attachNote() << " actual type: " << actualTypeList[i];
44  return diag;
45  }
46  }
47 
48  return success();
49 }
50 
51 static LogicalResult verifyArcSymbolUse(Operation *op, TypeRange inputs,
52  TypeRange results,
53  SymbolTableCollection &symbolTable) {
54  // Check that the arc attribute was specified.
55  auto arcName = op->getAttrOfType<FlatSymbolRefAttr>("arc");
56  // The arc attribute is verified by the tablegen generated verifier as it is
57  // an ODS defined attribute.
58  assert(arcName && "FlatSymbolRefAttr called 'arc' missing");
59  DefineOp arc = symbolTable.lookupNearestSymbolFrom<DefineOp>(op, arcName);
60  if (!arc)
61  return op->emitOpError() << "`" << arcName.getValue()
62  << "` does not reference a valid `arc.define`";
63 
64  // Verify that the operand and result types match the arc.
65  auto type = arc.getFunctionType();
66  if (failed(
67  verifyTypeListEquivalence(op, type.getInputs(), inputs, "operand")))
68  return failure();
69 
70  if (failed(
71  verifyTypeListEquivalence(op, type.getResults(), results, "result")))
72  return failure();
73 
74  return success();
75 }
76 
77 static bool isSupportedModuleOp(Operation *moduleOp) {
78  return llvm::isa<arc::ModelOp, hw::HWModuleLike>(moduleOp);
79 }
80 
81 /// Fetches the operation pointed to by `pointing` with name `symbol`, checking
82 /// that it is a supported model operation for simulation.
83 static Operation *getSupportedModuleOp(SymbolTableCollection &symbolTable,
84  Operation *pointing, StringAttr symbol) {
85  Operation *moduleOp = symbolTable.lookupNearestSymbolFrom(pointing, symbol);
86  if (!moduleOp) {
87  pointing->emitOpError("model not found");
88  return nullptr;
89  }
90 
91  if (!isSupportedModuleOp(moduleOp)) {
92  pointing->emitOpError("model symbol does not point to a supported model "
93  "operation, points to ")
94  << moduleOp->getName() << " instead";
95  return nullptr;
96  }
97 
98  return moduleOp;
99 }
100 
101 static std::optional<hw::ModulePort> getModulePort(Operation *moduleOp,
102  StringRef portName) {
103  auto findRightPort = [&](auto ports) -> std::optional<hw::ModulePort> {
104  const hw::ModulePort *port = llvm::find_if(
105  ports, [&](hw::ModulePort port) { return port.name == portName; });
106  if (port == ports.end())
107  return std::nullopt;
108  return *port;
109  };
110 
111  return TypeSwitch<Operation *, std::optional<hw::ModulePort>>(moduleOp)
112  .Case<arc::ModelOp>(
113  [&](arc::ModelOp modelOp) -> std::optional<hw::ModulePort> {
114  return findRightPort(modelOp.getIo().getPorts());
115  })
116  .Case<hw::HWModuleLike>(
117  [&](hw::HWModuleLike moduleLike) -> std::optional<hw::ModulePort> {
118  return findRightPort(moduleLike.getPortList());
119  })
120  .Default([](Operation *) { return std::nullopt; });
121 }
122 
123 //===----------------------------------------------------------------------===//
124 // DefineOp
125 //===----------------------------------------------------------------------===//
126 
127 ParseResult DefineOp::parse(OpAsmParser &parser, OperationState &result) {
128  auto buildFuncType =
129  [](Builder &builder, ArrayRef<Type> argTypes, ArrayRef<Type> results,
130  function_interface_impl::VariadicFlag,
131  std::string &) { return builder.getFunctionType(argTypes, results); };
132 
133  return function_interface_impl::parseFunctionOp(
134  parser, result, /*allowVariadic=*/false,
135  getFunctionTypeAttrName(result.name), buildFuncType,
136  getArgAttrsAttrName(result.name), getResAttrsAttrName(result.name));
137 }
138 
139 void DefineOp::print(OpAsmPrinter &p) {
140  function_interface_impl::printFunctionOp(
141  p, *this, /*isVariadic=*/false, "function_type", getArgAttrsAttrName(),
142  getResAttrsAttrName());
143 }
144 
145 LogicalResult DefineOp::verifyRegions() {
146  // Check that the body does not contain any side-effecting operations. We can
147  // simply iterate over the ops directly within the body; operations with
148  // regions, like scf::IfOp, implement the `HasRecursiveMemoryEffects` trait
149  // which causes the `isMemoryEffectFree` check to already recur into their
150  // regions.
151  for (auto &op : getBodyBlock()) {
152  if (isMemoryEffectFree(&op))
153  continue;
154 
155  // We don't use a op-error here because that leads to the whole arc being
156  // printed. This can be switched of when creating the context, but one
157  // might not want to switch that off for other error messages. Here it's
158  // definitely not desirable as arcs can be very big and would fill up the
159  // error log, making it hard to read. Currently, only the signature (first
160  // line) of the arc is printed.
161  auto diag = mlir::emitError(getLoc(), "body contains non-pure operation");
162  diag.attachNote(op.getLoc()).append("first non-pure operation here: ");
163  return diag;
164  }
165  return success();
166 }
167 
168 bool DefineOp::isPassthrough() {
169  if (getNumArguments() != getNumResults())
170  return false;
171 
172  return llvm::all_of(
173  llvm::zip(getArguments(), getBodyBlock().getTerminator()->getOperands()),
174  [](const auto &argAndRes) {
175  return std::get<0>(argAndRes) == std::get<1>(argAndRes);
176  });
177 }
178 
179 //===----------------------------------------------------------------------===//
180 // OutputOp
181 //===----------------------------------------------------------------------===//
182 
183 LogicalResult OutputOp::verify() {
184  auto *parent = (*this)->getParentOp();
185  TypeRange expectedTypes = parent->getResultTypes();
186  if (auto defOp = dyn_cast<DefineOp>(parent))
187  expectedTypes = defOp.getResultTypes();
188 
189  TypeRange actualTypes = getOperands().getTypes();
190  return verifyTypeListEquivalence(*this, expectedTypes, actualTypes, "output");
191 }
192 
193 //===----------------------------------------------------------------------===//
194 // StateOp
195 //===----------------------------------------------------------------------===//
196 
197 LogicalResult StateOp::verifySymbolUses(SymbolTableCollection &symbolTable) {
198  return verifyArcSymbolUse(*this, getInputs().getTypes(),
199  getResults().getTypes(), symbolTable);
200 }
201 
202 LogicalResult StateOp::verify() {
203  if (getLatency() < 1)
204  return emitOpError("latency must be a positive integer");
205 
206  if (!getOperation()->getParentOfType<ClockDomainOp>() && !getClock())
207  return emitOpError("outside a clock domain requires a clock");
208 
209  if (getOperation()->getParentOfType<ClockDomainOp>() && getClock())
210  return emitOpError("inside a clock domain cannot have a clock");
211 
212  return success();
213 }
214 
215 //===----------------------------------------------------------------------===//
216 // CallOp
217 //===----------------------------------------------------------------------===//
218 
219 LogicalResult CallOp::verifySymbolUses(SymbolTableCollection &symbolTable) {
220  return verifyArcSymbolUse(*this, getInputs().getTypes(),
221  getResults().getTypes(), symbolTable);
222 }
223 
224 bool CallOp::isClocked() { return false; }
225 
226 Value CallOp::getClock() { return Value{}; }
227 
228 void CallOp::eraseClock() {}
229 
230 uint32_t CallOp::getLatency() { return 0; }
231 
232 //===----------------------------------------------------------------------===//
233 // MemoryWritePortOp
234 //===----------------------------------------------------------------------===//
235 
236 SmallVector<Type> MemoryWritePortOp::getArcResultTypes() {
237  auto memType = cast<MemoryType>(getMemory().getType());
238  SmallVector<Type> resultTypes{memType.getAddressType(),
239  memType.getWordType()};
240  if (getEnable())
241  resultTypes.push_back(IntegerType::get(getContext(), 1));
242  if (getMask())
243  resultTypes.push_back(memType.getWordType());
244  return resultTypes;
245 }
246 
247 LogicalResult
248 MemoryWritePortOp::verifySymbolUses(SymbolTableCollection &symbolTable) {
249  return verifyArcSymbolUse(*this, getInputs().getTypes(), getArcResultTypes(),
250  symbolTable);
251 }
252 
253 LogicalResult MemoryWritePortOp::verify() {
254  if (getLatency() < 1)
255  return emitOpError("latency must be at least 1");
256 
257  if (!getOperation()->getParentOfType<ClockDomainOp>() && !getClock())
258  return emitOpError("outside a clock domain requires a clock");
259 
260  if (getOperation()->getParentOfType<ClockDomainOp>() && getClock())
261  return emitOpError("inside a clock domain cannot have a clock");
262 
263  return success();
264 }
265 
266 //===----------------------------------------------------------------------===//
267 // ClockDomainOp
268 //===----------------------------------------------------------------------===//
269 
270 LogicalResult ClockDomainOp::verifyRegions() {
271  return verifyTypeListEquivalence(*this, getBodyBlock().getArgumentTypes(),
272  getInputs().getTypes(), "input");
273 }
274 
275 //===----------------------------------------------------------------------===//
276 // RootInputOp
277 //===----------------------------------------------------------------------===//
278 
280  SmallString<32> buf("in_");
281  buf += getName();
282  setNameFn(getState(), buf);
283 }
284 
285 //===----------------------------------------------------------------------===//
286 // RootOutputOp
287 //===----------------------------------------------------------------------===//
288 
290  SmallString<32> buf("out_");
291  buf += getName();
292  setNameFn(getState(), buf);
293 }
294 
295 //===----------------------------------------------------------------------===//
296 // ModelOp
297 //===----------------------------------------------------------------------===//
298 
299 LogicalResult ModelOp::verify() {
300  if (getBodyBlock().getArguments().size() != 1)
301  return emitOpError("must have exactly one argument");
302  if (auto type = getBodyBlock().getArgument(0).getType();
303  !isa<StorageType>(type))
304  return emitOpError("argument must be of storage type");
305  for (const hw::ModulePort &port : getIo().getPorts())
306  if (port.dir == hw::ModulePort::Direction::InOut)
307  return emitOpError("inout ports are not supported");
308  return success();
309 }
310 
311 LogicalResult ModelOp::verifySymbolUses(SymbolTableCollection &symbolTable) {
312  if (!getInitialFn().has_value())
313  return success();
314 
315  auto referencedOp =
316  symbolTable.lookupNearestSymbolFrom(*this, getInitialFnAttr());
317  if (!referencedOp)
318  return emitError("Cannot find declaration of initializer function '")
319  << *getInitialFn() << "'.";
320  auto funcOp = dyn_cast<func::FuncOp>(referencedOp);
321  if (!funcOp) {
322  auto diag = emitError("Referenced initializer must be a 'func.func' op.");
323  diag.attachNote(referencedOp->getLoc()) << "Initializer declared here:";
324  return diag;
325  }
326  if (!llvm::equal(funcOp.getArgumentTypes(), getBody().getArgumentTypes())) {
327  auto diag = emitError("Arguments of initializer function must match "
328  "arguments of model body.");
329  diag.attachNote(referencedOp->getLoc()) << "Initializer declared here:";
330  return diag;
331  }
332  return success();
333 }
334 
335 //===----------------------------------------------------------------------===//
336 // LutOp
337 //===----------------------------------------------------------------------===//
338 
339 LogicalResult LutOp::verify() {
340  Location firstSideEffectOpLoc = UnknownLoc::get(getContext());
341  const WalkResult result = getBody().walk([&](Operation *op) {
342  if (auto memOp = dyn_cast<MemoryEffectOpInterface>(op)) {
343  SmallVector<SideEffects::EffectInstance<MemoryEffects::Effect>> effects;
344  memOp.getEffects(effects);
345 
346  if (!effects.empty()) {
347  firstSideEffectOpLoc = memOp->getLoc();
348  return WalkResult::interrupt();
349  }
350  }
351 
352  return WalkResult::advance();
353  });
354 
355  if (result.wasInterrupted())
356  return emitOpError("no operations with side-effects allowed inside a LUT")
357  .attachNote(firstSideEffectOpLoc)
358  << "first operation with side-effects here";
359 
360  return success();
361 }
362 
363 //===----------------------------------------------------------------------===//
364 // VectorizeOp
365 //===----------------------------------------------------------------------===//
366 
367 LogicalResult VectorizeOp::verify() {
368  if (getInputs().empty())
369  return emitOpError("there has to be at least one input vector");
370 
371  if (!llvm::all_equal(llvm::map_range(
372  getInputs(), [](OperandRange range) { return range.size(); })))
373  return emitOpError("all input vectors must have the same size");
374 
375  for (OperandRange range : getInputs()) {
376  if (!llvm::all_equal(range.getTypes()))
377  return emitOpError("all input vector lane types must match");
378 
379  if (range.empty())
380  return emitOpError("input vector must have at least one element");
381  }
382 
383  if (getResults().empty())
384  return emitOpError("must have at least one result");
385 
386  if (!llvm::all_equal(getResults().getTypes()))
387  return emitOpError("all result types must match");
388 
389  if (getResults().size() != getInputs().front().size())
390  return emitOpError("number results must match input vector size");
391 
392  return success();
393 }
394 
395 static FailureOr<unsigned> getVectorWidth(Type base, Type vectorized) {
396  if (isa<VectorType>(base))
397  return failure();
398 
399  if (auto vectorTy = dyn_cast<VectorType>(vectorized)) {
400  if (vectorTy.getElementType() != base)
401  return failure();
402 
403  return vectorTy.getDimSize(0);
404  }
405 
406  if (vectorized.getIntOrFloatBitWidth() < base.getIntOrFloatBitWidth())
407  return failure();
408 
409  if (vectorized.getIntOrFloatBitWidth() % base.getIntOrFloatBitWidth() == 0)
410  return vectorized.getIntOrFloatBitWidth() / base.getIntOrFloatBitWidth();
411 
412  return failure();
413 }
414 
415 LogicalResult VectorizeOp::verifyRegions() {
416  auto returnOp = cast<VectorizeReturnOp>(getBody().front().getTerminator());
417  TypeRange bodyArgTypes = getBody().front().getArgumentTypes();
418 
419  if (bodyArgTypes.size() != getInputs().size())
420  return emitOpError(
421  "number of block arguments must match number of input vectors");
422 
423  // Boundary and body are vectorized, or both are not vectorized
424  if (returnOp.getValue().getType() == getResultTypes().front()) {
425  for (auto [i, argTy] : llvm::enumerate(bodyArgTypes))
426  if (argTy != getInputs()[i].getTypes().front())
427  return emitOpError("if terminator type matches result type the "
428  "argument types must match the input types");
429 
430  return success();
431  }
432 
433  // Boundary is vectorized, body is not
434  if (auto width = getVectorWidth(returnOp.getValue().getType(),
435  getResultTypes().front());
436  succeeded(width)) {
437  for (auto [i, argTy] : llvm::enumerate(bodyArgTypes)) {
438  Type inputTy = getInputs()[i].getTypes().front();
439  FailureOr<unsigned> argWidth = getVectorWidth(argTy, inputTy);
440  if (failed(argWidth))
441  return emitOpError("block argument must be a scalar variant of the "
442  "vectorized operand");
443 
444  if (*argWidth != width)
445  return emitOpError("input and output vector width must match");
446  }
447 
448  return success();
449  }
450 
451  // Body is vectorized, boundary is not
452  if (auto width = getVectorWidth(getResultTypes().front(),
453  returnOp.getValue().getType());
454  succeeded(width)) {
455  for (auto [i, argTy] : llvm::enumerate(bodyArgTypes)) {
456  Type inputTy = getInputs()[i].getTypes().front();
457  FailureOr<unsigned> argWidth = getVectorWidth(inputTy, argTy);
458  if (failed(argWidth))
459  return emitOpError(
460  "block argument must be a vectorized variant of the operand");
461 
462  if (*argWidth != width)
463  return emitOpError("input and output vector width must match");
464 
465  if (getInputs()[i].size() > 1 && argWidth != getInputs()[i].size())
466  return emitOpError(
467  "when boundary not vectorized the number of vector element "
468  "operands must match the width of the vectorized body");
469  }
470 
471  return success();
472  }
473 
474  return returnOp.emitOpError(
475  "operand type must match parent op's result value or be a vectorized or "
476  "non-vectorized variant of it");
477 }
478 
479 bool VectorizeOp::isBoundaryVectorized() {
480  return getInputs().front().size() == 1;
481 }
482 bool VectorizeOp::isBodyVectorized() {
483  auto returnOp = cast<VectorizeReturnOp>(getBody().front().getTerminator());
484  if (isBoundaryVectorized() &&
485  returnOp.getValue().getType() == getResultTypes().front())
486  return true;
487 
488  if (auto width = getVectorWidth(getResultTypes().front(),
489  returnOp.getValue().getType());
490  succeeded(width))
491  return *width > 1;
492 
493  return false;
494 }
495 
496 //===----------------------------------------------------------------------===//
497 // SimInstantiateOp
498 //===----------------------------------------------------------------------===//
499 
500 void SimInstantiateOp::print(OpAsmPrinter &p) {
501  BlockArgument modelArg = getBody().getArgument(0);
502  auto modelType = cast<SimModelInstanceType>(modelArg.getType());
503 
504  p << " " << modelType.getModel() << " as ";
505  p.printRegionArgument(modelArg, {}, true);
506 
507  p.printOptionalAttrDictWithKeyword(getOperation()->getAttrs());
508 
509  p << " ";
510 
511  p.printRegion(getBody(), false);
512 }
513 
514 ParseResult SimInstantiateOp::parse(OpAsmParser &parser,
515  OperationState &result) {
516  StringAttr modelName;
517  if (failed(parser.parseSymbolName(modelName)))
518  return failure();
519 
520  if (failed(parser.parseKeyword("as")))
521  return failure();
522 
523  OpAsmParser::Argument modelArg;
524  if (failed(parser.parseArgument(modelArg, false, false)))
525  return failure();
526 
527  if (failed(parser.parseOptionalAttrDictWithKeyword(result.attributes)))
528  return failure();
529 
530  MLIRContext *ctxt = result.getContext();
531  modelArg.type =
533 
534  std::unique_ptr<Region> body = std::make_unique<Region>();
535  if (failed(parser.parseRegion(*body, {modelArg})))
536  return failure();
537 
538  result.addRegion(std::move(body));
539  return success();
540 }
541 
542 LogicalResult SimInstantiateOp::verifyRegions() {
543  Region &body = getBody();
544  if (body.getNumArguments() != 1)
545  return emitError("entry block of body region must have the model instance "
546  "as a single argument");
547  if (!llvm::isa<SimModelInstanceType>(body.getArgument(0).getType()))
548  return emitError("entry block argument type is not a model instance");
549  return success();
550 }
551 
552 LogicalResult
553 SimInstantiateOp::verifySymbolUses(SymbolTableCollection &symbolTable) {
554  Operation *moduleOp = getSupportedModuleOp(
555  symbolTable, getOperation(),
556  llvm::cast<SimModelInstanceType>(getBody().getArgument(0).getType())
557  .getModel()
558  .getAttr());
559  if (!moduleOp)
560  return failure();
561 
562  return success();
563 }
564 
565 //===----------------------------------------------------------------------===//
566 // SimSetInputOp
567 //===----------------------------------------------------------------------===//
568 
569 LogicalResult
570 SimSetInputOp::verifySymbolUses(SymbolTableCollection &symbolTable) {
571  Operation *moduleOp = getSupportedModuleOp(
572  symbolTable, getOperation(),
573  llvm::cast<SimModelInstanceType>(getInstance().getType())
574  .getModel()
575  .getAttr());
576  if (!moduleOp)
577  return failure();
578 
579  std::optional<hw::ModulePort> port = getModulePort(moduleOp, getInput());
580  if (!port)
581  return emitOpError("port not found on model");
582 
583  if (port->dir != hw::ModulePort::Direction::Input &&
585  return emitOpError("port is not an input port");
586 
587  if (port->type != getValue().getType())
588  return emitOpError(
589  "mismatched types between value and model port, port expects ")
590  << port->type;
591 
592  return success();
593 }
594 
595 //===----------------------------------------------------------------------===//
596 // SimGetPortOp
597 //===----------------------------------------------------------------------===//
598 
599 LogicalResult
600 SimGetPortOp::verifySymbolUses(SymbolTableCollection &symbolTable) {
601  Operation *moduleOp = getSupportedModuleOp(
602  symbolTable, getOperation(),
603  llvm::cast<SimModelInstanceType>(getInstance().getType())
604  .getModel()
605  .getAttr());
606  if (!moduleOp)
607  return failure();
608 
609  std::optional<hw::ModulePort> port = getModulePort(moduleOp, getPort());
610  if (!port)
611  return emitOpError("port not found on model");
612 
613  if (port->type != getValue().getType())
614  return emitOpError(
615  "mismatched types between value and model port, port expects ")
616  << port->type;
617 
618  return success();
619 }
620 
621 //===----------------------------------------------------------------------===//
622 // SimStepOp
623 //===----------------------------------------------------------------------===//
624 
625 LogicalResult SimStepOp::verifySymbolUses(SymbolTableCollection &symbolTable) {
626  Operation *moduleOp = getSupportedModuleOp(
627  symbolTable, getOperation(),
628  llvm::cast<SimModelInstanceType>(getInstance().getType())
629  .getModel()
630  .getAttr());
631  if (!moduleOp)
632  return failure();
633 
634  return success();
635 }
636 
637 #include "circt/Dialect/Arc/ArcInterfaces.cpp.inc"
638 
639 #define GET_OP_CLASSES
640 #include "circt/Dialect/Arc/Arc.cpp.inc"
static bool isSupportedModuleOp(Operation *moduleOp)
Definition: ArcOps.cpp:77
static LogicalResult verifyArcSymbolUse(Operation *op, TypeRange inputs, TypeRange results, SymbolTableCollection &symbolTable)
Definition: ArcOps.cpp:51
static LogicalResult verifyTypeListEquivalence(Operation *op, TypeRange expectedTypeList, TypeRange actualTypeList, StringRef elementName)
Definition: ArcOps.cpp:29
static FailureOr< unsigned > getVectorWidth(Type base, Type vectorized)
Definition: ArcOps.cpp:395
static Operation * getSupportedModuleOp(SymbolTableCollection &symbolTable, Operation *pointing, StringAttr symbol)
Fetches the operation pointed to by pointing with name symbol, checking that it is a supported model ...
Definition: ArcOps.cpp:83
static std::optional< hw::ModulePort > getModulePort(Operation *moduleOp, StringRef portName)
Definition: ArcOps.cpp:101
assert(baseType &&"element must be base type")
static Attribute getAttr(ArrayRef< NamedAttribute > attrs, StringRef name)
Get an attribute by name from a list of named attributes.
Definition: FIRRTLOps.cpp:3962
int32_t width
Definition: FIRRTL.cpp:36
static PortInfo getPort(ModuleTy &mod, size_t idx)
Definition: HWOps.cpp:1434
@ Input
Definition: HW.h:35
@ InOut
Definition: HW.h:35
static InstancePath empty
static Block * getBodyBlock(FModuleLike mod)
static LogicalResult verify(Value clock, bool eventExists, mlir::Location loc)
Definition: SVOps.cpp:2443
Direction get(bool isOutput)
Returns an output direction if isOutput is true, otherwise returns an input direction.
Definition: CalyxOps.cpp:55
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.
Definition: DebugAnalysis.h:21
function_ref< void(Value, StringRef)> OpAsmSetValueNameFn
Definition: LLVM.h:182