summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--include/cull_list.h5
-rw-r--r--src/cull_list.cpp40
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;