Rev 318 | Blame | Compare with Previous | Last modification | View Log | RSS feed
/******************************************************************************** proj2-group.c - implement a simple program that will find the number of* occurrances of a random "seed" in a randomly generated* array of n numbers. This uses the group communication* routines of MPI to transfer the data.** Copyright (c) 2006, Ira W. Snyder (devel@irasnyder.com)* All rights reserved.** Permission is hereby granted, free of charge, to any person obtaining a copy* of this software and associated documentation files (the "Software"), to* deal in the Software without restriction, including without limitation the* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or* sell copies of the Software, and to permit persons to whom the Software is* furnished to do so, subject to the following conditions:** The above copyright notice and this permission notice shall be included in* all copies or substantial portions of the Software.** THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS* IN THE SOFTWARE.******************************************************************************/#include <stdio.h>#include <stdlib.h>#include <unistd.h>#include <limits.h>#include <sys/timeb.h>#include <mpi.h>#define ROOT_NODE 0#ifdef SOLARIS/* NOTE: Solaris is broken, and RAND_MAX is NOT the maximum size* number that random() outputs. Here is the correct value. */#define RAND_MAX LONG_MAX#endifint find_seed_occurrances (int *a, int asize, int seed){int i, count=0;for (i=0; i<asize; i++)if (a[i] == seed)count++;return count;}int main (int argc, char *argv[]){int numprocs, myid, i;int n = -1;int c = -1;int size, seed, total, g_total;int *numbers;/* Declare timeb structs */struct timeb time1;struct timeb time2;MPI_Status status;MPI_Init (&argc, &argv);MPI_Comm_size (MPI_COMM_WORLD, &numprocs);MPI_Comm_rank (MPI_COMM_WORLD, &myid);if (myid == ROOT_NODE){/* Parse the command line options */opterr = 0;while ((c = getopt (argc, argv, "n:")) != -1){switch (c){/* Try to get -n option */case 'n':n = atoi (optarg);break;/* Catch bad options */case '?':if (isprint (optopt))fprintf (stderr, "Unknown option '-%c'.\n", optopt);elsefprintf (stderr, "Unknown option character '\\x%x'.\n", optopt);return 1;break;default:abort ();break;}}/* Check if the n value is valid */if (n<0){fprintf (stderr, "Please specify the '-n NUMBER' option\n");return 1;}/* Make sure that we have the ability to calculate the answer */if (n % numprocs != 0){fprintf (stderr, "n must be evenly divisble by numprocs\n");fprintf (stderr, "Please adjust -n and -np adequately\n");return 2;}/* Allocate memory for the random number array */if ((numbers = (int*) malloc (n * sizeof(int))) == NULL){puts ("Out of memory");return 1;}/* Seed the random number generator */srandom (time(NULL));/* Generate random numbers, staying in the range [0,n) */for (i=0; i<n; i++)numbers[i] = (int) (n * (random() / (RAND_MAX + 1.0)));/* Generate the seed value */seed = (int) (n * (random() / (RAND_MAX + 1.0)));/* Calculate slice size */size = n/numprocs;}/* Start the clock */if (myid == ROOT_NODE)ftime (&time1);/* Broadcast size to all nodes */MPI_Bcast (&size, 1, MPI_INT, ROOT_NODE, MPI_COMM_WORLD);if (myid != ROOT_NODE){if ((numbers = (int*) malloc (size * sizeof(int))) == NULL){puts ("Out of memory");return 1;}}/* Scatter the numbers array to all of the children */MPI_Scatter (numbers, size, MPI_INT, numbers, size, MPI_INT, ROOT_NODE, MPI_COMM_WORLD);/* Broadcast x to all of the children */MPI_Bcast (&seed, 1, MPI_INT, ROOT_NODE, MPI_COMM_WORLD);total = find_seed_occurrances (numbers, size, seed);/* Sum everyone's results */MPI_Reduce (&total, &g_total, 1, MPI_INT, MPI_SUM, ROOT_NODE, MPI_COMM_WORLD);/* Stop the clock */if (myid == ROOT_NODE)ftime (&time2);if (myid == ROOT_NODE){/* Print time taken */long time_taken = ((time2.time - time1.time) * 1000) + (time2.millitm - time1.millitm);printf ("Time taken: %d (milliseconds)\n", time_taken);/* Print the final status */printf ("Total occurrance of seed (%d) out of %d numbers: %d\n", seed, n, g_total);}free (numbers);MPI_Finalize ();return 0;}