/* * ORXONOX - the hottest 3D action shooter ever to exist * > www.orxonox.net < * * * License notice: * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * * Author: * Fabian 'x3n' Landau * Co-authors: * ... * */ /** @file @brief Implementation of the Identifier class. */ #include "Identifier.h" #include #include "util/StringUtils.h" #include "core/CoreIncludes.h" #include "core/config/ConfigValueContainer.h" #include "core/XMLPort.h" #include "core/object/ClassFactory.h" namespace orxonox { bool Identifier::initConfigValues_s = true; // ############################### // ### Identifier ### // ############################### /** @brief Constructor: No factory, no object created, new ObjectList and a unique networkID. */ Identifier::Identifier(const std::string& name, Factory* factory, bool bLoadable) { orxout(verbose, context::identifier) << "Create identifier for " << name << endl; static unsigned int classIDCounter = 0; this->classID_ = classIDCounter++; this->name_ = name; this->factory_ = factory; this->bLoadable_ = bLoadable; this->bInitialized_ = false; this->bIsVirtualBase_ = false; this->bHasConfigValues_ = false; // Default network ID is the class ID this->networkID_ = this->classID_; } /** @brief Destructor: Deletes the list containing the children. */ Identifier::~Identifier() { if (this->factory_) delete this->factory_; for (const InheritsFrom* manualDirectParent : this->manualDirectParents_) delete manualDirectParent; // erase this Identifier from all related identifiers for (const Identifier* parent : this->parents_) const_cast(parent)->children_.erase(this); for (const Identifier* directParent : this->directParents_) const_cast(directParent)->directChildren_.erase(this); for (const Identifier* child : this->children_) const_cast(child)->parents_.remove(this); for (const Identifier* directChild : this->directChildren_) const_cast(directChild)->directParents_.remove(this); for (const auto& mapEntry : this->configValues_) delete (mapEntry.second); for (const auto& mapEntry : this->xmlportParamContainers_) delete (mapEntry.second); for (const auto& mapEntry : this->xmlportObjectContainers_) delete (mapEntry.second); } /** @brief Creates an object of the type the Identifier belongs to. @return The new object */ Identifiable* Identifier::fabricate(Context* context) { if (this->factory_) { return this->factory_->fabricate(context); } else { orxout(user_error) << "An error occurred in Identifier.cc:" << endl; orxout(user_error) << "Cannot fabricate an object of type '" << this->name_ << "'. Class has no factory." << endl; orxout(user_error) << "Aborting..." << endl; abort(); return nullptr; } } /** @brief Sets the network ID to a new value and changes the entry in the ID-Identifier-map. */ void Identifier::setNetworkID(uint32_t id) { this->networkID_ = id; IdentifierManager::getInstance().addIdentifier(this); // add with new id } /** * @brief Used to define the direct parents of an Identifier of an abstract class. */ Identifier& Identifier::inheritsFrom(InheritsFrom* directParent) { if (this->directParents_.empty()) this->manualDirectParents_.push_back(directParent); else orxout(internal_error) << "Trying to manually add direct parent of " << this->getName() << " after the latter was already initialized" << endl; return *this; } /** * @brief Initializes the parents of this Identifier while creating the class hierarchy. * @param initializationTrace All identifiers that were recorded while creating an instance of this class (including nested classes and this identifier itself) */ void Identifier::initializeParents(const std::list& initializationTrace) { if (!IdentifierManager::getInstance().isCreatingHierarchy()) { orxout(internal_warning) << "Identifier::initializeParents() created outside of class hierarchy creation" << endl; return; } if (this->directParents_.empty()) { for (const Identifier* identifier : initializationTrace) if (identifier != this) this->parents_.push_back(identifier); } else orxout(internal_error) << "Trying to add parents to " << this->getName() << " after it was already initialized with manual calls to inheritsFrom()." << endl; } /** * @brief Finishes the initialization of this Identifier after creating the class hierarchy by wiring the (direct) parent/child references correctly. */ void Identifier::finishInitialization() { if (!IdentifierManager::getInstance().isCreatingHierarchy()) { orxout(internal_warning) << "Identifier::finishInitialization() created outside of class hierarchy creation" << endl; return; } if (this->isInitialized()) return; if (!this->parents_.empty()) { // parents defined -> this class was initialized by creating a sample instance and recording the trace of identifiers // initialize all parents for (const Identifier* parent : this->parents_) const_cast(parent)->finishInitialization(); // initialize parent // parents of parents are no direct parents of this identifier this->directParents_ = this->parents_; for (const Identifier* parent : this->parents_) for (const Identifier* parent_parent : const_cast(parent)->parents_) this->directParents_.remove(parent_parent); this->verifyIdentifierTrace(); } else if (!this->manualDirectParents_.empty()) { // no parents defined -> this class was manually initialized by calling inheritsFrom() // initialize all direct parents for (const InheritsFrom* manualDirectParent : this->manualDirectParents_) { Identifier* directParent = manualDirectParent->getParent(); this->directParents_.push_back(directParent); directParent->finishInitialization(); // initialize parent } // direct parents and their parents are also parents of this identifier (but only add them once) for (const Identifier* parent : this->directParents_) { for (const Identifier* parent_parent : const_cast(parent)->parents_) this->addIfNotExists(this->parents_, parent_parent); this->addIfNotExists(this->parents_, parent); } } else if (!this->isExactlyA(Class(Identifiable))) { // only Identifiable is allowed to have no parents (even tough it's currently not abstract) orxout(internal_error) << "Identifier " << this->getName() << " / " << this->getTypeInfo().name() << " is marked as abstract but has no direct parents defined" << endl; orxout(internal_error) << " If this class is not abstract, use RegisterClass(ThisClass);" << endl; orxout(internal_error) << " If this class is abstract, use RegisterAbstractClass(ThisClass).inheritsFrom(Class(BaseClass));" << endl; } // tell all parents that this identifier is a child for (const Identifier* parent : this->parents_) const_cast(parent)->children_.insert(this); // tell all direct parents that this identifier is a direct child for (const Identifier* directChild : this->directParents_) { const_cast(directChild)->directChildren_.insert(this); // Create the super-function dependencies directChild->createSuperFunctionCaller(); } this->bInitialized_ = true; } /** * Resets all information about the class hierarchy. The identifier is considered uninitialized afterwards. */ void Identifier::reset() { this->directParents_.clear(); this->parents_.clear(); this->directChildren_.clear(); this->children_.clear(); this->bInitialized_ = false; } /** * Verifies if the recorded trace of parent identifiers matches the expected trace according to the class hierarchy. If it doesn't match, the class * hierarchy is likely wrong, e.g. due to wrong inheritsFrom<>() definitions in abstract classes. */ void Identifier::verifyIdentifierTrace() const { std::list expectedIdentifierTrace; // if any parent class is virtual, it will be instantiated first, so we need to add them first. for (const Identifier* parent : this->parents_) { if (parent->isVirtualBase()) { for (const Identifier* parent_parent : const_cast(parent)->parents_) this->addIfNotExists(expectedIdentifierTrace, parent_parent); this->addIfNotExists(expectedIdentifierTrace, parent); } } // now all direct parents get created recursively. already added identifiers (e.g. virtual base classes) are not added again. for (const Identifier* directParent : this->directParents_) { for (const Identifier* parent_parent : const_cast(directParent)->parents_) this->addIfNotExists(expectedIdentifierTrace, parent_parent); this->addIfNotExists(expectedIdentifierTrace, directParent); } // check if the expected trace matches the actual trace (which was defined by creating a sample instance) if (expectedIdentifierTrace != this->parents_) { orxout(internal_warning) << this->getName() << " has an unexpected initialization trace:" << endl; orxout(internal_warning) << " Actual trace (after creating a sample instance):" << endl << " "; for (const Identifier* parent : this->parents_) orxout(internal_warning) << " " << parent->getName(); orxout(internal_warning) << endl; orxout(internal_warning) << " Expected trace (according to class hierarchy definitions):" << endl << " "; for (const Identifier* parent : expectedIdentifierTrace) orxout(internal_warning) << " " << parent->getName(); orxout(internal_warning) << endl; orxout(internal_warning) << " Direct parents (according to class hierarchy definitions):" << endl << " "; for (const Identifier* directParent : this->directParents_) orxout(internal_warning) << " " << directParent->getName(); orxout(internal_warning) << endl; } } /** * Adds @param identifierToAdd to @param list if this identifier is not already contained in the list. */ void Identifier::addIfNotExists(std::list& list, const Identifier* identifierToAdd) const { if (std::find(list.begin(), list.end(), identifierToAdd) == list.end()) list.push_back(identifierToAdd); } /** @brief Returns true, if the Identifier is at least of the given type. @param identifier The identifier to compare with */ bool Identifier::isA(const Identifier* identifier) const { return (identifier == this || (this->isChildOf(identifier))); } /** @brief Returns true, if the Identifier is exactly of the given type. @param identifier The identifier to compare with */ bool Identifier::isExactlyA(const Identifier* identifier) const { return (identifier == this); } /** @brief Returns true, if the assigned identifier is a child of the given identifier. @param identifier The identifier to compare with */ bool Identifier::isChildOf(const Identifier* identifier) const { return (std::find(this->parents_.begin(), this->parents_.end(), identifier) != this->parents_.end()); } /** @brief Returns true, if the assigned identifier is a direct child of the given identifier. @param identifier The identifier to compare with */ bool Identifier::isDirectChildOf(const Identifier* identifier) const { return (std::find(this->directParents_.begin(), this->directParents_.end(), identifier) != this->directParents_.end()); } /** @brief Returns true, if the assigned identifier is a parent of the given identifier. @param identifier The identifier to compare with */ bool Identifier::isParentOf(const Identifier* identifier) const { return (this->children_.find(identifier) != this->children_.end()); } /** @brief Returns true, if the assigned identifier is a direct parent of the given identifier. @param identifier The identifier to compare with */ bool Identifier::isDirectParentOf(const Identifier* identifier) const { return (this->directChildren_.find(identifier) != this->directChildren_.end()); } /** @brief Adds the ConfigValueContainer of a variable, given by the string of its name. @param varname The name of the variablee @param container The container */ void Identifier::addConfigValueContainer(const std::string& varname, ConfigValueContainer* container) { std::map::const_iterator it = this->configValues_.find(varname); if (it != this->configValues_.end()) { orxout(internal_warning) << "Overwriting config-value with name " << varname << " in class " << this->getName() << '.' << endl; delete (it->second); } this->bHasConfigValues_ = true; this->configValues_[varname] = container; } /** @brief Returns the ConfigValueContainer of a variable, given by the string of its name. @param varname The name of the variable @return The ConfigValueContainer */ ConfigValueContainer* Identifier::getConfigValueContainer(const std::string& varname) { std::map::const_iterator it = configValues_.find(varname); if (it != configValues_.end()) return it->second; else return nullptr; } /** @brief Returns a XMLPortParamContainer that loads a parameter of this class. @param paramname The name of the parameter @return The container */ XMLPortParamContainer* Identifier::getXMLPortParamContainer(const std::string& paramname) { std::map::const_iterator it = this->xmlportParamContainers_.find(paramname); if (it != this->xmlportParamContainers_.end()) return it->second; else return nullptr; } /** @brief Adds a new XMLPortParamContainer that loads a parameter of this class. @param paramname The name of the parameter @param container The container */ void Identifier::addXMLPortParamContainer(const std::string& paramname, XMLPortParamContainer* container) { std::map::const_iterator it = this->xmlportParamContainers_.find(paramname); if (it != this->xmlportParamContainers_.end()) { orxout(internal_warning) << "Overwriting XMLPortParamContainer in class " << this->getName() << '.' << endl; delete (it->second); } this->xmlportParamContainers_[paramname] = container; } /** @brief Returns a XMLPortObjectContainer that attaches an object to this class. @param sectionname The name of the section that contains the attachable objects @return The container */ XMLPortObjectContainer* Identifier::getXMLPortObjectContainer(const std::string& sectionname) { std::map::const_iterator it = this->xmlportObjectContainers_.find(sectionname); if (it != this->xmlportObjectContainers_.end()) return it->second; else return nullptr; } /** @brief Adds a new XMLPortObjectContainer that attaches an object to this class. @param sectionname The name of the section that contains the attachable objects @param container The container */ void Identifier::addXMLPortObjectContainer(const std::string& sectionname, XMLPortObjectContainer* container) { std::map::const_iterator it = this->xmlportObjectContainers_.find(sectionname); if (it != this->xmlportObjectContainers_.end()) { orxout(internal_warning) << "Overwriting XMLPortObjectContainer in class " << this->getName() << '.' << endl; delete (it->second); } this->xmlportObjectContainers_[sectionname] = container; } /** @brief Lists the names of all Identifiers in a std::set. @param out The outstream @param list The list (or set) of Identifiers @return The outstream */ std::ostream& operator<<(std::ostream& out, const std::set& list) { for (std::set::const_iterator it = list.begin(); it != list.end(); ++it) { if (it != list.begin()) out << ' '; out << (*it)->getName(); } return out; } }