1
0
mirror of git://sourceware.org/git/lvm2.git synced 2025-01-18 10:04:20 +03:00

dmeventd: remember number of log disablings

Individual events are handled through separate threads,
so once we have more then a single thread in this eventwait
sleeping, we got race on the dm_log setting, since
if one event is timeout out on alarm, while another is still waiting,
then dm log has been restored to NULL and the next sigalarm
has been reported as error.

Fix it by introducing counter which is protected via mutex,
and only when the last event is released, logging is restored.

TODO: libdm seems to have some static vars which may audit
for this type of use.
This commit is contained in:
Zdenek Kabelac 2014-03-10 09:47:44 +01:00
parent 460c19df62
commit 2a9b62c7f9
2 changed files with 15 additions and 2 deletions

View File

@ -1,5 +1,6 @@
Version 1.02.85 - Version 1.02.85 -
=================================== ===================================
Fix dmeventd logging with parallel wait event processing.
Reuse _node_send_messages() for validation of transaction_id in preload. Reuse _node_send_messages() for validation of transaction_id in preload.
Transaction_id could be lower by one only when messages are prepared. Transaction_id could be lower by one only when messages are prepared.
Do not call callback when preload fails. Do not call callback when preload fails.

View File

@ -661,6 +661,7 @@ static sigset_t _unblock_sigalrm(void)
/* Wait on a device until an event occurs. */ /* Wait on a device until an event occurs. */
static int _event_wait(struct thread_status *thread, struct dm_task **task) static int _event_wait(struct thread_status *thread, struct dm_task **task)
{ {
static unsigned _in_event_counter = 0;
sigset_t set; sigset_t set;
int ret = DM_WAIT_RETRY; int ret = DM_WAIT_RETRY;
struct dm_task *dmt; struct dm_task *dmt;
@ -677,12 +678,20 @@ static int _event_wait(struct thread_status *thread, struct dm_task **task)
!dm_task_set_event_nr(dmt, thread->event_nr)) !dm_task_set_event_nr(dmt, thread->event_nr))
goto out; goto out;
_lock_mutex();
/*
* Check if there are already some waiting events,
* in this case the logging is unmodified.
* TODO: audit libdm thread usage
*/
if (!_in_event_counter++)
dm_log_init(_no_intr_log);
_unlock_mutex();
/* /*
* This is so that you can break out of waiting on an event, * This is so that you can break out of waiting on an event,
* either for a timeout event, or to cancel the thread. * either for a timeout event, or to cancel the thread.
*/ */
set = _unblock_sigalrm(); set = _unblock_sigalrm();
dm_log_init(_no_intr_log);
errno = 0; errno = 0;
if (dm_task_run(dmt)) { if (dm_task_run(dmt)) {
thread->current_events |= DM_EVENT_DEVICE_ERROR; thread->current_events |= DM_EVENT_DEVICE_ERROR;
@ -706,7 +715,10 @@ static int _event_wait(struct thread_status *thread, struct dm_task **task)
} }
pthread_sigmask(SIG_SETMASK, &set, NULL); pthread_sigmask(SIG_SETMASK, &set, NULL);
dm_log_init(NULL); _lock_mutex();
if (--_in_event_counter == 0)
dm_log_init(NULL);
_unlock_mutex();
out: out:
if (ret == DM_WAIT_FATAL || ret == DM_WAIT_RETRY) { if (ret == DM_WAIT_FATAL || ret == DM_WAIT_RETRY) {