diff options
-rw-r--r-- | include/cull_list.h | 5 | ||||
-rw-r--r-- | src/cull_list.cpp | 40 |
2 files changed, 36 insertions, 9 deletions
diff --git a/include/cull_list.h b/include/cull_list.h index 2422f05b6..36a8886a0 100644 --- a/include/cull_list.h +++ b/include/cull_list.h @@ -85,6 +85,11 @@ class CullList * reference. */ std::map<userrec*,int> exempt; + + /** Check if a user pointer is valid + * (e.g. it exists in the user hash) + */ + bool IsValid(userrec* user); public: /** Constructor. * Clears the CullList::list and CullList::exempt diff --git a/src/cull_list.cpp b/src/cull_list.cpp index f1dd5b4d9..b06b80f7f 100644 --- a/src/cull_list.cpp +++ b/src/cull_list.cpp @@ -51,6 +51,17 @@ using namespace std; extern InspIRCd* ServerInstance; +bool CullList::IsValid(userrec* user) +{ + for (user_hash::iterator u = clientlist.begin(); u != clientlist.end(); u++) + { + userrec* u2 = (userrec*)*u; + if (user == u2) + return true; + } + return false; +} + CullItem::CullItem(userrec* u, std::string r) { this->user = u; @@ -79,6 +90,7 @@ void CullList::AddItem(userrec* user, std::string reason) { CullItem item(user,reason); list.push_back(item); + names.push_back(user->nick); exempt[user] = 1; } } @@ -90,17 +102,27 @@ int CullList::Apply() { std::vector<CullItem>::iterator a = list.begin(); userrec* u = a->GetUser(); - kill_link(u,a->GetReason().c_str()); - list.erase(list.begin()); - /* So that huge numbers of quits dont block, - * we yield back to our mainloop every 15 - * iterations. - * The DoOneIteration call basically acts - * like a software threading mechanism. + /* Because ServerInstance->DoOneIteration can + * take the user away from us in the middle of + * our operation, we should check to see if this + * pointer is still valid by iterating the hash. + * It's expensive, yes, but the DoOneIteration + * call stops it being horrendously bad. */ - if ((n++ % 15) == 0) + if (IsValid(u)) { - ServerInstance->DoOneIteration(false); + kill_link(u,a->GetReason().c_str()); + list.erase(list.begin()); + /* So that huge numbers of quits dont block, + * we yield back to our mainloop every 15 + * iterations. + * The DoOneIteration call basically acts + * like a software threading mechanism. + */ + if (((n++) % 15) == 0) + { + ServerInstance->DoOneIteration(false); + } } } return n; |