Solutions for Chapel course



 Table of contents

Part 1: basic language features

Solution to “Basic.1”

To see the evolution of the temperature at the top right corner of the plate, we just need to modify iout and jout. This corner correspond to the first row (iout=1) and the last column (jout=cols) of the plate.

$ chpl baseSolver.chpl -o baseSolver
$ sbatch serial.sh
$ tail -f solution.out
Temperature at iteration 0: 25.0
Temperature at iteration 20: 1.48171
Temperature at iteration 40: 0.767179
...
Temperature at iteration 460: 0.068973
Temperature at iteration 480: 0.0661081
Temperature at iteration 500: 0.0634717

Solution to “Basic.2”

To get the linear distribution, the 80 degrees must be divided by the number of rows or columns in our plate. So, the following couple of for loops at the start of time iteration will give us what we want:

// boundary conditions
for i in 1..rows do
  T[i,cols+1] = i*80.0/rows;   // right side
for j in 1..cols do
  T[rows+1,j] = j*80.0/cols;   // bottom side

Note that 80 degrees is written as a real number 80.0. The division of integers in Chapel returns an integer, then, as rows and cols are integers, we must have 80 as real so that the result is not truncated.

$ chpl baseSolver.chpl -o baseSolver
$ sbatch serial.sh
$ tail -f solution.out
Temperature at iteration 0: 25.0
Temperature at iteration 20: 2.0859
Temperature at iteration 40: 1.42663
...
Temperature at iteration 460: 0.826941
Temperature at iteration 480: 0.824959
Temperature at iteration 500: 0.823152

Solution to “Basic.3”

The idea is simple: after each iteration of the while loop, we must compare all elements of Tnew and T, find the greatest difference, and update delta with that value. The following nested for loops should do the job:

// update delta, the greatest difference between Tnew and T
delta = 0;
for i in 1..rows do {
  for j in 1..cols do {
    tmp = abs(Tnew[i,j] - T[i,j]);
    if tmp > delta then delta = tmp;
  }
}

Clearly there is no need to keep the difference at every single position in the array, we just need to update delta if we find a greater one.

$ chpl baseSolver.chpl -o baseSolver
$ sbatch serial.sh
$ tail -f solution.out
Temperature at iteration 0: 25.0
Temperature at iteration 20: 2.0859
Temperature at iteration 40: 1.42663
...
Temperature at iteration 460: 0.826941
Temperature at iteration 480: 0.824959
Temperature at iteration 500: 0.823152

Solution to “Basic.4”

For example, lets use a 650 x 650 grid and observe the evolution of the temperature at the position (200,300) for 10000 iterations or until the difference of temperature between iterations is less than 0.002; also, let’s print the temperature every 1000 iterations.

$ chpl --fast baseSolver.chpl -o baseSolver
$ ./baseSolver --rows=650 --cols=650 --iout=200 --jout=300 --niter=10000 --tolerance=0.002 --nout=1000
Temperature at iteration 0: 25.0
Temperature at iteration 1000: 25.0
Temperature at iteration 2000: 25.0
Temperature at iteration 3000: 25.0
Temperature at iteration 4000: 24.9998
Temperature at iteration 5000: 24.9984
Temperature at iteration 6000: 24.9935
Temperature at iteration 7000: 24.9819
Final temperature at the desired position after 7750 iterations is: 24.9671
The greatest difference in temperatures between the last two iterations was: 0.00199985

Solution to “Basic.5”

Without --fast the calculation will become slower by ~95X.

Part 2: task parallelism

Solution to “Task.1”

The following code is a possible solution:

var x = 1;
config var numthreads = 2;
var messages: [1..numthreads] string;
writeln('This is the main thread: x = ', x);
coforall threadid in 1..numthreads do {
  var c = threadid**2;
  messages[threadid] = 'this is thread ' + threadid:string + ': my value of c is ' + c:string + ' and x is ' + x:string;  // add to a string
}
writeln('This message will not appear until all threads are done ...');
for i in 1..numthreads do  // serial loop, will be printed in sequential order
  writeln(messages[i]);
$ chpl consecutive.chpl -o consecutive
$ sed -i -e 's|coforall --numthreads=5|consecutive --numthreads=5|' shared.sh
$ sbatch shared.sh
$ cat solution.out
This is the main thread: x = 10
This message will not appear until all threads are done ...
this is thread 1: my value of c is 1 and x is 10
this is thread 2: my value of c is 4 and x is 10
this is thread 3: my value of c is 9 and x is 10
this is thread 4: my value of c is 16 and x is 10
this is thread 5: my value of c is 25 and x is 10

Solution to “Task.2”

config const numthreads = 12;     // let's pretend we have 12 cores
const n = nelem / numthreads;     // number of elements per thread
const r = nelem - n*numthreads;   // these did not fit into the last thread
var lmax: [1..numthreads] real;   // local maximum for each thread

coforall threadid in 1..numthreads do {   // each iteration processed by a separate thread
  var start, finish: int;
  start  = (threadid-1)*n + 1;
  finish = (threadid-1)*n + n;
  if threadid == numthreads then finish += r;    // add r elements to the last thread
  for i in start..finish do
    if x[i] > lmax[threadid] then lmax[threadid] = x[i];
 }

for threadid in 1..numthreads do     // no need for a parallel loop here
  if lmax[threadid] > gmax then gmax = lmax[threadid];
$ chpl --fast gmax.chpl -o gmax
$ sed -i -e 's|coforall --numthreads=5|gmax|' shared.sh
$ sbatch shared.sh
$ cat solution.out
the maximum value in x is: 1.0

We use coforall to spawn threads that work concurrently in a fraction of the array. The trick here is to determine, based on the threadid, the initial and final indices that the thread will use. Each thread obtains the maximum in its fraction of the array, and finally, after the coforall is done, the main thread obtains the maximum of the array from the maximums of all threads.

Solution to “Task.3”

var x = 0;
writeln('This is the main thread, my value of x is ', x);

sync {
  begin {
     var x = 5;
     writeln('this is thread 1, my value of x is ', x);
  }
  begin writeln('this is thread 2, my value of x is ', x);
}

writeln('this message will not appear until all threads are done...');

Solution to “Task.4”

The code most likely will lock (although sometimes it might not), as we’ll be hitting a race condition. Refer to the diagram for explanation.

Solution to “Task.5”

You need two separate locks, and for simplicity increment them both:

var lock1, lock2: atomic int;
const numthreads = 5;
lock1.write(0);   // the main thread set lock to zero
lock2.write(0);   // the main thread set lock to zero
coforall id in 1..numthreads {
  writeln('greetings form thread ', id, '... I am waiting for all threads to say hello');
  lock1.add(1);              // thread id says hello and atomically adds 1 to lock
  lock1.waitFor(numthreads);   // then it waits for lock=numthreads (which will happen when all threads say hello)
  writeln('thread ', id, ' is done ...');
  lock2.add(1);
  lock2.waitFor(numthreads);
  writeln('thread ', id, ' is really done ...');
}

Part 3: data parallelism

Solution to “Data.1”

Change the line

for i in 1..n {

to

forall i in 1..n with (+ reduce total) {

Solution to “Data.2”

Run the code with

$ ./test -nl 4 --n=3
$ ./test -nl 4 --n=20

For n=3 we get fewer threads (7 in my case), for n=20 we still get 12 threads (the maximum available number of cores inside our job).

Solution to “Data.3”

Something along the lines of m = here.id:string + '-' + m.locale.id:string; should work.

In most cases m.locale.id should be the same as here.id (computation follows data distribution).

Solution to “Data.4”

It should be forall (i,j) in largerMesh[1..rows,1..cols] do (run on multiple locales in parallel) instead of forall (i,j) in mesh do (run in parallel on locale 0 only).

Another possible solution is forall (i,j) in Tnew.domain[1..rows,1..cols] do (run on multiple locales in parallel).

Also, we cannot have forall (i,j) in largerMesh do (run in parallel on multiple locales) as this would overwrite the boundaries.

Solution to “Data.5”

Just before temperature output (if count%nout == 0), insert the following:

  var total = 0.0;
  forall (i,j) in largerMesh[1..rows,1..cols] with (+ reduce total) do
    total += T[i,j];

and add total to the temperature output. It is decreasing as energy is leaving the system:

$ chpl --fast parallel3.chpl -o parallel3
$ ./parallel3 -nl 1 --rows=30 --cols=30 --niter=2000   # run this from inside distributed.sh
Temperature at iteration 0: 25.0
Temperature at iteration 20: 3.49566   21496.5
Temperature at iteration 40: 2.96535   21052.6
...
Temperature at iteration 1100: 2.5809   18609.5
Temperature at iteration 1120: 2.58087   18608.6
Temperature at iteration 1140: 2.58085   18607.7
Final temperature at the desired position [1,30] after 1148 iterations is: 2.58084
The largest temperature difference was 9.9534e-05
The simulation took 0.114942 seconds

Solution to “Data.6”

Here is one possible solution examining the locality of the finite-difference stencil:

var message: [largerMesh] string = 'empty';

and in the next line after computing Tnew[i,j] put

    message[i,j] = "%i".format(here.id) + message[i,j].locale.id + message[i-1,j].locale.id +
      message[i+1,j].locale.id + message[i,j-1].locale.id + message[i,j+1].locale.id + '  ';

and before the end of the while loop

  writeln(message);
  assert(1>2);

Then run it

$ chpl --fast parallel3.chpl -o parallel3
$ ./parallel3 -nl 4 --rows=8 --cols=8   # run this from inside distributed.sh