From b3456a884154f8ea03560a7242e2ba4a2d5a1f61 Mon Sep 17 00:00:00 2001 From: Koncord Date: Sat, 29 Dec 2018 11:09:13 +0800 Subject: [PATCH] [Server] Fix invalidation of iterators --- apps/openmw-mp/Cell.cpp | 13 ++++++---- apps/openmw-mp/Cell.hpp | 2 +- apps/openmw-mp/CellController.cpp | 43 +++++++++++++++++++++++-------- 3 files changed, 41 insertions(+), 17 deletions(-) diff --git a/apps/openmw-mp/Cell.cpp b/apps/openmw-mp/Cell.cpp index c02e5f808..2da4f3ea3 100644 --- a/apps/openmw-mp/Cell.cpp +++ b/apps/openmw-mp/Cell.cpp @@ -53,18 +53,21 @@ void Cell::addPlayer(Player *player) players.push_back(player); } -void Cell::removePlayer(Player *player) +void Cell::removePlayer(Player *player, bool cleanPlayer) { for (Iterator it = begin(); it != end(); it++) { if (*it == player) { - auto it2 = find(player->cells.begin(), player->cells.end(), this); - if (it2 != player->cells.end()) + if (cleanPlayer) { - LOG_APPEND(Log::LOG_INFO, "- Removing %s from Player %s", getDescription().c_str(), player->npc.mName.c_str()); + auto it2 = find(player->cells.begin(), player->cells.end(), this); + if (it2 != player->cells.end()) + { + LOG_APPEND(Log::LOG_INFO, "- Removing %s from Player %s", getDescription().c_str(), player->npc.mName.c_str()); - player->cells.erase(it2); + player->cells.erase(it2); + } } LOG_APPEND(Log::LOG_INFO, "- Removing %s from Cell %s", player->npc.mName.c_str(), getDescription().c_str()); diff --git a/apps/openmw-mp/Cell.hpp b/apps/openmw-mp/Cell.hpp index 04fc6b0e8..d2049aa11 100644 --- a/apps/openmw-mp/Cell.hpp +++ b/apps/openmw-mp/Cell.hpp @@ -28,7 +28,7 @@ public: Iterator end() const; void addPlayer(Player *player); - void removePlayer(Player *player); + void removePlayer(Player *player, bool cleanPlayer = true); void readActorList(unsigned char packetID, const mwmp::BaseActorList *newActorList); bool containsActor(int refNum, int mpNum); diff --git a/apps/openmw-mp/CellController.cpp b/apps/openmw-mp/CellController.cpp index 6f1b96838..dac45dc8d 100644 --- a/apps/openmw-mp/CellController.cpp +++ b/apps/openmw-mp/CellController.cpp @@ -136,26 +136,38 @@ void CellController::removeCell(Cell *cell) void CellController::removePlayer(Cell *cell, Player *player) { - cell->removePlayer(player); - - if (cell->players.empty()) - { - LOG_APPEND(Log::LOG_INFO, "- Cell %s has no players left", cell->getDescription().c_str()); - removeCell(cell); - } + //cell->removePlayer(player); } void CellController::deletePlayer(Player *player) { LOG_APPEND(Log::LOG_INFO, "- Iterating through Cells from Player %s", player->npc.mName.c_str()); - for (auto it = player->getCells()->begin(); player->getCells()->size() != 0; ++it) - removePlayer(*it, player); + std::vector toDelete; + + auto it = player->getCells()->begin(); + const auto endIter = player->getCells()->end(); + + for (; it != endIter; ++it) + { + Cell *c = *it; + c->removePlayer(player, false); + if (c->players.empty()) + toDelete.push_back(c); + } + + for (auto &&cell : toDelete) + { + LOG_APPEND(Log::LOG_INFO, "- Cell %s has no players left", cell->getDescription().c_str()); + removeCell(cell); + } } void CellController::update(Player *player) { - for (auto cell : player->cellStateChanges.cellStates) + std::vector toDelete; + + for (auto &&cell : player->cellStateChanges.cellStates) { if (cell.type == mwmp::CellState::LOAD) { @@ -171,7 +183,16 @@ void CellController::update(Player *player) c = getCellByXY(cell.cell.getGridX(), cell.cell.getGridY()); if (c != nullptr) - removePlayer(c, player); + { + c->removePlayer(player); + if (c->players.empty()) + toDelete.push_back(c); + } } } + for (auto &&cell : toDelete) + { + LOG_APPEND(Log::LOG_INFO, "- Cell %s has no players left", cell->getDescription().c_str()); + removeCell(cell); + } }