13#ifndef CIRCT_DIALECT_FIRRTL_FIRRTLANNOTATIONHELPER_H
14#define CIRCT_DIALECT_FIRRTL_FIRRTLANNOTATIONHELPER_H
21#include "llvm/ADT/TypeSwitch.h"
35 SmallVector<std::pair<StringRef, StringRef>>
instances;
43 void toVector(SmallVectorImpl<char> &out)
const;
46 std::string
str()
const {
49 return std::string(out);
68 template <
typename... T>
70 if (
auto opRef = dyn_cast<OpAnnoTarget>(
ref))
71 return isa<T...>(opRef.getOp());
78 os <<
"~" << path.
ref.
getModule()->getParentOfType<CircuitOp>().getName()
85 ->getParentOfType<FModuleLike>()
89 os <<
"/" << inst.getName() <<
":" << inst.getModuleName();
90 if (!path.
isOpOfType<FModuleOp, FExtModuleOp, InstanceOp>()) {
91 os <<
">" << path.
ref;
92 auto type = dyn_cast<FIRRTLBaseType>(path.
ref.
getType());
96 while (targetFieldID) {
98 .
Case<FVectorType>([&](FVectorType vector) {
99 auto index = vector.getIndexForFieldID(targetFieldID);
100 os <<
"[" << index <<
"]";
101 type = vector.getElementType();
102 targetFieldID -= vector.getFieldID(index);
104 .
template Case<BundleType>([&](BundleType bundle) {
105 auto index = bundle.getIndexForFieldID(targetFieldID);
106 os <<
"." << bundle.getElementName(index);
107 type = bundle.getElementType(index);
108 targetFieldID -= bundle.getFieldID(index);
110 .Default([&](
auto) { targetFieldID = 0; });
118 os << target.
getOp()->getAttrOfType<StringAttr>(
"name").getValue();
130 if (
auto op = dyn_cast<OpAnnoTarget>(target))
132 else if (
auto port = dyn_cast<PortAnnoTarget>(target))
135 os <<
"<<Unknown Anno Target>>";
155 TypeSwitch<Operation *>(op)
156 .Case<InstanceOp, MemOp, NodeOp, RegOp, RegResetOp, WireOp,
157 chirrtl::CombMemOp, chirrtl::SeqMemOp, chirrtl::MemoryPortOp,
158 chirrtl::MemoryDebugPortOp, PrintFOp>([&](
auto op) {
160 if (
auto name = op.getNameAttr(); name && !name.getValue().empty())
168 if (
auto name = oldOp->getAttrOfType<StringAttr>(
"name");
169 name && !name.getValue().empty())
176 targets.insert({mod.getPortNameAttr(portNo),
PortAnnoTarget(mod, portNo)});
181 void gatherTargets(FModuleLike mod);
183 llvm::DenseMap<StringRef, AnnoTarget>
targets;
191 auto it = targetCaches.find(module);
192 if (it == targetCaches.end())
193 it = targetCaches.try_emplace(module, module).first;
199 return getOrCreateCacheFor(module).getTargetForName(name);
208 auto mod = newOp->getParentOfType<FModuleOp>();
209 auto it = targetCaches.find(mod);
210 if (it == targetCaches.end())
212 it->getSecond().replaceOp(oldOp, newOp);
217 auto it = targetCaches.find(mod);
218 if (it == targetCaches.end())
220 it->getSecond().insertPort(mod, portNo);
225 auto mod = op->getParentOfType<FModuleOp>();
226 auto it = targetCaches.find(mod);
227 if (it == targetCaches.end())
229 it->getSecond().insertOp(op);
242std::optional<TokenAnnoTarget>
tokenizePath(StringRef origTarget);
252std::optional<AnnoPathValue>
resolvePath(StringRef rawPath, CircuitOp circuit,
312 hw::HierPathOp getOpFor(ArrayAttr attr);
315 return getOpFor(attr).getSymNameAttr();
319 return FlatSymbolRefAttr::get(getSymFor(attr));
326 DenseMap<ArrayAttr, hw::HierPathOp>
cache;
336 : circuit(circuit), symTbl(symTbl), addToWorklistFn(addToWorklistFn),
337 instancePathCache(instancePathCache), hierPathCache(circuit, symTbl),
338 noRefTypePorts(noRefTypePorts) {}
346 size_t numReusedHierPaths = 0;
356 return namespaces[module];
360 return IntegerAttr::get(IntegerType::get(circuit.getContext(), 64),
366 unsigned annotationID = 0;
389A
tryGetAs(DictionaryAttr &dict,
const Attribute &root, StringRef key,
390 Location loc, Twine className, Twine path = Twine()) {
392 auto value = dict.get(key);
394 SmallString<128> msg;
395 if (path.isTriviallyEmpty())
396 msg = (
"Annotation '" + className +
"' did not contain required key '" +
400 msg = (
"Annotation '" + className +
"' with path '" + path +
401 "' did not contain required key '" + key +
"'.")
403 mlir::emitError(loc, msg).attachNote()
404 <<
"The full Annotation is reproduced here: " << root <<
"\n";
408 auto valueA = dyn_cast_or_null<A>(value);
410 SmallString<128> msg;
411 if (path.isTriviallyEmpty())
412 msg = (
"Annotation '" + className +
413 "' did not contain the correct type for key '" + key +
"'.")
416 msg = (
"Annotation '" + className +
"' with path '" + path +
417 "' did not contain the correct type for key '" + key +
"'.")
419 mlir::emitError(loc, msg).attachNote()
420 <<
"The full Annotation is reproduced here: " << root <<
"\n";
429InstanceOp addPortsToModule(FModuleLike mod, InstanceOp instOnPath,
430 FIRRTLType portType, Direction dir,
432 InstancePathCache &instancePathcache,
433 CircuitTargetCache *targetCaches =
nullptr);
441 llvm::function_ref<std::optional<AnnoPathValue>(DictionaryAttr,
ApplyState &)>
443 llvm::function_ref<LogicalResult(
const AnnoPathValue &, DictionaryAttr,
451 const std::function<
void(llvm::Twine)> &errorHandler = {});
460std::optional<AnnoPathValue> stdResolve(DictionaryAttr anno, ApplyState &state);
463std::optional<AnnoPathValue> tryResolve(DictionaryAttr anno, ApplyState &state);
471LogicalResult applyWithoutTargetImpl(
const AnnoPathValue &target,
472 DictionaryAttr anno, ApplyState &state,
478template <
bool allowNonLocal,
bool allowPortAnnoTarget,
typename T,
483 if (isa<PortAnnoTarget>(target.
ref)) {
484 if (!allowPortAnnoTarget)
492template <
bool allowNonLocal,
typename T,
typename... Tr>
502template <
bool allowNonLocal = false>
This class implements the same functionality as TypeSwitch except that it uses firrtl::type_dyn_cast ...
FIRRTLTypeSwitch< T, ResultT > & Case(CallableT &&caseFn)
Add a case on the given type.
A tryGetAs(DictionaryAttr &dict, const Attribute &root, StringRef key, Location loc, Twine className, Twine path=Twine())
Implements the same behavior as DictionaryAttr::getAs<A> to return the value of a specific type assoc...
std::optional< AnnoPathValue > resolveEntities(TokenAnnoTarget path, CircuitOp circuit, SymbolTable &symTbl, CircuitTargetCache &cache)
Convert a parsed target string to a resolved target structure.
std::string canonicalizeTarget(StringRef target)
Return an input target string in canonical form.
LogicalResult applyGCTMemTaps(const AnnoPathValue &target, DictionaryAttr anno, ApplyState &state)
llvm::raw_ostream & operator<<(llvm::raw_ostream &os, const InstanceInfo::LatticeValue &value)
static LogicalResult applyWithoutTarget(const AnnoPathValue &target, DictionaryAttr anno, ApplyState &state)
An applier which puts the annotation on the target and drops the 'target' field from the annotation.
LogicalResult applyWiring(const AnnoPathValue &target, DictionaryAttr anno, ApplyState &state)
Consume SourceAnnotation and SinkAnnotation, storing into state.
LogicalResult applyTraceName(const AnnoPathValue &target, DictionaryAttr anno, ApplyState &state)
Expand a TraceNameAnnotation (which has don't touch semantics) into a TraceAnnotation (which does NOT...
LogicalResult registerAnnotationRecord(StringRef annoClass, AnnoRecord annoRecord, const std::function< void(llvm::Twine)> &errorHandler={})
Register external annotation records.
LogicalResult applyGCTDataTaps(const AnnoPathValue &target, DictionaryAttr anno, ApplyState &state)
std::optional< AnnoPathValue > resolvePath(StringRef rawPath, CircuitOp circuit, SymbolTable &symTbl, CircuitTargetCache &cache)
Resolve a string path to a named item inside a circuit.
LogicalResult applyGCTView(const AnnoPathValue &target, DictionaryAttr anno, ApplyState &state)
std::optional< TokenAnnoTarget > tokenizePath(StringRef origTarget)
Parse a FIRRTL annotation path into its constituent parts.
LogicalResult applyWithoutTargetImpl(const AnnoPathValue &target, DictionaryAttr anno, ApplyState &state, bool allowNonLocal)
===-------------------------------------------------------------------—===// Standard Utility Applier...
The InstanceGraph op interface, see InstanceGraphInterface.td for more details.
SmallVector< InstanceOp > instances
AnnoPathValue(const SmallVectorImpl< InstanceOp > &insts, AnnoTarget b, unsigned fieldIdx)
AnnoPathValue(CircuitOp op)
AnnoPathValue(Operation *op)
===-------------------------------------------------------------------—===// LowerAnnotations ===----...
llvm::function_ref< std::optional< AnnoPathValue >(DictionaryAttr, ApplyState &)> resolver
llvm::function_ref< LogicalResult(const AnnoPathValue &, DictionaryAttr, ApplyState &)> applier
Cache AnnoTargets for a module's named things.
AnnoTarget getTargetForName(StringRef name) const
Lookup the target for 'name', empty if not found.
AnnoTargetCache(FModuleLike mod)
AnnoTargetCache(AnnoTargetCache &&other)
void gatherTargets(FModuleLike mod)
Walk the module and add named things to 'targets'.
void insertOp(Operation *op)
AnnoTargetCache(const AnnoTargetCache &other)=default
void replaceOp(Operation *oldOp, Operation *newOp)
Replace oldOp with newOp in the target cache.
void insertPort(FModuleLike mod, size_t portNo)
Add a new module port to the target cache.
llvm::DenseMap< StringRef, AnnoTarget > targets
An annotation target is used to keep track of something that is targeted by an Annotation.
FIRRTLType getType() const
Get the type of the target.
Operation * getOp() const
FModuleLike getModule() const
Get the parent module of the target.
State threaded through functions for resolving and applying annotations.
HierPathCache hierPathCache
DenseSet< InstanceOp > wiringProblemInstRefs
DenseMap< StringAttr, LegacyWiringProblem > legacyWiringProblems
ApplyState(CircuitOp circuit, SymbolTable &symTbl, AddToWorklistFn addToWorklistFn, InstancePathCache &instancePathCache, bool noRefTypePorts)
SmallVector< WiringProblem > wiringProblems
AddToWorklistFn addToWorklistFn
InstancePathCache & instancePathCache
CircuitTargetCache targetCaches
hw::InnerSymbolNamespaceCollection namespaces
hw::InnerSymbolNamespace & getNamespace(FModuleLike module)
llvm::function_ref< void(DictionaryAttr)> AddToWorklistFn
Cache AnnoTargets for a circuit's modules, walked as needed.
const AnnoTargetCache & getOrCreateCacheFor(FModuleLike module)
Get cache for specified module, creating it as needed.
DenseMap< Operation *, AnnoTargetCache > targetCaches
void replaceOp(Operation *oldOp, Operation *newOp)
Replace oldOp with newOp in the target cache.
void invalidate()
Clear the cache completely.
void insertPort(FModuleLike mod, size_t portNo)
Add a new module port to the target cache.
void insertOp(Operation *op)
Add a new op to the target cache.
AnnoTarget lookup(FModuleLike module, StringRef name)
Lookup the target for 'name' in 'module'.
A cache of existing HierPathOps, mostly used to facilitate HierPathOp reuse.
DenseMap< ArrayAttr, hw::HierPathOp > cache
StringAttr getSymFor(ArrayAttr attr)
FlatSymbolRefAttr getRefFor(ArrayAttr attr)
const SymbolTable & getSymbolTable() const
SymbolTable & symbolTable
A representation of a legacy Wiring problem consisting of a signal source that should be connected to...
SmallVector< Value > sinks
Sink(s) to wire to.
Value source
A source to wire from.
A store of pending modifications to a FIRRTL module associated with solving one or more WiringProblem...
SmallVector< portInfoPair > portsToAdd
Ports that should be added to a module.
std::pair< size_t, PortInfo > portInfoPair
A pair of Wiring Problem index and port information.
SmallVector< uturnPair > uturns
A secondary value that may need to be hooked up.
DenseMap< size_t, Value > connectionMap
A mapping of a Value that should be connected to either a new port or a U-turn, for a specific Wiring...
std::pair< size_t, Value > uturnPair
A pair of Wiring Problem index and a U-turn Value that should be connected.
This represents an annotation targeting a specific operation.
This represents an annotation targeting a specific port of a module, memory, or instance.
unsigned getPortNo() const
Stores an index into an aggregate.
The parsed annotation path.
SmallVector< TargetToken > component
std::string str() const
Convert the annotation path to a string.
void toVector(SmallVectorImpl< char > &out) const
Append the annotation path to the given SmallString or SmallVector.
SmallVector< std::pair< StringRef, StringRef > > instances
A representation of a deferred Wiring problem consisting of a source that should be connected to a si...
Value source
A source to wire from.
Value sink
A sink to wire to.
RefTypeUsage refTypeUsage
The usage of ref type ports when solving this problem.
std::string newNameHint
A base name to use when generating new signals associated with this wiring problem.
A data structure that caches and provides absolute paths to module instances in the IR.