Full Transcript

Thread Synchronization Problems High Performance Computing Master in Applied Artificial Intelligence Nuno Lopes Thread Execution Ordering u The Mutual Exclusion mechanism makes it possible to guarantee that at no...

Thread Synchronization Problems High Performance Computing Master in Applied Artificial Intelligence Nuno Lopes Thread Execution Ordering u The Mutual Exclusion mechanism makes it possible to guarantee that at no time does more than one thread execute "simultaneously" code that manipulates the same resource. u Some problems are based on the ordering of thread execution. u Threads have an implicit execution order between them, even though they are concurrent. u Typical cases of synchronization: u Producer - Consumer u Reader - Writer u Barrier Producer-Consumer u Problem consists of two types of threads: Producer and Consumer. u Producer - produces resources (to be shared) u Production capacity depends on the availability of storing produced resources u Stores the produced resource in a shared buffer u Consumer - consumes produced resources u Condition: only consumed resources already produced can be consumed u Removes consumed resources from the shared buffer Reader-Writer u In this problem there are threads of two types: u Readers: threads that read data u Writers: threads that change the data u The restrictions of this problem are the following: u several readers can access the data concurrently, without risk of creating inconsistencies; u when a writer wants to alter the data there can be no other simultaneous access, reading or writing. Barrier u Problem of synchronization between threads. u Threads invoke barrier() function, which blocks the execution of the thread that calls it, until all threads are in this position. u When all threads execute this function, the function terminates (and unblocks) for all threads. u It acts as a synchronization point in the execution of each thread. Condition Variables Barrier Example using Active Waiting Motivation for Condition Variables u Sometimes the need arises in threads to check that a condition (global to all threads) is fulfilled. u However: u it is necessary to have access to a common global state; u but it is also necessary to wait for this state to be changed by another thread. Motivation for Condition Variables u Now, to make an access it is necessary to use Mutex, but as long as the thread has the Mutex it does not allow the others to change the (same) global state. u On the other hand, for the thread to wait for the state to change, in order to meet the condition, it is necessary to check if it has changed: u Active waiting! u How to do this synchronisation of a global state without active wait, nor block the threads ? Condition Variable u A condition variable is a structure that allows threads to suspend execution until a certain event (or condition) occurs. u When the event occurs, a signalling mechanism will "wake up" the locked threads to continue execution. u A condition variable must be associated with a mutual exclusion (mutex) mechanism. Condition Variables API u Passive waiting: pthread_cond_wait (cv, mt) u Signaling: pthread_cond_signal (cv) pthread_cond_broadcast (cv) u Creation and destruction: pthread_cond_init () pthread_cond_destroy () u Auxiliary Structure: pthread_cond_t Condition Variables API u Function for waiting for the event to occur: int pthread_cond_wait( pthread_cond_t* cond_var_p , pthread_mutex_t* mutex_p ); u This function has a combined and atomic function of the following functions: u pthread mutex unlock(mutex); u wait for signal on condition variable; u pthread mutex lock(mutex); Condition Variables API u Function to signal the occurrence of the event: For (any) locked thread: int pthread_cond_signal( pthread_cond_t* cond_var_p ); For all locked threads: int pthread_cond_broadcast () pthread_cond_t* cond_var_p ); Condition Variables API u Type of auxiliary structure : pthread_cond_t u Structure initialisation and termination functions int pthread_cond_init( pthread_cond_t* cond_p , const pthread_condattr_t* cond_attr_p ); int pthread_cond_destroy( pthread_cond_t* cond_p ); Condition Variables Usage Pattern u Problem of synchronization between threads, without using active wait. u Threads invoke wait() function, which only returns when (the) event is signalled. u All threads update global state (with mutex), and block. u The last thread, which detects the event or condition is correct, signals to all: broadcast() Barrier Example using Condition Variables (from Pacheco) Barrier Example using Condition Variables (from Pacheco) Barrier Pseudocode: Mutex_lock(mutex); counter++; if(counter == thread_count): counter = 0; Cond_broadcast( cond_var ); else: //(counter < thread_count) while (…); cond_wait( cond_var, mutex ) Mutex_unlock(mutex); Producer-Consumer Example Producer-Consumer UnBounded Buffer Example u An example of producer-consumer synchronization problem is the unbounded buffer: u Thread Producer generates items for a buffer (list) with limited size; u Thread Consumer extracts items from that same list. u Cases: u Consumer can only execute when there is at least 1 item in the list; u Producer can only execute if the list has space available for 1 more item (assuming limited size). Producer-Consumer UnBounded Buffer Example (pseudo-code) mutex_t mutex; // mutex_init() cond_t notEmptyforConsumer; // cond_init() cond_t notFullforProducer; // cond_init() void* buffer[]; int firstptr, lastptr, count; // = 0 int buf_size; // buffer size void* produce(void* item); void* consume(); Producer-Consumer UnBounded Buffer Example (pseudo-code) void* produce(void* item) { mutex_lock( &mutex); while (count == buffer_size) pthread_cond_wait( &mutex, &notFullforProducer ); buffer[lastptr] = item; lastptr = lastptr + 1 % buffer_size; count += 1; pthread_cond_signal( &notEmptyforConsumer ); mutex_unlock( &mutex); } // end produce Producer-Consumer UnBounded Buffer Example (pseudo-code) void* consume() { mutex_lock( &mutex); while (count == 0) pthread_cond_wait( &mutex, &notEmptyforConsumer); item = buffer[firstptr]; firstptr = firstptr + 1 % buffer_size; count -= 1; pthread_cond_signal( &notFullforProducer ); mutex_unlock( &mutex); return item; } // end consume

Use Quizgecko on...
Browser
Browser