5.PThreads – Thread-basierte Programmierung von Multicore-Systemen.

Slides:



Advertisements
Ähnliche Präsentationen
der Universität Oldenburg
Advertisements

Klassen - Verkettete Liste -
10.2 Wechselseitiger Ausschluss in Hardware
Forschungszentrum caesar
Agenda Sitzung 2 für den Programmaufbau
6. Der OpenMP Standard Direktiven-basiertes API zur Programmierung von Parallelrechnern mit gemeinsamem Speicher für FORTRAN, C und C++
Kapitel 3: Listen Lineare Liste: endliche Folge von Elementen eines Grundtyps (n>=0), leere Liste falls n=0 Listenelemente besitzen.
Vorlesung Echtzeitbetriebssysteme VI. Posix Übung
Seminar: "Einführung in C/C++" Einführung in die Programmiersprache C/C++ Donnerstag Andreas Döring SS 2004.
Kapitel 6.1 Nebenläufigkeit und wechselseitiger Ausschluss
Vorlesung Informatik 2 Algorithmen und Datenstrukturen (13 – Offenes Hashing) Prof. Th. Ottmann.
1 Named Pipes alias FIFO Haben einen Eintrag im Dateisystem und sind somit durch Zugriffsrechte identifizierbar Ermöglichen die Kommunikation zwischen.
EINI-I Einführung in die Informatik für Naturwissenschaftler und Ingenieure I Kapitel 9 Claudio Moraga; Gisbert Dittrich FBI Unido
EINI-I Einführung in die Informatik für Naturwissenschaftler und Ingenieure I Kapitel 7 Claudio Moraga, Gisbert Dittrich FBI Unido
EINI-I Einführung in die Informatik für Naturwissenschaftler und Ingenieure I Vorlesung 2 SWS WS 99/00 Gisbert Dittrich FBI Unido
EINI-I Einführung in die Informatik für Naturwissenschaftler und Ingenieure I Kapitel 10 Claudio Moraga; Gisbert Dittrich FBI Unido
Universität Dortmund, Lehrstuhl Informatik 1 EINI II Einführung in die Informatik für Naturwissenschaftler und Ingenieure.
PRJ 2007/1 Stefan Dissmann Motivation Problem: gleiche Datenstrukturen werden für verschiedene Objekte gebraucht: z.B. Listen von Studierenden, Kunden,
PKJ 2005/1 Stefan Dissmann Rückblick auf 2005 Was zuletzt in 2005 vorgestellt wurde: Klassen mit Attributen, Methoden und Konstruktoren Referenzen auf.
Vorlesung, Wintersemester 2009/10M. Schölzel 1 Optimierungstechniken in modernen Compilern Optimierungstechniken für SMP- Architekturen.
Projektplan: Fachgebiet Software Engineering Übersicht © Albert Zündorf, Kassel University.
Planung einfache Dateibehandlung (externe Dateien, Öffnen, Lesen/Schreiben, Schließen). Diskussion des Problems, die Wörter in einem gegebenen Text.
Weiteres Programm Studium des Breitendurchlaufs Hierzu
Wir müssen also überlegen: Implementierung der Knoten, Implementierung der Kanten, daraus: Implementierung des Graphen insgesamt. Annahme: die Knoteninhalte.
Einführung in die Programmierung Datensammlung
Einfach verkettete Listen
PRJ 2007/1 Stefan Dissmann Verkettete datenstruktur: Liste Problem: Liste, die eine beliebige Zahl von Elementen verwaltet Operationen: Erzeugen, Anfügen,
Einführung in die Informatik für Naturwissenschaftler und Ingenieure (alias Einführung in die Programmierung) (Vorlesung) Prof. Dr. Günter Rudolph Fachbereich.
Einführung in die Programmierung
Einführung in die Programmiersprache C 3.Tag Institut für Mathematische Optimierung - Technische Universität Braunschweig.
Javakurs FSS 2012 Lehrstuhl Stuckenschmidt
Einführung in die Programmierung Wintersemester 2012/13 Prof. Dr. Günter Rudolph Lehrstuhl für Algorithm Engineering Fakultät für Informatik TU Dortmund.
Einführung in die Programmierung Wintersemester 2009/10 Prof. Dr. Günter Rudolph Lehrstuhl für Algorithm Engineering Fakultät für Informatik TU Dortmund.
Einführung in die Informatik für Naturwissenschaftler und Ingenieure (alias Einführung in die Programmierung) (Vorlesung) Prof. Dr. Günter Rudolph Fakultät.
Einführung in die Programmierung
Einführung in die Programmierung
Einführung in die Programmiersprache C 4
Informatik 1 Letzte Übung.
Parallel Programming Condition Queues
Dynamische Datentypen
C-Einstieg. Agenda 1Vorbereitung 2Aufbau eines Programms 2.1Header 2.2 Methoden 2.3Main 3Datentypen & Variablen 4Operatoren(+, -, *, /) 5Logik 5.1IF 5.2Switch.
Thread Synchronisation in JAVA
Institut für Wirtschaftsinformatik – Software Engineering, JKU Linz 1 Algorithmen und Datenstrukturen Übungsmodul 2 Dr. W. Narzt u. Dr. A. Stritzinger.

Seminar aus Softwareentwicklung
BMEVIEEA100 Grundlagen der Programmierung
Vorlesung 3. if else Anweisung if (Ausdruck) Anweisung1 else Anweisung2 Ausdruck hat einen von 0 verschiedenen Wert, so wird Anweisung 1 ausgeführt. Bei.
Vorlesung 5. #include double fv(double x) { return sin(x); } int main(){ int i,N=10; double a=1,b=2,del,x,sum,f,integral; for(i=1,sum=0.0,del = (b-a)/N;i
Universität StuttgartInstitut für Wasserbau, Lehrstuhl für Hydrologie und Geohydrologie Copulas (1) András Bárdossy IWS Universität Stuttgart.
A) Erklären Sie den Datentyp char. b) Erklären Sie den Datentyp Struct c) Erklären Sie die Wirkungsweise des Operators & bei Anwendung im Zusammenhang.
Java Syntaxdiagramme Buchstabe A B Z a z ... Ziffer
1 // Laengste Zeile bestimmen // // Liest die Eingabe Zeichen fuer Zeichen, bis // das erscheint. // Die laengste Zeile wird ausgegeben (bei.
Echtzeitbetriebssysteme Übungen
Java-Kurs - 4. Übung Hausaufgabe Weitere Kontrollstrukturen
1 // Verkettete Liste 3 // demonstriert verkettete Listen und // Rekursion // (mit Entfernen eines Listenelementes) // #include struct Liste { int Element;
Threads in Java Threads  Sprachumfang von Java Der Java-Standard fordert nur die Unterstützung von Thread-Prioritäten. Es gibt keine Forderung bzgl.:
93 Das Monitorkonzept (nach Hoare/Brinch-Hansen 1974) Nur ein Prozess bzw. Thread kann zu einem bestimmten Zeitpunkt im Monitor aktiv sein => gegenseitiger.
1 // 9_1_Datei_IO //Elementare Verwendung von Dateien // Vorsicht: nicht robust, #include const int maxLen = 70; void Schreiben(char *, ofstream *); void.
1 // Verkettete Liste 2 // demonstriert verkettete Listen und // Rekursion // (Einfügen am "Fuß") // #include struct Liste { int Element; Liste *weiter;
, Dr. Wolfram Amme, Softwareentwicklung in Java, FSU Jena, WS 2005/06 1 Beispiel class SpreadSheet { int cellA1, cellA2, cellA3; synchronized.
Java Programme nur ein bisschen objektorientiert.
Tutorium Software-Engineering SS14 Florian Manghofer.
Pointer, Arrays und verkettete Listen. Mehrdimensionale Arrays  Pointer auf ein Array von Pointern  int32 **matrix = new int32*[3];  matrix: Zeiger.
Funktionen (Zweck und Eigenschaften) Funktionen sind Unterprogramme, die einen bestimmten Zweck erfüllen Sie zerlegen Probleme in kleine, abgeschlossene.
Center for Biotechnology Bielefeld Bioinformatics Service Netzwerk - Programmierung Threads Alexander Sczyrba Jan Krüger.
Graphische Benutzeroberfläche mit libforms
Hello World! Javakurs 2013 Arne Kappen
1.Event Queue.
Arguments to main() int main(int argc, char *argv[]) { int i;
Durchschnitt (verbal)
 Präsentation transkript:

5.PThreads – Thread-basierte Programmierung von Multicore-Systemen

216 Pthreads: POSIX Thread API Standard Thread API – IEEE POSIX c standard #include 3 Klassen von Routinen Threadverwaltung Erzeugung, Termination, Synchronisation, Setzen und Abfragen von Attributwerten Wechselseitiger Ausschluss (mutexes) Erzeugung, Löschen, Sperren und Entsperren von Mutexvariablen; Setzen und Modifikation von Mutex.-Attributen Bedingungsvariablen Erzeugung, Entfernen, Warten und Signalisieren auf Bedingungsvariablen, Setzen/Abfragen von Attributwerten

217 Namenskonventionen Alle Bezeichner beginnen mit pthread_. Präfix Funktionale Gruppe pthread _Threads selbst und Routinen pthread_attr_ Thread Attributobjekte pthread_mutex_ Mutexes pthread_mutexattr_ Mutex Attributobjekte pthread_cond_ Bedingungsvariablen pthread_condattr_ Attributobjekte zu Bedingungsvariablen Compilation von Programmen mit gcc -pthread

218 Threadverwaltung Thread-Erzeugung int pthread_create ( pthread_t*thread_handle, const pthread_attr_t *attribute, void * (*thread_function){void *}, void *arg); Thread-Synchronisation int pthread_join ( pthread_tthread, void **ptr); Thread-Termination Thread beendet Ausführung Aufruf von pthread_exit() Annulierung durch anderen Prozess mittels pthread_cancel Prozess terminiert

219 Beispiel: Hello World #include #define NUM_THREADS 5 void *PrintHello(void *threadid) { int tid; tid = (int)threadid; printf("Hello World! It's me, thread #%d!\n", tid); pthread_exit(NULL); } int main (int argc, char *argv[]) { pthread_t threads[NUM_THREADS]; int rc, t; for(t=0; t<NUM_THREADS; t++){ printf("In main: creating thread %d\n", t); rc = pthread_create(&threads[t], NULL, PrintHello, (void *)t); if (rc){ printf("ERROR; return code from pthread_create() is %d\n", rc); exit(-1); } } pthread_exit(NULL); }

220 Argumentübergabe Weitergabe eines Arguments an die Start-Routine eines Threads call-by-reference Übergabe mit einem cast zu (void *) Mehrere Argumente müssen in einer Struktur zusammengefasst werden. int *taskids[NUM_THREADS];... for(t=0;t<NUM_THREADS;t++) { taskids[t] = (int *) malloc(sizeof(int)); *taskids[t] = t; printf("Creating thread %d\n", t); rc = pthread_create(&threads[t], NULL, PrintHello, (void *) taskids[t]);... }

221 Beispiel: Übergabe mehrerer Argumente struct thread_data { intthread_id; int sum; char *message; }; struct thread_data thread_data_array[NUM_THREADS]; void *PrintHello(void *threadarg) {...; struct thread_data *my_data; my_data = (struct thread_data *) threadarg; taskid = my_data->thread_id; sum = my_data->sum; hello_msg = my_data->message;... } int main(int argc, char *argv[]) {... thread_data_array[t].thread_id = t; thread_data_array[t].sum = sum; thread_data_array[t].message = messages[t]; rc = pthread_create(&threads[t], NULL, PrintHello, (void *) &thread_data_array[t]);... }

222 Beispiel: Thread Synchronisation Das Attributargument von pthread_create () wird genutzt, um einen Thread als „joinable“ oder „detached“ (nicht joinable) zu erzeugen. typischer Ablauf: pthread_attr_t attr; /* Deklaration */... pthread_attr_init(&attr);/* Initialisierung */ pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_JOINABLE); /* Status joinable sicherstellen */... rc = pthread_create(&thread[t], &attr, BusyWork, NULL);... pthread_attr_destroy(&attr);.... rc = pthread_join(thread[t], &status); => Programm: join1.c

223 Fallstudie: Berechnung von pi #include... #define MAX_THREADS512 void *compute_pi (void *);... main() {... pthread_t p_threads[MAX_THREADS]; pthread_attr_t attr;... pthread_attr_init (&attr);... for (i=0; i< num_threads; i++) {hits[i][0] = i; pthread_create( &p_threads[i], &attr, compute_pi, (void *) &hits[i][0]); } for (i=0; i< num_threads; i++) {pthread_join( p_threads[i], NULL); total_hits += hits[i][0]; }... }

224 Beispiel: Berechnung von pi (Forts.) void *compute_pi (void *s) { int thread_no, i, *hit_pointer; double rand_no_x, rand_no_y; int hits; hit_pointer = (int *) s; thread_no = *hit_pointer; hits = 0; srand48(thread_no); for (i = 0; i < sample_points_per_thread; i++){ rand_no_x = (double)(rand_r(&thread_no))/(double)((2 <<30) - 1); rand_no_y = (double)(rand_r(&thread_no))/(double)((2 <<30) - 1); if (((rand_no_x - 0.5) * (rand_no_x - 0.5) + (rand_no_y - 0.5) * (rand_no_y - 0.5)) < 0.25) (*hit_pointer) ++; hits ++; } *hit_pointer = hits; } t

225 Wechselseitiger Ausschluss mittels binärer Semaphor-Variablen: mutex-locks Sperren int pthread_mutex_lock ( pthread_mutex_t *mutex_lock); Entsperren int pthread_mutex_unlock ( pthread_mutex_t *mutex_lock); Initialisieren int pthread_mutex_init ( pthread_mutex_t *mutex_lock, const pthread_mutexattr_t *lock_attr);

226 Beispielanwendung pthread_mutex_t minimum_value_lock;... main() {.... pthread_mutex_init(&minimum_value_lock, NULL);.... } void *find_min(void *list_ptr) {.... pthread_mutex_lock(&minimum_value_lock); if (my_min < minimum_value) minimum_value = my_min; pthread_mutex_unlock(&minimum_value_lock); } => Programme dotprod_serial.c, dotprod_mutex.c

227 Verringerung des Locking-Overheads pthread_mutex_trylock als Alternative zu pthread_mutex_lock pthread_mutex_trylock. int pthread_mutex_trylock ( pthread_mutex_t *mutex_lock); keine Blockade, sondern Rückgabe von EBUSY schneller, da keine Warteschlangenverwaltung

228 Beispiel: k Kopien eines Elementes finden void *find_entries(void *start_pointer) { struct database_record *next_record; int count; current_pointer = start_pointer; do {next_record = find_next_entry(current_pointer); count = output_record(next_record); } while (count < requested_number_of_records); } int output_record(struct database_record *record_ptr) { int count; pthread_mutex_lock(&output_count_lock); output_count ++; count = output_count; pthread_mutex_unlock(&output_count_lock); if (count <= requested_number_of_records) print_record(record_ptr); return (count); }

229 Beispiel:... mit trylock int output_record(struct database_record *record_ptr) { int count; int lock_status; lock_status=pthread_mutex_trylock(&output_count_lock); if (lock_status == EBUSY) { insert_into_local_list(record_ptr); return(0); } else { count = output_count; output_count += number_on_local_list + 1; pthread_mutex_unlock(&output_count_lock); print_records(record_ptr, local_list, requested_number_of_records - count); return(count + number_on_local_list + 1); }

230 Bedingungsvariablen zur einseitigen Synchronisation immer mit mutex-lock assoziiert Funktionen int pthread_cond_wait(pthread_cond_t *cond, pthread_mutex_t *mutex); int pthread_cond_signal(pthread_cond_t *cond); int pthread_cond_broadcast(pthread_cond_t *cond); int pthread_cond_init(pthread_cond_t *cond, const pthread_condattr_t *attr); int pthread_cond_destroy(pthread_cond_t *cond);

231 Beispiel: Producer-Consumer pthread_cond_t cond_queue_empty, cond_queue_full; pthread_mutex_t task_queue_cond_lock; int task_available; /* other data structures here */ main() { /* declarations and initializations */ task_available = 0; pthread_cond_init(&cond_queue_empty, NULL); pthread_cond_init(&cond_queue_full, NULL); pthread_mutex_init(&task_queue_cond_lock, NULL); /* create and join producer and consumer threads */ }

232 Beispiel (Forts.): Producer void *producer(void *producer_thread_data) { int inserted; while (!done()) { create_task(); pthread_mutex_lock(&task_queue_cond_lock); while (task_available == 1) pthread_cond_wait(&cond_queue_empty, &task_queue_cond_lock); insert_into_queue(); task_available = 1; pthread_cond_signal(&cond_queue_full); pthread_mutex_unlock(&task_queue_cond_lock); }

233 Beispiel (Forts.): Consumer void *consumer(void *consumer_thread_data) { while (!done()) { pthread_mutex_lock(&task_queue_cond_lock); while (task_available == 0) pthread_cond_wait(&cond_queue_full, &task_queue_cond_lock); my_task = extract_from_queue(); task_available = 0; pthread_cond_signal(&cond_queue_empty); pthread_mutex_unlock(&task_queue_cond_lock); process_task(my_task); }

234 Fallstudie: Lese-/Schreibsperren (read/write locks) erweiterter Synchronisationsmechanismus vgl. Leser-/Schreiber-Problem realisierbar mit mutex- und Bedingungsvariablen Beispielrealisierung einer RW-Sperrvariablen vom Typ mylib_rwlock_t Zähler readers für Leser Zähler writer für Schreiber (ist 0 oder 1) 2 Bedingungsvariablen readers_proceed writer_proceed Zähler pending_writers wartender Schreiber Mutex-Variable read_write_lock zum Schutz des Datenbereichs typedef struct { int readers; int writer; pthread_cond_t readers_proceed; pthread_cond_t writer_proceed; int pending_writers; pthread_mutex_t read_write_lock; } mylib_rwlock_t;

235 Initialisierung void mylib_rwlock_init (mylib_rwlock_t *l) { l->readers = l->writer = l->pending_writers = 0; pthread_mutex_init(&(l->read_write_lock),NULL); pthread_cond_init(&(l->readers_proceed),NULL); pthread_cond_init(&(l->writer_proceed), NULL); } Funktionen mylib_rwlock_rlock - Leserzugang mylib_rwlock_wlock - Schreiberzugang mylib_rwlock_unlock - Freigabe

236 Leserzugang void mylib_rwlock_rlock(mylib_rwlock_t *l) { /* if there is a write lock or pending writers, perform condition wait.. else increment count of readers and grant read lock */ pthread_mutex_lock(&(l->read_write_lock)); while ((l->pending_writers > 0) || (l->writer > 0)) pthread_cond_wait(&(l->readers_proceed), &(l->read_write_lock)); l -> readers ++; pthread_mutex_unlock(&(l->read_write_lock)); }

237 Schreiberzugang void mylib_rwlock_wlock(mylib_rwlock_t *l) { /* if there are readers or writers, increment pending writers count and wait. On being woken, decrement pending writers count and increment writer count */ pthread_mutex_lock(&(l->read_write_lock)); while ((l->writer > 0) || (l->readers > 0)) { l -> pending_writers ++; pthread_cond_wait(&(l->writer_proceed), &(l->read_write_lock)); } l->pending_writers--; l -> writer++; pthread_mutex_unlock(&(l->read_write_lock)); }

238 Freigabe void mylib_rwlock_unlock(mylib_rwlock_t *l) { /* if there is a write lock then unlock, else if there are read locks, decrement count of read locks. If the count is 0 and there is a pending writer, let it through, else if there are pending readers, let them all go through */ pthread_mutex_lock(&(l->read_write_lock)); if (l->writer > 0) l->writer = 0; else if (l->readers > 0) l->readers--; if ((l->readers == 0) && (l->pending_writers > 0)) pthread_cond_signal(&(l->writer_proceed)); else if (l->readers > 0) pthread_cond_broadcast(&(l->readers_proceed)); pthread_mutex_unlock(&(l->read_write_lock)); }

239 Fallstudie: Lineare Barriere typedef struct { pthread_mutex_t count_lock; pthread_cond_t ok_to_proceed; int count; } mylib_barrier_t; void mylib_init_barrier(mylib_barrier_t *b) { b -> count = 0; pthread_mutex_init(&(b -> count_lock), NULL); pthread_cond_init(&(b -> ok_to_proceed),NULL); }

240 void mylib_barrier (mylib_barrier_t *b, int num_threads) { pthread_mutex_lock(&(b -> count_lock)); b -> count ++; if (b->count == num_threads) { b->count = 0; pthread_cond_broadcast(&(b->ok_to_proceed)); } else while (pthread_cond_wait(&(b->ok_to_proceed), &(b->count_lock)) != 0); pthread_mutex_unlock(&(b -> count_lock)); }