mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-26 19:56:37 +00:00 
			
		
		
		
	Convert the CellRef record index lookup maps to use integer keys rather than strings.
- Morrowind load over 300,000 references, so even small inefficiencies add up to longer loading times. - std::map is used, but should try others, std::unordered_map or even std::vector
This commit is contained in:
		
							parent
							
								
									0de223c637
								
							
						
					
					
						commit
						86945d1912
					
				
					 5 changed files with 211 additions and 8 deletions
				
			
		|  | @ -104,7 +104,7 @@ namespace CSMWorld | ||||||
|             const ESM::Dialogue *mDialogue; // last loaded dialogue
 |             const ESM::Dialogue *mDialogue; // last loaded dialogue
 | ||||||
|             bool mBase; |             bool mBase; | ||||||
|             bool mProject; |             bool mProject; | ||||||
|             std::map<std::string, std::map<unsigned int, std::string> > mRefLoadCache; |             std::map<std::string, std::map<unsigned int, unsigned int> > mRefLoadCache; | ||||||
|             int mReaderIndex; |             int mReaderIndex; | ||||||
|             std::vector<std::string> mLoadedFiles; |             std::vector<std::string> mLoadedFiles; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -3,6 +3,7 @@ | ||||||
| #include <cmath> | #include <cmath> | ||||||
| 
 | 
 | ||||||
| CSMWorld::CellRef::CellRef() | CSMWorld::CellRef::CellRef() | ||||||
|  |   : mIdNum(0) | ||||||
| { | { | ||||||
|     mId.clear(); |     mId.clear(); | ||||||
|     mCell.clear(); |     mCell.clear(); | ||||||
|  | @ -22,6 +23,7 @@ CSMWorld::CellRef& CSMWorld::CellRef::operator= (CSMWorld::CellRef&& other) | ||||||
|     if (this != &other) |     if (this != &other) | ||||||
|     { |     { | ||||||
|         ESM::CellRef::operator= (other); |         ESM::CellRef::operator= (other); | ||||||
|  |         mIdNum = other.mIdNum; | ||||||
|         mId = std::move(other.mId); |         mId = std::move(other.mId); | ||||||
|         mCell = std::move(other.mCell); |         mCell = std::move(other.mCell); | ||||||
|         mOriginalCell = std::move(other.mOriginalCell); |         mOriginalCell = std::move(other.mOriginalCell); | ||||||
|  |  | ||||||
|  | @ -10,6 +10,8 @@ namespace CSMWorld | ||||||
|     /// \brief Wrapper for CellRef sub record
 |     /// \brief Wrapper for CellRef sub record
 | ||||||
|     struct CellRef : public ESM::CellRef |     struct CellRef : public ESM::CellRef | ||||||
|     { |     { | ||||||
|  |         unsigned int mIdNum; | ||||||
|  | 
 | ||||||
|         std::string mId; |         std::string mId; | ||||||
|         std::string mCell; |         std::string mCell; | ||||||
|         std::string mOriginalCell; |         std::string mOriginalCell; | ||||||
|  |  | ||||||
|  | @ -11,8 +11,37 @@ | ||||||
| #include "universalid.hpp" | #include "universalid.hpp" | ||||||
| #include "record.hpp" | #include "record.hpp" | ||||||
| 
 | 
 | ||||||
|  | namespace CSMWorld | ||||||
|  | { | ||||||
|  |     template<> | ||||||
|  |     void Collection<CellRef, IdAccessor<CellRef> >::removeRows (int index, int count) | ||||||
|  |     { | ||||||
|  |         mRecords.erase(mRecords.begin()+index, mRecords.begin()+index+count); | ||||||
|  | 
 | ||||||
|  |         // index map is updated in RefCollection::removeRows()
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     template<> | ||||||
|  |     void Collection<CellRef, IdAccessor<CellRef> >::insertRecord (std::unique_ptr<RecordBase> record, int index, | ||||||
|  |         UniversalId::Type type) | ||||||
|  |     { | ||||||
|  |         int size = static_cast<int>(mRecords.size()); | ||||||
|  |         if (index < 0 || index > size) | ||||||
|  |             throw std::runtime_error("index out of range"); | ||||||
|  | 
 | ||||||
|  |         std::unique_ptr<Record<CellRef> > record2(static_cast<Record<CellRef>*>(record.release())); | ||||||
|  | 
 | ||||||
|  |         if (index == size) | ||||||
|  |             mRecords.push_back(std::move(record2)); | ||||||
|  |         else | ||||||
|  |             mRecords.insert(mRecords.begin()+index, std::move(record2)); | ||||||
|  | 
 | ||||||
|  |         // index map is updated in RefCollection::insertRecord()
 | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void CSMWorld::RefCollection::load (ESM::ESMReader& reader, int cellIndex, bool base, | void CSMWorld::RefCollection::load (ESM::ESMReader& reader, int cellIndex, bool base, | ||||||
|     std::map<unsigned int, std::string>& cache, CSMDoc::Messages& messages) |     std::map<unsigned int, unsigned int>& cache, CSMDoc::Messages& messages) | ||||||
| { | { | ||||||
|     Record<Cell> cell = mCells.getRecord (cellIndex); |     Record<Cell> cell = mCells.getRecord (cellIndex); | ||||||
| 
 | 
 | ||||||
|  | @ -74,7 +103,7 @@ void CSMWorld::RefCollection::load (ESM::ESMReader& reader, int cellIndex, bool | ||||||
|         unsigned int  refNum = (ref.mRefNum.mIndex & 0x00ffffff) | |         unsigned int  refNum = (ref.mRefNum.mIndex & 0x00ffffff) | | ||||||
|             (ref.mRefNum.hasContentFile() ? ref.mRefNum.mContentFile : 0xff) << 24; |             (ref.mRefNum.hasContentFile() ? ref.mRefNum.mContentFile : 0xff) << 24; | ||||||
| 
 | 
 | ||||||
|         std::map<unsigned int, std::string>::iterator iter = cache.find(refNum); |         std::map<unsigned int, unsigned int>::iterator iter = cache.find(refNum); | ||||||
| 
 | 
 | ||||||
|         if (isDeleted) |         if (isDeleted) | ||||||
|         { |         { | ||||||
|  | @ -83,7 +112,11 @@ void CSMWorld::RefCollection::load (ESM::ESMReader& reader, int cellIndex, bool | ||||||
|                 CSMWorld::UniversalId id (CSMWorld::UniversalId::Type_Cell, |                 CSMWorld::UniversalId id (CSMWorld::UniversalId::Type_Cell, | ||||||
|                     mCells.getId (cellIndex)); |                     mCells.getId (cellIndex)); | ||||||
| 
 | 
 | ||||||
|                 messages.add (id, "Attempt to delete a non-existing reference"); |                 messages.add (id, "Attempt to delete a non-existing reference - RefNum index " | ||||||
|  |                         + std::to_string(ref.mRefNum.mIndex) + ", refID " + ref.mRefID + ", content file index " | ||||||
|  |                         + std::to_string(ref.mRefNum.mContentFile), | ||||||
|  |                         /*hint*/"", | ||||||
|  |                         CSMDoc::Message::Severity_Warning); | ||||||
|                 continue; |                 continue; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|  | @ -107,9 +140,10 @@ void CSMWorld::RefCollection::load (ESM::ESMReader& reader, int cellIndex, bool | ||||||
|         if (iter==cache.end()) |         if (iter==cache.end()) | ||||||
|         { |         { | ||||||
|             // new reference
 |             // new reference
 | ||||||
|  |             ref.mIdNum = mNextId; // FIXME: fragile
 | ||||||
|             ref.mId = getNewId(); |             ref.mId = getNewId(); | ||||||
| 
 | 
 | ||||||
|             cache.insert(std::make_pair(refNum, ref.mId)); |             cache.insert(std::make_pair(refNum, ref.mIdNum)); | ||||||
| 
 | 
 | ||||||
|             std::unique_ptr<Record<CellRef> > record(new Record<CellRef>); |             std::unique_ptr<Record<CellRef> > record(new Record<CellRef>); | ||||||
|             record->mState = base ? RecordBase::State_BaseOnly : RecordBase::State_ModifiedOnly; |             record->mState = base ? RecordBase::State_BaseOnly : RecordBase::State_ModifiedOnly; | ||||||
|  | @ -120,9 +154,27 @@ void CSMWorld::RefCollection::load (ESM::ESMReader& reader, int cellIndex, bool | ||||||
|         else |         else | ||||||
|         { |         { | ||||||
|             // old reference -> merge
 |             // old reference -> merge
 | ||||||
|             ref.mId = iter->second; |             int index = getIndex(iter->second); | ||||||
|  | #if 0 | ||||||
|  |             // ref.mRefNum.mIndex     : the key
 | ||||||
|  |             // iter->second           : previously cached idNum for the key
 | ||||||
|  |             // index                  : position of the record for that idNum
 | ||||||
|  |             // getRecord(index).get() : record in the index position
 | ||||||
|  |             assert(iter->second != getRecord(index).get().mIdNum); // sanity check
 | ||||||
| 
 | 
 | ||||||
|             int index = getIndex (ref.mId); |             // check if the plugin used the same RefNum index for a different record
 | ||||||
|  |             if (ref.mRefID != getRecord(index).get().mRefID) | ||||||
|  |             { | ||||||
|  |                 CSMWorld::UniversalId id(CSMWorld::UniversalId::Type_Cell, mCells.getId(cellIndex)); | ||||||
|  |                 messages.add(id, | ||||||
|  |                     "RefNum renamed from RefID \"" + getRecord(index).get().mRefID + "\" to \"" | ||||||
|  |                     + ref.mRefID + "\" (RefNum index " + std::to_string(ref.mRefNum.mIndex) + ")", | ||||||
|  |                     /*hint*/"", | ||||||
|  |                     CSMDoc::Message::Severity_Info); | ||||||
|  |             } | ||||||
|  | #endif | ||||||
|  |             ref.mId = getRecord(index).get().mId; | ||||||
|  |             ref.mIdNum = extractIdNum(ref.mId); | ||||||
| 
 | 
 | ||||||
|             std::unique_ptr<Record<CellRef> > record(new Record<CellRef>(getRecord(index))); |             std::unique_ptr<Record<CellRef> > record(new Record<CellRef>(getRecord(index))); | ||||||
|             record->mState = base ? RecordBase::State_BaseOnly : RecordBase::State_Modified; |             record->mState = base ? RecordBase::State_BaseOnly : RecordBase::State_Modified; | ||||||
|  | @ -139,3 +191,117 @@ std::string CSMWorld::RefCollection::getNewId() | ||||||
|     stream << "ref#" << mNextId++; |     stream << "ref#" << mNextId++; | ||||||
|     return stream.str(); |     return stream.str(); | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | unsigned int CSMWorld::RefCollection::extractIdNum (const std::string& id) const | ||||||
|  | { | ||||||
|  |     std::string::size_type separator = id.find_last_of('#'); | ||||||
|  | 
 | ||||||
|  |     if (separator == std::string::npos) | ||||||
|  |         throw std::runtime_error("invalid ref ID: " + id); | ||||||
|  | 
 | ||||||
|  |     return static_cast<unsigned int>(std::stoi(id.substr(separator+1))); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int CSMWorld::RefCollection::getIndex (unsigned int id) const | ||||||
|  | { | ||||||
|  |     int index = searchId(id); | ||||||
|  | 
 | ||||||
|  |     if (index == -1) | ||||||
|  |         throw std::runtime_error("invalid RefNum: " + std::to_string(id)); | ||||||
|  | 
 | ||||||
|  |     return index; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int CSMWorld::RefCollection::searchId (unsigned int id) const | ||||||
|  | { | ||||||
|  |     std::map<unsigned int, int>::const_iterator iter = mRefIndex.find(id); | ||||||
|  | 
 | ||||||
|  |     if (iter == mRefIndex.end()) | ||||||
|  |         return -1; | ||||||
|  | 
 | ||||||
|  |     return iter->second; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void CSMWorld::RefCollection::removeRows (int index, int count) | ||||||
|  | { | ||||||
|  |     Collection<CellRef, IdAccessor<CellRef> >::removeRows(index, count); // erase records only
 | ||||||
|  | 
 | ||||||
|  |     std::map<unsigned int, int>::iterator iter = mRefIndex.begin(); | ||||||
|  |     while (iter != mRefIndex.end()) | ||||||
|  |     { | ||||||
|  |         if (iter->second>=index) | ||||||
|  |         { | ||||||
|  |             if (iter->second >= index+count) | ||||||
|  |             { | ||||||
|  |                 iter->second -= count; | ||||||
|  |                 ++iter; | ||||||
|  |             } | ||||||
|  |             else | ||||||
|  |                 mRefIndex.erase(iter++); | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |             ++iter; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void  CSMWorld::RefCollection::appendBlankRecord (const std::string& id, UniversalId::Type type) | ||||||
|  | { | ||||||
|  |     std::unique_ptr<Record<CellRef> > record2(new Record<CellRef>); | ||||||
|  | 
 | ||||||
|  |     record2->mState = Record<CellRef>::State_ModifiedOnly; | ||||||
|  |     record2->mModified.blank(); | ||||||
|  | 
 | ||||||
|  |     record2->get().mId = id; | ||||||
|  |     record2->get().mIdNum = extractIdNum(id); | ||||||
|  | 
 | ||||||
|  |     Collection<CellRef, IdAccessor<CellRef> >::appendRecord(std::move(record2)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void CSMWorld::RefCollection::cloneRecord (const std::string& origin, | ||||||
|  |                                            const std::string& destination, | ||||||
|  |                                            const UniversalId::Type type) | ||||||
|  | { | ||||||
|  |    std::unique_ptr<Record<CellRef> > copy(new Record<CellRef>); | ||||||
|  | 
 | ||||||
|  |    copy->mModified = getRecord(origin).get(); | ||||||
|  |    copy->mState = RecordBase::State_ModifiedOnly; | ||||||
|  | 
 | ||||||
|  |    copy->get().mId = destination; | ||||||
|  |    copy->get().mIdNum = extractIdNum(destination); | ||||||
|  | 
 | ||||||
|  |    insertRecord(std::move(copy), getAppendIndex(destination, type)); // call RefCollection::insertRecord()
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int CSMWorld::RefCollection::searchId (const std::string& id) const | ||||||
|  | { | ||||||
|  |     return searchId(extractIdNum(id)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void CSMWorld::RefCollection::appendRecord (std::unique_ptr<RecordBase> record, UniversalId::Type type) | ||||||
|  | { | ||||||
|  |     int index = getAppendIndex(/*id*/"", type); // for CellRef records id is ignored
 | ||||||
|  | 
 | ||||||
|  |     mRefIndex.insert(std::make_pair(static_cast<Record<CellRef>*>(record.get())->get().mIdNum, index)); | ||||||
|  | 
 | ||||||
|  |     Collection<CellRef, IdAccessor<CellRef> >::insertRecord(std::move(record), index, type); // add records only
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void CSMWorld::RefCollection::insertRecord (std::unique_ptr<RecordBase> record, int index, | ||||||
|  |     UniversalId::Type type) | ||||||
|  | { | ||||||
|  |     int size = getAppendIndex(/*id*/"", type); // for CellRef records id is ignored
 | ||||||
|  |     unsigned int idNum = static_cast<Record<CellRef>*>(record.get())->get().mIdNum; | ||||||
|  | 
 | ||||||
|  |     Collection<CellRef, IdAccessor<CellRef> >::insertRecord(std::move(record), index, type); // add records only
 | ||||||
|  | 
 | ||||||
|  |     if (index < size-1) | ||||||
|  |     { | ||||||
|  |         for (std::map<unsigned int, int>::iterator iter(mRefIndex.begin()); iter != mRefIndex.end(); ++iter) | ||||||
|  |         { | ||||||
|  |             if (iter->second >= index) | ||||||
|  |                 ++(iter->second); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     mRefIndex.insert(std::make_pair(idNum, index)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @ -14,12 +14,27 @@ namespace CSMWorld | ||||||
|     struct Cell; |     struct Cell; | ||||||
|     class UniversalId; |     class UniversalId; | ||||||
| 
 | 
 | ||||||
|  |     template<> | ||||||
|  |     void Collection<CellRef, IdAccessor<CellRef> >::removeRows (int index, int count); | ||||||
|  | 
 | ||||||
|  |     template<> | ||||||
|  |     void Collection<CellRef, IdAccessor<CellRef> >::insertRecord (std::unique_ptr<RecordBase> record, int index, | ||||||
|  |         UniversalId::Type type); | ||||||
|  | 
 | ||||||
|     /// \brief References in cells
 |     /// \brief References in cells
 | ||||||
|     class RefCollection : public Collection<CellRef> |     class RefCollection : public Collection<CellRef> | ||||||
|     { |     { | ||||||
|             Collection<Cell>& mCells; |             Collection<Cell>& mCells; | ||||||
|  |             std::map<unsigned int, int> mRefIndex; | ||||||
|  | 
 | ||||||
|             int mNextId; |             int mNextId; | ||||||
| 
 | 
 | ||||||
|  |             unsigned int extractIdNum(const std::string& id) const; | ||||||
|  | 
 | ||||||
|  |             int getIndex (unsigned int id) const; | ||||||
|  | 
 | ||||||
|  |             int searchId (unsigned int id) const; | ||||||
|  | 
 | ||||||
|         public: |         public: | ||||||
|             // MSVC needs the constructor for a class inheriting a template to be defined in header
 |             // MSVC needs the constructor for a class inheriting a template to be defined in header
 | ||||||
|             RefCollection (Collection<Cell>& cells) |             RefCollection (Collection<Cell>& cells) | ||||||
|  | @ -27,10 +42,28 @@ namespace CSMWorld | ||||||
|             {} |             {} | ||||||
| 
 | 
 | ||||||
|             void load (ESM::ESMReader& reader, int cellIndex, bool base, |             void load (ESM::ESMReader& reader, int cellIndex, bool base, | ||||||
|                 std::map<unsigned int, std::string>& cache, CSMDoc::Messages& messages); |                 std::map<unsigned int, unsigned int>& cache, CSMDoc::Messages& messages); | ||||||
|             ///< Load a sequence of references.
 |             ///< Load a sequence of references.
 | ||||||
| 
 | 
 | ||||||
|             std::string getNewId(); |             std::string getNewId(); | ||||||
|  | 
 | ||||||
|  |             virtual void removeRows (int index, int count); | ||||||
|  | 
 | ||||||
|  |             virtual void appendBlankRecord (const std::string& id, | ||||||
|  |                                             UniversalId::Type type = UniversalId::Type_None); | ||||||
|  | 
 | ||||||
|  |             virtual void cloneRecord (const std::string& origin, | ||||||
|  |                                       const std::string& destination, | ||||||
|  |                                       const UniversalId::Type type); | ||||||
|  | 
 | ||||||
|  |             virtual int searchId (const std::string& id) const; | ||||||
|  | 
 | ||||||
|  |             virtual void appendRecord (std::unique_ptr<RecordBase> record, | ||||||
|  |                                        UniversalId::Type type = UniversalId::Type_None); | ||||||
|  | 
 | ||||||
|  |             virtual void insertRecord (std::unique_ptr<RecordBase> record, | ||||||
|  |                                        int index, | ||||||
|  |                                        UniversalId::Type type = UniversalId::Type_None); | ||||||
|     }; |     }; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue