AUTH's THMMY "Parallel and distributed systems" course assignments.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

165 lines
4.9 KiB

  1. /*!
  2. * \file
  3. * \brief Main application file for PDS HW2 (MPI)
  4. *
  5. * \author
  6. * Christos Choutouridis AEM:8997
  7. * <cchoutou@ece.auth.gr>
  8. */
  9. #include <exception>
  10. #include <iostream>
  11. #include <algorithm>
  12. #include "utils.hpp"
  13. #include "config.h"
  14. #include "distsort.hpp"
  15. // Global session data
  16. session_t session;
  17. MPI_t<> mpi;
  18. distBuffer_t Data;
  19. Log logger;
  20. Timing timer;
  21. /*!
  22. * A small command line argument parser
  23. * \return The status of the operation
  24. */
  25. bool get_options(int argc, char* argv[]){
  26. bool status =true;
  27. // iterate over the passed arguments
  28. for (int i=1 ; i<argc ; ++i) {
  29. std::string arg(argv[i]); // get current argument
  30. if (arg == "-q" || arg == "--array-size") {
  31. if (i+1 < argc) {
  32. session.arraySize = 1 << atoi(argv[++i]);
  33. }
  34. else {
  35. status = false;
  36. }
  37. }
  38. else if (arg == "--ndebug") {
  39. session.ndebug = true;
  40. }
  41. else if (arg == "-t" || arg == "--timing") {
  42. session.timing = true;
  43. }
  44. else if (arg == "-v" || arg == "--verbose") {
  45. session.verbose = true;
  46. }
  47. else if (arg == "-h" || arg == "--help") {
  48. std::cout << "distbitonic/distbubbletonic - A distributed bitonic sort\n\n";
  49. std::cout << "distbitonic -q <> [--ndebug] [-v]\n";
  50. std::cout << "distbitonic -h\n";
  51. std::cout << "distbubbletonic -q <> [--ndebug] [-v]\n";
  52. std::cout << "distbubbletonic -h\n";
  53. std::cout << '\n';
  54. std::cout << "Options:\n\n";
  55. std::cout << " -q | --array-size <size>\n";
  56. std::cout << " Selects the array size according to size = 2^q\n\n";
  57. std::cout << " --ndebug\n";
  58. std::cout << " Skip debug breakpoint when on debug build.\n\n";
  59. std::cout << " -t | --timing\n";
  60. std::cout << " Request timing measurements output to stdout.\n\n";
  61. std::cout << " -v | --verbose\n";
  62. std::cout << " Request a more verbose output to stdout.\n\n";
  63. std::cout << " -h | --help\n";
  64. std::cout << " Prints this and exit.\n\n";
  65. std::cout << "Examples:\n\n";
  66. std::cout << " mpirun -np 4 distbitonic -q 24\n";
  67. std::cout << " Runs distbitonic in 4 MPI processes with 2^24 array points each\n\n";
  68. std::cout << " mpirun -np 16 distbubbletonic -q 20\n";
  69. std::cout << " Runs distbubbletonic in 16 MPI processes with 2^20 array points each\n\n";
  70. exit(0);
  71. }
  72. else { // parse error
  73. std::cout << "Invocation error. Try -h for details.\n";
  74. status = false;
  75. }
  76. }
  77. return status;
  78. }
  79. #if !defined TESTING
  80. int main(int argc, char* argv[]) try {
  81. // Initialize MPI environment
  82. mpi.init(&argc, &argv);
  83. // try to read command line (after MPI parsing)
  84. if (!get_options(argc, argv))
  85. exit(1);
  86. logger << "MPI environment initialized." <<
  87. " Rank: " << mpi.rank() <<
  88. " Size: " << mpi.size() <<
  89. logger.endl;
  90. #if defined DEBUG
  91. #if defined TESTING
  92. /*
  93. * In case of a debug build we will wait here until sleep_wait
  94. * will reset via debugger. In order to do that the user must attach
  95. * debugger to all processes. For example:
  96. * $> mpirun -np 2 ./<program path>
  97. * $> ps aux | grep <program>
  98. * $> gdb <program> <PID1>
  99. * $> gdb <program> <PID2>
  100. */
  101. volatile bool sleep_wait = false;
  102. #else
  103. volatile bool sleep_wait = true;
  104. #endif
  105. while (sleep_wait && !session.ndebug)
  106. sleep(1);
  107. #endif
  108. logger << "Initialize local array of " << session.arraySize << " elements" << logger.endl;
  109. std::srand(unsigned(std::time(nullptr)));
  110. Data.resize(session.arraySize);
  111. std::generate(Data.begin(), Data.end(), std::rand);
  112. if (mpi.rank() == 0)
  113. logger << "Starting distributed sorting ... ";
  114. timer.start();
  115. #if CODE_VERSION == BUBBLETONIC
  116. distBubbletonic(Data, mpi.size());
  117. #else
  118. distBitonic (Data, mpi.size());
  119. #endif
  120. timer.stop();
  121. if (mpi.rank() == 0)
  122. logger << " Done." << logger.endl;
  123. std::string timeMsg = "rank " + std::to_string(mpi.rank());
  124. timer.print_dt(timeMsg.c_str());
  125. std::cout << "[Data]: Rank " << mpi.rank() << ": [" << (int)Data.front() << " .. " << (int)Data.back() << "]" << std::endl;
  126. mpi.finalize();
  127. return 0;
  128. }
  129. catch (std::exception& e) {
  130. //we probably pollute the user's screen. Comment `cerr << ...` if you don't like it.
  131. std::cerr << "Error: " << e.what() << '\n';
  132. exit(1);
  133. }
  134. #else
  135. #include <gtest/gtest.h>
  136. #include <exception>
  137. GTEST_API_ int main(int argc, char **argv) try {
  138. testing::InitGoogleTest(&argc, argv);
  139. return RUN_ALL_TESTS();
  140. }
  141. catch (std::exception& e) {
  142. std::cout << "Exception: " << e.what() << '\n';
  143. }
  144. #endif