45 uncompress(dst.data(), &dstSize, compressed.data(), compressed.size());
47 throw std::runtime_error(
"zlib uncompress failed with rc=" +
49 return std::string(
reinterpret_cast<char *
>(dst.data()), dstSize);
61 auto offsetIter = client.implOptions.find(
"offset");
62 if (offsetIter == client.implOptions.end())
63 throw std::runtime_error(
"MMIO client missing 'offset' option");
64 const Constant *offset = std::any_cast<Constant>(&offsetIter->second);
66 throw std::runtime_error(
67 "MMIO client 'offset' option must be a constant");
68 const uint64_t *offsetVal = std::any_cast<uint64_t>(&offset->
value);
70 throw std::runtime_error(
71 "MMIO client 'offset' option must be an integer");
72 if (*offsetVal >= 1ull << 32)
73 throw std::runtime_error(
"MMIO client offset mustn't exceed 32 bits");
75 auto sizeIter = client.implOptions.find(
"size");
76 if (sizeIter == client.implOptions.end())
77 throw std::runtime_error(
"MMIO client missing 'size' option");
78 const Constant *size = std::any_cast<Constant>(&sizeIter->second);
80 throw std::runtime_error(
"MMIO client 'size' option must be a constant");
81 const uint64_t *sizeVal = std::any_cast<uint64_t>(&size->
value);
83 throw std::runtime_error(
"MMIO client 'size' option must be an integer");
84 if (*sizeVal >= 1ull << 32)
85 throw std::runtime_error(
"MMIO client size mustn't exceed 32 bits");
87 static_cast<uint32_t
>(*offsetVal),
static_cast<uint32_t
>(*sizeVal)};
95 auto regionIter =
regions.find(
id);
96 if (regionIter ==
regions.end())
103class MMIOPassThrough :
public MMIO {
107 :
MMIO(parent->getConnection(), idPath, clients), parent(parent) {}
108 uint64_t read(uint32_t addr)
const override {
return parent->read(addr); }
109 void write(uint32_t addr, uint64_t data)
override {
110 parent->write(addr, data);
121 if (service !=
typeid(
MMIO))
123 return new MMIOPassThrough(clients,
id,
this);
131 :
ServicePort(id, nullptr, {}), parent(parent), desc(desc) {}
133 if (addr >= desc.size)
134 throw std::runtime_error(
"MMIO read out of bounds: " +
toHex(addr));
135 return parent->read(desc.base + addr);
138 if (addr >= desc.size)
139 throw std::runtime_error(
"MMIO write out of bounds: " +
toHex(addr));
140 parent->write(desc.base + addr, data);
149 throw std::runtime_error(
"Invalid magic number: " +
toHex(reg));
167 throw std::runtime_error(
"Unsupported ESI header version: " +
168 std::to_string(version));
170 uint64_t size =
mmio->
read(manifestPtr);
171 uint64_t numWords = (size + 7) / 8;
172 std::vector<uint64_t> manifestWords(numWords);
173 for (
size_t i = 0; i < numWords; ++i)
174 manifestWords[i] =
mmio->
read(manifestPtr + 8 + (i * 8));
176 std::vector<uint8_t> manifest;
177 for (
size_t i = 0; i < size; ++i) {
178 uint64_t word = manifestWords[i / 8];
179 manifest.push_back(word >> (8 * (i % 8)));
190 if (
auto f = details.find(
"service"); f != details.end()) {
206 if (
auto f = details.find(
"service"); f != details.end())
208 symbol = std::any_cast<std::string>(f->second).substr(1);
218 return new ToHost(
id.back(), type, ports);
219 return new FromHost(
id.back(), type, ports);
225 return new ToHost(
id,
type, {{std::string(
"data"), data}});
230 throw std::runtime_error(
"ToHost channel is already connected");
231 if (channels.size() != 1)
232 throw std::runtime_error(
"ChannelService ToHost must have exactly one "
234 dataPort = &getRawRead(
"data");
241 throw std::runtime_error(
242 "ToHost channel must be 'connect'ed before reading");
243 return dataPort->readAsync();
249 return new FromHost(
id, type, {{std::string(
"data"), data}});
254 throw std::runtime_error(
"FromHost channel is already connected");
255 if (channels.size() != 1)
256 throw std::runtime_error(
"ChannelService FromHost must have exactly one "
258 dataPort = &getRawWrite(
"data");
265 throw std::runtime_error(
266 "FromHost channel must be 'connect'ed before writing");
267 dataPort->write(data);
274 if (
auto f = details.find(
"service"); f != details.end())
276 symbol = std::any_cast<std::string>(f->second).substr(1);
282 return new Function(
id.back(), type,
290 id,
type, {{std::string(
"arg"),
arg}, {std::string(
"result"),
result}});
296 throw std::runtime_error(
"Function is already connected");
297 if (channels.size() != 2)
298 throw std::runtime_error(
"FuncService must have exactly two channels");
299 arg = &getRawWrite(
"arg");
301 result = &getRawRead(
"result");
306std::future<MessageData>
309 throw std::runtime_error(
"Function must be 'connect'ed before calling");
310 std::scoped_lock<std::mutex> lock(callMutex);
312 return result->readAsync();
318 if (
auto f = details.find(
"service"); f != details.end())
320 symbol = std::any_cast<std::string>(f->second).substr(1);
339 return new Callback(acc,
id, type, {{
"arg", arg}, {
"result", result}});
344 if (channels.size() != 2)
345 throw std::runtime_error(
"CallService must have exactly two channels");
346 result = &getRawWrite(
"result");
348 arg = &getRawRead(
"arg");
351 arg->connect([
this, callback](
MessageData argMsg) ->
bool {
352 MessageData resultMsg = callback(std::move(argMsg));
353 this->result->write(std::move(resultMsg));
359 acc.getServiceThread()->addListener(
361 MessageData resultMsg = callback(std::move(argMsg));
362 this->result->write(std::move(resultMsg));
374 if (prefix.size() > 0)
377 if (client.implOptions.contains(
"type") &&
378 std::any_cast<std::string>(client.implOptions.at(
"type")) !=
"mmio")
380 AppIDPath fullClientPath = prefix + client.relPath;
381 auto offsetIter = client.implOptions.find(
"offset");
382 if (offsetIter == client.implOptions.end()) {
384 "mmio client " + fullClientPath.
toStr() +
385 " missing 'offset' option, skipping");
388 const Constant *offset = std::any_cast<Constant>(&offsetIter->second);
389 if (offset ==
nullptr) {
391 "Telemetry",
"mmio client " + fullClientPath.
toStr() +
392 " 'offset' option must be a constant, skipping");
395 const uint64_t *offsetVal = std::any_cast<uint64_t>(&offset->
value);
396 if (offsetVal ==
nullptr) {
398 "Telemetry",
"mmio client " + fullClientPath.
toStr() +
399 " 'offset' option must be an integer, skipping");
415 mmioPath.push_back(
AppID(
"__telemetry_mmio"));
418 throw std::runtime_error(
"TelemetryService: could not resolve port " +
419 id.toStr() +
". Got as far as " +
423 throw std::runtime_error(
"TelemetryService: port " +
id.toStr() +
424 " is not a MMIO region");
432 auto *port =
new Metric(
id.back(), type, {},
this,
434 ? std::optional<uint64_t>(offsetIter->second)
441 std::string implName,
452 std::optional<uint64_t> offset)
453 :
ServicePort(id, type, channels), telemetryService(telemetryService),
454 mmio(nullptr), offset(offset) {}
458 if (!offset.has_value())
459 throw std::runtime_error(
"Telemetry offset not found for " +
id.
toString());
460 mmio = telemetryService->getMMIORegion();
461 assert(
mmio &&
"TelemetryService: MMIO region not found");
465 return std::async(std::launch::async, [
this]() {
466 uint64_t data = readInt();
472 assert(offset.has_value() &&
473 "Telemetry offset must be set. Checked in connect().");
474 assert(
mmio &&
"TelemetryService: MMIO region not set");
480 ports[entry.first] = entry.second;
482 child->getTelemetryPorts(ports);
487 std::string implName,
492 return new FuncService(
id, *acc, details, clients);
506 if (svcName ==
"esi.service.std.func")
508 if (svcName ==
"esi.service.std.call")
510 if (svcName ==
"esi.service.std.channel")
assert(baseType &&"element must be base type")
constexpr uint32_t MAX_MANIFEST_SIZE
Abstract class representing a connection to an accelerator.
ServiceClass * getService(AppIDPath id={}, std::string implName={}, ServiceImplDetails details={}, HWClientDetails clients={})
Get a typed reference to a particular service type.
Accelerator & getAccelerator()
virtual const BundleEngineMap & getEngineMapFor(AppIDPath id)
Logger & getLogger() const
std::string toStr() const
PortMap requestPorts(const AppIDPath &idPath, const BundleType *bundleType) const
Request ports for all the channels in a bundle.
Services provide connections to 'bundles' – collections of named, unidirectional communication channe...
Bundles represent a collection of channels.
std::pair< const Type *, Direction > findChannel(std::string name) const
BundlePort * resolvePort(const AppIDPath &path, AppIDPath &lastLookup) const
Attempt to resolve a path to a port.
virtual void warning(const std::string &subsystem, const std::string &msg, const std::map< std::string, std::any > *details=nullptr)
Report a warning.
A logical chunk of data representing serialized data.
static MessageData from(T &t)
Cast from a type to its raw bytes.
A ChannelPort which reads data from the accelerator.
A ChannelPort which sends data to the accelerator.
A function call which gets attached to a service port.
static Callback * get(AcceleratorConnection &acc, AppID id, const BundleType *type, WriteChannelPort &result, ReadChannelPort &arg)
Callback(AcceleratorConnection &acc, AppID id, const BundleType *, PortMap channels)
void connect(std::function< MessageData(const MessageData &)> callback, bool quick=false)
Connect a callback to code which will be executed when the accelerator invokes the callback.
Service for servicing function calls from the accelerator.
virtual std::string getServiceSymbol() const override
CallService(AcceleratorConnection &acc, AppIDPath id, ServiceImplDetails details)
virtual BundlePort * getPort(AppIDPath id, const BundleType *type) const override
Get specialized port for this service to attach to the given appid path.
A port which writes data to the accelerator (from_host).
static FromHost * get(AppID id, const BundleType *type, WriteChannelPort &dataPort)
void write(const MessageData &data)
A port which reads data from the accelerator (to_host).
std::future< MessageData > read()
static ToHost * get(AppID id, const BundleType *type, ReadChannelPort &dataPort)
Service for raw communication channels to/from the accelerator.
virtual std::string getServiceSymbol() const override
virtual BundlePort * getPort(AppIDPath id, const BundleType *type) const override
Get specialized port for this service to attach to the given appid path.
ChannelService(AppIDPath id, AcceleratorConnection &, ServiceImplDetails details, HWClientDetails clients)
A service for which there are no standard services registered.
virtual BundlePort * getPort(AppIDPath id, const BundleType *type) const override
Get specialized port for this service to attach to the given appid path.
std::string serviceSymbol
CustomService(AppIDPath idPath, AcceleratorConnection &, const ServiceImplDetails &details, const HWClientDetails &clients)
A function call which gets attached to a service port.
std::future< MessageData > call(const MessageData &arg)
static Function * get(AppID id, BundleType *type, WriteChannelPort &arg, ReadChannelPort &result)
Service for calling functions.
virtual std::string getServiceSymbol() const override
virtual BundlePort * getPort(AppIDPath id, const BundleType *type) const override
Get specialized port for this service to attach to the given appid path.
FuncService(AppIDPath id, AcceleratorConnection &, ServiceImplDetails details, HWClientDetails clients)
virtual std::string getServiceSymbol() const override
static constexpr std::string_view StdName
std::optional< uint64_t > getCycleCount() const override
Get the current cycle count of the accelerator system's core clock.
virtual std::vector< uint8_t > getCompressedManifest() const override
Return the zlib compressed JSON system manifest.
std::optional< uint64_t > getCoreClockFrequency() const override
Get the "core" clock frequency of the accelerator system in Hz.
uint32_t getEsiVersion() const override
Get the ESI version number to check version compatibility.
MMIOSysInfo(const MMIO *)
A "slice" of some parent MMIO space.
virtual uint64_t read(uint32_t addr) const
Read a 64-bit value from this region, not the global address space.
MMIORegion(AppID id, MMIO *parent, RegionDescriptor desc)
virtual void write(uint32_t addr, uint64_t data)
Write a 64-bit value to this region, not the global address space.
virtual uint64_t read(uint32_t addr) const =0
Read a 64-bit value from the global MMIO space.
MMIO(AcceleratorConnection &, const AppIDPath &idPath, const HWClientDetails &clients)
virtual BundlePort * getPort(AppIDPath id, const BundleType *type) const override
Get a MMIO region port for a particular region descriptor.
std::map< AppIDPath, RegionDescriptor > regions
MMIO base address table.
static constexpr std::string_view StdName
virtual Service * getChildService(Service::Type service, AppIDPath id={}, std::string implName={}, ServiceImplDetails details={}, HWClientDetails clients={}) override
If the service is a MMIO service, return a region of the MMIO space which peers into ours.
virtual std::string getServiceSymbol() const override
Add a custom interface to a service client at a particular point in the design hierarchy.
static Service::Type lookupServiceType(const std::string &)
Resolve a service type from a string.
static Service * createService(AcceleratorConnection *acc, Service::Type svcType, AppIDPath id, std::string implName, ServiceImplDetails details, HWClientDetails clients)
Create a service instance from the given details.
Parent class of all APIs modeled as 'services'.
AcceleratorConnection & getConnection() const
const std::type_info & Type
virtual Service * getChildService(Service::Type service, AppIDPath id={}, std::string implName={}, ServiceImplDetails details={}, HWClientDetails clients={})
Create a "child" service of this service.
AcceleratorConnection & conn
Information about the Accelerator system.
virtual std::string getJsonManifest() const
Return the JSON-formatted system manifest.
virtual std::vector< uint8_t > getCompressedManifest() const =0
Return the zlib compressed JSON system manifest.
virtual std::string getServiceSymbol() const override
A telemetry port which gets attached to a service port.
void connect()
Connect to a particular telemetry port. Offset should be non-nullopt.
std::future< MessageData > read()
Metric(AppID id, const BundleType *type, PortMap channels, const TelemetryService *telemetryService, std::optional< uint64_t > offset)
Service for retrieving telemetry data from the accelerator.
std::list< TelemetryService * > children
std::map< AppIDPath, Metric * > getTelemetryPorts()
MMIO::MMIORegion * getMMIORegion() const
std::map< AppIDPath, Metric * > telemetryPorts
std::map< AppIDPath, uint64_t > portAddressAssignments
static constexpr std::string_view StdName
TelemetryService(AppIDPath id, AcceleratorConnection &, ServiceImplDetails details, HWClientDetails clients)
virtual std::string getServiceSymbol() const override
virtual BundlePort * getPort(AppIDPath id, const BundleType *type) const override
Get specialized port for this service to attach to the given appid path.
virtual Service * getChildService(Service::Type service, AppIDPath id={}, std::string implName={}, ServiceImplDetails details={}, HWClientDetails clients={}) override
Create a "child" service of this service.
std::string toString(const std::any &a)
'Stringify' a std::any. This is used to log std::any values by some loggers.
constexpr uint32_t CoreFreqOffset
constexpr uint64_t MagicNumber
std::map< std::string, std::any > ServiceImplDetails
std::string toHex(void *val)
constexpr uint64_t MagicNumberOffset
constexpr uint32_t MetadataOffset
constexpr uint32_t CycleCountOffset
constexpr uint32_t ManifestPtrOffset
std::map< std::string, ChannelPort & > PortMap
constexpr uint64_t VersionNumberOffset
std::vector< HWClientDetail > HWClientDetails
A description of a hardware client.
Describe a region (slice) of MMIO space.