/** * @file * * @copyright * @verbatim Copyright @ 2017 Audi Electronics Venture GmbH. All rights reserved. This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0. If a copy of the MPL was not distributed with this file, You can obtain one at https://mozilla.org/MPL/2.0/. If it is not possible or desirable to put the notice in a particular file, then You may include the notice in a location (such as a LICENSE file in a relevant directory) where a recipient would be likely to look for such a notice. You may add additional accurate notices of copyright ownership. @endverbatim */ #include "ddlcontainer.h" #include "ddl_type.h" #include "ddlunit.h" #include "ddlbaseunit.h" #include "ddlprefix.h" #include "ddldatatype.h" #include "ddlenum.h" #include "ddlcomplex.h" #include "ddlstream.h" #include "ddlstreammetatype.h" namespace ddl { template struct LessCompare { bool operator()(const T* obj1, const T* obj2) { return obj1->getName() < obj2->getName(); } bool operator()(const T* obj, const std::string& name) { return obj->getName() < name; } bool operator()(const std::string& name, const T* obj) { return name < obj->getName(); } }; template DDLContainerNoClone::DDLContainerNoClone(bool sorted): _sorted(sorted) { } template void DDLContainerNoClone::insert(T* elem, int pos) { if (_sorted) { _pointers.insert(std::lower_bound(_pointers.begin(), _pointers.end(), elem, LessCompare()), elem); } else { const typename std::vector::size_type sz_pos = static_cast::size_type>(pos); if (0 <= sz_pos && sz_pos < _pointers.size()) { _pointers.insert(_pointers.begin() + sz_pos, elem); } else { _pointers.push_back(elem); } } } template typename DDLContainerNoClone::iterator DDLContainerNoClone::findIt(const std::string& name) { if (_sorted) { iterator it = std::lower_bound(this->begin(), this->end(), name, LessCompare()); if (this->end() == it || name != (*it)->getName()) { // not found return this->end(); } return it; } else { iterator it = std::find_if(this->begin(), this->end(), DDLCompareFunctor<>(name)); if (this->end() == it) { // not found return this->end(); } return it; } } template const T* DDLContainerNoClone::find(const std::string& name) const { if (_sorted) { return find_sorted(name); } return find_unsorted(name); } template T* DDLContainerNoClone::find(const std::string& name) { if (_sorted) { return find_sorted(name); } return find_unsorted(name); } template const T* DDLContainerNoClone::find_unsorted(const std::string& name) const { const_iterator it = std::find_if(this->begin(), this->end(), DDLCompareFunctor<>(name)); if (this->end() == it) { // not found return NULL; } return *it; } template const T* DDLContainerNoClone::find_sorted(const std::string& name) const { const_iterator it = std::lower_bound(this->begin(), this->end(), name, LessCompare()); if (this->end() == it || name != (*it)->getName()) { // not found return find_unsorted(name); } return *it; } template T* DDLContainerNoClone::find_unsorted(const std::string& name) { iterator it = std::find_if(this->begin(), this->end(), DDLCompareFunctor<>(name)); if (end() == it) { // not found return NULL; } return *it; } template T* DDLContainerNoClone::find_sorted(const std::string& name) { iterator it = std::lower_bound(this->begin(), this->end(), name, LessCompare()); if (end() == it || name != (*it)->getName()) { // not found auto obj = find_unsorted(name); if (obj != nullptr) { sort(); } return obj; } return *it; } template void DDLContainerNoClone::copyFromRef(DDLContainerNoClone& other) { _pointers.resize(other.size()); std::copy(other.begin(), other.end(), _pointers.begin()); if (_sorted && !other._sorted) { sort(); } } template void DDLContainerNoClone::deleteAll() { for (iterator it = begin(); it != end(); ++it) { DDL::deleteChild(*it); } clear(); } template void DDLContainerNoClone::sort() { std::sort(_pointers.begin(), _pointers.end(), LessCompare()); } template bool DDLContainerNoClone::isSorted() const { return _sorted; } // Note on our iterators: // we do not want to expose the std::vector and std::vector::iterator implementation to the user // so we use the simplest kind of iterator, a pointer. template typename DDLContainerNoClone::iterator DDLContainerNoClone::begin() { #if defined(WIN32) && ((_MSC_VER < 1600) || defined(_DEBUG)) // have a look at http://www.open-std.org/jtc1/sc22/wg21/docs/lwg-defects.html#464 and // http://stackoverflow.com/questions/3829788/using-operator-on-empty-stdvector // to find out the reason for this special implementation (VS2008 and lower) if (_pointers.empty()) { return &_empty_dummy; } else { return &*_pointers.begin(); } #else return &*_pointers.begin(); #endif } template typename DDLContainerNoClone::const_iterator DDLContainerNoClone::begin() const { #if defined(WIN32) && ((_MSC_VER < 1600) || defined(_DEBUG)) // have a look at http://www.open-std.org/jtc1/sc22/wg21/docs/lwg-defects.html#464 and // http://stackoverflow.com/questions/3829788/using-operator-on-empty-stdvector // to find out the reason for this special implementation (VS2008 and lower) if (_pointers.empty()) { return &_empty_dummy; } else { return &*_pointers.begin(); } #else return &*_pointers.begin(); #endif } template typename DDLContainerNoClone::iterator DDLContainerNoClone::end() { #if defined(WIN32) && ((_MSC_VER < 1600) || defined(_DEBUG)) // have a look at http://www.open-std.org/jtc1/sc22/wg21/docs/lwg-defects.html#464 and // http://stackoverflow.com/questions/3829788/using-operator-on-empty-stdvector // to find out the reason for this special implementation (VS2008 and lower) if (_pointers.empty()) { return &_empty_dummy; } else { return &*_pointers.begin() + _pointers.size(); } #else return &*_pointers.end(); #endif } template typename DDLContainerNoClone::const_iterator DDLContainerNoClone::end() const { #if defined(WIN32) && ((_MSC_VER < 1600) || defined(_DEBUG)) // have a look at http://www.open-std.org/jtc1/sc22/wg21/docs/lwg-defects.html#464 and // http://stackoverflow.com/questions/3829788/using-operator-on-empty-stdvector // to find out the reason for this special implementation (VS2008 and lower) if (_pointers.empty()) { return &_empty_dummy; } else { return &*_pointers.begin() + _pointers.size(); } #else return &*_pointers.end(); #endif } template void DDLContainerNoClone::clear() { _pointers.clear(); } template typename DDLContainerNoClone::iterator DDLContainerNoClone::erase(iterator it) { typename std::vector::iterator it_helper = _pointers.erase(_pointers.begin() + (it - begin())); #if defined(WIN32) && ((_MSC_VER < 1600) || defined(_DEBUG)) // have a look at http://www.open-std.org/jtc1/sc22/wg21/docs/lwg-defects.html#464 and // http://stackoverflow.com/questions/3829788/using-operator-on-empty-stdvector // to find out the reason for this special implementation (VS2008 and lower) if (it_helper == _pointers.end()) { return &_empty_dummy; } else { return &*it_helper; } #else return &*it_helper; #endif } template typename DDLContainerNoClone::iterator DDLContainerNoClone::erase(iterator pos_first, iterator pos_last) { if (pos_first == pos_last) { return end(); } #if defined(WIN32) && ((_MSC_VER < 1600) || defined(_DEBUG)) // have a look at http://www.open-std.org/jtc1/sc22/wg21/docs/lwg-defects.html#464 and // http://stackoverflow.com/questions/3829788/using-operator-on-empty-stdvector // to find out the reason for this special implementation (VS2008 and lower) // another bug in MSVS Debug http://connect.microsoft.com/VisualStudio/feedback/details/557029/visual-c-iterator-debugging-incorrectly-raises-assertion-on-correct-use-of-return-value-of-std-vector-erase // prevents us from using the erase(start, end) version. typename std::vector::iterator it_helper = _pointers.begin() + (pos_first - begin()); for (uint64_t nCount = pos_last - pos_first; nCount > 0; --nCount) { it_helper = _pointers.erase(it_helper); } if (it_helper == _pointers.end()) { return &_empty_dummy; } else { return &*it_helper; } #else typename std::vector::iterator it_helper = _pointers.erase(_pointers.begin() + (pos_first - begin()), _pointers.begin() + (pos_last - begin())); return &*it_helper; #endif } template bool DDLContainerNoClone::empty() const { return _pointers.empty(); } template unsigned int DDLContainerNoClone::size() const { return static_cast(_pointers.size()); } template T*& DDLContainerNoClone::operator[] (unsigned int pos) { return _pointers[pos]; } template T* const& DDLContainerNoClone::operator[] (unsigned int pos) const { return _pointers[pos]; } template T*& DDLContainerNoClone::at(unsigned int pos) { return _pointers.at(pos); } template T* const& DDLContainerNoClone::at(unsigned int pos) const { return _pointers.at(pos); } template DDLContainer::DDLContainer(bool sorted): DDLContainerNoClone(sorted) { } template void DDLContainer::cloneFrom(const DDLContainer& other) { this->clear(); for (typename DDLContainer::const_iterator it = other.begin(); it != other.end(); ++it) { this->insert(DDL::clone(*it)); } } template class DDLContainerNoClone; template class DDLContainerNoClone; template class DDLContainerNoClone; template class DDLContainerNoClone; template class DDLContainerNoClone; template class DDLContainerNoClone; template class DDLContainerNoClone; template class DDLContainerNoClone; template class DDLContainerNoClone; #if defined(WIN32) && ((_MSC_VER < 1600) || defined(_DEBUG)) IDDL* DDLContainerNoClone::_empty_dummy = NULL; DDLUnit* DDLContainerNoClone::_empty_dummy = NULL; DDLBaseunit* DDLContainerNoClone::_empty_dummy = NULL; DDLPrefix* DDLContainerNoClone::_empty_dummy = NULL; DDLDataType* DDLContainerNoClone::_empty_dummy = NULL; DDLEnum* DDLContainerNoClone::_empty_dummy = NULL; DDLComplex* DDLContainerNoClone::_empty_dummy = NULL; DDLStream* DDLContainerNoClone::_empty_dummy = NULL; DDLStreamMetaType* DDLContainerNoClone::_empty_dummy = NULL; #endif template class DDLContainer; template class DDLContainer; template class DDLContainer; template class DDLContainer; template class DDLContainer; template class DDLContainer; template class DDLContainer; template class DDLContainer; }