[6607] | 1 | /*! |
---|
| 2 | \file declare_ref_func.hpp |
---|
| 3 | \author Ha NGUYEN |
---|
| 4 | \date 02 Dec 2014 |
---|
| 5 | \since 02 Dec 2014 |
---|
| 6 | |
---|
| 7 | \brief Macros to add functions used to solve references in class CDomain, CAxis and CField |
---|
| 8 | */ |
---|
| 9 | #ifndef __XIOS_DECLARE_REF_FUNC_HPP__ |
---|
| 10 | #define __XIOS_DECLARE_REF_FUNC_HPP__ |
---|
| 11 | |
---|
| 12 | // Declarations |
---|
| 13 | |
---|
| 14 | #define DECLARE_REF_FUNC(type, name_) \ |
---|
| 15 | public: \ |
---|
| 16 | void solveRefInheritance(bool apply = true); \ |
---|
| 17 | void removeRefInheritance(); \ |
---|
| 18 | bool hasDirect##type##Reference(void) const; \ |
---|
| 19 | C##type* getDirect##type##Reference(void) const; \ |
---|
| 20 | const StdString get##type##OutputName(void) const; \ |
---|
| 21 | void setAttributesReference(bool apply = true); \ |
---|
| 22 | bool hasRefTo(C##type* ref) const; \ |
---|
| 23 | \ |
---|
| 24 | private: \ |
---|
| 25 | std::vector<C##type*> refObjects; |
---|
| 26 | |
---|
| 27 | // Definitions |
---|
| 28 | |
---|
| 29 | #define DEFINE_REF_FUNC(type, name_) \ |
---|
| 30 | void C##type::solveRefInheritance(bool apply) \ |
---|
| 31 | { \ |
---|
| 32 | std::set<C##type*> tmpRefObjects; \ |
---|
| 33 | C##type* refer_ptr = this; \ |
---|
| 34 | std::vector<C##type*>().swap(refObjects); \ |
---|
| 35 | refObjects.push_back(this); \ |
---|
| 36 | \ |
---|
| 37 | while (refer_ptr->hasDirect##type##Reference()) \ |
---|
| 38 | { \ |
---|
| 39 | tmpRefObjects.insert(refer_ptr); \ |
---|
| 40 | \ |
---|
| 41 | refer_ptr = refer_ptr->getDirect##type##Reference(); \ |
---|
| 42 | \ |
---|
| 43 | if (tmpRefObjects.end() != tmpRefObjects.find(refer_ptr)) \ |
---|
| 44 | { \ |
---|
| 45 | ERROR("void C" #type "::solveRefInheritance(bool apply)", \ |
---|
| 46 | << "Circular dependency stopped for " #name_ " object " \ |
---|
| 47 | << "with id = \"" << refer_ptr->getId() << "\"."); \ |
---|
| 48 | } \ |
---|
| 49 | \ |
---|
| 50 | refObjects.push_back(refer_ptr); \ |
---|
| 51 | SuperClassAttribute::setAttributes(refer_ptr, apply); \ |
---|
| 52 | } \ |
---|
| 53 | if (this->hasAttribute("name") && this->name.isEmpty()) \ |
---|
| 54 | this->name.setValue(this->get##type##OutputName()); \ |
---|
| 55 | } \ |
---|
| 56 | \ |
---|
| 57 | void C##type::setAttributesReference(bool apply) \ |
---|
| 58 | { \ |
---|
| 59 | for (int i = 1; i < refObjects.size(); ++i) \ |
---|
| 60 | refObjects[i]->setAttributes(refObjects[i-1], apply); \ |
---|
| 61 | if (refObjects.size() > 1) \ |
---|
| 62 | refObjects[refObjects.size()-1]->removeRefInheritance(); \ |
---|
| 63 | } \ |
---|
| 64 | \ |
---|
| 65 | void C##type::removeRefInheritance() \ |
---|
| 66 | { \ |
---|
| 67 | if (!this->name_##_ref.isEmpty()) \ |
---|
| 68 | this->name_##_ref.reset(); \ |
---|
| 69 | } \ |
---|
| 70 | \ |
---|
| 71 | bool C##type::hasDirect##type##Reference(void) const \ |
---|
| 72 | { \ |
---|
| 73 | return (!this->name_##_ref.isEmpty() && \ |
---|
| 74 | C##type::has(this->name_##_ref)); \ |
---|
| 75 | } \ |
---|
| 76 | \ |
---|
| 77 | C##type* C##type::getDirect##type##Reference(void) const \ |
---|
| 78 | { \ |
---|
| 79 | if (this->name_##_ref.isEmpty()) \ |
---|
| 80 | ERROR("C" #type "* C" #type "::getDirect" #type "Reference(void)", \ |
---|
| 81 | << "The " #name_ " with id = '" << getId() << "'" \ |
---|
| 82 | << " has no " #name_ "_ref."); \ |
---|
| 83 | \ |
---|
| 84 | if (!C##type::has(this->name_##_ref)) \ |
---|
| 85 | ERROR("C" #type "* C" #type "::getDirect" #type "Reference(void)", \ |
---|
| 86 | << this->name_##_ref \ |
---|
| 87 | << " refers to an unknown " #name_ " id."); \ |
---|
| 88 | \ |
---|
| 89 | return C##type::get(this->name_##_ref); \ |
---|
| 90 | } \ |
---|
| 91 | \ |
---|
| 92 | const StdString C##type::get##type##OutputName(void) const \ |
---|
| 93 | { \ |
---|
| 94 | if (!this->name.isEmpty()) return this->name; \ |
---|
| 95 | else if (this->hasAutoGeneratedId() && hasDirect##type##Reference()) \ |
---|
| 96 | { \ |
---|
| 97 | const C##type* refer_ptr = this, *tmp_ptr; \ |
---|
| 98 | StdString nameRef = this->name_##_ref; \ |
---|
| 99 | std::set<const C##type*> tmpRefObjects; \ |
---|
| 100 | while (refer_ptr->hasAutoGeneratedId() && \ |
---|
| 101 | (C##type::has(nameRef))) \ |
---|
| 102 | { \ |
---|
| 103 | tmpRefObjects.insert(refer_ptr); \ |
---|
| 104 | tmp_ptr = refer_ptr; \ |
---|
| 105 | refer_ptr = tmp_ptr->getDirect##type##Reference(); \ |
---|
| 106 | if (refer_ptr->hasAutoGeneratedId() && \ |
---|
| 107 | refer_ptr->hasDirect##type##Reference()) \ |
---|
| 108 | nameRef = refer_ptr->name_##_ref; \ |
---|
| 109 | else { \ |
---|
| 110 | nameRef = refer_ptr->getId(); break; \ |
---|
| 111 | } \ |
---|
| 112 | if (tmpRefObjects.end() != tmpRefObjects.find(refer_ptr)) \ |
---|
| 113 | { \ |
---|
| 114 | ERROR("const StdString& C" #type "::get" #type "OutputName(void) const ", \ |
---|
| 115 | << "Circular dependency stopped for " #name_ " object " \ |
---|
| 116 | << "with id = \"" << refer_ptr->getId() << "\"."); \ |
---|
| 117 | } \ |
---|
| 118 | } \ |
---|
| 119 | return nameRef; \ |
---|
| 120 | } \ |
---|
| 121 | else \ |
---|
| 122 | return getId(); \ |
---|
| 123 | } \ |
---|
| 124 | \ |
---|
| 125 | bool C##type::hasRefTo(C##type* ref) const \ |
---|
| 126 | { \ |
---|
| 127 | bool found = false; \ |
---|
| 128 | for (int idx = 0; idx < refObjects.size(); ++idx) \ |
---|
| 129 | if (ref == refObjects[idx]) { found = true; break; } \ |
---|
| 130 | return found; \ |
---|
| 131 | } |
---|
| 132 | |
---|
| 133 | #endif // __XIOS_DECLARE_REF_FUNC_HPP__ |
---|