Be sure you have read, or at least skimmed, readings from the relevant updated appendices.
Your mission for this assignment is to improve the programs you wrote for Homework 2, to write versions in Java and OpenCL, and to measure their performance and accuracy more systematically.
(5 points)
Your first step will be to write a thread-safe random number generator, i.e., one that can be called from multiple threads concurrently without ill effects. To keep this part manageable, I suggest that you just use the technique mentioned in class, LCG (Linear Congruential Generator). The Wikipedia article has a pretty good discussion, but briefly:
This algorithm generates a pseudorandom sequence
from a seed , constants , , and , and
a simple recurrence relation:
(If for some reason you want to try a different algorithm, check with me first -- there may well be better choices, but there are probably worse choices too.)
You will need two implementations of whatever algorithm you choose, one in C and one in Java. Exactly how you package the algorithm is somewhat up to you, but you want functions analogous to srand() and rand(), and there needs to be some way to deal with the ``state'' of the sequence being generated (the current or next ) in a way that makes it possible for each thread to have its own state (rather than there being one hidden global state, as with srand() and rand()).
For C, what I think makes sense is to represent the saved state as an int64_t and define two functions that take a pointer to a state as a parameter:
void rand_set_seed(long seed, rand_state_t *state);
int64_t rand_next(rand_state_t *state);
const int64_t RANDMAX = (1LL << 48) - 1;
(Note that this is .)
For Java, you'll probably want to define a class analogous to java.util.Random, but much simpler, with just a RANDMAX constant, a constructor, and a next method. I'm guessing many of you don't know Java, so to get you started here are a skeleton class (called SimpleLCG) and a test program that uses it: SimpleLCG.java, TestSimpleLCG.java. TestSimpleLCG gets a seed and number of samples from the command line. Run it without arguments and it will remind you what arguments it wants.
The next step is to replace the current code for generating random numbers in two starter programs, one in C and one in Java, with your RNG code:
(5 points)
Replace the current code for generating random numbers in the two starter sequential programs with calls to your RNG. (If you didn't already test your RNG code, you might temporarily put in some debug-print statements to be sure it's generating reasonable output.) The two programs (C and Java) should now produce the same output (except for execution time).
(5 points)
(You only need to do this for one of your sequential programs, since they should give the same results.) Experiment until you find a seed that seems to give reasonable results, and then measure the relationship between accuracy (difference between the computed value of and the constant as defined in the math library) and number of samples: Generate output for at least six different values of ``number of samples'' (I recommend starting with a medium-size number and then repeatedly doubling it, rather than increasing by a fixed amount). Plot the results, by hand or with whatever program you like. (I use gnuplot. Short introduction/example below.) You can repeat this for more than one seed and plot all sets of results if you like.
(30 points)
Your mission for this step is to produce parallel programs for our four programming environments: C with OpenMP, C with MPI, Java, and C with OpenCL.
As we noted in class, having all UEs (processes or threads) generate points using the same RNG and seed is not useful. You have two options for dealing with this:
Hints for using leapfrogging:
For Java, a class still makes sense, but I think its constructor should take two more arguments, the number of ``streams'' (UEs for us) and which stream this object is for. You could put the code to generate the modified constants in the constructor, and I think it's fairly straightforward to do and to get right if in computing the constants you use BigInteger for intermediate values and only convert to Longs at the end (when the ``'' step gives you a result you know will fit).
For C, I thought it made sense to make the ``state'' a struct and introduce one more function
void rand_init_state(int p, int id, rand_state_t *state);
that computes and saves the values for the modified constants.
(5 points)
(You only need to do this for one of your parallel programs, since they should give the same results for the same number of units of execution, where ``units of execution'' is threads for OpenMP and Java, processes for MPI, and work items for OpenCL.) Experiment until you find a seed and number of samples that seem to give good results, and then measure the relationship between accuracy (difference between the computed value of and the constant as defined in the math library) and number of UEs. Generate output for at least six different values of ``number of UEs'' (I recommend powers of two, starting with one). (Since for OpenCL the number of work items has to be a multiple of the minimum work-group size, it might be interesting to make a second plot showing that minimum value and then several multiples of it.) Plot the results, again by hand or with whatever program you like.
(5 points)
To get meaningful results for performance, you will likely need far more samples than are needed to give reasonably accuracy, though you can use whatever seed seemed to be most effective. Find a number of samples for which both of your sequential programs take at least 2 seconds, and measure execution times for both sequential programs and all four parallel programs. For the parallel programs, measure execution time using different numbers of UEs (start with one and double until you notice that execution time is no longer decreasing). I strongly encourage you to do this on the machines that to me seem most suitable in terms of being able to ``scale up'' to interesting numbers of UEs: For OpenMP and Java, that would be Dione, for MPI, the Pandora cluster, and for OpenCL, Deimos or one of the Atlas machines. You should do each measurement more than once; if you get wildly different results it probably means you are competing with other work on the machine and should try again another time or using another machine or machines.
Plot the results, again by hand or with whatever program you like:
I talked about the plotting tool gnuplot in class one day (10/23). Here are files for a simple example along the lines of what you need to do for this assignment (plot parallel times as a function of UEs, also showing sequential time):
Turn in the following:
Submit your program source code, plots, and input data by sending mail to bmassing@cs.trinity.edu with each file as an attachment. Please use a subject line that mentions the course and the assignment (e.g., ``csci 3366 hw 3'' or ``parallel hw 3'').
Send program source as attachments. You can include other information in the body of the message or attach files in any format readable on our Linux lab machines (plain text, PDF, something openable with OpenOffice, etc.). No links to Google Docs, please.
Include the Honor Code pledge or just the word ``pledged'', plus at least one of the following about collaboration and help (as many as apply).1Text in italics is explanatory or something for you to fill in. For programming assignments, this should go in the body of the e-mail or in a plain-text file honor-code.txt (no word-processor files please).
Include a brief essay (a sentence or two is fine, though you can write as much as you like) telling me what about the assignment you found interesting, difficult, or otherwise noteworthy. For programming assignments, it should go in the body of the e-mail or in a plain-text file essay.txt (no word-processor files please).