pthread condition variables on Linux, odd behaviour

前端 未结 4 844
情深已故
情深已故 2021-01-20 01:49

I\'m synchronizing reader and writer processes on Linux.

I have 0 or more process (the readers) that need to sleep until they are woken up, read a resource, go back

相关标签:
4条回答
  • 2021-01-20 02:14

    what the last poster said is correct. the KEY to the whole cond-variable situation working correctly is that the cond-var is NOT signalled prior to it being waited on. its strictly a signal that is to be used when others (single or multiple) are waiting. when no one is waiting, its effectively a NOP. which, btw, is NOT how i believe it SHOULD work, but how it DOES work.

    larry

    0 讨论(0)
  • 2021-01-20 02:17

    Do you test for some condition before your process actually call pthread_cond_wait() ? I am asking because, it's a very common mistake : Your process must not call wait() unless you are sure that some process will call signal() (or broadcast()) later.

    concidering this code (from pthread_cond_wait man page) :

              pthread_mutex_lock(&mut);
              while (x <= y) {
                      pthread_cond_wait(&cond, &mut);
              }
              /* operate on x and y */
              pthread_mutex_unlock(&mut);
    

    If your omit the while test, and just signal from another process whenever your (x <= y) condition is true, it won't work since the signal only wakes up the process the are already waiting. If signal() called before the other process calls wait() the signal will be lost and the waiting process will be waiting forever.

    EDIT : About the while loop. When you are signaling one process from another process it is set on the ''ready list'' but not necessarily scheduled and your condition (x <= y) may be change again since no one holds the lock. That's why you need to check for your condition each time you are about to wait. It should always be wakeup -> check if the condition is still true -> do work.

    hope it's clear.

    0 讨论(0)
  • 2021-01-20 02:31

    The documentation says that it should work... are you sure it's the same conditional value that the rest of the threads are looking at?

    This is the example code from opengroup.org:

    pthread_cond_wait(mutex, cond):
        value = cond->value; /* 1 */
        pthread_mutex_unlock(mutex); /* 2 */
        pthread_mutex_lock(cond->mutex); /* 10 */
        if (value == cond->value) { /* 11 */
            me->next_cond = cond->waiter;
            cond->waiter = me;
            pthread_mutex_unlock(cond->mutex);
            unable_to_run(me);
        } else
            pthread_mutex_unlock(cond->mutex); /* 12 */
        pthread_mutex_lock(mutex); /* 13 */
    
    
    pthread_cond_signal(cond):
        pthread_mutex_lock(cond->mutex); /* 3 */
        cond->value++; /* 4 */
        if (cond->waiter) { /* 5 */
            sleeper = cond->waiter; /* 6 */
            cond->waiter = sleeper->next_cond; /* 7 */
            able_to_run(sleeper); /* 8 */
        }
        pthread_mutex_unlock(cond->mutex); /* 9 */
    
    0 讨论(0)
  • 2021-01-20 02:32

    Have you set the PTHREAD_PROCESS_SHARED attribute on both your condvar and mutex?

    For Linux consult the following man pages:

    • pthread_mutexattr_init (with sample)
    • pthread_mutexattr_setpshared
    • pthread_condattr_init
    • pthread_condattr_setpshared

    Methods, types, constants etc. are normally defined in /usr/include/pthread.h, /usr/include/nptl/pthread.h.

    0 讨论(0)
提交回复
热议问题