aboutsummaryrefslogtreecommitdiffstats
path: root/src/kvstorage_file.c
diff options
context:
space:
mode:
authorVsevolod Stakhov <vsevolod@rambler-co.ru>2011-11-28 16:23:21 +0300
committerVsevolod Stakhov <vsevolod@rambler-co.ru>2011-11-28 16:23:21 +0300
commit287067006bf2b44b9a68f3dd9d24414f4902e225 (patch)
treea6a162bda65c7c094a4432062516917af6ae2682 /src/kvstorage_file.c
parent8cff136c17d945280161d7f5224dc09341035bd9 (diff)
downloadrspamd-287067006bf2b44b9a68f3dd9d24414f4902e225.tar.gz
rspamd-287067006bf2b44b9a68f3dd9d24414f4902e225.zip
Fix race condition.
Diffstat (limited to 'src/kvstorage_file.c')
-rw-r--r--src/kvstorage_file.c8
1 files changed, 7 insertions, 1 deletions
diff --git a/src/kvstorage_file.c b/src/kvstorage_file.c
index e15b22daf..fe401a268 100644
--- a/src/kvstorage_file.c
+++ b/src/kvstorage_file.c
@@ -198,6 +198,7 @@ file_process_queue (struct rspamd_kv_backend *backend)
g_slice_free1 (len * sizeof (gint), fds);
/* Clean the queue */
+ g_hash_table_remove_all (db->ops_hash);
cur = db->ops_queue->head;
while (cur) {
op = cur->data;
@@ -214,7 +215,6 @@ file_process_queue (struct rspamd_kv_backend *backend)
cur = g_list_next (cur);
}
- g_hash_table_remove_all (db->ops_hash);
g_queue_clear (db->ops_queue);
return TRUE;
@@ -304,10 +304,13 @@ rspamd_file_insert (struct rspamd_kv_backend *backend, gpointer key, guint keyle
/* We found another op with such key in this queue */
if (op->op == FILE_OP_DELETE || (op->elt->flags & KV_ELT_NEED_FREE) != 0) {
/* Also clean memory */
+ g_hash_table_steal (db->ops_hash, &search_elt);
g_slice_free1 (ELT_SIZE (op->elt), op->elt);
}
op->op = FILE_OP_INSERT;
op->elt = elt;
+ elt->flags |= KV_ELT_DIRTY;
+ g_hash_table_insert (db->ops_hash, elt, op);
}
else {
op = g_slice_alloc (sizeof (struct file_op));
@@ -343,10 +346,13 @@ rspamd_file_replace (struct rspamd_kv_backend *backend, gpointer key, guint keyl
/* We found another op with such key in this queue */
if (op->op == FILE_OP_DELETE || (op->elt->flags & KV_ELT_NEED_FREE) != 0) {
/* Also clean memory */
+ g_hash_table_steal (db->ops_hash, &search_elt);
g_slice_free1 (ELT_SIZE (op->elt), op->elt);
}
op->op = FILE_OP_REPLACE;
op->elt = elt;
+ elt->flags |= KV_ELT_DIRTY;
+ g_hash_table_insert (db->ops_hash, elt, op);
}
else {
op = g_slice_alloc (sizeof (struct file_op));