/*! * \file * \brief Main application file for PDS HW2 (MPI) * * \author * Christos Choutouridis AEM:8997 * <cchoutou@ece.auth.gr> */ #include <exception> #include <iostream> #include <algorithm> #include <random> #include "utils.hpp" #include "config.h" #include "distsort.hpp" // Global session data config_t config; MPI_t<> mpi; distBuffer_t Data; Log logger; Timing Ttotal; /*! * A small command line argument parser * \return The status of the operation */ bool get_options(int argc, char* argv[]){ bool status =true; // iterate over the passed arguments for (int i=1 ; i<argc ; ++i) { std::string arg(argv[i]); // get current argument if (arg == "-q" || arg == "--array-size") { if (i+1 < argc) { config.arraySize = 1 << atoi(argv[++i]); } else { status = false; } } else if (arg == "--pipeline") { if (i+1 < argc) { auto stages = atoi(argv[++i]); if (isPowerOfTwo(stages) && stages <= static_cast<int>(MAX_PIPELINE_SIZE)) config.pipeline = stages; else status = false; } else { status = false; } } else if (arg == "--validation") { config.validation = true; } else if (arg == "--perf") { config.perf = true; } else if (arg == "--ndebug") { config.ndebug = true; } else if (arg == "-v" || arg == "--verbose") { config.verbose = true; } else if (arg == "-h" || arg == "--help") { std::cout << "distbitonic/distbubbletonic - A distributed bitonic sort\n\n"; std::cout << "distbitonic -q <N> [--pipeline N] [--validation] [--ndebug] [-v]\n"; std::cout << "distbitonic -h\n"; std::cout << "distbubbletonic -q <N> [--pipeline N] [--validation] [--ndebug] [-v]\n"; std::cout << "distbubbletonic -h\n"; std::cout << '\n'; std::cout << "Options:\n\n"; std::cout << " -q | --array-size <N>\n"; std::cout << " Selects the array size according to size = 2^N\n\n"; std::cout << " --pipeline <N>\n"; std::cout << " Request a pipeline of <N> stages for exchange-minmax\n"; std::cout << " N must be power of 2 up to " << MAX_PIPELINE_SIZE << "\n\n"; std::cout << " --validation\n"; std::cout << " Request a full validation at the end, performed by process rank 0\n\n"; std::cout << " --perf\n"; std::cout << " Request performance timing measurements to stdout.\n\n"; std::cout << " --ndebug\n"; std::cout << " Skip debug breakpoint when on debug build.\n\n"; std::cout << " -v | --verbose\n"; std::cout << " Request a more verbose output to stdout.\n\n"; std::cout << " -h | --help\n"; std::cout << " Prints this and exit.\n\n"; std::cout << "Examples:\n\n"; std::cout << " mpirun -np 4 distbitonic -q 24\n"; std::cout << " Runs distbitonic in 4 MPI processes with 2^24 array points each\n\n"; std::cout << " mpirun -np 16 distbubbletonic -q 20\n"; std::cout << " Runs distbubbletonic in 16 MPI processes with 2^20 array points each\n\n"; exit(0); } else { // parse error std::cout << "Invocation error. Try -h for details.\n"; status = false; } } return status; } /*! * A simple validator for the entire distributed process * * @tparam ShadowedDataT A Shadowed buffer type with random access iterator. * * @param data [ShadowedDataT] The local to MPI process * @param Processes [mpi_id_t] The total number of MPI processes * @param rank [mpi_id_t] The current process id * * @return [bool] True if all are sorted and in total ascending order */ template<typename ShadowedDataT> bool validator(ShadowedDataT& data, mpi_id_t Processes, mpi_id_t rank) { using value_t = typename ShadowedDataT::value_type; bool ret = true; // Have faith! // Local results value_t lmin = data.front(); value_t lmax = data.back(); value_t lsort = static_cast<value_t>(std::is_sorted(data.begin(), data.end())); // Gather min/max/sort to rank 0 std::vector<value_t> mins(Processes); std::vector<value_t> maxes(Processes); std::vector<value_t> sorts(Processes); MPI_Datatype datatype = MPI_TypeMapper<value_t>::getType(); MPI_Gather(&lmin, 1, datatype, mins.data(), 1, datatype, 0, MPI_COMM_WORLD); MPI_Gather(&lmax, 1, datatype, maxes.data(), 1, datatype, 0, MPI_COMM_WORLD); MPI_Gather(&lsort, 1, datatype, sorts.data(), 1, datatype, 0, MPI_COMM_WORLD); // Check all results if (rank == 0) { for (mpi_id_t r = 1; r < Processes; ++r) { if (sorts[r] == 0) ret = false; if (maxes[r - 1] > mins[r]) ret = false; } } return ret; } #if !defined TESTING /*! * @return Returns 0, but.... we may throw or exit(1) */ int main(int argc, char* argv[]) try { // Initialize MPI environment mpi.init(&argc, &argv); // try to read command line (after MPI parsing) if (!get_options(argc, argv)) exit(1); logger << "MPI environment initialized." << " Rank: " << mpi.rank() << " Size: " << mpi.size() << logger.endl; #if defined DEBUG #if defined TESTING /* * In case of a debug build we will wait here until sleep_wait * will reset via debugger. In order to do that the user must attach * debugger to all processes. For example: * $> mpirun -np 2 ./<program path> * $> ps aux | grep <program> * $> gdb <program> <PID1> * $> gdb <program> <PID2> */ volatile bool sleep_wait = false; #else volatile bool sleep_wait = true; #endif while (sleep_wait && !config.ndebug) sleep(1); #endif // Initialize local data logger << "Initialize local array of " << config.arraySize << " elements" << logger.endl; std::random_device rd; // Mersenne seeded from hw if possible. range: [type_min, type_max] std::mt19937 gen(rd()); std::uniform_int_distribution<distValue_t > dis( std::numeric_limits<distValue_t>::min(), std::numeric_limits<distValue_t>::max() ); // Fill vector Data.resize(config.arraySize); std::generate(Data.begin(), Data.end(), [&]() { return dis(gen); }); // Run distributed sort if (mpi.rank() == 0) logger << "Starting distributed sorting ... "; Ttotal.start(); #if CODE_VERSION == BUBBLETONIC distBubbletonic(Data, mpi.size(), mpi.rank()); #else distBitonic (Data, mpi.size(), mpi.rank()); #endif Ttotal.stop(); if (mpi.rank() == 0) logger << " Done." << logger.endl; // Print-outs and validation if (config.perf) { Ttotal.print_duration("Total ", mpi.rank()); TfullSort.print_duration("Full-Sort ", mpi.rank()); Texchange.print_duration("Exchange ", mpi.rank()); Tminmax.print_duration("Min-Max ", mpi.rank()); TelbowSort.print_duration("Elbow-Sort", mpi.rank()); } if (config.validation) { // If requested, we have the chance to fail! if (mpi.rank() == 0) std::cout << "Results validation ..."; bool val = validator(Data, mpi.size(), mpi.rank()); if (mpi.rank() == 0) std::cout << ((val) ? "\x1B[32m [PASS] \x1B[0m\n" : " \x1B[32m [FAIL] \x1B[0m\n"); } mpi.finalize(); return 0; } catch (std::exception& e) { //we probably pollute the user's screen. Comment `cerr << ...` if you don't like it. std::cerr << "Error: " << e.what() << '\n'; exit(1); } #else #include <gtest/gtest.h> #include <exception> /*! * The testing version of our program */ GTEST_API_ int main(int argc, char **argv) try { testing::InitGoogleTest(&argc, argv); return RUN_ALL_TESTS(); } catch (std::exception& e) { std::cout << "Exception: " << e.what() << '\n'; } #endif