diff options
author | Kitsune Ral <Kitsune-Ral@users.sf.net> | 2018-06-16 20:50:01 +0900 |
---|---|---|
committer | Kitsune Ral <Kitsune-Ral@users.sf.net> | 2018-06-16 20:50:01 +0900 |
commit | 5a34e11132b612361439c7e1243ff2f3ac959213 (patch) | |
tree | 893a172160a16fdfc4a6017faa9a51d1dfb5d860 /lib | |
parent | 42b1e859f10d352eb1b8ad5ad5226500f65d9dfe (diff) | |
download | libquotient-5a34e11132b612361439c7e1243ff2f3ac959213.tar.gz libquotient-5a34e11132b612361439c7e1243ff2f3ac959213.zip |
Connection::doInDirectChat: be more careful with cleanup
d->directChats was cleaned up while it was iterated upon, obviously
leading to Undefined Behaviour. Closes #214.
Diffstat (limited to 'lib')
-rw-r--r-- | lib/connection.cpp | 12 |
1 files changed, 10 insertions, 2 deletions
diff --git a/lib/connection.cpp b/lib/connection.cpp index 29bb9564..d92071df 100644 --- a/lib/connection.cpp +++ b/lib/connection.cpp @@ -486,6 +486,7 @@ void Connection::doInDirectChat(const QString& userId, // There can be more than one DC; find the first valid, and delete invalid // (left/forgotten) ones along the way. const auto* u = user(userId); + DirectChatsMap removals; for (auto it = d->directChats.find(u); it != d->directChats.end() && it.key() == u; ++it) { @@ -509,8 +510,15 @@ void Connection::doInDirectChat(const QString& userId, }); } qCWarning(MAIN) << "Direct chat with" << userId << "known as room" - << roomId << "is not valid, discarding it"; - removeFromDirectChats(roomId); + << roomId << "is not valid and will be discarded"; + // Postpone actual deletion until we finish iterating d->directChats. + removals.insert(it.key(), it.value()); + } + if (!removals.isEmpty()) + { + for (auto it = removals.cbegin(); it != removals.cend(); ++it) + d->directChats.remove(it.key(), it.value()); + d->broadcastDirectChatUpdates({}, removals); } auto j = createDirectChat(userId); |