Search
Cílem cvičení je implementovat program s třemi paralelně běžícími vlákny. První vlákno je určeno pro zpracování vstupu (čtení stisknuté klávesy), druhé vlákno aktualizuje výstup (jednořádkový) a třetí vlákno implementuje časovač, který po uplynutí definované periody zvýší hodnotu proměnné (čítače). Perioda může být nastavována uživatelem, stiskem definovaných kláves. Použití vláken je založeno na knihovně pthread (POSIX threads) a příklad demonstruje použití základních konstrukcí pthread knihovny.
pthread
void* thread1(void*); void* thread2(void*);
thread1()
counter
void* thread1(void *v) { bool q = false; /* from the <stdbool.h> */ while (!q) { usleep(100 * 1000); /* from the <unistd.h> */ counter += 1; } return 0; }
thread2()
void* thread2(void *v) { bool q = false; while (!q) { printf(*"\rCounter %10i", counter); } return 0; }
pthread_create()
counter = 0; pthread_t thrs[2]; pthread_create(&thrs[0], NULL, thread1, NULL); pthread_create(&thrs[1], NULL, thread2, NULL);
Jakmile je vlákno vytvořeno, je spuštěno a začne vykonávat tělo předané funkce.
Enter
pthread_join()
getchar(); for (int i = 0; i < 2; ++i) { pthread_join(thrs[i], NULL); }
V případě, kdy je nutné přistupovat k proměnné sdílené více vlákny je zpravidla žádoucí zabranit souběhu definováním kritické sekce zámkem (mutexem), který dovolí vstup do sekce pouze jedinému vláknu. Kritická sekce může být např. implementována s využitím zámku pthread_mutex_t.
pthread_mutex_t
mtx
pthread_mutex_init()
pthread_mutex_t mtx; pthread_mutex_init(&mtx, NULL);
pthead_mutex_lock()
pthread_mutex_unlock()
void *thread1(void *v) { ... pthread_mutex_lock(&mtx); counter += 1; pthread_mutex_unlock(&mtx); ... }
quit
bool quit = false; ... gechar(); pthread_mutex_lock(&mtx); quit = true; pthread_mutex_unlock(&mtx); ...
void* thread1(void *v) { ... while (!q) { ... pthread_mutex_lock(&mtx); ... q = quit; ....
Aktuální implementace aktualizace výpisu v cyklu thread2 je výpočetně velmi náročná, neboť cyklus je opakován tak rychle jak dovolují prostředky CPU. V našem případě je taková extermní frekvence aktualizace zbytečná a neefektivní (např. při běhu notebooku/tabletu na baterii se zbytečně čerpá omezená energie a navíc zařízení zahříváme). V podstatě nám stačí aktualizovat výpis pouze v případě, kdy dojde ke změně hodnoty proměnné counter Takovou synchronizaci vláken můžeme implementovat mechanismem podmíněných proměnných (conditional variable), tj. proměnná typu pthread_cond_t, kterou použijeme k pozastavení vykonávání vlákna voláním pthread_cond_wait(). Takto čekající vlákno můžeme následně probudit voláním pthread_cond_singnal() nebo pthread_cond_broadcast().
thread2
pthread_cond_t
pthread_cond_wait()
pthread_cond_singnal()
pthread_cond_broadcast()
condvar
pthread_cond_t condvar; ... pthread_cond_init(&condvar, NULL);
void* thread2(void *d) { ... pthread_mutex_lock(&mtx); ... pthread_cond_wait(&condvar, &mtx); ... pthread_mutex_unlock(&mtx); ...
pthread_cond_signal()
void* thread1(void *d) { ... pthread_mutex_lock(&mtx); counter += 1; ... pthread_cond_signal(&condvar); ... pthread_mutex_unlock(&mtx); ...
flush
stdout
fflush(stdout);
main()
raw
void set_raw(int set) { static struct termios tio, tioOld; tcgetattr(STDIN_FILENO, &tio); if (set) { tioOld = tio; //backup cfmakeraw(&tio); tio.c_lflag &= ~ECHO; // assure echo is disabled tio.c_oflag |= OPOST; // enable output postprocessing tcsetattr(STDIN_FILENO, TCSANOW, &tio); } else { tcsetattr(STDIN_FILENO, TCSANOW, &tioOld); } }
"\rAlarm period: %10i Alarm counter: %10i"
V další části cvičení pokračujte implementací domácího úkolu hw09.