Download presentation
Presentation is loading. Please wait.
1
Arquitectura de Sistemas Paralelos e Distribuídos Paulo Marques Dep. Eng. Informática – Universidade de Coimbra pmarques@dei.uc.pt Ago/2007 4. OpenMP
2
2 OpenMP – Recap Shared Memory Model with Threads Typically used on UMA or NUMA machines Compiler directive based Can use serial code Even so, it is necessary to use some library routines Jointly defined and endorsed by a group of major computer hardware and software vendors. Portable / multi-platform, including Unix and Windows Available in C/C++ and Fortran implementations Can be very easy and simple to use - provides for “incremental parallelism”
3
3 OpenMP Programming Model Fork-Join Parallelism Master thread creates a number of working threads as needed Typically used on heavy-duty for cycles: parallel loops A sequential program naturally evolves into a parallel one Parallel Regions Master Thread Worker Threads
4
4 How is OpenMP Typically Used? OpenMP is mostly used to parallelize time consuming loops The iterations of each “for” loop are divided among different threads void main() { double result[SIZE]; for (int i=0; i<SIZE; i++) { huge_computation(i, result[i]); } void main() { double result[SIZE]; #pragma omp parallel for for (int i=0; i<SIZE; i++) { huge_computation(i, result[i]); }
5
5 OpenMP – Some Considerations OpenMP is a shared-memory model By default, all threads share the same variables Beware: This can lead to race conditions There are primitives for synchronizing threads Beware: Synchronization is quite expensive An alternative is to duplicate data, avoiding variable sharing between threads. That can, nevertheless, also be expensive. There are also primitives for this.
6
6 Syntax OpenMP constructs can either be: Compiler directives Library function calls Compiler directives apply to structured blocks A structured block consists of a block of code with only one entry point at the top and an exit point at the bottom. The only branch statement allowed in a structured block is an exit() call. In C/C++, the compiler directives take the form: #pragma omp construct [clause [clause] …]
7
7 Simple “Hello World” program (hello_omp.c) #include #define NUM_THREADS 4 int main() { omp_set_num_threads(NUM_THREADS); #pragma omp parallel { int id = omp_get_thread_num(); printf("Hello, I'm thread %d\n", id); } return 0; } OpenMP function call OpenMP compiler directive
8
8 Compiling & Running the Example Compiling [pmarques@ingrid ~/best] source /opt/intel/compiler10/ia32/bin/iccvars.sh [pmarques@ingrid ~/best] icc -openmp hello_omp.c -o hello_omp Running [pmarques@ingrid ~/best]./hello_omp Hello, I'm thread 0 Hello, I'm thread 1 Hello, I'm thread 3 Hello, I'm thread 2
9
9 Actually... In the last example, you will probably see something like... [pmarques@ingrid ~/best]./hello_omp Hello, I'm thrHello, I'm thread 1Hello, I'm thread 3 Aborted You must be careful about what you call in parallel. The correct version of the previous program would be (hello_omp2.c):... #pragma omp parallel { int id = omp_get_thread_num(); #pragma omp critical printf("Hello, I'm thread %d\n", id); }...
10
10 Constructs OpenMP’s constructs fall into 5 categories Parallel regions Work-sharing Data environment Synchronization Runtime functions/Environment variables We are going to see a good deal of them!
11
11 Parallel Regions Threads are created using the #pragma omp parallel directive omp_set_num_threads() sets the number of threads for the next region omp_get_thread_num() gets the number of the current thread Variables are shared across threads double A[1000]; omp_set_num_threads(3); #pragma omp parallel { int id = omp_get_thread_num(); calc(id, A); } printf(“Done\n”); omp_set_num_threads(3); double A[1000]; calc(0,A) calc(1,A)calc(2,A) printf(“Done\n”); A is shared across threads There is a barrier at the end of the block
12
12 An Important Detail… Normally, you specify a number of threads that correspond to the number of processors you have OpenMP lets you use two running modes: Dynamic (the default): The number of threads in a parallel region can vary from one parallel region to another Setting the maximum number of threads only sets the maximum: you can get less! Static: The number of threads is always the same and specified by the programmer
13
13 Work-Sharing Constructs – for The work-sharing construct “for”, divides the iterations of a loop among several threads Shorthand for parallel fors: #pragma omp parallel for #pragma omp parallel { … #pragma omp for for (int i=0; i<N; i++) { heavy_stuff(i); } By default there is a barrier at the end of the “omp for”. It can be turned off with the “nowait” clause. (Use with care!)
14
14 And Now, Something Interesting! // Matrix Multiplication (1024x1024) #include #define N 1024 double A[N][N], B[N][N], C[N][N]; int main() { time_t t1, t2; time(&t1); for (int i=0; i<N; i++) { for (int j=0; j<N; j++) { C[i][j] = 0.0; for (int k=0; k<N; k++) C[i][j]+= A[i][k]*B[k][j]; } time(&t2); printf(“Multiplying %dx%d matrixes took %d sec\n”, N, N, t2-t1); }
15
15 Matrix Multiplication, OpenMP version (mult_omp.c) #include... int main() { time_t t1, t2; time(&t1); omp_set_num_threads(omp_get_num_proc()); #pragma omp parallel for for (int i=0; i<N; i++) { for (int j=0; j<N; j++) { C[i][j] = 0.0; for (int k=0; k<N; k++) C[i][j]+= A[i][k]*B[k][j]; } time(&t2); printf(“Multiplying %dx%d matrixes took %d sec\n”, N, N, t2-t1); }
16
16 Results Ingrid, a dual-Xeon P4 2GHz machine with support for Hyper-Threading In practice, it thinks it has 4 processors! Serial version: 63 seconds! Parallel version: 8 seconds! This is a 7.8x speedup on a dual processor machine!!! Not bad for 3 lines of code Can you explain this astonishing result? (If you think this question is silly, either you know too much or too little.)
17
17 omp parallel for You can add a schedule clause: #pragma omp parallel for schedule(…) schedule(static [,chunk]) Each thread is given out a certain number of blocks of size chunk schedule(dynamic [,chunk]) Each thread grabs chunk iterations from a queue as soon as it finishes its current work schedule(guided [,chunk]) Each thread dynamically gets blocks of iterations. The size starts out by being big and shrinks down to chunk size schedule(runtime) Any of the above, as specified in the OMP_SCHEDULE environment variable
18
18 Sections Work-Sharing Construct sections give a different structured block to each thread By default there is a barrier at the end of the construct. As in the “for” construct, you can turn it off by specifying the “nowait” flag Given to a thread #pragma omp parallel #pragma omp sections { x_calculation(); #pragma omp section y_calculation(); #pragma omp section z_calculation(); } Given to a thread
19
19 Scope of OpenMP Constructs In OpenMP there is the notion of Lexical (or static) extent and Dynamic extent Lexical Extent: corresponds to the static region being defined by the omp pragma Dynamic Extent: corresponds to what is seen at runtime. I.e. the dynamic extent can be larger than the lexical static extent dynamic extent main.c … #pragma omp parallel { f(); } f.c … void f() { #pragma omp for for (int i=0; i<1000; i++) g(i); } static extent dynamic extent
20
20 Data Environment Constructs Default Storage Shared Memory Model: most variables are shared by default Global variables are SHARED among threads Not everything is shared: Stack variables in routines called in parallel regions (these are thread private) Automatic variables within a statement block (which are also private)
21
21 Changing Storage Attributes It is possible to change the way threads see each variable within a parallel block For this, the following attributes are used: shared (the default) private firstprivate lastprivate threadprivate Note that shared, private, firstprivate and threadprivate can apply to any parallel block, while lastprivate only applies to parallel fors
22
22 private clause private(var) creates a local copy of var for each thread The value of var is not initialized the private copy of var is in no way connected with the original variable. (It’s like it’s a new variable with the same name!) int total; total = 0; #pragma omp parallel for private(total) for (int i=0; i<10; i++) total += i; printf(“%d\n”, total); Each thread gets a copy of total. But the initial value of total is unknown The value of total is unknown at this point
23
23 A note on usage… Note that private (and firstprivate, which we will see next), are mostly used for internal variables needed for a computation Example: calculate all the square roots from 1 to 10000 (N) in parallel double result[N]; double aux; #pragma omp parallel for private(aux) for (int i=0; i<N; i++) { aux = sqrt(i); result[i] = aux; } double result[N]; double aux; #pragma omp parallel for for (int i=0; i<N; i++) { aux = sqrt(i); result[i] = aux; } Right! Wrong!
24
24 firstprivate clause It’s the same as private, but the initial value of the variable is passed to each thread int total; total = 0; #pragma omp parallel for firstprivate(total) for (int i=0; i<10; i++) total += i; printf(“%d\n”, total); Each thread gets a copy of total, with the initial value of 0 The value of total is not valid at this point (same as private).
25
25 lastprivate clause Same as private, but passes the value of the last iteration of a parallel for to the variable of the same name it represents Note that it is the value of the last iteration In a parallel for, it’s possible to specify firstprivate and lastprivate int total; total = 0; #pragma omp parallel for firstprivate(total) lastprivate(total) for (int i=0; i<10; i++) total += i; printf(“%d\n”, total); Tricky: what’s the value printed out? Certainly not what we want…
26
26 A Little Reminder c is shared by all threads In the parallel block, each thread has a copy of a and b a is 1 when each thread starts the value of b is not initialized when each thread starts After the block, the values of a and b are undefined int a, b, c; a = b = c = 1; #pragma omp parallel firstprivate(a) private(b) { … }
27
27 threadprivate clause Makes global data (global variables) an integral part of each thread This is different from making them private With private, global variables are masked With threadprivate, each thread gets a persistent copy of a global variable that survives across parallel blocks Threadprivate variables can be initialized with the “copyin” clause
28
28 reduction clause A very important clause Allows values of a parallel block to be combined after the block reduction(operator:list) The variables in list must appear in the enclosing parallel region Inside a parallel or work-sharing construct: A local copy of each variable in list is made, and initialized according to the operator being specified (e.g. 0 for +, 1 for *, etc.) Local copies of the variable are reduced to a single global copy at the end of the construct
29
29 Example of reduction #include int main() { int total = 0; int i; #pragma omp parallel for firstprivate(total) reduction(+:total) for (i=0; i<10; i++) total+=i; printf("total=%d\n", total); return 0; } Always prints out 45!
30
30 Synchronization OpenMP supports the following constructs for synchronization atomic critical section barrier flush (not covered here) ordered (not covered here) (single) (master) not exclusively synchronization
31
31 Atomic It’s a type of critical section – only one thread can execute at a time It can only be used for guaranteeing correct memory updates The statement (memory update) must have the form: varop= expr, where op is + * - / & ^ | >, or ++var, --var, var++, var--. int x, b; x = 0; #pragma omp parallel private(b) { b = calculate(i); #pragma omp atomic x+= b; }
32
32 Critical Section Only one thread at a time can enter a critical section. Critical sections can be named: #pragma omp critical (name) int x, b; #pragma omp parallel private(b) { b = calculate(i); #pragma omp critical { x+= b; write_to_file(b); }
33
33 Barrier Each thread waits until all threads have arrived at the barrier int A[1000]; #pragma omp parallel { big_calculation_1(A); #pragma omp barrier big_calculation_2(A); }
34
34 Single The single construct is used to implement a block that is executed only by one thread. All other threads wait (barrier) until that one finishes that section. #pragma omp parallel { perform_calculationA(omp_get_thread_num()); #pragma omp single { printf(“Now at the middle of the calculation…\n”); } perform_calculationB(omp_get_thread_num()); }
35
35 Master The master construct denotes a structured block that is only executed by the master thread. The other threads just ignore it. (no implied barriers) #pragma omp parallel { #pragma omp master { printf(“Now starting main calculation…\n”); } perform_calculation(omp_get_thread_num()); }
36
36 OpenMP Library Routines Locks omp_init_lock(), omp_set_lock(), omp_unset_lock(), omp_tst_lock() Runtime Check/Modify number of threads omp_set_num_threads(), omp_get_num_threads(), omp_get_thread_num(), omp_get_max_threads() Turn on/off dynamic mode omp_get_dynamic(), omp_set_dynamic() Number of processors in the system omp_get_num_procs()
37
37 Fixing the Number of Threads in a Program a) Turn off dynamic mode b) Set the number of threads int main() { omp_set_dynamic(false); omp_set_num_threads(omp_get_num_procs()); … }
38
38 Environment Variables Some variables control the way OpenMP works by default OMP_SHEDULE parameters for schedule(dynamic) OMP_NUM_THREADS default number of threads OMP_DYNAMIC (TRUE/FALSE) use of dynamic mode OMP_NESTED (TRUE/FALSE) whether parallel regions can be nested
39
39 And Now, a Little Example… Our familiar N-Queens Problem Source code slightly changed so that the first two levels are directly checked using a nested “for” loop (this saves a lot of trouble)
40
40 Serial Version of N-Queens (nqueens.c) int n_queens(int size) { // The board int board[MAX_SIZE]; // Total solutions for this level int solutions = 0; // Try to place a queen in each line of the column for (int a=0; a<size; a++) { for (int b=0; b<size; b++) { if ((a==b) || (a==b-1) || (a==b+1)) continue; // Place queens board[0] = a; board[1] = b; // Check the rest solutions += place_queen(2, board, size); } return solutions; }
41
41 Parallel Version of N-Queens (nqueens_omp.c) int n_queens(int size) { // The board int board[MAX_SIZE]; // Total solutions for this level int solutions = 0; #pragma omp parallel for reduction(+:solutions) private(board) // Try to place a queen in each line of the column for (int a=0; a<size; a++) { for (int b=0; b<size; b++) { // The first two queens cannot be in the same line or diagonal if ((a==b) || (a==b-1) || (a==b+1)) continue; // Place queens board[0] = a; board[1] = b; // Check the rest solutions += place_queen(2, board, size); } return solutions; }
42
42 Results on Ingrid A 2.4 speedup for a dual-processor with hyper-threading Although very good, why isn’t this result as good as matrix-mult with OMP? [pmarques@ingrid ~/best]./nqueens Running NQueens size=14 Queens 14: total solutions: 365596 It took 12 sec [pmarques@ingrid ~/best]./nqueens_omp Running NQueens size=14 Using 4 threads Queens 14: total solutions: 365596 It took 5 sec
Similar presentations
© 2025 SlidePlayer.com. Inc.
All rights reserved.