CIRCT 22.0.0git
Loading...
Searching...
No Matches
Manifest.cpp
Go to the documentation of this file.
1//===- Manifest.cpp - Metadata on the accelerator -------------------------===//
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// DO NOT EDIT!
10// This file is distributed as part of an ESI package. The source for this file
11// should always be modified within CIRCT (lib/dialect/ESI/runtime/cpp/).
12//
13//===----------------------------------------------------------------------===//
14
15#include "esi/Manifest.h"
16#include "esi/Accelerator.h"
17#include "esi/Services.h"
18
19#if defined(__GNUC__)
20#pragma GCC diagnostic push
21#pragma GCC diagnostic ignored "-Wcovered-switch-default"
22#endif
23#include <nlohmann/json.hpp>
24#if defined(__GNUC__)
25#pragma GCC diagnostic pop
26#endif
27#include <sstream>
28
29using namespace ::esi;
30
31// This is a proxy class to the manifest JSON. It is used to avoid having to
32// include the JSON parser in the header. Forward references don't work since
33// nlohmann::json is a rather complex template.
34//
35// Plus, it allows us to hide some implementation functions from the header
36// file.
38 friend class ::esi::Manifest;
39
40public:
41 Impl(Context &ctxt, const std::string &jsonManifest);
42
43 auto at(const std::string &key) const { return manifestJson.at(key); }
44
45 // Get the module info (if any) for the module instance in 'json'.
46 std::optional<ModuleInfo> getModInfo(const nlohmann::json &) const;
47
48 /// Go through the "service_decls" section of the manifest and populate the
49 /// services table as appropriate.
50 void scanServiceDecls(AcceleratorConnection &, const nlohmann::json &,
51 ServiceTable &) const;
52
53 void createEngine(AcceleratorConnection &, AppIDPath appID,
54 const nlohmann::json &) const;
55
56 /// Get a Service for the service specified in 'json'. Update the
57 /// activeServices table. TODO: re-using this for the engines section is a
58 /// terrible hack. Figure out a better way.
59 services::Service *getService(AppIDPath idPath, AcceleratorConnection &,
60 const nlohmann::json &,
61 ServiceTable &activeServices,
62 bool isEngine = false) const;
63
64 /// Get all the services in the description of an instance. Update the active
65 /// services table.
66 std::vector<services::Service *>
67 getServices(AppIDPath idPath, AcceleratorConnection &, const nlohmann::json &,
68 ServiceTable &activeServices) const;
69
70 /// Get the bundle ports for the instance at 'idPath' and specified in
71 /// 'instJson'. Look them up in 'activeServies'.
72 std::vector<std::unique_ptr<BundlePort>>
73 getBundlePorts(AcceleratorConnection &acc, AppIDPath idPath,
74 const ServiceTable &activeServices,
75 const nlohmann::json &instJson) const;
76
77 /// Build the set of child instances (recursively) for the module instance
78 /// description.
79 std::vector<std::unique_ptr<Instance>>
80 getChildInstances(AppIDPath idPath, AcceleratorConnection &acc,
81 const ServiceTable &activeServices,
82 const nlohmann::json &instJson) const;
83
84 /// Get a single child instance. Implicitly copy the active services table so
85 /// that it can be safely updated for the child's branch of the tree.
86 std::unique_ptr<Instance>
87 getChildInstance(AppIDPath idPath, AcceleratorConnection &acc,
88 ServiceTable activeServices,
89 const nlohmann::json &childJson) const;
90
91 /// Parse all the types and populate the types table.
92 void populateTypes(const nlohmann::json &typesJson);
93
94 /// Get the ordered list of types from the manifest.
95 const std::vector<const Type *> &getTypeTable() const { return _typeTable; }
96
97 /// Build a dynamic API for the Accelerator connection 'acc' based on the
98 /// manifest stored herein.
99 std::unique_ptr<Accelerator>
100 buildAccelerator(AcceleratorConnection &acc) const;
101
102 const Type *parseType(const nlohmann::json &typeJson);
103
104 const std::map<std::string, const ModuleInfo> &getSymbolInfo() const {
105 return symbolInfoCache;
106 }
107
108private:
110 std::vector<const Type *> _typeTable;
111
112 std::optional<const Type *> getType(Type::ID id) const {
113 return ctxt.getType(id);
114 }
115
116 std::any getAny(const nlohmann::json &value) const;
117 void parseModuleMetadata(ModuleInfo &info, const nlohmann::json &mod) const;
118 void parseModuleConsts(ModuleInfo &info, const nlohmann::json &mod) const;
119
120 // The parsed json.
121 nlohmann::json manifestJson;
122 // Cache the module info for each symbol.
123 std::map<std::string, const ModuleInfo> symbolInfoCache;
124};
125
126//===----------------------------------------------------------------------===//
127// Simple JSON -> object parsers.
128//===----------------------------------------------------------------------===//
129
130static std::optional<AppID> parseID(const nlohmann::json &jsonID) {
131 if (!jsonID.is_object())
132 return std::nullopt;
133 std::optional<uint32_t> idx;
134 if (jsonID.contains("index"))
135 idx = jsonID.at("index").get<uint32_t>();
136 if (jsonID.contains("name") && jsonID.size() <= 2)
137 return AppID(jsonID.at("name").get<std::string>(), idx);
138 return std::nullopt;
139}
140
141static AppID parseIDChecked(const nlohmann::json &jsonID) {
142 std::optional<AppID> id = parseID(jsonID);
143 if (!id)
144 throw std::runtime_error("Malformed manifest: invalid appID");
145 return *id;
146}
147static AppIDPath parseIDPath(const nlohmann::json &jsonIDPath) {
148 AppIDPath ret;
149 for (auto &idJson : jsonIDPath)
150 ret.push_back(parseIDChecked(idJson));
151 return ret;
152}
153
154static ServicePortDesc parseServicePort(const nlohmann::json &jsonPort) {
155 return ServicePortDesc{jsonPort.at("serviceName").get<std::string>(),
156 jsonPort.at("port").get<std::string>()};
157}
158
159/// Convert the json value to a 'std::any', which can be exposed outside of this
160/// file.
161std::any Manifest::Impl::getAny(const nlohmann::json &value) const {
162 auto getObject = [this](const nlohmann::json &json) -> std::any {
163 std::map<std::string, std::any> ret;
164 for (auto &e : json.items())
165 ret[e.key()] = getAny(e.value());
166
167 // If this can be converted to a constant, do so.
168 if (ret.size() != 2 || !ret.contains("type") || !ret.contains("value"))
169 return ret;
170 std::any value = ret.at("value");
171 std::any typeID = ret.at("type");
172 if (typeID.type() != typeid(std::string))
173 return ret;
174 std::optional<const Type *> type =
175 getType(std::any_cast<std::string>(type));
176 if (!type)
177 return ret;
178 // TODO: Check or guide the conversion of the value to the type based on the
179 // type.
180 return Constant{value, type};
181 };
182
183 auto getArray = [this](const nlohmann::json &json) -> std::any {
184 std::vector<std::any> ret;
185 for (auto &e : json)
186 ret.push_back(getAny(e));
187 return ret;
188 };
189
190 auto getValue = [&](const nlohmann::json &innerValue) -> std::any {
191 if (innerValue.is_string())
192 return innerValue.get<std::string>();
193 else if (innerValue.is_number_unsigned())
194 return innerValue.get<uint64_t>();
195 else if (innerValue.is_number_integer())
196 return innerValue.get<int64_t>();
197 else if (innerValue.is_number_float())
198 return innerValue.get<double>();
199 else if (innerValue.is_boolean())
200 return innerValue.get<bool>();
201 else if (innerValue.is_null())
202 return innerValue.get<std::nullptr_t>();
203 else if (innerValue.is_object())
204 return getObject(innerValue);
205 else if (innerValue.is_array())
206 return getArray(innerValue);
207 else
208 throw std::runtime_error("Unknown type in manifest: " +
209 innerValue.dump(2));
210 };
211
212 std::optional<AppID> appid = parseID(value);
213 if (appid)
214 return *appid;
215 if (!value.is_object() || !value.contains("type") || !value.contains("value"))
216 return getValue(value);
217 return Constant{getValue(value.at("value")), getType(value.at("type"))};
218}
219
221 const nlohmann::json &mod) const {
222 for (auto &extra : mod.items())
223 if (extra.key() != "name" && extra.key() != "summary" &&
224 extra.key() != "version" && extra.key() != "repo" &&
225 extra.key() != "commitHash")
226 info.extra[extra.key()] = getAny(extra.value());
227
228 auto value = [&](const std::string &key) -> std::optional<std::string> {
229 auto f = mod.find(key);
230 if (f == mod.end())
231 return std::nullopt;
232 return f.value();
233 };
234 info.name = value("name");
235 info.summary = value("summary");
236 info.version = value("version");
237 info.repo = value("repo");
238 info.commitHash = value("commitHash");
239}
240
242 const nlohmann::json &mod) const {
243 for (auto &item : mod.items()) {
244 std::any value = getAny(item.value());
245 auto *c = std::any_cast<Constant>(&value);
246 if (c)
247 info.constants[item.key()] = *c;
248 else
249 // If the value isn't a "proper" constant, present it as one with no type.
250 info.constants[item.key()] = Constant{value, std::nullopt};
251 }
252}
253
254//===----------------------------------------------------------------------===//
255// Manifest::Impl class implementation.
256//===----------------------------------------------------------------------===//
257
258Manifest::Impl::Impl(Context &ctxt, const std::string &manifestStr)
259 : ctxt(ctxt) {
260 manifestJson = nlohmann::ordered_json::parse(manifestStr);
261
262 try {
263 // Populate the types table first since anything else might need it.
264 populateTypes(manifestJson.at("types"));
265
266 // Populate the symbol info cache.
267 for (auto &mod : manifestJson.at("modules")) {
268 ModuleInfo info;
269 if (mod.contains("symInfo"))
270 parseModuleMetadata(info, mod.at("symInfo"));
271 if (mod.contains("symConsts"))
272 parseModuleConsts(info, mod.at("symConsts"));
273 symbolInfoCache.insert(make_pair(mod.at("symbol"), info));
274 }
275 } catch (const std::exception &e) {
276 std::string msg = "malformed manifest: " + std::string(e.what());
277 if (manifestJson.at("apiVersion") == 0)
278 msg += " (schema version 0 is not considered stable)";
279 throw std::runtime_error(msg);
280 }
281}
282
283std::unique_ptr<Accelerator>
284Manifest::Impl::buildAccelerator(AcceleratorConnection &acc) const {
285 ServiceTable activeSvcs;
286
287 auto designJson = manifestJson.at("design");
288
289 // Create all of the engines at the top level of the design.
290 // TODO: support engines at lower levels.
291 auto enginesIter = designJson.find("engines");
292 if (enginesIter != designJson.end())
293 for (auto &engineDesc : enginesIter.value())
294 createEngine(acc, {}, engineDesc);
295
296 // Get the initial active services table. Update it as we descend down.
297 auto svcDecls = manifestJson.at("serviceDeclarations");
298 scanServiceDecls(acc, svcDecls, activeSvcs);
299
300 // Get the services instantiated at the top level.
301 std::vector<services::Service *> services =
302 getServices({}, acc, designJson, activeSvcs);
303
304 // Get the ports at the top level.
305 auto ports = getBundlePorts(acc, {}, activeSvcs, designJson);
306
307 return std::make_unique<Accelerator>(
308 getModInfo(designJson),
309 getChildInstances({}, acc, activeSvcs, designJson), services,
310 std::move(ports));
311}
312
313std::optional<ModuleInfo>
314Manifest::Impl::getModInfo(const nlohmann::json &json) const {
315 auto instOfIter = json.find("instOf");
316 if (instOfIter == json.end())
317 return std::nullopt;
318 auto f = symbolInfoCache.find(instOfIter.value());
319 if (f != symbolInfoCache.end())
320 return f->second;
321 return std::nullopt;
322}
323
324/// TODO: Hack. This method is a giant hack to reuse the getService method for
325/// engines. It works, but it ain't pretty and it ain't right.
326void Manifest::Impl::createEngine(AcceleratorConnection &acc, AppIDPath idPath,
327 const nlohmann::json &eng) const {
328 ServiceTable dummy;
329 getService(idPath, acc, eng, dummy, /*isEngine=*/true);
330}
331
332void Manifest::Impl::scanServiceDecls(AcceleratorConnection &acc,
333 const nlohmann::json &svcDecls,
334 ServiceTable &activeServices) const {
335 for (auto &svcDecl : svcDecls) {
336 // Get the implementation details.
337 ServiceImplDetails svcDetails;
338 for (auto &detail : svcDecl.items())
339 svcDetails[detail.key()] = getAny(detail.value());
340
341 // Create the service.
342 auto serviceNameIter = svcDecl.find("serviceName");
343 std::string serviceName;
344 if (serviceNameIter != svcDecl.end())
345 serviceName = serviceNameIter.value();
346 services::Service::Type svcId =
347 services::ServiceRegistry::lookupServiceType(serviceName);
348 auto svc = acc.getService(svcId, /*id=*/{}, /*implName=*/"",
349 /*details=*/svcDetails, /*clients=*/{});
350 if (svc)
351 activeServices[svcDecl.at("symbol")] = svc;
352 }
353}
354
355std::vector<std::unique_ptr<Instance>>
356Manifest::Impl::getChildInstances(AppIDPath idPath, AcceleratorConnection &acc,
357 const ServiceTable &activeServices,
358 const nlohmann::json &instJson) const {
359 std::vector<std::unique_ptr<Instance>> ret;
360 auto childrenIter = instJson.find("children");
361 if (childrenIter == instJson.end())
362 return ret;
363 for (auto &child : childrenIter.value())
364 ret.emplace_back(getChildInstance(idPath, acc, activeServices, child));
365 return ret;
366}
367
368std::unique_ptr<Instance>
369Manifest::Impl::getChildInstance(AppIDPath idPath, AcceleratorConnection &acc,
370 ServiceTable activeServices,
371 const nlohmann::json &child) const {
372 AppID childID = parseIDChecked(child.at("appID"));
373 idPath.push_back(childID);
374
375 std::vector<services::Service *> services =
376 getServices(idPath, acc, child, activeServices);
377
378 auto children = getChildInstances(idPath, acc, activeServices, child);
379 auto ports = getBundlePorts(acc, idPath, activeServices, child);
380 return std::make_unique<Instance>(parseIDChecked(child.at("appID")),
381 getModInfo(child), std::move(children),
382 services, std::move(ports));
383}
384
385services::Service *Manifest::Impl::getService(AppIDPath idPath,
386 AcceleratorConnection &acc,
387 const nlohmann::json &svcJson,
388 ServiceTable &activeServices,
389 bool isEngine) const {
390
391 AppID id = parseIDChecked(svcJson.at("appID"));
392 idPath.push_back(id);
393
394 // Get all the client info, including the implementation details.
395 HWClientDetails clientDetails;
396 for (auto &client : svcJson.at("clientDetails")) {
397 HWClientDetail clientDetail;
398 for (auto &detail : client.items()) {
399 if (detail.key() == "relAppIDPath")
400 clientDetail.relPath = parseIDPath(detail.value());
401 else if (detail.key() == "servicePort")
402 clientDetail.port = parseServicePort(detail.value());
403 else if (detail.key() == "channelAssignments") {
404 for (auto &chan : detail.value().items()) {
405 ChannelAssignment chanAssign;
406 for (auto &assign : chan.value().items())
407 if (assign.key() == "type")
408 chanAssign.type = assign.value();
409 else
410 chanAssign.implOptions[assign.key()] = getAny(assign.value());
411 clientDetail.channelAssignments[chan.key()] = chanAssign;
412 }
413 } else
414 clientDetail.implOptions[detail.key()] = getAny(detail.value());
415 }
416 clientDetails.push_back(clientDetail);
417 }
418
419 // Get the implementation details.
420 ServiceImplDetails svcDetails;
421 std::string implName;
422 std::string service;
423 for (auto &detail : svcJson.items()) {
424 if (detail.key() == "appID" || detail.key() == "clientDetails")
425 continue;
426 if (detail.key() == "serviceImplName")
427 implName = detail.value();
428 else if (detail.key() == "service")
429 service = detail.value().get<std::string>();
430 else
431 svcDetails[detail.key()] = getAny(detail.value());
432 }
433
434 // Create the service.
435 services::Service *svc = nullptr;
436 auto activeServiceIter = activeServices.find(service);
437 if (activeServiceIter != activeServices.end()) {
438 services::Service::Type svcType =
439 services::ServiceRegistry::lookupServiceType(
440 activeServiceIter->second->getServiceSymbol());
441 svc = activeServiceIter->second->getChildService(svcType, idPath, implName,
442 svcDetails, clientDetails);
443 } else {
444 services::Service::Type svcType =
445 services::ServiceRegistry::lookupServiceType(service);
446 if (isEngine)
447 acc.createEngine(implName, idPath, svcDetails, clientDetails);
448 else
449 svc =
450 acc.getService(svcType, idPath, implName, svcDetails, clientDetails);
451 }
452
453 if (svc)
454 // Update the active services table.
455 activeServices[service] = svc;
456 return svc;
457}
458
459std::vector<services::Service *>
460Manifest::Impl::getServices(AppIDPath idPath, AcceleratorConnection &acc,
461 const nlohmann::json &svcsJson,
462 ServiceTable &activeServices) const {
463 std::vector<services::Service *> ret;
464 auto svcsIter = svcsJson.find("services");
465 if (svcsIter == svcsJson.end())
466 return ret;
467
468 for (auto &svc : svcsIter.value())
469 ret.emplace_back(getService(idPath, acc, svc, activeServices));
470 return ret;
471}
472
473std::vector<std::unique_ptr<BundlePort>>
474Manifest::Impl::getBundlePorts(AcceleratorConnection &acc, AppIDPath idPath,
475 const ServiceTable &activeServices,
476 const nlohmann::json &instJson) const {
477 std::vector<std::unique_ptr<BundlePort>> ret;
478 auto clientPortsIter = instJson.find("clientPorts");
479 if (clientPortsIter == instJson.end())
480 return ret;
481
482 for (auto &content : clientPortsIter.value()) {
483 // Lookup the requested service in the active services table.
484 std::string serviceName = "";
485 if (auto f = content.find("servicePort"); f != content.end())
486 serviceName = parseServicePort(f.value()).name;
487 auto svcIter = activeServices.find(serviceName);
488 if (svcIter == activeServices.end()) {
489 // If a specific service isn't found, search for the default service
490 // (typically provided by a BSP).
491 if (svcIter = activeServices.find(""); svcIter == activeServices.end())
492 throw std::runtime_error(
493 "Malformed manifest: could not find active service '" +
494 serviceName + "'");
495 }
496 services::Service *svc = svcIter->second;
497
498 std::string typeName = content.at("typeID");
499 auto type = getType(typeName);
500 if (!type)
501 throw std::runtime_error(
502 "Malformed manifest: could not find port type '" + typeName + "'");
503 const BundleType *bundleType = dynamic_cast<const BundleType *>(*type);
504 if (!bundleType)
505 throw std::runtime_error("Malformed manifest: type '" + typeName +
506 "' is not a bundle type");
507
508 idPath.push_back(parseIDChecked(content.at("appID")));
509
510 BundlePort *svcPort = svc->getPort(idPath, bundleType);
511 if (svcPort)
512 ret.emplace_back(svcPort);
513 // Since we share idPath between iterations, pop the last element before the
514 // next iteration.
515 idPath.pop_back();
516 }
517 return ret;
518}
519
520namespace {
521const Type *parseType(const nlohmann::json &typeJson, Context &ctxt);
522
523BundleType *parseBundleType(const nlohmann::json &typeJson, Context &cache) {
524 assert(typeJson.at("mnemonic") == "bundle");
525
526 std::vector<std::tuple<std::string, BundleType::Direction, const Type *>>
527 channels;
528 for (auto &chanJson : typeJson["channels"]) {
529 std::string dirStr = chanJson.at("direction");
530 BundleType::Direction dir;
531 if (dirStr == "to")
532 dir = BundleType::Direction::To;
533 else if (dirStr == "from")
534 dir = BundleType::Direction::From;
535 else
536 throw std::runtime_error("Malformed manifest: unknown direction '" +
537 dirStr + "'");
538 channels.emplace_back(chanJson.at("name"), dir,
539 parseType(chanJson["type"], cache));
540 }
541 return new BundleType(typeJson.at("id"), channels);
542}
543
544ChannelType *parseChannelType(const nlohmann::json &typeJson, Context &cache) {
545 assert(typeJson.at("mnemonic") == "channel");
546 return new ChannelType(typeJson.at("id"),
547 parseType(typeJson.at("inner"), cache));
548}
549
550Type *parseInt(const nlohmann::json &typeJson, Context &cache) {
551 assert(typeJson.at("mnemonic") == "int");
552 std::string sign = typeJson.at("signedness");
553 uint64_t width = typeJson.at("hwBitwidth");
554 Type::ID id = typeJson.at("id");
555
556 if (sign == "signed")
557 return new SIntType(id, width);
558 else if (sign == "unsigned")
559 return new UIntType(id, width);
560 else if (sign == "signless" && width == 0)
561 // By convention, a zero-width signless integer is a void type.
562 return new VoidType(id);
563 else if (sign == "signless" && width > 0)
564 return new BitsType(id, width);
565 else
566 throw std::runtime_error("Malformed manifest: unknown sign '" + sign + "'");
567}
568
569StructType *parseStruct(const nlohmann::json &typeJson, Context &cache) {
570 assert(typeJson.at("mnemonic") == "struct");
571 std::vector<std::pair<std::string, const Type *>> fields;
572 for (auto &fieldJson : typeJson["fields"])
573 fields.emplace_back(fieldJson.at("name"),
574 parseType(fieldJson["type"], cache));
575 return new StructType(typeJson.at("id"), fields);
576}
577
578ArrayType *parseArray(const nlohmann::json &typeJson, Context &cache) {
579 assert(typeJson.at("mnemonic") == "array");
580 uint64_t size = typeJson.at("size");
581 return new ArrayType(typeJson.at("id"),
582 parseType(typeJson.at("element"), cache), size);
583}
584
585WindowType *parseWindow(const nlohmann::json &typeJson, Context &cache) {
586 assert(typeJson.at("mnemonic") == "window");
587 std::string name = typeJson.at("name");
588 const Type *intoType = parseType(typeJson.at("into"), cache);
589 const Type *loweredType = parseType(typeJson.at("loweredType"), cache);
590
591 // Parse the frames information.
592 std::vector<WindowType::Frame> frames;
593 for (auto &frameJson : typeJson.at("frames")) {
594 WindowType::Frame frame;
595 frame.name = frameJson.at("name");
596 for (auto &fieldJson : frameJson.at("fields")) {
597 WindowType::Field field;
598 field.name = fieldJson.at("name");
599 if (fieldJson.contains("numItems"))
600 field.numItems = fieldJson.at("numItems");
601 if (fieldJson.contains("bulkCountWidth"))
602 field.bulkCountWidth = fieldJson.at("bulkCountWidth");
603 frame.fields.push_back(field);
604 }
605 frames.push_back(frame);
606 }
607
608 return new WindowType(typeJson.at("id"), name, intoType, loweredType, frames);
609}
610
611ListType *parseList(const nlohmann::json &typeJson, Context &cache) {
612 assert(typeJson.at("mnemonic") == "list");
613 return new ListType(typeJson.at("id"),
614 parseType(typeJson.at("element"), cache));
615}
616
617using TypeParser = std::function<Type *(const nlohmann::json &, Context &)>;
618const std::map<std::string_view, TypeParser> typeParsers = {
619 {"bundle", parseBundleType},
620 {"channel", parseChannelType},
621 {"std::any", [](const nlohmann::json &typeJson,
622 Context &cache) { return new AnyType(typeJson.at("id")); }},
623 {"int", parseInt},
624 {"struct", parseStruct},
625 {"array", parseArray},
626 {"window", parseWindow},
627 {"list", parseList},
628
629};
630
631// Parse a type if it doesn't already exist in the cache.
632const Type *parseType(const nlohmann::json &typeJson, Context &cache) {
633 std::string id;
634 if (typeJson.is_string())
635 id = typeJson.get<std::string>();
636 else
637 id = typeJson.at("id");
638 if (std::optional<const Type *> t = cache.getType(id))
639 return *t;
640 if (typeJson.is_string())
641 throw std::runtime_error("malformed manifest: unknown type '" + id + "'");
642
643 Type *t;
644 std::string mnemonic = typeJson.at("mnemonic");
645 auto f = typeParsers.find(mnemonic);
646 if (f != typeParsers.end())
647 t = f->second(typeJson, cache);
648 else
649 // Types we don't know about are opaque.
650 t = new Type(id);
651
652 // Insert into the cache.
653 cache.registerType(t);
654 return t;
655}
656} // namespace
657
658const Type *Manifest::Impl::parseType(const nlohmann::json &typeJson) {
659 return ::parseType(typeJson, ctxt);
660}
661
662void Manifest::Impl::populateTypes(const nlohmann::json &typesJson) {
663 for (auto &typeJson : typesJson)
664 _typeTable.push_back(parseType(typeJson));
665}
666
667//===----------------------------------------------------------------------===//
668// Manifest class implementation.
669//===----------------------------------------------------------------------===//
670
671Manifest::Manifest(Context &ctxt, const std::string &jsonManifest)
672 : impl(new Impl(ctxt, jsonManifest)) {}
673
674Manifest::~Manifest() { delete impl; }
675
676uint32_t Manifest::getApiVersion() const {
677 return impl->at("apiVersion").get<uint32_t>();
678}
679
680std::vector<ModuleInfo> Manifest::getModuleInfos() const {
681 std::vector<ModuleInfo> ret;
682 for (auto &[symbol, info] : impl->getSymbolInfo())
683 ret.push_back(info);
684 return ret;
685}
686
687Accelerator *Manifest::buildAccelerator(AcceleratorConnection &acc) const {
688 try {
689 return acc.takeOwnership(impl->buildAccelerator(acc));
690 } catch (const std::exception &e) {
691 std::string msg = "malformed manifest: " + std::string(e.what());
692 if (getApiVersion() == 0)
693 msg += " (schema version 0 is not considered stable)";
694 throw std::runtime_error(msg);
695 }
696}
697
698const std::vector<const Type *> &Manifest::getTypeTable() const {
699 return impl->getTypeTable();
700}
701
702//===----------------------------------------------------------------------===//
703// POCO helpers.
704//===----------------------------------------------------------------------===//
705
706// Print a module info, including the extra metadata.
707std::ostream &operator<<(std::ostream &os, const ModuleInfo &m) {
708 auto printAny = [&os](std::any a) {
709 if (std::any_cast<Constant>(&a))
710 a = std::any_cast<Constant>(a).value;
711
712 const std::type_info &t = a.type();
713 if (t == typeid(std::string))
714 os << std::any_cast<std::string>(a);
715 else if (t == typeid(int64_t))
716 os << std::any_cast<int64_t>(a);
717 else if (t == typeid(uint64_t))
718 os << std::any_cast<uint64_t>(a);
719 else if (t == typeid(double))
720 os << std::any_cast<double>(a);
721 else if (t == typeid(bool))
722 os << std::any_cast<bool>(a);
723 else if (t == typeid(std::nullptr_t))
724 os << "null";
725 else
726 os << "unknown";
727 };
728
729 if (m.name)
730 os << *m.name << " ";
731 if (m.version)
732 os << *m.version << " ";
733 if (m.name || m.version)
734 os << std::endl;
735 if (m.repo || m.commitHash) {
736 os << " Version control: ";
737 if (m.repo)
738 os << *m.repo;
739 if (m.commitHash)
740 os << "@" << *m.commitHash;
741 os << std::endl;
742 }
743 if (m.summary)
744 os << " " << *m.summary;
745 os << "\n";
746
747 if (!m.constants.empty()) {
748 os << " Constants:\n";
749 for (auto &e : m.constants) {
750 os << " " << e.first << ": ";
751 printAny(e.second);
752 os << "\n";
753 }
754 }
755
756 if (!m.extra.empty()) {
757 os << " Extra metadata:\n";
758 for (auto &e : m.extra) {
759 os << " " << e.first << ": ";
760 printAny(e.second);
761 os << "\n";
762 }
763 }
764 return os;
765}
766
767namespace esi {
768AppIDPath AppIDPath::operator+(const AppIDPath &b) const {
769 AppIDPath ret = *this;
770 ret.insert(ret.end(), b.begin(), b.end());
771 return ret;
772}
773
774AppIDPath AppIDPath::parent() const {
775 AppIDPath ret = *this;
776 if (!ret.empty())
777 ret.pop_back();
778 return ret;
779}
780
781std::string AppIDPath::toStr() const {
782 std::ostringstream os;
783 os << *this;
784 return os.str();
785}
786
787bool operator<(const AppID &a, const AppID &b) {
788 if (a.name != b.name)
789 return a.name < b.name;
790 return a.idx < b.idx;
791}
792bool operator<(const AppIDPath &a, const AppIDPath &b) {
793 if (a.size() != b.size())
794 return a.size() < b.size();
795 for (size_t i = 0, e = a.size(); i < e; ++i)
796 if (a[i] != b[i])
797 return a[i] < b[i];
798 return false;
799}
800
801std::ostream &operator<<(std::ostream &os, const AppID &id) {
802 os << id.name;
803 if (id.idx)
804 os << "[" << *id.idx << "]";
805 return os;
806}
807std::ostream &operator<<(std::ostream &os, const AppIDPath &path) {
808 for (size_t i = 0, e = path.size(); i < e; ++i) {
809 if (i > 0)
810 os << '.';
811 os << path[i];
812 }
813 return os;
814}
815
816} // namespace esi
assert(baseType &&"element must be base type")
static ParseResult parseType(Type &result, StringRef name, AsmParser &parser)
Parse a type defined by this dialect.
static std::optional< AppID > parseID(const nlohmann::json &jsonID)
Definition Manifest.cpp:130
static AppIDPath parseIDPath(const nlohmann::json &jsonIDPath)
Definition Manifest.cpp:147
static AppID parseIDChecked(const nlohmann::json &jsonID)
Definition Manifest.cpp:141
static ServicePortDesc parseServicePort(const nlohmann::json &jsonPort)
Definition Manifest.cpp:154
static llvm::raw_ostream & operator<<(llvm::raw_ostream &os, const bool a)
This class provides a thread-safe interface to access the analysis results.
const std::vector< const Type * > & getTypeTable() const
Get the ordered list of types from the manifest.
Definition Manifest.cpp:95
std::optional< const Type * > getType(Type::ID id) const
Definition Manifest.cpp:112
const Type * parseType(const nlohmann::json &typeJson)
Definition Manifest.cpp:658
auto at(const std::string &key) const
Definition Manifest.cpp:43
void populateTypes(const nlohmann::json &typesJson)
Parse all the types and populate the types table.
Definition Manifest.cpp:662
std::optional< ModuleInfo > getModInfo(const nlohmann::json &) const
Definition Manifest.cpp:314
void createEngine(AcceleratorConnection &, AppIDPath appID, const nlohmann::json &) const
TODO: Hack.
Definition Manifest.cpp:326
std::vector< std::unique_ptr< BundlePort > > getBundlePorts(AcceleratorConnection &acc, AppIDPath idPath, const ServiceTable &activeServices, const nlohmann::json &instJson) const
Get the bundle ports for the instance at 'idPath' and specified in 'instJson'.
Definition Manifest.cpp:474
std::vector< std::unique_ptr< Instance > > getChildInstances(AppIDPath idPath, AcceleratorConnection &acc, const ServiceTable &activeServices, const nlohmann::json &instJson) const
Build the set of child instances (recursively) for the module instance description.
Definition Manifest.cpp:356
nlohmann::json manifestJson
Definition Manifest.cpp:121
std::vector< services::Service * > getServices(AppIDPath idPath, AcceleratorConnection &, const nlohmann::json &, ServiceTable &activeServices) const
Get all the services in the description of an instance.
Definition Manifest.cpp:460
std::map< std::string, const ModuleInfo > symbolInfoCache
Definition Manifest.cpp:123
services::Service * getService(AppIDPath idPath, AcceleratorConnection &, const nlohmann::json &, ServiceTable &activeServices, bool isEngine=false) const
Get a Service for the service specified in 'json'.
Definition Manifest.cpp:385
const std::map< std::string, const ModuleInfo > & getSymbolInfo() const
Definition Manifest.cpp:104
std::any getAny(const nlohmann::json &value) const
Convert the json value to a 'std::any', which can be exposed outside of this file.
Definition Manifest.cpp:161
void parseModuleMetadata(ModuleInfo &info, const nlohmann::json &mod) const
Definition Manifest.cpp:220
std::unique_ptr< Accelerator > buildAccelerator(AcceleratorConnection &acc) const
Build a dynamic API for the Accelerator connection 'acc' based on the manifest stored herein.
Definition Manifest.cpp:284
void parseModuleConsts(ModuleInfo &info, const nlohmann::json &mod) const
Definition Manifest.cpp:241
void scanServiceDecls(AcceleratorConnection &, const nlohmann::json &, ServiceTable &) const
Go through the "service_decls" section of the manifest and populate the services table as appropriate...
Definition Manifest.cpp:332
Context & ctxt
Definition Manifest.cpp:109
std::vector< const Type * > _typeTable
Definition Manifest.cpp:110
std::unique_ptr< Instance > getChildInstance(AppIDPath idPath, AcceleratorConnection &acc, ServiceTable activeServices, const nlohmann::json &childJson) const
Get a single child instance.
Definition Manifest.cpp:369
Impl(Context &ctxt, int port)
Start a server on the given port. -1 means to let the OS pick a port.
void info(Twine message)
Definition LSPUtils.cpp:20
Definition esi.py:1
std::string name
Definition Common.h:36
std::optional< uint32_t > idx
Definition Common.h:37