From 3de9e2d7f2168fe23adacdb62f79ab2e1de9c5a5 Mon Sep 17 00:00:00 2001 From: Matthew Barnes Date: Thu, 12 Mar 2009 18:27:23 +0000 Subject: [PATCH] - Add patch for RH bug #568332 (thread leak in fsync() rate limiting). --- ...a-server-2.25.92-sqlite-fsync-rework.patch | 456 ++++++++++++++++++ evolution-data-server.spec | 7 + 2 files changed, 463 insertions(+) create mode 100644 evolution-data-server-2.25.92-sqlite-fsync-rework.patch diff --git a/evolution-data-server-2.25.92-sqlite-fsync-rework.patch b/evolution-data-server-2.25.92-sqlite-fsync-rework.patch new file mode 100644 index 0000000..252a02b --- /dev/null +++ b/evolution-data-server-2.25.92-sqlite-fsync-rework.patch @@ -0,0 +1,456 @@ +diff -up evolution-data-server-2.25.92/camel/camel-db.c.sqlite-fsync-rework evolution-data-server-2.25.92/camel/camel-db.c +--- evolution-data-server-2.25.92/camel/camel-db.c.sqlite-fsync-rework 2009-02-04 12:28:12.000000000 -0500 ++++ evolution-data-server-2.25.92/camel/camel-db.c 2009-03-12 14:25:00.000000000 -0400 +@@ -36,239 +36,114 @@ + + #include "camel-debug.h" + +-/* how long to wait before invoking sync on the file; in miliseconds */ +-#define SYNC_TIMEOUT 5000 ++/* how long to wait before invoking sync on the file */ ++#define SYNC_TIMEOUT_SECONDS 5 + + static sqlite3_vfs *old_vfs = NULL; + + GStaticRecMutex only_once_lock = G_STATIC_REC_MUTEX_INIT; +-GStaticRecMutex sync_queue_lock = G_STATIC_REC_MUTEX_INIT; +-#define LockQueue() g_static_rec_mutex_lock (&sync_queue_lock) +-#define UnlockQueue() g_static_rec_mutex_unlock (&sync_queue_lock) +- +-/* 'sync_queue' is using keys sqlite3_file to sync_queue_data structures. +- Access to this is guarded with LockQueue/UnlockQueue function. */ +-static GHashTable *sync_queue = NULL; +- +-typedef struct _sync_queue_data { +- guint timeout_source; /* id of the source */ +- GThread *running_thread; + +- int sync_flags; +-} sync_queue_data; +- +-struct CamelSqlite3File +-{ ++typedef struct { + sqlite3_file parent; + sqlite3_file *old_vfs_file; /* pointer to old_vfs' file */ +-}; ++ GAsyncQueue *queue; ++ GThread *thread; ++ guint timeout_id; ++ gint flags; ++} CamelSqlite3File; + + static int +-call_old_file_Sync (sqlite3_file *pFile, int flags) ++call_old_file_Sync (CamelSqlite3File *cFile, int flags) + { +- struct CamelSqlite3File *cFile; +- + g_return_val_if_fail (old_vfs != NULL, SQLITE_ERROR); +- g_return_val_if_fail (pFile != NULL, SQLITE_ERROR); ++ g_return_val_if_fail (cFile != NULL, SQLITE_ERROR); + +- cFile = (struct CamelSqlite3File *)pFile; + g_return_val_if_fail (cFile->old_vfs_file->pMethods != NULL, SQLITE_ERROR); + return cFile->old_vfs_file->pMethods->xSync (cFile->old_vfs_file, flags); + } + +-static gboolean prepare_to_run_sync_in_thread (gpointer pFile); ++/* This special flag tells the sync request thread to exit. ++ * Just have to make sure it does not collide with SQLite's ++ * own synchronization flags (SQLITE_SYNC_xxx). */ ++#define SYNC_THREAD_EXIT 0x100000 + + static gpointer +-run_sync_in_thread (gpointer pFile) ++sync_request_thread_cb (CamelSqlite3File *cFile) + { +- int sync_flags = 0; +- sync_queue_data *data; +- +- g_return_val_if_fail (pFile != NULL, NULL); +- g_return_val_if_fail (sync_queue != NULL, NULL); ++ gpointer data; ++ gint flags = 0; + +- LockQueue (); +- data = g_hash_table_lookup (sync_queue, pFile); +- if (data) { +- /* sync_flags can change while we are running */ +- sync_flags = data->sync_flags; +- data->sync_flags = 0; +- } +- UnlockQueue (); +- +- /* this should not happen, once we are in a thread, the datas are ours */ +- g_return_val_if_fail (data != NULL, NULL); +- +- /* do the sync itself, but do not block the sync_queue; +- any error here is silently ignored. */ +- call_old_file_Sync (/*sqlite3_file*/pFile, sync_flags); +- +- LockQueue (); +- if (data->timeout_source == -1) { +- /* new sync request arrived meanwhile, indicate thread finished... */ +- data->running_thread = NULL; +- /* ...and reschedule */ +- data->timeout_source = g_timeout_add (SYNC_TIMEOUT, prepare_to_run_sync_in_thread, pFile); +- } else { +- /* remove it from a sync_queue and free memory */ +- g_hash_table_remove (sync_queue, pFile); +- g_free (data); +- } +- UnlockQueue (); ++ g_async_queue_ref (cFile->queue); + +- return NULL; +-} ++ while (TRUE) { ++ /* Block until a request arrives. */ ++ data = g_async_queue_pop (cFile->queue); + +-static gboolean +-prepare_to_run_sync_in_thread (gpointer pFile) +-{ +- sync_queue_data *data; ++ /* Make sure we can safely deference. */ ++ if (data == NULL) ++ continue; + +- g_return_val_if_fail (pFile != NULL, FALSE); +- g_return_val_if_fail (sync_queue != NULL, FALSE); ++ /* Extract flags and discard request. */ ++ flags = *((gint *) data); ++ g_slice_free (gint, data); + +- LockQueue (); ++ /* Check for exit request. */ ++ if (flags & SYNC_THREAD_EXIT) ++ break; + +- data = g_hash_table_lookup (sync_queue, pFile); +- /* check if still tracking this file and if didn't get rescheduled */ +- if (data && data->timeout_source == g_source_get_id (g_main_current_source ())) { +- /* run the thread */ +- data->running_thread = g_thread_create (run_sync_in_thread, pFile, TRUE, NULL); +- data->timeout_source = 0; ++ /* Got a boneafide sync request. ++ * Do it, but ignore errors. */ ++ call_old_file_Sync (cFile, flags); + } + +- UnlockQueue (); ++ /* Clear the exit flag. */ ++ flags &= ~SYNC_THREAD_EXIT; + +- return FALSE; +-} +- +-/* +- Adds sync on this file to the queue. Flags are just bit-OR-ed, +- which will not hopefully hurt. In case the file is waiting for +- it's sync, we just postpone it once again. +- In case file is syncing just in call of this, we schedule other +- sync after that. +- */ +-static void +-queue_sync (sqlite3_file *pFile, int flags) +-{ +- sync_queue_data *data; +- +- g_return_if_fail (pFile != NULL); +- g_return_if_fail (flags != 0); +- +- LockQueue (); +- +- if (!sync_queue) +- sync_queue = g_hash_table_new (g_direct_hash, g_direct_equal); +- +- data = g_hash_table_lookup (sync_queue, pFile); +- if (data) { +- /* There is a sync request for this file already. */ +- if (data->running_thread) { +- /* -1 indicates to reschedule after the actual sync finishes */ +- data->timeout_source = -1; +- /* start with new flags - thread set it to 0; next time just add others */ +- data->sync_flags = data->sync_flags | flags; +- } else { +- data->sync_flags = data->sync_flags | flags; ++ /* One more for the road? */ ++ if (getenv ("CAMEL_NO_SYNC_ON_CLOSE") == NULL) ++ call_old_file_Sync (cFile, flags); + +- /* reschedule */ +- g_source_remove (data->timeout_source); +- data->timeout_source = g_timeout_add (SYNC_TIMEOUT, prepare_to_run_sync_in_thread, pFile); +- } +- } else { +- data = g_malloc0 (sizeof (sync_queue_data)); +- data->sync_flags = flags; +- data->running_thread = NULL; +- data->timeout_source = g_timeout_add (SYNC_TIMEOUT, prepare_to_run_sync_in_thread, pFile); ++ g_async_queue_unref (cFile->queue); + +- g_hash_table_insert (sync_queue, pFile, data); +- } +- +- UnlockQueue (); ++ return NULL; + } + +-/* +- If file is not in a queue, it does nothing, otherwise it removes +- it from a queue, and calls sync immediately. +- If file is syncing at the moment, it waits until the previous sync finishes. +- */ +-static void +-dequeue_sync (sqlite3_file *pFile) +-{ +- sync_queue_data *data; +- +- g_return_if_fail (pFile != NULL); +- +- LockQueue (); +- +- if (!sync_queue) { +- /* closing file which wasn't requested to sync, and none +- before it too. */ +- UnlockQueue (); +- return; +- } +- +- data = g_hash_table_lookup (sync_queue, pFile); +- if (data) { +- int sync_flags = data->sync_flags; +- +- if (data->timeout_source) { +- if (data->timeout_source != -1) +- g_source_remove (data->timeout_source); +- data->timeout_source = 0; +- } +- +- if (data->running_thread) { +- GThread *thread = data->running_thread; ++static gboolean ++sync_push_request (CamelSqlite3File *cFile) ++{ ++ gint *data; + +- /* do not do anything later */ +- data = NULL; ++ /* The queue itself does not need to be locked yet, ++ * but we use its mutex to safely manipulate flags. */ ++ g_async_queue_lock (cFile->queue); + +- /* unlock here, thus the thread can hold the lock again */ +- UnlockQueue (); ++ /* We can't just cast the flags to a pointer because ++ * g_async_queue_push() won't take NULLs, and flags ++ * may be zero. So we have to allocate memory to ++ * send an integer. Bother. */ ++ data = g_slice_new (gint); ++ *data = cFile->flags; ++ cFile->flags = 0; + +- /* it's running at the moment, wait for a finish. +- it'll remove structure from a sync_queue too. */ +- g_thread_join (thread); +- } else { +- g_hash_table_remove (sync_queue, pFile); +- } ++ g_async_queue_push_unlocked (cFile->queue, data); + +- if (data) { +- static gboolean no_sync_on_close = FALSE, iKnow = FALSE; ++ cFile->timeout_id = 0; + +- if (!iKnow) { +- iKnow = TRUE; +- no_sync_on_close = getenv ("CAMEL_NO_SYNC_ON_CLOSE") != NULL; +- } +- +- /* do not block queue on while syncing */ +- UnlockQueue (); +- +- if (!no_sync_on_close) { +- /* sync on close */ +- call_old_file_Sync (pFile, sync_flags); +- } ++ g_async_queue_unlock (cFile->queue); + +- g_free (data); +- } +- } else { +- UnlockQueue (); +- } ++ return FALSE; + } + + #define def_subclassed(_nm, _params, _call) \ + static int \ + camel_sqlite3_file_ ## _nm _params \ + { \ +- struct CamelSqlite3File *cFile; \ ++ CamelSqlite3File *cFile; \ + \ + g_return_val_if_fail (old_vfs != NULL, SQLITE_ERROR); \ + g_return_val_if_fail (pFile != NULL, SQLITE_ERROR); \ + \ +- cFile = (struct CamelSqlite3File *) pFile; \ ++ cFile = (CamelSqlite3File *) pFile; \ + g_return_val_if_fail (cFile->old_vfs_file->pMethods != NULL, SQLITE_ERROR); \ + return cFile->old_vfs_file->pMethods->_nm _call; \ + } +@@ -293,15 +168,41 @@ def_subclassed (xDeviceCharacteristics, + static int + camel_sqlite3_file_xClose (sqlite3_file *pFile) + { +- struct CamelSqlite3File *cFile; ++ CamelSqlite3File *cFile; + int res; + + g_return_val_if_fail (old_vfs != NULL, SQLITE_ERROR); + g_return_val_if_fail (pFile != NULL, SQLITE_ERROR); + +- dequeue_sync (pFile); ++ cFile = (CamelSqlite3File *) pFile; ++ ++ /* The queue itself does not need to be locked yet, ++ * but we use its mutex to safely manipulate flags. */ ++ g_async_queue_lock (cFile->queue); ++ ++ /* Tell the sync request thread to exit. It may do ++ * one last sync before exiting, so preserve any sync ++ * flags that have accumulated. */ ++ cFile->flags |= SYNC_THREAD_EXIT; ++ ++ /* Cancel any pending sync requests. */ ++ if (cFile->timeout_id > 0) ++ g_source_remove (cFile->timeout_id); ++ ++ /* Unlock the queue before pushing the exit request. */ ++ g_async_queue_unlock (cFile->queue); ++ ++ /* Push the exit request. */ ++ sync_push_request (cFile); ++ ++ /* Wait for the thread to exit. */ ++ g_thread_join (cFile->thread); ++ cFile->thread = NULL; ++ ++ /* Now we can safely destroy the queue. */ ++ g_async_queue_unref (cFile->queue); ++ cFile->queue = NULL; + +- cFile = (struct CamelSqlite3File *) pFile; + if (cFile->old_vfs_file->pMethods) + res = cFile->old_vfs_file->pMethods->xClose (cFile->old_vfs_file); + else +@@ -316,10 +217,26 @@ camel_sqlite3_file_xClose (sqlite3_file + static int + camel_sqlite3_file_xSync (sqlite3_file *pFile, int flags) + { ++ CamelSqlite3File *cFile; ++ + g_return_val_if_fail (old_vfs != NULL, SQLITE_ERROR); + g_return_val_if_fail (pFile != NULL, SQLITE_ERROR); + +- queue_sync (pFile, flags); ++ cFile = (CamelSqlite3File *) pFile; ++ ++ /* The queue itself does not need to be locked yet, ++ * but we use its mutex to safely manipulate flags. */ ++ g_async_queue_lock (cFile->queue); ++ ++ /* If a sync request is already scheduled, accumulate flags. */ ++ cFile->flags |= flags; ++ ++ if (cFile->timeout_id == 0) ++ cFile->timeout_id = g_timeout_add_seconds ( ++ SYNC_TIMEOUT_SECONDS, (GSourceFunc) ++ sync_push_request, cFile); ++ ++ g_async_queue_unlock (cFile->queue); + + return SQLITE_OK; + } +@@ -328,14 +245,24 @@ static int + camel_sqlite3_vfs_xOpen (sqlite3_vfs *pVfs, const char *zPath, sqlite3_file *pFile, int flags, int *pOutFlags) + { + static sqlite3_io_methods io_methods = {0}; +- struct CamelSqlite3File *cFile; ++ CamelSqlite3File *cFile; ++ GError *error = NULL; + int res; + + g_return_val_if_fail (old_vfs != NULL, -1); + g_return_val_if_fail (pFile != NULL, -1); + +- cFile = (struct CamelSqlite3File *)pFile; ++ cFile = (CamelSqlite3File *)pFile; + cFile->old_vfs_file = g_malloc0 (old_vfs->szOsFile); ++ cFile->queue = g_async_queue_new (); ++ ++ /* Spawn a joinable thread to listen for sync requests. */ ++ cFile->thread = g_thread_create ( ++ (GThreadFunc) sync_request_thread_cb, cFile, TRUE, &error); ++ if (error != NULL) { ++ g_warning ("%s", error->message); ++ g_error_free (error); ++ } + + res = old_vfs->xOpen (old_vfs, zPath, cFile->old_vfs_file, flags, pOutFlags); + +@@ -370,33 +297,23 @@ camel_sqlite3_vfs_xOpen (sqlite3_vfs *pV + return res; + } + +-static void ++static gpointer + init_sqlite_vfs (void) + { + static sqlite3_vfs vfs = { 0 }; + +- g_static_rec_mutex_lock (&only_once_lock); +- if (old_vfs) { +- g_static_rec_mutex_unlock (&only_once_lock); +- return; +- } +- + old_vfs = sqlite3_vfs_find (NULL); +- if (!old_vfs) { +- g_static_rec_mutex_unlock (&only_once_lock); +- g_return_if_fail (old_vfs != NULL); +- return; +- } ++ g_return_val_if_fail (old_vfs != NULL, NULL); + + memcpy (&vfs, old_vfs, sizeof (sqlite3_vfs)); + +- vfs.szOsFile = sizeof (struct CamelSqlite3File); ++ vfs.szOsFile = sizeof (CamelSqlite3File); + vfs.zName = "camel_sqlite3_vfs"; + vfs.xOpen = camel_sqlite3_vfs_xOpen; + + sqlite3_vfs_register (&vfs, 1); + +- g_static_rec_mutex_unlock (&only_once_lock); ++ return NULL; + } + + #define d(x) if (camel_debug("sqlite")) x +@@ -453,12 +370,13 @@ cdb_sql_exec (sqlite3 *db, const char* s + CamelDB * + camel_db_open (const char *path, CamelException *ex) + { ++ static GOnce vfs_once = G_ONCE_INIT; + CamelDB *cdb; + sqlite3 *db; + char *cache; + int ret; + +- init_sqlite_vfs (); ++ g_once (&vfs_once, (GThreadFunc) init_sqlite_vfs, NULL); + + CAMEL_DB_USE_SHARED_CACHE; + diff --git a/evolution-data-server.spec b/evolution-data-server.spec index 0216462..142e7d4 100644 --- a/evolution-data-server.spec +++ b/evolution-data-server.spec @@ -56,6 +56,9 @@ Patch12: evolution-data-server-1.11.5-fix-64bit-acinclude.patch # Revert GNOME bug #573240 Patch13: evolution-data-server-2.25.92-unbreak-imap.patch +# RH bug #489696 / GNOME bug #568322 +Patch14: evolution-data-server-2.25.92-sqlite-fsync-rework.patch + ### Build Dependencies ### BuildRequires: GConf2-devel @@ -140,6 +143,7 @@ This package contains developer documentation for %{name}. %patch11 -p1 -b .camel-folder-summary-crash %patch12 -p1 -b .fix-64bit-acinclude %patch13 -p1 -b .unbreak-imap +%patch14 -p1 -b .sqlite-fsync-rework mkdir -p krb5-fakeprefix/include mkdir -p krb5-fakeprefix/lib @@ -385,6 +389,9 @@ rm -rf $RPM_BUILD_ROOT %{_datadir}/gtk-doc/html/libedataserverui %changelog +* Thu Mar 12 2009 Matthew Barnes - 2.25.92-3.fc11 +- Add patch for RH bug #568332 (thread leak in fsync() rate limiting). + * Sat Mar 07 2009 Matthew Barnes - 2.25.92-2.fc11 - Add patch to revert GNOME bug #573240 (IMAP message loading regressions).