diff --git a/extras/Hadrons/Environment.hpp b/extras/Hadrons/Environment.hpp index 3b1d45f8..637962b1 100644 --- a/extras/Hadrons/Environment.hpp +++ b/extras/Hadrons/Environment.hpp @@ -116,6 +116,10 @@ public: Ts && ... args); void setObjectModule(const unsigned int objAddress, const int modAddress); + template + T * getDerivedObject(const unsigned int address) const; + template + T * getDerivedObject(const std::string name) const; template T * getObject(const unsigned int address) const; template @@ -186,7 +190,7 @@ Holder::Holder(T *pt) template T & Holder::get(void) const { - return &objPt_.get(); + return *objPt_.get(); } template @@ -231,7 +235,7 @@ void Environment::createDerivedObject(const std::string name, object_[address].Ls = Ls; object_[address].data.reset(new Holder(new T(std::forward(args)...))); object_[address].size = MemoryProfiler::stats->maxAllocated - initMem; - object_[address].type = &typeid(T); + object_[address].type = &typeid(B); if (MemoryProfiler::stats == &memStats) { MemoryProfiler::stats = nullptr; @@ -241,7 +245,7 @@ void Environment::createDerivedObject(const std::string name, else if ((object_[address].storage != Storage::cache) or (object_[address].storage != storage) or (object_[address].name != name) or - (object_[address].type != &typeid(T))) + (object_[address].type != &typeid(B))) { HADRON_ERROR(Definition, "object '" + name + "' already allocated"); } @@ -256,21 +260,37 @@ void Environment::createObject(const std::string name, createDerivedObject(name, storage, Ls, std::forward(args)...); } -template -T * Environment::getObject(const unsigned int address) const +template +T * Environment::getDerivedObject(const unsigned int address) const { if (hasObject(address)) { if (hasCreatedObject(address)) { - if (auto h = dynamic_cast *>(object_[address].data.get())) + if (auto h = dynamic_cast *>(object_[address].data.get())) { - return h->getPt(); + if (&typeid(T) == &typeid(B)) + { + return dynamic_cast(h->getPt()); + } + else + { + if (auto hder = dynamic_cast(h->getPt())) + { + return hder; + } + else + { + HADRON_ERROR(Definition, "object with address " + std::to_string(address) + + " cannot be casted to '" + typeName(&typeid(T)) + + "' (has type '" + typeName(&typeid(h->get())) + "')"); + } + } } else { HADRON_ERROR(Definition, "object with address " + std::to_string(address) + - " does not have type '" + typeName(&typeid(T)) + + " does not have type '" + typeName(&typeid(B)) + "' (has type '" + getObjectType(address) + "')"); } } @@ -286,6 +306,18 @@ T * Environment::getObject(const unsigned int address) const } } +template +T * Environment::getDerivedObject(const std::string name) const +{ + return getDerivedObject(getObjectAddress(name)); +} + +template +T * Environment::getObject(const unsigned int address) const +{ + return getDerivedObject(address); +} + template T * Environment::getObject(const std::string name) const { diff --git a/extras/Hadrons/Module.hpp b/extras/Hadrons/Module.hpp index 018a26f7..85c27472 100644 --- a/extras/Hadrons/Module.hpp +++ b/extras/Hadrons/Module.hpp @@ -91,6 +91,9 @@ static ns##mod##ModuleRegistrar ns##mod##ModuleRegistrarInstance; #define envGet(type, name)\ *env().template getObject(name) +#define envGetDerived(base, type, name)\ +*env().template getDerivedObject(name) + #define envGetTmp(type, var)\ type &var = *env().template getObject(getName() + "_tmp_" + #var)