CIRCT 22.0.0git
Loading...
Searching...
No Matches
Accelerator.h
Go to the documentation of this file.
1//===- Accelerator.h - Base ESI runtime API ---------------------*- C++ -*-===//
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//
9// Basic ESI APIs. The 'Accelerator' class is the superclass for all accelerator
10// backends. It should (usually) provide enough functionality such that users do
11// not have to interact with the platform-specific backend implementation with
12// the exception of connecting to the accelerator.
13//
14// DO NOT EDIT!
15// This file is distributed as part of an ESI package. The source for this file
16// should always be modified within CIRCT.
17//
18//===----------------------------------------------------------------------===//
19
20// NOLINTNEXTLINE(llvm-header-guard)
21#ifndef ESI_ACCELERATOR_H
22#define ESI_ACCELERATOR_H
23
24#include "esi/Context.h"
25#include "esi/Design.h"
26#include "esi/Engines.h"
27#include "esi/Manifest.h"
28#include "esi/Ports.h"
29#include "esi/Services.h"
30
31#include <functional>
32#include <map>
33#include <memory>
34#include <string>
35#include <typeinfo>
36
37namespace esi {
38// Forward declarations.
39class AcceleratorServiceThread;
40
41//===----------------------------------------------------------------------===//
42// Constants used by low-level APIs.
43//===----------------------------------------------------------------------===//
44
45constexpr uint32_t MetadataOffset = 8;
46constexpr uint64_t MagicNumberLo = 0xE5100E51;
47constexpr uint64_t MagicNumberHi = 0x207D98E5;
48constexpr uint64_t MagicNumber = MagicNumberLo | (MagicNumberHi << 32);
49constexpr uint32_t ExpectedVersionNumber = 0;
50
51//===----------------------------------------------------------------------===//
52// Accelerator design hierarchy root.
53//===----------------------------------------------------------------------===//
54
55/// Top level accelerator class. Maintains a shared pointer to the manifest,
56/// which owns objects used in the design hierarchy owned by this class. Since
57/// this class owns the entire design hierarchy, when it gets destroyed the
58/// entire design hierarchy gets destroyed so all of the instances, ports, etc.
59/// are no longer valid pointers.
60class Accelerator : public HWModule {
61public:
62 Accelerator() = delete;
63 Accelerator(const Accelerator &) = delete;
64 ~Accelerator() = default;
65 Accelerator(std::optional<ModuleInfo> info,
66 std::vector<std::unique_ptr<Instance>> children,
67 std::vector<services::Service *> services,
68 std::vector<std::unique_ptr<BundlePort>> &ports)
69 : HWModule(info, std::move(children), services, ports) {}
70};
71
72//===----------------------------------------------------------------------===//
73// Connection to the accelerator and its services.
74//===----------------------------------------------------------------------===//
75
76/// Abstract class representing a connection to an accelerator. Actual
77/// connections (e.g. to a co-simulation or actual device) are implemented by
78/// subclasses. No methods in here are thread safe.
80public:
82 virtual ~AcceleratorConnection();
83 Context &getCtxt() const { return ctxt; }
84 Logger &getLogger() const { return ctxt.getLogger(); }
85
86 /// Disconnect from the accelerator cleanly.
87 virtual void disconnect();
88
89 /// Return a pointer to the accelerator 'service' thread (or threads). If the
90 /// thread(s) are not running, they will be started when this method is
91 /// called. `std::thread` is used. If users don't want the runtime to spin up
92 /// threads, don't call this method. `AcceleratorServiceThread` is owned by
93 /// AcceleratorConnection and governed by the lifetime of the this object.
95
97 /// Get a typed reference to a particular service type. Caller does *not* take
98 /// ownership of the returned pointer -- the Accelerator object owns it.
99 /// Pointer lifetime ends with the Accelerator lifetime.
100 template <typename ServiceClass>
101 ServiceClass *getService(AppIDPath id = {}, std::string implName = {},
102 ServiceImplDetails details = {},
103 HWClientDetails clients = {}) {
104 return dynamic_cast<ServiceClass *>(
105 getService(typeid(ServiceClass), id, implName, details, clients));
106 }
107 /// Calls `createService` and caches the result. Subclasses can override if
108 /// they want to use their own caching mechanism.
109 virtual Service *getService(Service::Type service, AppIDPath id = {},
110 std::string implName = {},
111 ServiceImplDetails details = {},
112 HWClientDetails clients = {});
113
114 /// Assume ownership of an accelerator object. Ties the lifetime of the
115 /// accelerator to this connection. Returns a raw pointer to the object.
116 Accelerator *takeOwnership(std::unique_ptr<Accelerator> accel);
117
118 /// Create a new engine for channel communication with the accelerator. The
119 /// default is to call the global `createEngine` to get an engine which has
120 /// registered itself. Individual accelerator connection backends can override
121 /// this to customize behavior.
122 virtual void createEngine(const std::string &engineTypeName, AppIDPath idPath,
123 const ServiceImplDetails &details,
124 const HWClientDetails &clients);
126 return clientEngines[id];
127 }
128
130 if (!ownedAccelerator)
131 throw std::runtime_error(
132 "AcceleratorConnection does not own an accelerator");
133 return *ownedAccelerator;
134 }
135
136protected:
137 /// If `createEngine` is overridden, this method should be called to register
138 /// the engine and all of the channels it services.
139 void registerEngine(AppIDPath idPath, std::unique_ptr<Engine> engine,
140 const HWClientDetails &clients);
141
142 /// Called by `getServiceImpl` exclusively. It wraps the pointer returned by
143 /// this in a unique_ptr and caches it. Separate this from the
144 /// wrapping/caching since wrapping/caching is an implementation detail.
146 std::string implName,
147 const ServiceImplDetails &details,
148 const HWClientDetails &clients) = 0;
149
150 /// Collection of owned engines.
151 std::map<AppIDPath, std::unique_ptr<Engine>> ownedEngines;
152 /// Mapping of clients to their servicing engines.
153 std::map<AppIDPath, BundleEngineMap> clientEngines;
154
155private:
156 /// ESI accelerator context.
158
159 /// Cache services via a unique_ptr so they get free'd automatically when
160 /// Accelerator objects get deconstructed.
161 using ServiceCacheKey = std::tuple<std::string, AppIDPath>;
162 std::map<ServiceCacheKey, std::unique_ptr<Service>> serviceCache;
163
164 std::unique_ptr<AcceleratorServiceThread> serviceThread;
165
166 /// Accelerator object owned by this connection.
167 std::unique_ptr<Accelerator> ownedAccelerator;
168};
169
170namespace registry {
171
172namespace internal {
173
174/// Backends can register themselves to be connected via a connection string.
175using BackendCreate = std::function<std::unique_ptr<AcceleratorConnection>(
176 Context &, std::string)>;
177void registerBackend(const std::string &name, BackendCreate create);
178
179// Helper struct to
180template <typename TAccelerator>
182 RegisterAccelerator(const char *name) {
183 registerBackend(name, &TAccelerator::connect);
184 }
185};
186
187#define REGISTER_ACCELERATOR(Name, TAccelerator) \
188 static ::esi::registry::internal::RegisterAccelerator<TAccelerator> \
189 __register_accel____LINE__(Name)
190
191} // namespace internal
192} // namespace registry
193
194/// Background thread which services various requests. Currently, it listens on
195/// ports and calls callbacks for incoming messages on said ports.
197public:
200
201 /// When there's data on any of the listenPorts, call the callback. Callable
202 /// from any thread.
203 void
204 addListener(std::initializer_list<ReadChannelPort *> listenPorts,
205 std::function<void(ReadChannelPort *, MessageData)> callback);
206
207 /// Poll this module.
208 void addPoll(HWModule &module);
209
210 /// Instruct the service thread to stop running.
211 void stop();
212
213private:
214 struct Impl;
215 std::unique_ptr<Impl> impl;
216};
217} // namespace esi
218
219#endif // ESI_ACCELERATOR_H
Abstract class representing a connection to an accelerator.
Definition Accelerator.h:79
std::tuple< std::string, AppIDPath > ServiceCacheKey
Cache services via a unique_ptr so they get free'd automatically when Accelerator objects get deconst...
virtual Service * createService(Service::Type service, AppIDPath idPath, std::string implName, const ServiceImplDetails &details, const HWClientDetails &clients)=0
Called by getServiceImpl exclusively.
services::Service Service
Definition Accelerator.h:96
ServiceClass * getService(AppIDPath id={}, std::string implName={}, ServiceImplDetails details={}, HWClientDetails clients={})
Get a typed reference to a particular service type.
Context & getCtxt() const
Definition Accelerator.h:83
std::map< AppIDPath, BundleEngineMap > clientEngines
Mapping of clients to their servicing engines.
Context & ctxt
ESI accelerator context.
void registerEngine(AppIDPath idPath, std::unique_ptr< Engine > engine, const HWClientDetails &clients)
If createEngine is overridden, this method should be called to register the engine and all of the cha...
std::map< ServiceCacheKey, std::unique_ptr< Service > > serviceCache
std::unique_ptr< AcceleratorServiceThread > serviceThread
std::unique_ptr< Accelerator > ownedAccelerator
Accelerator object owned by this connection.
virtual void disconnect()
Disconnect from the accelerator cleanly.
std::map< AppIDPath, std::unique_ptr< Engine > > ownedEngines
Collection of owned engines.
Accelerator & getAccelerator()
virtual void createEngine(const std::string &engineTypeName, AppIDPath idPath, const ServiceImplDetails &details, const HWClientDetails &clients)
Create a new engine for channel communication with the accelerator.
virtual const BundleEngineMap & getEngineMapFor(AppIDPath id)
Logger & getLogger() const
Definition Accelerator.h:84
AcceleratorServiceThread * getServiceThread()
Return a pointer to the accelerator 'service' thread (or threads).
Accelerator * takeOwnership(std::unique_ptr< Accelerator > accel)
Assume ownership of an accelerator object.
Background thread which services various requests.
std::unique_ptr< Impl > impl
void stop()
Instruct the service thread to stop running.
void addPoll(HWModule &module)
Poll this module.
void addListener(std::initializer_list< ReadChannelPort * > listenPorts, std::function< void(ReadChannelPort *, MessageData)> callback)
When there's data on any of the listenPorts, call the callback.
Top level accelerator class.
Definition Accelerator.h:60
Accelerator(std::optional< ModuleInfo > info, std::vector< std::unique_ptr< Instance > > children, std::vector< services::Service * > services, std::vector< std::unique_ptr< BundlePort > > &ports)
Definition Accelerator.h:65
~Accelerator()=default
Accelerator()=delete
Accelerator(const Accelerator &)=delete
Since engines can support multiple channels BUT not necessarily all of the channels in a bundle,...
Definition Engines.h:73
AcceleratorConnections, Accelerators, and Manifests must all share a context.
Definition Context.h:34
Logger & getLogger()
Definition Context.h:69
Represents either the top level or an instance of a hardware module.
Definition Design.h:47
const std::optional< ModuleInfo > info
Definition Design.h:97
const std::vector< std::unique_ptr< BundlePort > > ports
Definition Design.h:101
const std::vector< std::unique_ptr< Instance > > children
Definition Design.h:98
const std::vector< services::Service * > services
Definition Design.h:100
A logical chunk of data representing serialized data.
Definition Common.h:113
A ChannelPort which reads data from the accelerator.
Definition Ports.h:124
Parent class of all APIs modeled as 'services'.
Definition Services.h:57
const std::type_info & Type
Definition Services.h:59
void registerBackend(const std::string &name, BackendCreate create)
std::function< std::unique_ptr< AcceleratorConnection >(Context &, std::string)> BackendCreate
Backends can register themselves to be connected via a connection string.
Definition esi.py:1
constexpr uint64_t MagicNumber
Definition Accelerator.h:48
std::map< std::string, std::any > ServiceImplDetails
Definition Common.h:108
constexpr uint64_t MagicNumberHi
Definition Accelerator.h:47
constexpr uint32_t ExpectedVersionNumber
Definition Accelerator.h:49
constexpr uint32_t MetadataOffset
Definition Accelerator.h:45
constexpr uint64_t MagicNumberLo
Definition Accelerator.h:46
std::vector< HWClientDetail > HWClientDetails
Definition Common.h:107