diff --git a/src/vt-tv/api/info.h b/src/vt-tv/api/info.h index 09da27954..7721270b5 100644 --- a/src/vt-tv/api/info.h +++ b/src/vt-tv/api/info.h @@ -65,13 +65,11 @@ namespace vt::tv { * */ struct Info { - Info( std::unordered_map in_object_info, - std::unordered_map in_ranks - ) : object_info_(std::move(in_object_info)), - ranks_(std::move(in_ranks)) - { } + std::unordered_map in_ranks) + : object_info_(std::move(in_object_info)), + ranks_(std::move(in_ranks)) { } Info() = default; @@ -81,9 +79,8 @@ struct Info { * \param[in] object_info object information to merge with existing data * \param[in] r the rank work */ - void addInfo( - std::unordered_map object_info, Rank r - ) { + void + addInfo(std::unordered_map object_info, Rank r) { for (auto x : object_info) { object_info_.try_emplace(x.first, std::move(x.second)); } @@ -139,9 +136,11 @@ struct Info { uint64_t n_phases = 0; if (this->ranks_.size() > 0) { n_phases = this->ranks_.at(0).getNumPhases(); - for (NodeType rank_id = 1; rank_id < static_cast(this->ranks_.size()); rank_id++) { + for (NodeType rank_id = 1; + rank_id < static_cast(this->ranks_.size()); rank_id++) { if (ranks_.at(rank_id).getNumPhases() != n_phases) { - throw std::runtime_error("Number of phases must be consistent across ranks"); + throw std::runtime_error( + "Number of phases must be consistent across ranks"); } } } @@ -156,26 +155,28 @@ struct Info { /* ----------------------------------- Getters ----------------------------------- */ - /** + /** * \brief Converts a QOI from QOIVariantTypes to double */ - double convertQOIVariantTypeToDouble_(const QOIVariantTypes &variant) const { + double convertQOIVariantTypeToDouble_(const QOIVariantTypes& variant) const { if (std::holds_alternative(variant)) { return static_cast(std::get(variant)); } else if (std::holds_alternative(variant)) { return std::get(variant); } else if (std::holds_alternative(variant)) { - throw std::runtime_error("QOI type must be numerical (received std::string)."); + throw std::runtime_error( + "QOI type must be numerical (received std::string)."); } else { - throw std::runtime_error("Invalid QOI type received (must be numerical)."); + throw std::runtime_error( + "Invalid QOI type received (must be numerical)."); } } - - /** + /** * \brief Returns a getter to a specified rank QOI */ - std::function getRankQOIGetter(const std::string &rank_qoi) const { + std::function + getRankQOIGetter(const std::string& rank_qoi) const { std::function qoi_getter; if (rank_qoi == "load") { qoi_getter = [&](Rank rank, PhaseType phase) { @@ -183,7 +184,8 @@ struct Info { }; } else if (rank_qoi == "received_volume") { qoi_getter = [&](Rank rank, PhaseType phase) { - return convertQOIVariantTypeToDouble_(getRankReceivedVolume(rank, phase)); + return convertQOIVariantTypeToDouble_( + getRankReceivedVolume(rank, phase)); }; } else if (rank_qoi == "sent_volume") { qoi_getter = [&](Rank rank, PhaseType phase) { @@ -195,11 +197,13 @@ struct Info { }; } else if (rank_qoi == "number_of_migratable_objects") { qoi_getter = [&](Rank rank, PhaseType phase) { - return convertQOIVariantTypeToDouble_(getRankNumMigratableObjects(rank, phase)); + return convertQOIVariantTypeToDouble_( + getRankNumMigratableObjects(rank, phase)); }; } else if (rank_qoi == "migratable_load") { qoi_getter = [&](Rank rank, PhaseType phase) { - return convertQOIVariantTypeToDouble_(getRankMigratableLoad(rank, phase)); + return convertQOIVariantTypeToDouble_( + getRankMigratableLoad(rank, phase)); }; } else if (rank_qoi == "sentinel_load") { qoi_getter = [&](Rank rank, PhaseType phase) { @@ -220,10 +224,11 @@ struct Info { return qoi_getter; } - /** + /** * \brief Returns a getter to a specified object QOI */ - std::function getObjectQoiGetter(const std::string &object_qoi) const { + std::function + getObjectQoiGetter(const std::string& object_qoi) const { std::function qoi_getter; if (object_qoi == "load") { qoi_getter = [&](ObjectWork obj) { @@ -252,7 +257,8 @@ struct Info { } else { // Look in attributes and user_defined (will throw an error if QOI doesn't exist) qoi_getter = [&](ObjectWork obj) { - return convertQOIVariantTypeToDouble_(getObjectAttributeOrUserDefined(obj, object_qoi)); + return convertQOIVariantTypeToDouble_( + getObjectAttributeOrUserDefined(obj, object_qoi)); }; } return qoi_getter; @@ -260,23 +266,25 @@ struct Info { /* --------------------------------- Rank Getters --------------------------------- */ - /** + /** * \brief Get QOI of a given rank at a given phase * * \return a map of QOI per rank */ - double getRankQOIAtPhase(ElementIDType rank_id, PhaseType phase, std::string rank_qoi) const { + double getRankQOIAtPhase( + ElementIDType rank_id, PhaseType phase, std::string rank_qoi) const { auto qoi_getter = getRankQOIGetter(rank_qoi); auto const& rank = this->ranks_.at(rank_id); return qoi_getter(rank, phase); } - /** + /** * \brief Get QOI of a given rank across all phases * * \return a map of QOI per rank */ - std::unordered_map getAllQOIAtRank(ElementIDType rank_id, std::string rank_qoi) const { + std::unordered_map + getAllQOIAtRank(ElementIDType rank_id, std::string rank_qoi) const { std::unordered_map rank_qois; auto qoi_getter = getRankQOIGetter(rank_qoi); auto const& rank = this->ranks_.at(rank_id); @@ -293,7 +301,8 @@ struct Info { * * \return a map of QOI per rank */ - std::unordered_map getAllRankQOIAtPhase(PhaseType phase, std::string rank_qoi) const { + std::unordered_map + getAllRankQOIAtPhase(PhaseType phase, std::string rank_qoi) const { std::unordered_map rank_qois; auto qoi_getter = getRankQOIGetter(rank_qoi); for (auto const& [rank_id, rank] : this->ranks_) { @@ -310,7 +319,8 @@ struct Info { * * \return the object QOI */ - double getObjectQoi(ElementIDType obj_id, PhaseType phase, std::string obj_qoi) const { + double getObjectQoi( + ElementIDType obj_id, PhaseType phase, std::string obj_qoi) const { auto qoi_getter = getObjectQoiGetter(obj_qoi); auto const& objects = this->getPhaseObjects(phase); auto const& obj = objects.at(obj_id); @@ -325,7 +335,8 @@ struct Info { * * \return the objects */ - std::unordered_map getRankObjects(ElementIDType rank_id, PhaseType phase) const { + std::unordered_map + getRankObjects(ElementIDType rank_id, PhaseType phase) const { std::unordered_map objects; // Get Rank info for specified rank @@ -338,7 +349,8 @@ struct Info { auto const& phase_work_at_rank = phase_history_at_rank.find(phase); // Get all objects at specified phase - auto const& object_work_at_phase_at_rank = phase_work_at_rank->second.getObjectWork(); + auto const& object_work_at_phase_at_rank = + phase_work_at_rank->second.getObjectWork(); for (auto const& [elm_id, obj_work] : object_work_at_phase_at_rank) { objects.insert(std::make_pair(elm_id, obj_work)); @@ -354,7 +366,8 @@ struct Info { * * \return the objects */ - std::unordered_map getPhaseObjects(PhaseType phase) const { + std::unordered_map + getPhaseObjects(PhaseType phase) const { // fmt::print("Phase: {}\n", phase); // Map of objects at given phase @@ -372,7 +385,8 @@ struct Info { // Get phase work at specified phase auto const& phase_work = phase_history.find(phase); if (phase_work == phase_history.end()) { - auto ex = "info::getPhaseObjects: Phase " + std::to_string(phase) + " doesn't exist for rank " + std::to_string(rank); + auto ex = "info::getPhaseObjects: Phase " + std::to_string(phase) + + " doesn't exist for rank " + std::to_string(rank); throw std::runtime_error(ex); } @@ -404,17 +418,20 @@ struct Info { auto n_phases = this->getNumPhases(); if (selected_phase_ != std::numeric_limits::max()) { - auto const& objects = this->getPhaseObjects(selected_phase_); - for (auto const& [obj_id, obj_work] : objects) { - auto obj_max_v = obj_work.getMaxVolume(); - if (obj_max_v > ov_max) ov_max = obj_max_v; - } + auto const& objects = this->getPhaseObjects(selected_phase_); + for (auto const& [obj_id, obj_work] : objects) { + auto obj_max_v = obj_work.getMaxVolume(); + if (obj_max_v > ov_max) + ov_max = obj_max_v; + } } else { - for (PhaseType phase = 0; phase < n_phases; phase++) {{ + for (PhaseType phase = 0; phase < n_phases; phase++) { + { auto const& objects = this->getPhaseObjects(phase); for (auto const& [obj_id, obj_work] : objects) { auto obj_max_v = obj_work.getMaxVolume(); - if (obj_max_v > ov_max) ov_max = obj_max_v; + if (obj_max_v > ov_max) + ov_max = obj_max_v; } } } @@ -436,14 +453,16 @@ struct Info { auto const& objects = this->getPhaseObjects(selected_phase_); for (auto const& [obj_id, obj_work] : objects) { auto obj_load = obj_work.getLoad(); - if (obj_load > ol_max) ol_max = obj_load; + if (obj_load > ol_max) + ol_max = obj_load; } } else { for (PhaseType phase = 0; phase < n_phases; phase++) { auto const& objects = this->getPhaseObjects(phase); for (auto const& [obj_id, obj_work] : objects) { auto obj_load = obj_work.getLoad(); - if (obj_load > ol_max) ol_max = obj_load; + if (obj_load > ol_max) + ol_max = obj_load; } } } @@ -457,7 +476,8 @@ struct Info { * * \return the objects */ - std::unordered_map createPhaseObjectsMapping(PhaseType phase) { + std::unordered_map + createPhaseObjectsMapping(PhaseType phase) { // fmt::print("Phase: {}\n", phase); // Map of objects at given phase @@ -492,7 +512,6 @@ struct Info { * \return the objects */ std::set getAllObjectIDs() const { - // Map of objects at given phase std::set objects; @@ -538,81 +557,92 @@ struct Info { // Vector of tuples of communications to add: {side_to_be_modified, id1, id2, bytes} for an id1 -> id2 communication (1 sends to 2, 2 receives from 1) // if type is "sender", communication has to be added to sent communications for object id1 // if type is "recipient", communication has to be added to received communications for object id2 - std::vector> communications_to_add; + std::vector> + communications_to_add; auto phase_objects = createPhaseObjectsMapping(phase); // Checking all communications for object A in all objects of all ranks at given phase: A <- ... and A -> ... for (auto& [A_id, object_work] : phase_objects) { -// fmt::print("- Object ID: {}\n", A_id); + // fmt::print("- Object ID: {}\n", A_id); auto sent = object_work.getSent(); -// fmt::print(" Has {} sent communications", sent.size()); + // fmt::print(" Has {} sent communications", sent.size()); auto received = object_work.getReceived(); -// fmt::print(" and {} received communications.\n", received.size()); + // fmt::print(" and {} received communications.\n", received.size()); // Going through A -> ... communications -// fmt::print(" Checking sent communications:\n"); + // fmt::print(" Checking sent communications:\n"); for (auto& [B_id, bytes] : sent) { -// fmt::print(" Communication sent to object {} of {} bytes:\n", B_id, bytes); + // fmt::print(" Communication sent to object {} of {} bytes:\n", B_id, bytes); // check if B exists for the A -> B communication if (phase_objects.find(B_id) != phase_objects.end()) { -// fmt::print(" Found recipient object {} when searching for communication sent by object {} of {} bytes.\n", B_id, A_id, bytes); + // fmt::print(" Found recipient object {} when searching for communication sent by object {} of {} bytes.\n", B_id, A_id, bytes); auto to_object_work = phase_objects.at(B_id); auto target_received = to_object_work.getReceived(); -// fmt::print( "Object {} has {} received communications.\n", B_id, target_received.size()); + // fmt::print( "Object {} has {} received communications.\n", B_id, target_received.size()); // Check if B has symmetric B <- A received communication if (target_received.find(A_id) != target_received.end()) { -// fmt::print( " Object {} already has received communication from object {}.\n", B_id, A_id); + // fmt::print( " Object {} already has received communication from object {}.\n", B_id, A_id); } else { -// fmt::print( " Object {} doesn't have received communication from object {}. Pushing to list of communications to add.\n", B_id, A_id); - communications_to_add.push_back(std::make_tuple("recipient", A_id, B_id, bytes)); + // fmt::print( " Object {} doesn't have received communication from object {}. Pushing to list of communications to add.\n", B_id, A_id); + communications_to_add.push_back( + std::make_tuple("recipient", A_id, B_id, bytes)); } } else { - fmt::print(" /!\\ Didn't find recipient object {} when searching for communication sent by object {} of {} bytes.\n", B_id, A_id, bytes); + fmt::print( + " /!\\ Didn't find recipient object {} when searching for " + "communication sent by object {} of {} bytes.\n", + B_id, A_id, bytes); } } // Going through A <- ... communications -// fmt::print(" Checking received communications:\n"); - for (auto& [B_id, bytes] : received) { // Going through A <- ... communications -// fmt::print(" Communication received from object {} of {} bytes:\n", B_id, bytes); + // fmt::print(" Checking received communications:\n"); + for (auto& [B_id, bytes] : + received) { // Going through A <- ... communications + // fmt::print(" Communication received from object {} of {} bytes:\n", B_id, bytes); // check if B exists for the A <- B communication if (phase_objects.find(B_id) != phase_objects.end()) { -// fmt::print(" Found sender object {} when searching for communication received by object {} of {} bytes.\n", B_id, A_id, bytes); + // fmt::print(" Found sender object {} when searching for communication received by object {} of {} bytes.\n", B_id, A_id, bytes); auto from_object_work = phase_objects.at(B_id); auto target_sent = from_object_work.getSent(); -// fmt::print( "Object {} has {} sent communications.\n", B_id, target_sent.size()); + // fmt::print( "Object {} has {} sent communications.\n", B_id, target_sent.size()); // Check if B has symmetric B -> A received communication if (target_sent.find(A_id) != target_sent.end()) { -// fmt::print( " Object {} already has sent communication to object {}.\n", B_id, A_id); + // fmt::print( " Object {} already has sent communication to object {}.\n", B_id, A_id); } else { -// fmt::print( " Object {} doesn't have sent communication to object {}. Pushing to list of communications to add.\n", B_id, A_id); - communications_to_add.push_back(std::make_tuple("sender", B_id, A_id, bytes)); + // fmt::print( " Object {} doesn't have sent communication to object {}. Pushing to list of communications to add.\n", B_id, A_id); + communications_to_add.push_back( + std::make_tuple("sender", B_id, A_id, bytes)); } } else { -// fmt::print(" /!\\ Didn't find sender object {} when searching for communication received by object {} of {} bytes.\n", B_id, A_id, bytes); + // fmt::print(" /!\\ Didn't find sender object {} when searching for communication received by object {} of {} bytes.\n", B_id, A_id, bytes); } } } // loop through ranks and add communications // fmt::print("Updating communications for phase {}.\n", phase); - for (auto &[rank_id, rank]: ranks_) { + for (auto& [rank_id, rank] : ranks_) { // fmt::print(" Checking objects in rank {}.\n", rank_id); - auto &phaseWork = rank.getPhaseWork(); - auto &phaseWorkAtPhase = phaseWork.at(phase); - auto &objects = phaseWorkAtPhase.getObjectWork(); - for (auto &[obj_id, obj_work]: objects) { + auto& phaseWork = rank.getPhaseWork(); + auto& phaseWorkAtPhase = phaseWork.at(phase); + auto& objects = phaseWorkAtPhase.getObjectWork(); + for (auto& [obj_id, obj_work] : objects) { // fmt::print(" Checking if object {} needs to be updated.\n", obj_id); // fmt::print(" Communications to update:\n"); uint64_t i = 0; - for (auto &[object_to_update, sender_id, recipient_id, bytes]: communications_to_add) { + for (auto& [object_to_update, sender_id, recipient_id, bytes] : + communications_to_add) { // fmt::print(" {} needs to be updated in {} -> {} communication of {} bytes.\n", object_to_update, - // sender_id, recipient_id, bytes); + // sender_id, recipient_id, bytes); if (object_to_update == "sender" && sender_id == obj_id) { // fmt::print(" Sender to be updated is object on this rank. Updating.\n"); - rank.addObjectSentCommunicationAtPhase(phase, obj_id, recipient_id, bytes); + rank.addObjectSentCommunicationAtPhase( + phase, obj_id, recipient_id, bytes); communications_to_add.erase(communications_to_add.begin() + i); - } else if (object_to_update == "recipient" && recipient_id == obj_id) { + } else if ( + object_to_update == "recipient" && recipient_id == obj_id) { // fmt::print(" Recipient to be updated is object on this rank. Updating.\n"); - rank.addObjectReceivedCommunicationAtPhase(phase, obj_id, sender_id, bytes); + rank.addObjectReceivedCommunicationAtPhase( + phase, obj_id, sender_id, bytes); communications_to_add.erase(communications_to_add.begin() + i); } if (communications_to_add.empty()) { @@ -637,7 +667,8 @@ struct Info { for (uint64_t rank = 0; rank < this->ranks_.size(); rank++) { auto rank_max_load = this->getRank(rank).getLoad(phase); - if (rank_max_load > max_load) max_load = rank_max_load; + if (rank_max_load > max_load) + max_load = rank_max_load; load_sum += this->getRank(rank).getLoad(phase); } double load_avg = load_sum / this->ranks_.size(); @@ -658,7 +689,7 @@ struct Info { * \return the id */ QOIVariantTypes getObjectID(ObjectWork object) const { - return static_cast(object.getID()); + return static_cast(object.getID()); } /** @@ -669,9 +700,9 @@ struct Info { * \return the rank id */ QOIVariantTypes getObjectRankID(ObjectWork object) const { - auto obj_id = object.getID(); - auto obj_info = object_info_.at(obj_id); - return obj_info.getHome(); + auto obj_id = object.getID(); + auto obj_info = object_info_.at(obj_id); + return obj_info.getHome(); } /** @@ -682,7 +713,7 @@ struct Info { * \return the load */ QOIVariantTypes getObjectLoad(ObjectWork object) const { - return object.getLoad(); + return object.getLoad(); } /** @@ -696,7 +727,7 @@ struct Info { return object.getReceivedVolume(); } - /** + /** * \brief Get the sent volume of an object at a given phase * * \param[in] object the current object @@ -707,7 +738,7 @@ struct Info { return object.getSentVolume(); } - /** + /** * \brief Get the max volume of an object at a given phase * * \param[in] object the current object @@ -725,7 +756,8 @@ struct Info { * * \return the requested attribute or user_defined QOI */ - QOIVariantTypes getObjectAttributeOrUserDefined(ObjectWork object, std::string object_qoi) const { + QOIVariantTypes getObjectAttributeOrUserDefined( + ObjectWork object, std::string object_qoi) const { auto obj_attributes = object.getAttributes(); if (obj_attributes.count(object_qoi) > 0) { return obj_attributes.at(object_qoi); @@ -742,7 +774,7 @@ struct Info { /* -------------------- Rank QOI getters -------------------- */ - /** + /** * \brief Get id of a given rank * * \param[in] rank the rank @@ -750,9 +782,11 @@ struct Info { * * \return the rank id */ - QOIVariantTypes getRankID(Rank rank) const { return static_cast(rank.getRankID()); } + QOIVariantTypes getRankID(Rank rank) const { + return static_cast(rank.getRankID()); + } - /** + /** * \brief Get load of a given rank * * \param[in] rank the rank @@ -760,7 +794,9 @@ struct Info { * * \return the rank load */ - QOIVariantTypes getRankLoad(Rank rank, PhaseType phase) const { return rank.getLoad(phase); } + QOIVariantTypes getRankLoad(Rank rank, PhaseType phase) const { + return rank.getLoad(phase); + } /** * \brief Get the received volume of a rank at a given phase @@ -771,7 +807,6 @@ struct Info { * \return the received volume */ QOIVariantTypes getRankReceivedVolume(Rank rank, PhaseType phase) const { - auto received_volume = 0.; auto const& phase_objects = rank.getPhaseWork().at(phase).getObjectWork(); for (auto const& [obj_id, obj_work] : phase_objects) { @@ -818,7 +853,8 @@ struct Info { * * \return the number of migratable objects */ - QOIVariantTypes getRankNumMigratableObjects(Rank rank, PhaseType phase) const { + QOIVariantTypes + getRankNumMigratableObjects(Rank rank, PhaseType phase) const { auto num_migratable_objects = 0; auto const& phase_objects = rank.getPhaseWork().at(phase).getObjectWork(); for (auto const& [obj_id, _] : phase_objects) { diff --git a/src/vt-tv/api/object_communicator.h b/src/vt-tv/api/object_communicator.h index 93974237d..e67d987bc 100644 --- a/src/vt-tv/api/object_communicator.h +++ b/src/vt-tv/api/object_communicator.h @@ -64,7 +64,6 @@ struct Object; * \brief A class holding received and sent messages for an object. */ struct ObjectCommunicator { - ObjectCommunicator() = default; /** @@ -72,9 +71,7 @@ struct ObjectCommunicator { * * \param[in] id_in the object id */ - explicit ObjectCommunicator(ElementIDType id_in) - : object_id_(id_in) - {} + explicit ObjectCommunicator(ElementIDType id_in) : object_id_(id_in) { } /** * \brief Construct an \c ObjectCommunicator with edges @@ -84,13 +81,11 @@ struct ObjectCommunicator { * \param[in] sent_in send edges */ ObjectCommunicator( - ElementIDType id_in, - std::multimap recv_in, - std::multimap sent_in - ) : object_id_(id_in), + ElementIDType id_in, std::multimap recv_in, + std::multimap sent_in) + : object_id_(id_in), received_(recv_in), - sent_(sent_in) - { } + sent_(sent_in) { } /** * \brief Get the id of object for this communicator @@ -114,7 +109,7 @@ struct ObjectCommunicator { std::vector results; for (auto it = range.first; it != range.second; ++it) { - results.push_back(it->second); + results.push_back(it->second); } return results; @@ -133,7 +128,7 @@ struct ObjectCommunicator { std::vector results; for (auto it = range.first; it != range.second; ++it) { - results.push_back(it->second); + results.push_back(it->second); } return results; @@ -149,8 +144,7 @@ struct ObjectCommunicator { this->received_.insert(std::make_pair(from_id, bytes)); if (from_id == this->object_id_) { fmt::print( - "Object {} receiving communication from myself\n", this->object_id_ - ); + "Object {} receiving communication from myself\n", this->object_id_); } } @@ -164,37 +158,32 @@ struct ObjectCommunicator { this->sent_.insert(std::make_pair(to_id, bytes)); if (to_id == this->object_id_) { fmt::print( - "Object {} sending communication to myself\n", this->object_id_ - ); + "Object {} sending communication to myself\n", this->object_id_); } } - /** + /** * \brief maximum bytes received or sent at this communicator */ double getMaxVolume() const { // Search for the maximum value in received and sent (0. if sets are empty) double max_recv = !this->received_.empty() ? - std::max_element( - this->received_.begin(), - this->received_.end(), - [](const auto& a, const auto& b) { - return a.second < b.second; - })->second : - 0.0; + std::max_element( + this->received_.begin(), this->received_.end(), + [](const auto& a, const auto& b) { return a.second < b.second; }) + ->second : + 0.0; double max_sent = !this->sent_.empty() ? - std::max_element( - this->sent_.begin(), - this->sent_.end(), - [](const auto& a, const auto& b) { - return a.second < b.second; - })->second : - 0.0; + std::max_element( + this->sent_.begin(), this->sent_.end(), + [](const auto& a, const auto& b) { return a.second < b.second; }) + ->second : + 0.0; // Return the max return std::max(max_recv, max_sent); -} + } /** * \brief Get the total received communication volume for this communicator @@ -235,7 +224,7 @@ struct ObjectCommunicator { } private: - ElementIDType object_id_; /**< The object id */ + ElementIDType object_id_; /**< The object id */ std::multimap received_; /**< The received edges */ std::multimap sent_; /**< The sent edges */ }; diff --git a/src/vt-tv/api/object_info.h b/src/vt-tv/api/object_info.h index 6298b91fa..5bc6d3155 100644 --- a/src/vt-tv/api/object_info.h +++ b/src/vt-tv/api/object_info.h @@ -57,7 +57,6 @@ namespace vt::tv { * ranks or phases. */ struct ObjectInfo { - ObjectInfo() = default; /** @@ -69,15 +68,12 @@ struct ObjectInfo { * \param[in] in_index the index for the object */ ObjectInfo( - ElementIDType in_id, - NodeType in_home, - bool in_migratable, - std::vector const& in_index - ) : id_(in_id), + ElementIDType in_id, NodeType in_home, bool in_migratable, + std::vector const& in_index) + : id_(in_id), home_(in_home), migratable_(in_migratable), - index_(in_index) - { } + index_(in_index) { } /** * \brief Get the object's ID diff --git a/src/vt-tv/api/object_work.h b/src/vt-tv/api/object_work.h index 17136e156..c735a4ba5 100644 --- a/src/vt-tv/api/object_work.h +++ b/src/vt-tv/api/object_work.h @@ -60,7 +60,6 @@ namespace vt::tv { * \brief Holds work for an object for a given phase */ struct ObjectWork { - ObjectWork() = default; /** @@ -72,18 +71,16 @@ struct ObjectWork { * \param[in] in_user_defined the user-defined fields in json */ ObjectWork( - ElementIDType in_id, - TimeType in_whole_phase_load, + ElementIDType in_id, TimeType in_whole_phase_load, std::unordered_map in_subphase_loads, std::unordered_map in_user_defined = {}, - std::unordered_map in_attributes = {} - ) : id_(in_id), + std::unordered_map in_attributes = {}) + : id_(in_id), whole_phase_load_(in_whole_phase_load), subphase_loads_(std::move(in_subphase_loads)), user_defined_(std::move(in_user_defined)), communicator_(id_), - attributes_(std::move(in_attributes)) - { } + attributes_(std::move(in_attributes)) { } /** * \brief Get element ID @@ -165,9 +162,7 @@ struct ObjectWork { /** * \brief Get maximum bytes received or sent at this object */ - double getMaxVolume() const { - return communicator_.getMaxVolume(); - } + double getMaxVolume() const { return communicator_.getMaxVolume(); } /** * \brief Get the total received communication volume for this object @@ -183,9 +178,7 @@ struct ObjectWork { * * \return total sent communication volume */ - double getSentVolume() const { - return communicator_.getTotalSentVolume(); - } + double getSentVolume() const { return communicator_.getTotalSentVolume(); } /** * \brief Serializer for data @@ -215,7 +208,6 @@ struct ObjectWork { ObjectCommunicator communicator_; // QOIs to be visualized std::unordered_map attributes_; - }; } /* end namespace vt::tv */ diff --git a/src/vt-tv/api/phase_work.h b/src/vt-tv/api/phase_work.h index 54b823039..d674ee515 100644 --- a/src/vt-tv/api/phase_work.h +++ b/src/vt-tv/api/phase_work.h @@ -57,7 +57,6 @@ namespace vt::tv { * \brief The work for a given phase */ struct PhaseWork { - PhaseWork() = default; /** @@ -68,10 +67,9 @@ struct PhaseWork { */ PhaseWork( PhaseType in_phase, - std::unordered_map in_objects - ) : phase_(in_phase), - objects_(std::move(in_objects)) - { } + std::unordered_map in_objects) + : phase_(in_phase), + objects_(std::move(in_objects)) { } /** * \brief Get the phase ID @@ -105,14 +103,17 @@ struct PhaseWork { * * \return void */ - void setCommunications(ElementIDType o_id, ObjectCommunicator& c) { objects_.at(o_id).setCommunications(c); }; + void setCommunications(ElementIDType o_id, ObjectCommunicator& c) { + objects_.at(o_id).setCommunications(c); + }; /** * \brief add a received communication to an object in this phase * * \return void */ - void addObjectReceivedCommunication(ElementIDType o_id, ElementIDType from_id, double bytes) { + void addObjectReceivedCommunication( + ElementIDType o_id, ElementIDType from_id, double bytes) { objects_.at(o_id).addReceivedCommunications(from_id, bytes); }; @@ -121,7 +122,8 @@ struct PhaseWork { * * \return void */ - void addObjectSentCommunication(ElementIDType o_id, ElementIDType to_id, double bytes) { + void addObjectSentCommunication( + ElementIDType o_id, ElementIDType to_id, double bytes) { objects_.at(o_id).addSentCommunications(to_id, bytes); }; @@ -133,7 +135,8 @@ struct PhaseWork { for (auto const& [obj_id, obj_work] : this->objects_) { auto obj_max_v = obj_work.getMaxVolume(); - if (obj_max_v > ov_max) ov_max = obj_max_v; + if (obj_max_v > ov_max) + ov_max = obj_max_v; } return ov_max; } diff --git a/src/vt-tv/api/rank.h b/src/vt-tv/api/rank.h index 4a5adfc1b..608810c07 100644 --- a/src/vt-tv/api/rank.h +++ b/src/vt-tv/api/rank.h @@ -54,7 +54,6 @@ namespace vt::tv { * \brief All the data for a given \c Rank */ struct Rank { - Rank() = default; /** @@ -64,13 +63,11 @@ struct Rank { * \param[in] in_phase_info all the phase info */ Rank( - NodeType in_rank, - std::unordered_map in_phase_info, - std::unordered_map in_attributes = {} - ) : rank_(in_rank), + NodeType in_rank, std::unordered_map in_phase_info, + std::unordered_map in_attributes = {}) + : rank_(in_rank), phase_info_(std::move(in_phase_info)), - attributes_(std::move(in_attributes)) - { } + attributes_(std::move(in_attributes)) { } /** * \brief Get the rank ID @@ -98,7 +95,9 @@ struct Rank { * * \return the load */ - double getLoad(PhaseType phase) const { return phase_info_.at(phase).getLoad(); } + double getLoad(PhaseType phase) const { + return phase_info_.at(phase).getLoad(); + } /** * \brief Get attribute fields @@ -114,15 +113,20 @@ struct Rank { * * \return the number of objects */ - uint64_t getNumObjects(PhaseType phase) const { return phase_info_.at(phase).getObjectWork().size(); } + uint64_t getNumObjects(PhaseType phase) const { + return phase_info_.at(phase).getObjectWork().size(); + } /** * \brief add a received communication to an object at a given phase * * \return void */ - void addObjectReceivedCommunicationAtPhase(PhaseType phase_id, ElementIDType o_id, ElementIDType from_id, double bytes) { - phase_info_.at(phase_id).addObjectReceivedCommunication(o_id, from_id, bytes); + void addObjectReceivedCommunicationAtPhase( + PhaseType phase_id, ElementIDType o_id, ElementIDType from_id, + double bytes) { + phase_info_.at(phase_id).addObjectReceivedCommunication( + o_id, from_id, bytes); }; /** @@ -130,7 +134,8 @@ struct Rank { * * \return void */ - void addObjectSentCommunicationAtPhase(PhaseType phase_id, ElementIDType o_id, ElementIDType to_id, double bytes) { + void addObjectSentCommunicationAtPhase( + PhaseType phase_id, ElementIDType o_id, ElementIDType to_id, double bytes) { phase_info_.at(phase_id).addObjectSentCommunication(o_id, to_id, bytes); }; diff --git a/src/vt-tv/render/render.cc b/src/vt-tv/render/render.cc index 3e8f0dadd..da06ccdd8 100644 --- a/src/vt-tv/render/render.cc +++ b/src/vt-tv/render/render.cc @@ -52,29 +52,31 @@ namespace vt { namespace tv { Render::Render(Info in_info) -: info_(in_info) // std:move ? -, n_ranks_(in_info.getNumRanks()) -, n_phases_(in_info.getNumPhases()) -{ + : info_(in_info) // std:move ? + , + n_ranks_(in_info.getNumRanks()), + n_phases_(in_info.getNumPhases()) { // If selected_phase is not provided, use all phases selected_phase_ = std::numeric_limits::max(); // Generically set rank grid dimensions according to the total number of ranks grid_size_[2] = 1; // we assume 2D representation - uint64_t sqrt_n_ranks = static_cast(std::sqrt(static_cast(n_ranks_))); - if(sqrt_n_ranks * sqrt_n_ranks == n_ranks_) { + uint64_t sqrt_n_ranks = + static_cast(std::sqrt(static_cast(n_ranks_))); + if (sqrt_n_ranks * sqrt_n_ranks == n_ranks_) { // n_ranks is a perfect square grid_size_[0] = sqrt_n_ranks; grid_size_[1] = sqrt_n_ranks; } else { // n_ranks is not a perfect square - grid_size_[0] = sqrt_n_ranks; // floor + grid_size_[0] = sqrt_n_ranks; // floor grid_size_[1] = sqrt_n_ranks + 1; // ceil } - for (uint64_t d = 0 ; d < 3 ; d++) { - if(grid_size_[d] > 1) rank_dims_.insert(d); + for (uint64_t d = 0; d < 3; d++) { + if (grid_size_[d] > 1) + rank_dims_.insert(d); } max_o_per_dim_ = 0; @@ -97,8 +99,9 @@ Render::Render(Info in_info) std::array jitterDims; for (uint64_t d = 0; d < 3; d++) { if (auto f = this->rank_dims_.find(d); f != this->rank_dims_.end()) { - jitterDims[d] = ((double)std::rand()/RAND_MAX - 0.5) * object_jitter_; - } else jitterDims[d] = 0; + jitterDims[d] = ((double)std::rand() / RAND_MAX - 0.5) * object_jitter_; + } else + jitterDims[d] = 0; } jitter_dims_.insert(std::make_pair(objectID, jitterDims)); } @@ -110,39 +113,32 @@ Render::Render(Info in_info) }; Render::Render( - std::array in_qoi_request, - bool in_continuous_object_qoi, - Info& in_info, - std::array in_grid_size, - double in_object_jitter, - std::string in_output_dir, - std::string in_output_file_stem, - double in_resolution, - bool in_save_meshes, - bool in_save_pngs, - PhaseType in_selected_phase -) -: rank_qoi_(in_qoi_request[0]) -, object_qoi_(in_qoi_request[2]) -, continuous_object_qoi_(in_continuous_object_qoi) -, info_(in_info) -, n_ranks_(in_info.getNumRanks()) -, n_phases_(in_info.getNumPhases()) -, grid_size_(in_grid_size) -, object_jitter_(in_object_jitter) -, output_dir_(in_output_dir) -, output_file_stem_(in_output_file_stem) -, grid_resolution_(in_resolution) -, save_meshes_(in_save_meshes) -, save_pngs_(in_save_pngs) -, selected_phase_(in_selected_phase) -{ + std::array in_qoi_request, bool in_continuous_object_qoi, + Info& in_info, std::array in_grid_size, double in_object_jitter, + std::string in_output_dir, std::string in_output_file_stem, + double in_resolution, bool in_save_meshes, bool in_save_pngs, + PhaseType in_selected_phase) + : rank_qoi_(in_qoi_request[0]), + object_qoi_(in_qoi_request[2]), + continuous_object_qoi_(in_continuous_object_qoi), + info_(in_info), + n_ranks_(in_info.getNumRanks()), + n_phases_(in_info.getNumPhases()), + grid_size_(in_grid_size), + object_jitter_(in_object_jitter), + output_dir_(in_output_dir), + output_file_stem_(in_output_file_stem), + grid_resolution_(in_resolution), + save_meshes_(in_save_meshes), + save_pngs_(in_save_pngs), + selected_phase_(in_selected_phase) { // initialize number of ranks n_ranks_ = info_.getNumRanks(); // initialize rank dimensions according to given grid - for (uint64_t d = 0 ; d < 3 ; d++) { - if(grid_size_[d] > 1) rank_dims_.insert(d); + for (uint64_t d = 0; d < 3; d++) { + if (grid_size_[d] > 1) + rank_dims_.insert(d); } max_o_per_dim_ = 0; @@ -154,7 +150,7 @@ Render::Render( this->info_.normalizeEdges(selected_phase_); } else { for (PhaseType phase = 0; phase < this->n_phases_; phase++) { - this->info_.normalizeEdges(phase); + this->info_.normalizeEdges(phase); } } @@ -165,8 +161,9 @@ Render::Render( std::array jitterDims; for (uint64_t d = 0; d < 3; d++) { if (auto f = this->rank_dims_.find(d); f != this->rank_dims_.end()) { - jitterDims[d] = ((double)std::rand()/RAND_MAX - 0.5) * object_jitter_; - } else jitterDims[d] = 0; + jitterDims[d] = ((double)std::rand() / RAND_MAX - 0.5) * object_jitter_; + } else + jitterDims[d] = 0; } jitter_dims_.insert(std::make_pair(objectID, jitterDims)); } @@ -182,12 +179,13 @@ double Render::computeMaxObjectVolume_() { return ov_max; } -std::variant, std::set>> Render::computeObjectQoiRange_() { +std::variant, std::set>> +Render::computeObjectQoiRange_() { // Initialize object QOI range attributes double oq_max = -1 * std::numeric_limits::infinity(); double oq_min = std::numeric_limits::infinity(); double oq; - std::set> oq_all; + std::set> oq_all; // Update the QOI range auto updateQoiRange = [&](auto const& objects, PhaseType phase) { @@ -201,13 +199,15 @@ std::variant, std::set>> Rend } else { oq_all.insert(oq); } - if(oq_all.size() > 20) { + if (oq_all.size() > 20) { oq_all.clear(); continuous_object_qoi_ = true; } } - if (oq > oq_max) oq_max = oq; - if (oq < oq_min) oq_min = oq; + if (oq > oq_max) + oq_max = oq; + if (oq < oq_min) + oq_min = oq; } }; @@ -247,27 +247,29 @@ std::pair Render::computeRankQoiRange_() { rank_qoi_map = this->info_.getAllQOIAtRank(rank_id, this->rank_qoi_); // Get max qoi for this rank across all phases - auto prmax = std::max_element - ( + auto prmax = std::max_element( std::begin(rank_qoi_map), std::end(rank_qoi_map), - [] (const std::pair& p1, const std::pair& p2) { + []( + const std::pair& p1, + const std::pair& p2) { return p1.second < p2.second; - } - ); + }); rqmax_for_phase = prmax->second; // Get min qoi for this rank across all phases - auto prmin = std::max_element - ( + auto prmin = std::max_element( std::begin(rank_qoi_map), std::end(rank_qoi_map), - [] (const std::pair& p1, const std::pair& p2) { + []( + const std::pair& p1, + const std::pair& p2) { return p1.second > p2.second; - } - ); + }); rqmin_for_phase = prmin->second; - if (rqmax_for_phase > rq_max) rq_max = rqmax_for_phase; - if (rqmin_for_phase < rq_min) rq_min = rqmin_for_phase; + if (rqmax_for_phase > rq_max) + rq_max = rqmax_for_phase; + if (rqmin_for_phase < rq_min) + rq_min = rqmin_for_phase; } // return range @@ -277,18 +279,22 @@ std::pair Render::computeRankQoiRange_() { double Render::computeRankQoiAverage_(PhaseType phase, std::string qoi) { // Initialize rank QOI range attributes double rq_sum = 0.0; - auto const& rank_loads_at_phase = this->info_.getAllRankQOIAtPhase(phase, qoi); - for (auto const& [rank, rank_load] : rank_loads_at_phase){ + auto const& rank_loads_at_phase = + this->info_.getAllRankQOIAtPhase(phase, qoi); + for (auto const& [rank, rank_load] : rank_loads_at_phase) { rq_sum += rank_load; } return rq_sum / rank_loads_at_phase.size(); } -std::map> Render::createObjectMapping_(PhaseType phase) { - std::map> object_mapping; +std::map> +Render::createObjectMapping_(PhaseType phase) { + std::map> + object_mapping; // Add each rank and its corresponding objects at the given phase to the object mapping for (uint64_t rank_id = 0; rank_id < this->n_ranks_; rank_id++) { - object_mapping.insert(std::make_pair(rank_id, this->info_.getRankObjects(rank_id, phase))); + object_mapping.insert( + std::make_pair(rank_id, this->info_.getRankObjects(rank_id, phase))); } return object_mapping; } @@ -305,18 +311,19 @@ vtkNew Render::createRankMesh_(PhaseType phase) { rank_arr->SetNumberOfTuples(this->n_ranks_); for (uint64_t rank_id = 0; rank_id < this->n_ranks_; rank_id++) { - std::array cartesian = this->globalIDToCartesian_(rank_id, this->grid_size_); + std::array cartesian = + this->globalIDToCartesian_(rank_id, this->grid_size_); std::array offsets = { cartesian[0] * this->grid_resolution_, cartesian[1] * this->grid_resolution_, - cartesian[2] * this->grid_resolution_ - }; + cartesian[2] * this->grid_resolution_}; // Insert point based on cartesian coordinates rank_points_->SetPoint(rank_id, offsets[0], offsets[1], offsets[2]); auto objects = this->info_.getRankObjects(rank_id, phase); - auto rank_qoi_val = this->info_.getRankQOIAtPhase(rank_id, phase, this->rank_qoi_); + auto rank_qoi_val = + this->info_.getRankQOIAtPhase(rank_id, phase, this->rank_qoi_); rank_arr->SetTuple1(rank_id, rank_qoi_val); } @@ -327,7 +334,9 @@ vtkNew Render::createRankMesh_(PhaseType phase) { return pd_mesh; } -bool compareObjects(const std::pair& p1, const std::pair& p2) { +bool compareObjects( + const std::pair& p1, + const std::pair& p2) { ElementIDType lhsID = p1.first.getID(); ElementIDType rhsID = p2.first.getID(); uint64_t migratableLhs = p1.second; @@ -379,18 +388,17 @@ vtkNew Render::createObjectMesh_(PhaseType phase) { // Iterate through object mapping for (auto const& [rankID, objects] : object_mapping) { - std::array ijk = this->globalIDToCartesian_(rankID, this->grid_size_); + std::array ijk = + this->globalIDToCartesian_(rankID, this->grid_size_); std::array offsets = { - ijk[0] * this->grid_resolution_, - ijk[1] * this->grid_resolution_, - ijk[2] * this->grid_resolution_ - }; + ijk[0] * this->grid_resolution_, ijk[1] * this->grid_resolution_, + ijk[2] * this->grid_resolution_}; // Compute local object block parameters uint64_t n_o_rank = objects.size(); - uint64_t n_o_per_dim = ceil( pow( n_o_rank, 1.0 / this->rank_dims_.size()) ); + uint64_t n_o_per_dim = ceil(pow(n_o_rank, 1.0 / this->rank_dims_.size())); if (n_o_per_dim > this->max_o_per_dim_) { this->max_o_per_dim_ = n_o_per_dim; } @@ -401,7 +409,8 @@ vtkNew Render::createObjectMesh_(PhaseType phase) { for (uint64_t d = 0; d < 3; d++) { if (auto f = this->rank_dims_.find(d); f != this->rank_dims_.end()) { rank_size[d] = n_o_per_dim; - } else rank_size[d] = 1; + } else + rank_size[d] = 1; } std::array centering = {0, 0, 0}; @@ -432,17 +441,14 @@ vtkNew Render::createObjectMesh_(PhaseType phase) { std::array currentPointPosition = {0, 0, 0}; int d = 0; for (auto c : this->globalIDToCartesian_(i, rank_size)) { - currentPointPosition[d] = offsets[d] - centering[d] + ( - jitter_dims_.at(objectWork.getID())[d] + c) * o_resolution; + currentPointPosition[d] = offsets[d] - centering[d] + + (jitter_dims_.at(objectWork.getID())[d] + c) * o_resolution; d++; } points->SetPoint( - point_index, - currentPointPosition[0], - currentPointPosition[1], - currentPointPosition[2] - ); + point_index, currentPointPosition[0], currentPointPosition[1], + currentPointPosition[2]); // Set object attributes ElementIDType obj_id = objectWork.getID(); @@ -468,16 +474,17 @@ vtkNew Render::createObjectMesh_(PhaseType phase) { lineValuesArray->SetName("bytes"); vtkNew lines; uint64_t n_e = 0; - std::map, std::tuple> edge_values; + std::map< + std::tuple, std::tuple> + edge_values; fmt::print(" Creating inter-object communication edges\n"); - for(auto& [pt_index, k, v] : sent_volumes) { + for (auto& [pt_index, k, v] : sent_volumes) { // sort the point index and the object id in the "ij" tuple std::tuple ij; if (pt_index <= objectid_to_index.at(k)) { ij = {pt_index, objectid_to_index.at(k)}; - } - else { + } else { ij = {objectid_to_index.at(k), pt_index}; } @@ -487,10 +494,10 @@ vtkNew Render::createObjectMesh_(PhaseType phase) { auto current_edge = std::get<0>(edge_values.at(ij)); auto current_v = std::get<1>(edge_values.at(ij)); edge_values.at(ij) = {current_edge, current_v + v}; - lineValuesArray->SetTuple1(std::get<0>(edge_values.at(ij)), std::get<1>(edge_values.at(ij))); + lineValuesArray->SetTuple1( + std::get<0>(edge_values.at(ij)), std::get<1>(edge_values.at(ij))); // fmt::print("\tupdating edge {} ({}--{}): {}\n", current_edge, std::get<0>(ij), std::get<1>(ij), current_v+v); - } - else { + } else { // If it doesn't, we create it // fmt::print("\tcreating edge {} ({}--{}): {}\n", n_e, std::get<0>(ij), std::get<1>(ij), v); edge_value = {n_e, v}; @@ -519,7 +526,8 @@ vtkNew Render::createObjectMesh_(PhaseType phase) { return pd_mesh; } -void Render::getRgbFromTab20Colormap_(int index, double& r, double& g, double& b) { +void Render::getRgbFromTab20Colormap_( + int index, double& r, double& g, double& b) { const std::vector> tab20_cmap = { {0.12156862745098039, 0.4666666666666667, 0.7058823529411765}, {0.6823529411764706, 0.7803921568627451, 0.9098039215686274}, @@ -540,25 +548,29 @@ void Render::getRgbFromTab20Colormap_(int index, double& r, double& g, double& b {0.7372549019607844, 0.7411764705882353, 0.13333333333333333}, {0.8588235294117647, 0.8588235294117647, 0.5529411764705883}, {0.09019607843137255, 0.7450980392156863, 0.8117647058823529}, - {0.6196078431372549, 0.8549019607843137, 0.8980392156862745} - }; + {0.6196078431372549, 0.8549019607843137, 0.8980392156862745}}; if (index < 0 || static_cast(index) >= tab20_cmap.size()) { throw std::runtime_error("Index out of bounds for tab20 colormap."); } std::tie(r, g, b) = tab20_cmap[index]; } -/*static*/ vtkSmartPointer Render::createColorTransferFunction_( - std::variant, std::set>> attribute_range, ColorType ct -) { - vtkSmartPointer ctf = vtkSmartPointer::New(); +/*static*/ vtkSmartPointer +Render::createColorTransferFunction_( + std::variant, std::set>> + attribute_range, + ColorType ct) { + vtkSmartPointer ctf = + vtkSmartPointer::New(); ctf->SetNanColorRGBA(1., 1., 1., 0.); ctf->UseBelowRangeColorOn(); ctf->UseAboveRangeColorOn(); // Make discrete when requested - if(std::holds_alternative>>(attribute_range)) { - const std::set>& values = std::get>>(attribute_range); + if (std::holds_alternative>>( + attribute_range)) { + const std::set>& values = + std::get>>(attribute_range); // Handle the set type ctf->DiscretizeOn(); @@ -567,7 +579,9 @@ void Render::getRgbFromTab20Colormap_(int index, double& r, double& g, double& b ctf->SetNumberOfIndexedColors(n_colors); int i = 0; for (auto v : values) { - std::visit([&ctf](auto&& val) { ctf->SetAnnotation(val, std::to_string(val)); }, v); + std::visit( + [&ctf](auto&& val) { ctf->SetAnnotation(val, std::to_string(val)); }, + v); // Use discrete color map double r, g, b; getRgbFromTab20Colormap_(i, r, g, b); @@ -577,8 +591,10 @@ void Render::getRgbFromTab20Colormap_(int index, double& r, double& g, double& b } ctf->Build(); return ctf; - } else if (std::holds_alternative>(attribute_range)) { - const std::pair& range = std::get>(attribute_range); + } else if (std::holds_alternative>( + attribute_range)) { + const std::pair& range = + std::get>(attribute_range); switch (ct) { case BlueToRed: { ctf->SetColorSpaceToDiverging(); @@ -595,13 +611,13 @@ void Render::getRgbFromTab20Colormap_(int index, double& r, double& g, double& b double const mid_point1 = (range.second - range.first) * 0.25; double const mid_point2 = (range.second - range.first) * 0.75; - ctf->AddRGBPoint(range.first, 0.0, 0.0, 1.0); // Blue + ctf->AddRGBPoint(range.first, 0.0, 0.0, 1.0); // Blue ctf->AddRGBPoint(mid_point1, 0.0, 1.0, 0.0); // Green ctf->AddRGBPoint(mid_point2, 1.0, 0.8, 0.0); // Orange - ctf->AddRGBPoint(range.second, 1.0, 0.0, 0.0); // Red + ctf->AddRGBPoint(range.second, 1.0, 0.0, 0.0); // Red - ctf->SetBelowRangeColor(0.0, 1.0, 1.0); // Cyan - ctf->SetAboveRangeColor(1.0, 1.0, 0.0); // Yellow + ctf->SetBelowRangeColor(0.0, 1.0, 1.0); // Cyan + ctf->SetAboveRangeColor(1.0, 1.0, 0.0); // Yellow break; } case WhiteToBlack: { @@ -628,13 +644,10 @@ void Render::getRgbFromTab20Colormap_(int index, double& r, double& g, double& b } /*static*/ vtkSmartPointer Render::createScalarBarActor_( - vtkSmartPointer mapper, - const std::string& title, - double x, double y, - uint64_t font_size, - std::set> values -) { - vtkSmartPointer scalar_bar_actor = vtkSmartPointer::New(); + vtkSmartPointer mapper, const std::string& title, double x, + double y, uint64_t font_size, std::set> values) { + vtkSmartPointer scalar_bar_actor = + vtkSmartPointer::New(); scalar_bar_actor->SetLookupTable(mapper->GetLookupTable()); // Set default parameters @@ -667,7 +680,7 @@ void Render::getRgbFromTab20Colormap_(int index, double& r, double& g, double& b properties.push_back(scalar_bar_actor->GetLabelTextProperty()); properties.push_back(scalar_bar_actor->GetAnnotationTextProperty()); - for (vtkTextProperty* prop : properties){ + for (vtkTextProperty* prop : properties) { prop->SetColor(0.0, 0.0, 0.0); prop->ItalicOff(); prop->BoldOff(); @@ -684,8 +697,7 @@ void Render::getRgbFromTab20Colormap_(int index, double& r, double& g, double& b } /* static */ std::array Render::globalIDToCartesian_( - uint64_t flat_id, std::array grid_sizes -) { + uint64_t flat_id, std::array grid_sizes) { std::array cartesian = {0, 0, 0}; // Sanity check uint64_t n01 = grid_sizes[0] * grid_sizes[1]; @@ -706,17 +718,18 @@ void Render::getRgbFromTab20Colormap_(int index, double& r, double& g, double& b /* static */ vtkSmartPointer Render::setupRenderer_() { vtkSmartPointer renderer = vtkSmartPointer::New(); - renderer->SetBackground(1.0, 1.0, 1.0); // Set background to white + renderer->SetBackground(1.0, 1.0, 1.0); // Set background to white renderer->GetActiveCamera()->ParallelProjectionOn(); return renderer; } /* static */ vtkSmartPointer Render::createRanksMapper_( vtkPolyData* rank_mesh, - std::variant, std::set>> rank_qoi_range -) { + std::variant, std::set>> + rank_qoi_range) { // Create square glyphs at ranks - vtkSmartPointer rank_glyph = vtkSmartPointer::New(); + vtkSmartPointer rank_glyph = + vtkSmartPointer::New(); rank_glyph->SetGlyphTypeToSquare(); rank_glyph->SetScale(0.95); rank_glyph->FilledOn(); @@ -729,27 +742,34 @@ void Render::getRgbFromTab20Colormap_(int index, double& r, double& g, double& b // Lower glyphs slightly for visibility vtkSmartPointer z_lower = vtkSmartPointer::New(); z_lower->Translate(0.0, 0.0, -0.01); - vtkSmartPointer trans = vtkSmartPointer::New(); + vtkSmartPointer trans = + vtkSmartPointer::New(); trans->SetTransform(z_lower); trans->SetInputConnection(rank_glypher->GetOutputPort()); // Create mapper for rank glyphs - vtkSmartPointer rank_mapper = vtkSmartPointer::New(); + vtkSmartPointer rank_mapper = + vtkSmartPointer::New(); rank_mapper->SetInputConnection(trans->GetOutputPort()); - rank_mapper->SetLookupTable(createColorTransferFunction_(rank_qoi_range, BlueToRed)); + rank_mapper->SetLookupTable( + createColorTransferFunction_(rank_qoi_range, BlueToRed)); // Check the type held by the variant qoi range and set the scalar range appropriately if (std::holds_alternative>(rank_qoi_range)) { auto range_pair = std::get>(rank_qoi_range); rank_mapper->SetScalarRange(range_pair.first, range_pair.second); - } else if (std::holds_alternative>>(rank_qoi_range)) { - const auto& range_set = std::get>>(rank_qoi_range); + } else if (std::holds_alternative>>( + rank_qoi_range)) { + const auto& range_set = + std::get>>(rank_qoi_range); if (!range_set.empty()) { auto range_begin = *range_set.begin(); auto range_end = *range_set.rbegin(); if (std::holds_alternative(range_begin)) { - rank_mapper->SetScalarRange(std::get(range_begin), std::get(range_end)); + rank_mapper->SetScalarRange( + std::get(range_begin), std::get(range_end)); } else { - rank_mapper->SetScalarRange(std::get(range_begin), std::get(range_end)); + rank_mapper->SetScalarRange( + std::get(range_begin), std::get(range_end)); } } else { rank_mapper->SetScalarRange(0., 0.); @@ -763,37 +783,25 @@ void Render::getRgbFromTab20Colormap_(int index, double& r, double& g, double& b } void Render::renderPNG( - PhaseType phase, - vtkPolyData* rank_mesh, - vtkPolyData* object_mesh, - uint64_t edge_width, - double glyph_factor, - uint64_t win_size, - uint64_t font_size, - std::string output_dir, - std::string output_file_stem -) { + PhaseType phase, vtkPolyData* rank_mesh, vtkPolyData* object_mesh, + uint64_t edge_width, double glyph_factor, uint64_t win_size, + uint64_t font_size, std::string output_dir, std::string output_file_stem) { // Setup rendering space vtkSmartPointer renderer = setupRenderer_(); // Create rank mapper for later use and create corresponding rank actor - std::variant, std::set>> rank_qoi_variant(rank_qoi_range_); - vtkSmartPointer rank_mapper = createRanksMapper_( - rank_mesh, - rank_qoi_variant - ); + std::variant, std::set>> + rank_qoi_variant(rank_qoi_range_); + vtkSmartPointer rank_mapper = + createRanksMapper_(rank_mesh, rank_qoi_variant); vtkSmartPointer rank_actor = vtkSmartPointer::New(); rank_actor->SetMapper(rank_mapper); // Create qoi scale legend for ranks std::string rank_qoi_scale_title = "Rank " + this->rank_qoi_; - vtkSmartPointer rank_qoi_scale_actor = createScalarBarActor_( - rank_mapper, - rank_qoi_scale_title, - 0.5, - 0.9, - font_size - ); + vtkSmartPointer rank_qoi_scale_actor = + createScalarBarActor_( + rank_mapper, rank_qoi_scale_title, 0.5, 0.9, font_size); rank_qoi_scale_actor->DrawBelowRangeSwatchOn(); rank_qoi_scale_actor->SetBelowRangeAnnotation("<"); rank_qoi_scale_actor->DrawAboveRangeSwatchOn(); @@ -803,9 +811,10 @@ void Render::renderPNG( renderer->AddActor(rank_actor); renderer->AddActor2D(rank_qoi_scale_actor); - if(this->object_qoi_ != "") { + if (this->object_qoi_ != "") { // Create white to black lookup table - vtkSmartPointer bw_lut = vtkSmartPointer::New(); + vtkSmartPointer bw_lut = + vtkSmartPointer::New(); bw_lut->SetTableRange(0.0, this->object_volume_max_); bw_lut->SetSaturationRange(0, 0); bw_lut->SetHueRange(0, 0); @@ -814,7 +823,8 @@ void Render::renderPNG( bw_lut->Build(); // Create mapper for inter-object edges - vtkSmartPointer edge_mapper = vtkSmartPointer::New(); + vtkSmartPointer edge_mapper = + vtkSmartPointer::New(); edge_mapper->SetInputData(object_mesh); edge_mapper->SetScalarModeToUseCellData(); edge_mapper->SetScalarRange(0.0, this->object_volume_max_); @@ -824,13 +834,15 @@ void Render::renderPNG( vtkSmartPointer edge_actor = vtkSmartPointer::New(); edge_actor->SetMapper(edge_mapper); edge_actor->GetProperty()->SetLineWidth(edge_width); - vtkSmartPointer volume_actor = createScalarBarActor_(edge_mapper, "Inter-Object Volume", 0.04, 0.04, font_size); + vtkSmartPointer volume_actor = createScalarBarActor_( + edge_mapper, "Inter-Object Volume", 0.04, 0.04, font_size); // Add communications visualization to renderer renderer->AddActor(edge_actor); renderer->AddActor2D(volume_actor); // Compute square root of object loads - vtkSmartPointer sqrtL = vtkSmartPointer::New(); + vtkSmartPointer sqrtL = + vtkSmartPointer::New(); sqrtL->SetInputData(object_mesh); sqrtL->AddScalarArrayName("load"); std::string sqrtL_str = "sqrt(load)"; @@ -843,11 +855,12 @@ void Render::renderPNG( // Glyph sentinel and migratable objects separately: 0 is for non-migratable objects, 1 for migratable std::map> glyph_mappers = { {0.0, vtkSmartPointer::New()}, - {1.0, vtkSmartPointer::New()} - }; - std::map glyph_types = {{0.0, "Square"}, {1.0, "Circle"}}; + {1.0, vtkSmartPointer::New()}}; + std::map glyph_types = { + {0.0, "Square"}, {1.0, "Circle"}}; for (const auto& [k, v] : glyph_types) { - vtkSmartPointer thresh = vtkSmartPointer::New(); + vtkSmartPointer thresh = + vtkSmartPointer::New(); thresh->SetInputData(sqrtL_out); thresh->ThresholdBetween(k, k); thresh->Update(); @@ -859,8 +872,9 @@ void Render::renderPNG( thresh_out->GetPointData()->SetActiveScalars(sqrtL_str.c_str()); // Glyph by square root of object quantity of interest - vtkSmartPointer glyph = vtkSmartPointer::New(); - if(v == "Square") { + vtkSmartPointer glyph = + vtkSmartPointer::New(); + if (v == "Square") { glyph->SetGlyphTypeToSquare(); } else if (v == "Circle") { glyph->SetGlyphTypeToCircle(); @@ -876,25 +890,32 @@ void Render::renderPNG( glypher->SetScaleModeToScaleByScalar(); glypher->SetScaleFactor(glyph_factor); glypher->Update(); - glypher->GetOutput()->GetPointData()->SetActiveScalars(this->object_qoi_.c_str()); + glypher->GetOutput()->GetPointData()->SetActiveScalars( + this->object_qoi_.c_str()); - vtkSmartPointer zRaise = vtkSmartPointer::New(); + vtkSmartPointer zRaise = + vtkSmartPointer::New(); zRaise->Translate(0.0, 0.0, 0.01); - vtkSmartPointer trans = vtkSmartPointer::New(); + vtkSmartPointer trans = + vtkSmartPointer::New(); trans->SetTransform(zRaise); trans->SetInputData(glypher->GetOutput()); glyph_mappers.at(k)->SetInputConnection(trans->GetOutputPort()); - glyph_mappers.at(k)->SetLookupTable(createColorTransferFunction_(this->object_qoi_range_)); + glyph_mappers.at(k)->SetLookupTable( + createColorTransferFunction_(this->object_qoi_range_)); - if (std::holds_alternative>(this->object_qoi_range_)) { - auto range = std::get>(this->object_qoi_range_); + if (std::holds_alternative>( + this->object_qoi_range_)) { + auto range = + std::get>(this->object_qoi_range_); // Manually set scalar range so either mapper (migratable vs non-migratable) can be used for the scalar bar range glyph_mappers.at(k)->SetScalarRange(range.first, range.second); } - vtkSmartPointer object_glyph_actor = vtkSmartPointer::New(); + vtkSmartPointer object_glyph_actor = + vtkSmartPointer::New(); object_glyph_actor->SetMapper(glyph_mappers.at(k)); // Add objects visualization to renderer @@ -903,23 +924,23 @@ void Render::renderPNG( if (glyph_mappers.at(1.0)) { std::string object_qoi_name = "Object " + this->object_qoi_; - std::set> values = {}; + std::set> values = {}; // Check continuity of object qoi - if (std::holds_alternative>(this->object_qoi_range_)) { + if (std::holds_alternative>( + this->object_qoi_range_)) { values = {}; - } else if (std::holds_alternative>>(this->object_qoi_range_)) { - values = std::get>>(this->object_qoi_range_); + } else if (std::holds_alternative>>( + this->object_qoi_range_)) { + values = std::get>>( + this->object_qoi_range_); } else { - throw std::runtime_error("Unexpected type in object_qoi_range variant."); + throw std::runtime_error( + "Unexpected type in object_qoi_range variant."); } - vtkSmartPointer object_qoi_scalar_bar_actor = createScalarBarActor_( - glyph_mappers.at(1.0), - object_qoi_name.c_str(), - 0.52, - 0.04, - font_size, - values - ); + vtkSmartPointer object_qoi_scalar_bar_actor = + createScalarBarActor_( + glyph_mappers.at(1.0), object_qoi_name.c_str(), 0.52, 0.04, font_size, + values); renderer->AddActor2D(object_qoi_scalar_bar_actor); } } @@ -932,9 +953,11 @@ void Render::renderPNG( } else { ss << "Phase: " << phase << "/" << (this->n_phases_ - 1) << "\n"; } - ss << "Load Imbalance: " << std::fixed << std::setprecision(2) << this->info_.getImbalance(phase); + ss << "Load Imbalance: " << std::fixed << std::setprecision(2) + << this->info_.getImbalance(phase); // Setup text actor - vtkSmartPointer text_actor = vtkSmartPointer::New(); + vtkSmartPointer text_actor = + vtkSmartPointer::New(); text_actor->SetInput(ss.str().c_str()); vtkTextProperty* textProp = text_actor->GetTextProperty(); textProp->SetColor(0.0, 0.0, 0.0); @@ -966,7 +989,8 @@ void Render::renderPNG( // Export the PNG image vtkNew writer; writer->SetInputConnection(w2i->GetOutputPort()); - std::string png_filename = output_dir + output_file_stem + std::to_string(phase) + ".png"; + std::string png_filename = + output_dir + output_file_stem + std::to_string(phase) + ".png"; writer->SetFileName(png_filename.c_str()); writer->SetCompressionLevel(2); writer->Write(); @@ -980,67 +1004,63 @@ void Render::generate(uint64_t font_size, uint64_t win_size) { auto range_pair = std::get>(object_qoi_range_); double object_qoi_min = range_pair.first; double object_qoi_max = range_pair.second; - fmt::print("Rank {} range: {}, {}\n", rank_qoi_, rank_qoi_min, rank_qoi_max); - fmt::print("Object {} range: {}, {}\n", object_qoi_, object_qoi_min, object_qoi_max); + fmt::print( + "Rank {} range: {}, {}\n", rank_qoi_, rank_qoi_min, rank_qoi_max); + fmt::print( + "Object {} range: {}, {}\n", object_qoi_, object_qoi_min, object_qoi_max); } fmt::print("selected phase={}\n", selected_phase_); auto createMeshAndRender = [&](PhaseType phase) { - vtkNew object_mesh = this->createObjectMesh_(phase); - vtkNew rank_mesh = this->createRankMesh_(phase); - - if (save_meshes_) { - fmt::print("== Writing object mesh for phase {}\n", phase); - vtkNew writer; - std::string object_mesh_filename = output_dir_ + output_file_stem_ + "_object_mesh_" + std::to_string(phase) + ".vtp"; - writer->SetFileName(object_mesh_filename.c_str()); - writer->SetInputData(object_mesh); - writer->Write(); - - fmt::print("== Writing rank mesh for phase {}\n", phase); - vtkNew writer2; - std::string rank_mesh_filneame = output_dir_ + output_file_stem_ + "_rank_mesh_" + std::to_string(phase) + ".vtp"; - writer2->SetFileName(rank_mesh_filneame.c_str()); - writer2->SetInputData(rank_mesh); - writer2->Write(); - } - - if (save_pngs_) { - fmt::print("== Rendering visualization PNG for phase {}\n", phase); + vtkNew object_mesh = this->createObjectMesh_(phase); + vtkNew rank_mesh = this->createRankMesh_(phase); + + if (save_meshes_) { + fmt::print("== Writing object mesh for phase {}\n", phase); + vtkNew writer; + std::string object_mesh_filename = output_dir_ + output_file_stem_ + + "_object_mesh_" + std::to_string(phase) + ".vtp"; + writer->SetFileName(object_mesh_filename.c_str()); + writer->SetInputData(object_mesh); + writer->Write(); + + fmt::print("== Writing rank mesh for phase {}\n", phase); + vtkNew writer2; + std::string rank_mesh_filneame = output_dir_ + output_file_stem_ + + "_rank_mesh_" + std::to_string(phase) + ".vtp"; + writer2->SetFileName(rank_mesh_filneame.c_str()); + writer2->SetInputData(rank_mesh); + writer2->Write(); + } - std::pair obj_qoi_range; - try { - obj_qoi_range = std::get>(this->object_qoi_range_); - } - catch(const std::exception& e) { - std::cerr << e.what() << '\n'; - obj_qoi_range = {0, 1}; - } + if (save_pngs_) { + fmt::print("== Rendering visualization PNG for phase {}\n", phase); - uint64_t window_size = win_size; - uint64_t edge_width = 0.03 * window_size / *std::max_element(this->grid_size_.begin(), this->grid_size_.end()); - double glyph_factor = 0.8 * this->grid_resolution_ / ( - (this->max_o_per_dim_ + 1) - * std::sqrt(object_load_max_)); - fmt::print(" Image size: {}x{}px\n", win_size, win_size); - fmt::print(" Font size: {}pt\n", font_size); - this->renderPNG( - phase, - rank_mesh, - object_mesh, - edge_width, - glyph_factor, - window_size, - font_size, - output_dir_, - output_file_stem_ - ); + std::pair obj_qoi_range; + try { + obj_qoi_range = + std::get>(this->object_qoi_range_); + } catch (const std::exception& e) { + std::cerr << e.what() << '\n'; + obj_qoi_range = {0, 1}; } + + uint64_t window_size = win_size; + uint64_t edge_width = 0.03 * window_size / + *std::max_element(this->grid_size_.begin(), this->grid_size_.end()); + double glyph_factor = 0.8 * this->grid_resolution_ / + ((this->max_o_per_dim_ + 1) * std::sqrt(object_load_max_)); + fmt::print(" Image size: {}x{}px\n", win_size, win_size); + fmt::print(" Font size: {}pt\n", font_size); + this->renderPNG( + phase, rank_mesh, object_mesh, edge_width, glyph_factor, window_size, + font_size, output_dir_, output_file_stem_); + } }; if (selected_phase_ != std::numeric_limits::max()) { - createMeshAndRender(selected_phase_); + createMeshAndRender(selected_phase_); } else { for (PhaseType phase = 0; phase < this->n_phases_; phase++) { createMeshAndRender(phase); @@ -1048,4 +1068,4 @@ void Render::generate(uint64_t font_size, uint64_t win_size) { } } -}} /* end namesapce vt::tv */ +}} // namespace vt::tv diff --git a/src/vt-tv/render/render.h b/src/vt-tv/render/render.h index 66e229f26..ed1c826ac 100644 --- a/src/vt-tv/render/render.h +++ b/src/vt-tv/render/render.h @@ -107,12 +107,7 @@ namespace vt { namespace tv { */ struct Render { private: - enum ColorType { - Default, - BlueToRed, - HotSpot, - WhiteToBlack - }; + enum ColorType { Default, BlueToRed, HotSpot, WhiteToBlack }; // quantities of interest std::string rank_qoi_ = "load"; @@ -134,7 +129,8 @@ struct Render { PhaseType selected_phase_ = std::numeric_limits::max(); // numeric parameters - std::variant, std::set>> object_qoi_range_; + std::variant, std::set>> + object_qoi_range_; std::pair rank_qoi_range_; // Maximum object attribute values @@ -158,7 +154,8 @@ struct Render { * * \return object qoi range */ - std::variant, std::set>> computeObjectQoiRange_(); + std::variant, std::set>> + computeObjectQoiRange_(); /** * \brief Compute range of rank qoi. @@ -183,7 +180,8 @@ struct Render { * * \return mapping */ - std::map> createObjectMapping_(PhaseType phase); + std::map> + createObjectMapping_(PhaseType phase); /** * \brief Map ranks to polygonal mesh. @@ -203,26 +201,26 @@ struct Render { */ vtkNew createObjectMesh_(PhaseType phase); - static void getRgbFromTab20Colormap_(int index, double& r, double& g, double& b); + static void + getRgbFromTab20Colormap_(int index, double& r, double& g, double& b); - static vtkSmartPointer createColorTransferFunction_( - std::variant, std::set>> attribute_range, ColorType ct = ColorType::Default - ); + static vtkSmartPointer + createColorTransferFunction_( + std::variant, std::set>> + attribute_range, + ColorType ct = ColorType::Default); static vtkSmartPointer createScalarBarActor_( - vtkSmartPointer mapper, - const std::string& title, - double x, double y, - uint64_t font_size = 50, - std::set> values = {} - ); + vtkSmartPointer mapper, const std::string& title, double x, + double y, uint64_t font_size = 50, + std::set> values = {}); static vtkSmartPointer setupRenderer_(); static vtkSmartPointer createRanksMapper_( vtkPolyData* rank_mesh, - std::variant, std::set>> rank_qoi_range - ); + std::variant, std::set>> + rank_qoi_range); /** * \brief Map global index to its Cartesian grid coordinates. @@ -232,9 +230,8 @@ struct Render { * * \return i,j,k Cartesian coordinates */ - static std::array globalIDToCartesian_( - uint64_t flat_id, std::array grid_sizes - ); + static std::array + globalIDToCartesian_(uint64_t flat_id, std::array grid_sizes); public: /** @@ -258,18 +255,12 @@ struct Render { * \param[in] in_selected_phase the phase selected (if max then render all) */ Render( - std::array in_qoi_request, - bool in_continuous_object_qoi, - Info& in_info, - std::array in_grid_size, - double in_object_jitter, - std::string in_output_dir, - std::string in_output_file_stem, - double in_resolution, - bool in_save_meshes, + std::array in_qoi_request, bool in_continuous_object_qoi, + Info& in_info, std::array in_grid_size, + double in_object_jitter, std::string in_output_dir, + std::string in_output_file_stem, double in_resolution, bool in_save_meshes, bool in_save_pngs, - PhaseType in_selected_phase = std::numeric_limits::max() - ); + PhaseType in_selected_phase = std::numeric_limits::max()); /** * @brief Export a visualization PNG from meshes. @@ -286,16 +277,9 @@ struct Render { * @return A smart pointer to the resulting render window. */ void renderPNG( - PhaseType phase, - vtkPolyData* rank_mesh, - vtkPolyData* object_mesh, - uint64_t edge_width, - double glyph_factor, - uint64_t win_size, - uint64_t font_size, - std::string output_dir, - std::string output_file_stem - ); + PhaseType phase, vtkPolyData* rank_mesh, vtkPolyData* object_mesh, + uint64_t edge_width, double glyph_factor, uint64_t win_size, + uint64_t font_size, std::string output_dir, std::string output_file_stem); void generate(uint64_t font_size = 50, uint64_t win_size = 2000); }; diff --git a/src/vt-tv/utility/compressor.cc b/src/vt-tv/utility/compressor.cc index 35eb7e96e..d17ccc2e7 100644 --- a/src/vt-tv/utility/compressor.cc +++ b/src/vt-tv/utility/compressor.cc @@ -48,8 +48,7 @@ namespace vt::tv::utility { Compressor::Compressor(int in_quality, int in_window_bits, std::size_t buf_size) : quality_(in_quality), window_bits_(in_window_bits), - buf_size_(buf_size) -{ + buf_size_(buf_size) { // for now, use the default allocator enc_ = BrotliEncoderCreateInstance(nullptr, nullptr, nullptr); if (!enc_) { @@ -61,8 +60,7 @@ Compressor::Compressor(int in_quality, int in_window_bits, std::size_t buf_size) } if ( window_bits_ < BROTLI_MIN_WINDOW_BITS or - window_bits_ > BROTLI_MAX_WINDOW_BITS - ) { + window_bits_ > BROTLI_MAX_WINDOW_BITS) { window_bits_ = BROTLI_MAX_WINDOW_BITS; } // Set the quality parameter for the encoder diff --git a/src/vt-tv/utility/compressor.h b/src/vt-tv/utility/compressor.h index e4b4ea95c..180dfb7fc 100644 --- a/src/vt-tv/utility/compressor.h +++ b/src/vt-tv/utility/compressor.h @@ -57,7 +57,6 @@ namespace vt::tv::utility { * \brief A streaming compressor for outputting data with brotli compression */ struct Compressor { - /** * \brief Construct a compressor * @@ -66,8 +65,7 @@ struct Compressor { * \param[in] buf_size the size of the temporary buffer */ explicit Compressor( - int in_quality, int in_window_bits, std::size_t buf_size = 1 << 20 - ); + int in_quality, int in_window_bits, std::size_t buf_size = 1 << 20); ~Compressor(); @@ -106,16 +104,15 @@ struct Compressor { */ template bool writeImpl( - StreamLike& s, uint8_t const* buffer, std::size_t const size, - bool finish_ - ); + StreamLike& s, uint8_t const* buffer, std::size_t const size, bool finish_); private: - int quality_ = 0; /**< The compressor quality */ - int window_bits_ = 0; /**< The compressor lgwin bits */ - std::size_t buf_size_ = 0; /**< The output buffer size */ - BrotliEncoderState* enc_ = nullptr; /**< Underlying encoder state */ - std::unique_ptr out_buf_ = nullptr; /**< The temporary output buffer */ + int quality_ = 0; /**< The compressor quality */ + int window_bits_ = 0; /**< The compressor lgwin bits */ + std::size_t buf_size_ = 0; /**< The output buffer size */ + BrotliEncoderState* enc_ = nullptr; /**< Underlying encoder state */ + std::unique_ptr out_buf_ = + nullptr; /**< The temporary output buffer */ }; } /* end namespace vt::tv::utility */ diff --git a/src/vt-tv/utility/compressor.impl.h b/src/vt-tv/utility/compressor.impl.h index 4a949f430..c03b71db2 100644 --- a/src/vt-tv/utility/compressor.impl.h +++ b/src/vt-tv/utility/compressor.impl.h @@ -51,15 +51,15 @@ namespace vt::tv::utility { template -bool Compressor::write(StreamLike& s, uint8_t const* buffer, std::size_t const size) { +bool Compressor::write( + StreamLike& s, uint8_t const* buffer, std::size_t const size) { constexpr auto finish_writing = false; return writeImpl(s, buffer, size, finish_writing); } template bool Compressor::writeImpl( - StreamLike& s, uint8_t const* buffer, std::size_t const size, bool finish_ -) { + StreamLike& s, uint8_t const* buffer, std::size_t const size, bool finish_) { assert(enc_ && "Must have a valid compressor"); uint8_t const* cur = buffer; @@ -71,8 +71,7 @@ bool Compressor::writeImpl( auto op = finish_ ? BROTLI_OPERATION_FINISH : BROTLI_OPERATION_PROCESS; bool ret = BrotliEncoderCompressStream( - enc_, op, &rem, &cur, &avail_out, &next_out, nullptr - ); + enc_, op, &rem, &cur, &avail_out, &next_out, nullptr); if (!ret) { assert(false && "Failed to stream compression data out\n"); return false; diff --git a/src/vt-tv/utility/decompression_input_container.cc b/src/vt-tv/utility/decompression_input_container.cc index fd26d11c7..74da3b610 100644 --- a/src/vt-tv/utility/decompression_input_container.cc +++ b/src/vt-tv/utility/decompression_input_container.cc @@ -48,9 +48,8 @@ namespace vt::tv::utility { DecompressionInputContainer::DecompressionInputContainer( - std::string const& filename, std::size_t in_chunk_size -) : chunk_size_(in_chunk_size) -{ + std::string const& filename, std::size_t in_chunk_size) + : chunk_size_(in_chunk_size) { std::ifstream is(filename, std::ios::binary); assert(is.good()); d_ = std::make_unique>(std::move(is)); diff --git a/src/vt-tv/utility/decompression_input_container.h b/src/vt-tv/utility/decompression_input_container.h index c59273534..9d1881021 100644 --- a/src/vt-tv/utility/decompression_input_container.h +++ b/src/vt-tv/utility/decompression_input_container.h @@ -70,11 +70,10 @@ struct DecompressionInputContainer { * \param[in] in_chunk_size the chunk size to read in increments */ explicit DecompressionInputContainer( - std::string const& filename, std::size_t in_chunk_size = 1 << 16 - ); + std::string const& filename, std::size_t in_chunk_size = 1 << 16); /// Tag type for non-file constructor - struct AnyStreamTag {}; + struct AnyStreamTag { }; /** * \brief Construct with anything that resembles a stream @@ -84,8 +83,7 @@ struct DecompressionInputContainer { */ template explicit DecompressionInputContainer( - AnyStreamTag, StreamLike stream, std::size_t in_chunk_size = 1 << 16 - ); + AnyStreamTag, StreamLike stream, std::size_t in_chunk_size = 1 << 16); /** * \brief Advance by one @@ -102,11 +100,12 @@ struct DecompressionInputContainer { char const& getCurrent() const; private: - std::size_t chunk_size_ = 0; /**< The chunk size */ - std::unique_ptr d_ = nullptr; /**< The decompressor */ - std::unique_ptr output_buf_ = nullptr; /**< The temp output buffer */ - std::size_t mutable cur_ = 0; /**< Current position */ - std::size_t mutable len_ = 0; /**< Max position in buf */ + std::size_t chunk_size_ = 0; /**< The chunk size */ + std::unique_ptr d_ = nullptr; /**< The decompressor */ + std::unique_ptr output_buf_ = + nullptr; /**< The temp output buffer */ + std::size_t mutable cur_ = 0; /**< Current position */ + std::size_t mutable len_ = 0; /**< Max position in buf */ }; } /* end namespace vt::tv::utility */ diff --git a/src/vt-tv/utility/decompression_input_container.impl.h b/src/vt-tv/utility/decompression_input_container.impl.h index 7533e45fc..1e7eca610 100644 --- a/src/vt-tv/utility/decompression_input_container.impl.h +++ b/src/vt-tv/utility/decompression_input_container.impl.h @@ -48,9 +48,8 @@ namespace vt::tv::utility { template DecompressionInputContainer::DecompressionInputContainer( - AnyStreamTag, StreamLike stream, std::size_t in_chunk_size -) : chunk_size_(in_chunk_size) -{ + AnyStreamTag, StreamLike stream, std::size_t in_chunk_size) + : chunk_size_(in_chunk_size) { d_ = std::make_unique>(std::move(stream)); output_buf_ = std::make_unique(chunk_size_); len_ = d_->read(output_buf_.get(), chunk_size_); diff --git a/src/vt-tv/utility/decompressor.h b/src/vt-tv/utility/decompressor.h index b17555b1a..f2022c7c3 100644 --- a/src/vt-tv/utility/decompressor.h +++ b/src/vt-tv/utility/decompressor.h @@ -61,7 +61,6 @@ namespace vt::tv::utility { */ template struct Decompressor : DecompressorBase { - /** * \brief Construct the decompressor * @@ -80,7 +79,8 @@ struct Decompressor : DecompressorBase { * * \return how many bytes it actually read into the buffer */ - std::size_t read(uint8_t* output_buffer, std::size_t bytes_to_output) override; + std::size_t + read(uint8_t* output_buffer, std::size_t bytes_to_output) override; /** * \brief Whether we are done with decompressing the file @@ -99,11 +99,11 @@ struct Decompressor : DecompressorBase { private: Readable r_; - BrotliDecoderState* dec_ = nullptr; /**< Underlying decoder state */ - std::size_t in_buf_len_ = 0; /**< Input buffer max length (chunk) */ - std::unique_ptr buf_in_; /**< Temporary input buffer to read */ - uint8_t const* next_in_ = nullptr; /**< Next input pointer */ - std::size_t avail_in_ = 0; /**< Available length of input data */ + BrotliDecoderState* dec_ = nullptr; /**< Underlying decoder state */ + std::size_t in_buf_len_ = 0; /**< Input buffer max length (chunk) */ + std::unique_ptr buf_in_; /**< Temporary input buffer to read */ + uint8_t const* next_in_ = nullptr; /**< Next input pointer */ + std::size_t avail_in_ = 0; /**< Available length of input data */ }; } /* end namespace vt::tv::utility */ diff --git a/src/vt-tv/utility/decompressor.impl.h b/src/vt-tv/utility/decompressor.impl.h index 713d3bd63..1a5d4a3a7 100644 --- a/src/vt-tv/utility/decompressor.impl.h +++ b/src/vt-tv/utility/decompressor.impl.h @@ -55,8 +55,7 @@ namespace vt::tv::utility { template Decompressor::Decompressor(Readable in_r, std::size_t buf_len_) : r_(std::move(in_r)), - in_buf_len_(buf_len_) -{ + in_buf_len_(buf_len_) { // for now, use the default allocator dec_ = BrotliDecoderCreateInstance(nullptr, nullptr, nullptr); if (!dec_) { @@ -83,8 +82,7 @@ Decompressor::~Decompressor() { template std::size_t Decompressor::read( - uint8_t* output_buffer, std::size_t bytes_to_output -) { + uint8_t* output_buffer, std::size_t bytes_to_output) { uint8_t* next_out = output_buffer; std::size_t avail_out = bytes_to_output; @@ -96,42 +94,41 @@ std::size_t Decompressor::read( // While we have input data and room to output decompressed data... while (avail_out > 0 and not success) { BrotliDecoderResult res = BrotliDecoderDecompressStream( - dec_, &avail_in_, &next_in_, &avail_out, &next_out, nullptr - ); + dec_, &avail_in_, &next_in_, &avail_out, &next_out, nullptr); switch (res) { - case BROTLI_DECODER_RESULT_NEEDS_MORE_INPUT: - { + case BROTLI_DECODER_RESULT_NEEDS_MORE_INPUT: { // We need to read more to continue decoding - assert(avail_in_ == 0 && "Brotli asked for more input even though we still had some."); + assert( + avail_in_ == 0 && + "Brotli asked for more input even though we still had some."); bool const has_more_input = getMoreInput(); if (not has_more_input) { - assert(false && "Brotli asked for more input but the file terminated early."); + assert( + false && + "Brotli asked for more input but the file terminated early."); } break; } - case BROTLI_DECODER_RESULT_NEEDS_MORE_OUTPUT: - { + case BROTLI_DECODER_RESULT_NEEDS_MORE_OUTPUT: { // this is an expected condition assert( - avail_out == 0 && "Brotli asked for more output even though we still had some." - ); + avail_out == 0 && + "Brotli asked for more output even though we still had some."); break; } - case BROTLI_DECODER_RESULT_SUCCESS: - { + case BROTLI_DECODER_RESULT_SUCCESS: { if (avail_in_ != 0 or getMoreInput()) { - assert(false && "Brotli terminated early before reading the whole file!"); + assert( + false && "Brotli terminated early before reading the whole file!"); } success = true; break; } - case BROTLI_DECODER_RESULT_ERROR: - { + case BROTLI_DECODER_RESULT_ERROR: { // we have hit an unknown error, print the code and corresponding message! auto error_code = BrotliDecoderGetErrorCode(dec_); auto error_str = fmt::format( - "code={}, msg={}\n", error_code, BrotliDecoderErrorString(error_code) - ); + "code={}, msg={}\n", error_code, BrotliDecoderErrorString(error_code)); fmt::print(error_str); assert(false); break; diff --git a/src/vt-tv/utility/decompressor_base.h b/src/vt-tv/utility/decompressor_base.h index 16a348071..4830d1c54 100644 --- a/src/vt-tv/utility/decompressor_base.h +++ b/src/vt-tv/utility/decompressor_base.h @@ -59,7 +59,8 @@ namespace vt::tv::utility { struct DecompressorBase { virtual ~DecompressorBase() = default; virtual bool done() const = 0; - virtual std::size_t read(uint8_t* output_buffer, std::size_t bytes_to_output) = 0; + virtual std::size_t + read(uint8_t* output_buffer, std::size_t bytes_to_output) = 0; }; } /* end namespace vt::tv::utility */ diff --git a/src/vt-tv/utility/input_iterator.h b/src/vt-tv/utility/input_iterator.h index 91b642ccc..b2f26b482 100644 --- a/src/vt-tv/utility/input_iterator.h +++ b/src/vt-tv/utility/input_iterator.h @@ -64,9 +64,7 @@ struct InputIterator { using iterator_category = std::input_iterator_tag; InputIterator() = default; - explicit InputIterator(DecompressionInputContainer const* in_c) - : c_(in_c) - { } + explicit InputIterator(DecompressionInputContainer const* in_c) : c_(in_c) { } InputIterator& operator++() { if (not c_->advance()) { @@ -76,13 +74,9 @@ struct InputIterator { return *this; } - bool operator!=(InputIterator const& rhs) const { - return rhs.c_ != c_; - } + bool operator!=(InputIterator const& rhs) const { return rhs.c_ != c_; } - reference operator*() const { - return c_->getCurrent(); - } + reference operator*() const { return c_->getCurrent(); } private: /// The underlying container with the data. diff --git a/src/vt-tv/utility/json_generator.cc b/src/vt-tv/utility/json_generator.cc index 4b367b41c..ab98d8266 100644 --- a/src/vt-tv/utility/json_generator.cc +++ b/src/vt-tv/utility/json_generator.cc @@ -82,9 +82,9 @@ std::unique_ptr JSONGenerator::generateJSON() const { for (auto const& [key, val] : user_defined) { // can't capture structured binding in C++17 (wait for 20!) auto const& key2 = key; - std::visit([&](auto&& arg) { - j["tasks"][task_index]["user_defined"][key2] = arg; - }, val); + std::visit( + [&](auto&& arg) { j["tasks"][task_index]["user_defined"][key2] = arg; }, + val); } // @todo: add communications @@ -93,10 +93,12 @@ std::unique_ptr JSONGenerator::generateJSON() const { return std::make_unique(std::move(j)); } -void JSONGenerator::outputObjectMetaData(nlohmann::json& j, ElementIDType id) const { +void JSONGenerator::outputObjectMetaData( + nlohmann::json& j, ElementIDType id) const { auto const& object_info_map = info_.getObjectInfo(); - assert(object_info_map.find(id) != object_info_map.end() && "Object must exist"); + assert( + object_info_map.find(id) != object_info_map.end() && "Object must exist"); auto const& object_info = object_info_map.find(id)->second; j["type"] = "object"; diff --git a/src/vt-tv/utility/json_generator.h b/src/vt-tv/utility/json_generator.h index d260eb13a..0d8c63e2e 100644 --- a/src/vt-tv/utility/json_generator.h +++ b/src/vt-tv/utility/json_generator.h @@ -59,7 +59,6 @@ namespace vt::tv::utility { * \brief Generates JSON from vt-tv data structures */ struct JSONGenerator { - /** * \brief Construct the reader * @@ -68,8 +67,7 @@ struct JSONGenerator { JSONGenerator(Info const& in_info, NodeType in_rank, PhaseType in_phase) : info_(in_info), rank_(in_rank), - phase_(in_phase) - { } + phase_(in_phase) { } /** * \brief Generate JSON for given rank and phase diff --git a/src/vt-tv/utility/json_reader.cc b/src/vt-tv/utility/json_reader.cc index 5f46ec489..8972b5ef9 100644 --- a/src/vt-tv/utility/json_reader.cc +++ b/src/vt-tv/utility/json_reader.cc @@ -142,8 +142,7 @@ std::unique_ptr JSONReader::parse() { if ( task["entity"].find("collection_id") != task["entity"].end() and - task["entity"].find("index") != task["entity"].end() - ) { + task["entity"].find("index") != task["entity"].end()) { auto cid = task["entity"]["collection_id"]; auto idx = task["entity"]["index"]; if (cid.is_number() && idx.is_array()) { @@ -183,7 +182,8 @@ std::unique_ptr JSONReader::parse() { } } - std::unordered_map readed_user_defined; + std::unordered_map + readed_user_defined; if (task.find("user_defined") != task.end()) { auto user_defined = task["user_defined"]; if (user_defined.is_object()) { @@ -207,9 +207,8 @@ std::unique_ptr JSONReader::parse() { objects.try_emplace( object, ObjectWork{ - object, time, std::move(subphase_loads), std::move(readed_user_defined), std::move(readed_attributes) - } - ); + object, time, std::move(subphase_loads), + std::move(readed_user_defined), std::move(readed_attributes)}); } } } @@ -243,7 +242,10 @@ std::unique_ptr JSONReader::parse() { if (to_it != objects.end()) { to_it->second.addReceivedCommunications(from_id, bytes); } else { - fmt::print("Warning: Communication {} -> {}: neither sender nor recipient was found in objects.\n", from_id, to_id); + fmt::print( + "Warning: Communication {} -> {}: neither sender nor " + "recipient was found in objects.\n", + from_id, to_id); } } } diff --git a/src/vt-tv/utility/json_reader.h b/src/vt-tv/utility/json_reader.h index 6bcac6c8d..0c274a441 100644 --- a/src/vt-tv/utility/json_reader.h +++ b/src/vt-tv/utility/json_reader.h @@ -60,13 +60,10 @@ namespace vt::tv::utility { * \brief Reader for JSON in the LBDataType format. */ struct JSONReader { - /** * \brief Construct the reader */ - JSONReader(NodeType in_rank) - : rank_(in_rank) - { } + JSONReader(NodeType in_rank) : rank_(in_rank) { } /** * \brief Check if the file is compressed or not diff --git a/src/vt-tv/utility/output_adaptor.h b/src/vt-tv/utility/output_adaptor.h index a20a7da35..8e05600e7 100644 --- a/src/vt-tv/utility/output_adaptor.h +++ b/src/vt-tv/utility/output_adaptor.h @@ -58,7 +58,6 @@ namespace vt::tv::utility { */ template struct OutputAdaptor : nlohmann::detail::output_adapter_protocol { - /** * \brief Construct with a stream * @@ -67,17 +66,14 @@ struct OutputAdaptor : nlohmann::detail::output_adapter_protocol { */ OutputAdaptor(StreamLike& in_os, bool compress) : c_((compress ? std::make_unique(8, 20) : nullptr)), - os_(in_os) - { } + os_(in_os) { } /** * \brief Write a single character to the output * * \param[in] c the character */ - void write_character(CharType c) override { - write_characters(&c, 1); - } + void write_character(CharType c) override { write_characters(&c, 1); } /** * \brief Write an array of characters to the output @@ -103,8 +99,8 @@ struct OutputAdaptor : nlohmann::detail::output_adapter_protocol { } private: - std::unique_ptr c_ = nullptr; /**< The compressor */ - StreamLike& os_; /**< The output stream */ + std::unique_ptr c_ = nullptr; /**< The compressor */ + StreamLike& os_; /**< The output stream */ }; } /* end namespace vt::tv::utility */ diff --git a/src/vt-tv/utility/parse_render.cc b/src/vt-tv/utility/parse_render.cc index 0c5734afa..c8282c148 100644 --- a/src/vt-tv/utility/parse_render.cc +++ b/src/vt-tv/utility/parse_render.cc @@ -46,12 +46,12 @@ #include "vt-tv/render/render.h" #include "vt-tv/api/info.h" - #include namespace vt::tv::utility { -void ParseRender::parseAndRender(PhaseType phase_id, std::unique_ptr info) { +void ParseRender::parseAndRender( + PhaseType phase_id, std::unique_ptr info) { try { // Load the yaml file YAML::Node config = YAML::LoadFile(filename_); @@ -71,7 +71,8 @@ void ParseRender::parseAndRender(PhaseType phase_id, std::unique_ptr info) input_dir += '/'; } - int64_t n_ranks = config["input"]["n_ranks"].as(); // signed for omp parallel for + int64_t n_ranks = + config["input"]["n_ranks"].as(); // signed for omp parallel for // Read JSON file and input data std::filesystem::path p = input_dir; @@ -79,49 +80,45 @@ void ParseRender::parseAndRender(PhaseType phase_id, std::unique_ptr info) info = std::make_unique(); - #ifdef VT_TV_OPENMP_ENABLED - #if VT_TV_OPENMP_ENABLED - #ifdef VT_TV_N_THREADS - const int threads = VT_TV_N_THREADS; - #else - const int threads = 2; - #endif // VT_TV_N_THREADS - omp_set_num_threads(threads); - fmt::print("vt-tv: Using {} threads\n", threads); - # pragma omp parallel for - #endif - #endif // VT_TV_OPENMP_ENABLED - for (int64_t rank = 0; rank < n_ranks; rank++) { - fmt::print("Reading file for rank {}\n", rank); - utility::JSONReader reader{static_cast(rank)}; - reader.readFile(input_dir + "data." + std::to_string(rank) + ".json"); - auto tmpInfo = reader.parse(); - #ifdef VT_TV_OPENMP_ENABLED - #if VT_TV_OPENMP_ENABLED - #pragma omp critical - #endif - #endif - { - info->addInfo(tmpInfo->getObjectInfo(), tmpInfo->getRank(rank)); - } - } +#ifdef VT_TV_OPENMP_ENABLED +#if VT_TV_OPENMP_ENABLED +#ifdef VT_TV_N_THREADS + const int threads = VT_TV_N_THREADS; +#else + const int threads = 2; +#endif // VT_TV_N_THREADS + omp_set_num_threads(threads); + fmt::print("vt-tv: Using {} threads\n", threads); +#pragma omp parallel for +#endif +#endif // VT_TV_OPENMP_ENABLED + for (int64_t rank = 0; rank < n_ranks; rank++) { + fmt::print("Reading file for rank {}\n", rank); + utility::JSONReader reader{static_cast(rank)}; + reader.readFile(input_dir + "data." + std::to_string(rank) + ".json"); + auto tmpInfo = reader.parse(); +#ifdef VT_TV_OPENMP_ENABLED +#if VT_TV_OPENMP_ENABLED +#pragma omp critical +#endif +#endif + { info->addInfo(tmpInfo->getObjectInfo(), tmpInfo->getRank(rank)); } + } } std::array qoi_request = { - config["viz"]["rank_qoi"].as("load"), - "", - config["viz"]["object_qoi"].as("load") - }; + config["viz"]["rank_qoi"].as("load"), "", + config["viz"]["object_qoi"].as("load")}; bool save_meshes = config["viz"]["save_meshes"].as(true); bool save_pngs = config["viz"]["save_pngs"].as(true); - bool continuous_object_qoi = config["viz"]["force_continuous_object_qoi"].as(true); + bool continuous_object_qoi = + config["viz"]["force_continuous_object_qoi"].as(true); std::array grid_size = { config["viz"]["x_ranks"].as(), config["viz"]["y_ranks"].as(), - config["viz"]["z_ranks"].as(1) - }; + config["viz"]["z_ranks"].as(1)}; double object_jitter = config["viz"]["object_jitter"].as(0.5); @@ -168,16 +165,17 @@ void ParseRender::parseAndRender(PhaseType phase_id, std::unique_ptr info) // Instantiate render Render r( - qoi_request, continuous_object_qoi, *std::move(info), grid_size, object_jitter, - output_dir, output_file_stem, 1.0, save_meshes, save_pngs, phase_id - ); + qoi_request, continuous_object_qoi, *std::move(info), grid_size, + object_jitter, output_dir, output_file_stem, 1.0, save_meshes, save_pngs, + phase_id); if (save_meshes || save_pngs) { r.generate(font_size, win_size); } } catch (std::exception const& e) { - std::cout << "Error reading the configuration file: " << e.what() << std::endl; + std::cout << "Error reading the configuration file: " << e.what() + << std::endl; } } diff --git a/src/vt-tv/utility/parse_render.h b/src/vt-tv/utility/parse_render.h index 9011f1e1c..1e8b9f972 100644 --- a/src/vt-tv/utility/parse_render.h +++ b/src/vt-tv/utility/parse_render.h @@ -53,7 +53,7 @@ #ifdef VT_TV_OPENMP_ENABLED #if VT_TV_OPENMP_ENABLED - #include +#include #endif #endif namespace vt::tv::utility { @@ -64,15 +64,12 @@ namespace vt::tv::utility { * \brief Parse YAML file and render based on configuration */ struct ParseRender { - /** * \brief Construct the class * * \param[in] in_filename the yaml file name to read */ - ParseRender(std::string const& in_filename) - : filename_(in_filename) - { } + ParseRender(std::string const& in_filename) : filename_(in_filename) { } /** * \brief Parse yaml file and render @@ -84,8 +81,7 @@ struct ParseRender { */ void parseAndRender( PhaseType phase_id = std::numeric_limits::max(), - std::unique_ptr info = nullptr - ); + std::unique_ptr info = nullptr); private: std::string filename_; diff --git a/src/vt-tv/utility/qoi_serializer.h b/src/vt-tv/utility/qoi_serializer.h index 27228c826..db5fe2e57 100644 --- a/src/vt-tv/utility/qoi_serializer.h +++ b/src/vt-tv/utility/qoi_serializer.h @@ -49,34 +49,34 @@ #include #include -namespace nlohmann -{ - template <> - struct adl_serializer<::vt::tv::QOIVariantTypes> { - using VariantTypes = ::vt::tv::QOIVariantTypes; - using ElementIDType = ::vt::tv::ElementIDType; +namespace nlohmann { +template <> +struct adl_serializer<::vt::tv::QOIVariantTypes> { + using VariantTypes = ::vt::tv::QOIVariantTypes; + using ElementIDType = ::vt::tv::ElementIDType; - // Produce compilation error if variant types were modified - static_assert(std::is_same_v>); - static_assert(std::is_same_v>); - static_assert(std::is_same_v>); + // Produce compilation error if variant types were modified + static_assert( + std::is_same_v>); + static_assert( + std::is_same_v>); + static_assert( + std::is_same_v>); - static void to_json(json &j, const VariantTypes &value) { - std::visit([&](auto const &arg) - { j = arg; }, - value); - } + static void to_json(json& j, const VariantTypes& value) { + std::visit([&](auto const& arg) { j = arg; }, value); + } - static void from_json(const json &j, VariantTypes &value) { - if (j.is_number_integer()) { - value = j.get(); - } else if (j.is_number_float()) { - value = j.get(); - } else if (j.is_string()) { - value = j.get(); - } + static void from_json(const json& j, VariantTypes& value) { + if (j.is_number_integer()) { + value = j.get(); + } else if (j.is_number_float()) { + value = j.get(); + } else if (j.is_string()) { + value = j.get(); } - }; + } +}; } /* end namespace nlohmann */ diff --git a/tests/unit/api/basic_serializer.h b/tests/unit/api/basic_serializer.h index aac06e806..ae48bc997 100644 --- a/tests/unit/api/basic_serializer.h +++ b/tests/unit/api/basic_serializer.h @@ -48,18 +48,18 @@ namespace vt::tv::tests::unit::api { * This basic serializer stores items as a map and enable the pipe operator to add * items to the inner map. This is useful to test api calls to serialization methods. */ - template class BasicSerializer { - public: - void operator | (const TSerializable& item) { - items[n] = item; - n++; - } +public: + void operator|(const TSerializable& item) { + items[n] = item; + n++; + } + + std::map items = std::map(); - std::map items = std::map(); - private: - int n = 0; +private: + int n = 0; }; -} \ No newline at end of file +} // namespace vt::tv::tests::unit::api \ No newline at end of file diff --git a/tests/unit/api/test_info.cc b/tests/unit/api/test_info.cc index 4c9a15bae..d6a35307a 100644 --- a/tests/unit/api/test_info.cc +++ b/tests/unit/api/test_info.cc @@ -49,16 +49,19 @@ namespace vt::tv::tests::unit::api { -struct InfoTestParam -{ - public: - - InfoTestParam(uint64_t in_num_objects, int16_t in_num_ranks, size_t in_num_phases): - num_objects(in_num_objects), num_ranks(in_num_ranks), num_phases(in_num_phases) { - } - - friend std::ostream& operator<< (std::ostream& stream, const InfoTestParam& param) { - stream << "{num_objects: " << std::to_string(param.num_objects) << ", num_ranks: " << std::to_string(param.num_ranks) << ", num_phases: " << std::to_string(param.num_phases) << "}"; +struct InfoTestParam { +public: + InfoTestParam( + uint64_t in_num_objects, int16_t in_num_ranks, size_t in_num_phases) + : num_objects(in_num_objects), + num_ranks(in_num_ranks), + num_phases(in_num_phases) { } + + friend std::ostream& + operator<<(std::ostream& stream, const InfoTestParam& param) { + stream << "{num_objects: " << std::to_string(param.num_objects) + << ", num_ranks: " << std::to_string(param.num_ranks) + << ", num_phases: " << std::to_string(param.num_phases) << "}"; return stream; } @@ -70,14 +73,15 @@ struct InfoTestParam /** * Provides unit tests for the vt::tv::api::Info class */ -class InfoTest :public ::testing::TestWithParam { }; +class InfoTest : public ::testing::TestWithParam { }; /** * Test Info:getNumRanks returns same number of ranks as defined in the sample */ TEST_P(InfoTest, test_get_num_ranks) { - InfoTestParam const & param = GetParam(); - Info info = Generator::makeInfo(param.num_objects, param.num_ranks, param.num_phases); + InfoTestParam const& param = GetParam(); + Info info = + Generator::makeInfo(param.num_objects, param.num_ranks, param.num_phases); EXPECT_EQ(info.getNumRanks(), param.num_ranks); } @@ -85,8 +89,9 @@ TEST_P(InfoTest, test_get_num_ranks) { * Test Info:getNumPhases */ TEST_P(InfoTest, test_get_num_phases) { - InfoTestParam const & param = GetParam(); - Info info = Generator::makeInfo(param.num_objects, param.num_ranks, param.num_phases); + InfoTestParam const& param = GetParam(); + Info info = + Generator::makeInfo(param.num_objects, param.num_ranks, param.num_phases); EXPECT_EQ(info.getNumPhases(), param.num_phases); } @@ -95,8 +100,9 @@ TEST_P(InfoTest, test_get_num_phases) { * Test Info:getRankIDs */ TEST_P(InfoTest, test_get_rank_ids) { - InfoTestParam const & param = GetParam(); - Info info = Generator::makeInfo(param.num_objects, param.num_ranks, param.num_phases); + InfoTestParam const& param = GetParam(); + Info info = + Generator::makeInfo(param.num_objects, param.num_ranks, param.num_phases); fmt::print("{}={}", param.num_phases, info.getNumPhases()); auto rank_ids = std::vector(); @@ -104,18 +110,22 @@ TEST_P(InfoTest, test_get_rank_ids) { rank_ids.push_back(rank_id); } - ASSERT_THAT(info.getRankIDs(), ::testing::UnorderedElementsAreArray(rank_ids)); + ASSERT_THAT( + info.getRankIDs(), ::testing::UnorderedElementsAreArray(rank_ids)); } /** * Test Info:getObjectQoiGetter */ TEST_P(InfoTest, test_get_object_qoi_getter) { - InfoTestParam const & param = GetParam(); - Info info = Generator::makeInfo(param.num_objects, param.num_ranks, param.num_phases); - - auto qoi_list = std::vector({"load", "received_volume", "sent_volume", "max_volume", "id", "rank_id", "non-existent"}); - for (auto const& qoi: qoi_list) { + InfoTestParam const& param = GetParam(); + Info info = + Generator::makeInfo(param.num_objects, param.num_ranks, param.num_phases); + + auto qoi_list = std::vector( + {"load", "received_volume", "sent_volume", "max_volume", "id", "rank_id", + "non-existent"}); + for (auto const& qoi : qoi_list) { auto qoi_getter = info.getObjectQoiGetter(qoi); } } @@ -124,7 +134,7 @@ TEST_P(InfoTest, test_get_object_qoi_getter) { * Test Info:getAllObjectIDs returns same items as defined in the sample */ TEST_P(InfoTest, test_get_all_object_ids) { - InfoTestParam const & param = GetParam(); + InfoTestParam const& param = GetParam(); auto objects = Generator::makeObjects(param.num_objects); auto ranks = Generator::makeRanks(objects, param.num_ranks, param.num_phases); @@ -135,35 +145,31 @@ TEST_P(InfoTest, test_get_all_object_ids) { auto const& actual = info.getAllObjectIDs(); EXPECT_EQ(expected.size(), actual.size()); - for (auto& pair: expected) { + for (auto& pair : expected) { auto object_id = pair.first; - EXPECT_NE(actual.find(pair.first), actual.end()) << "Cannot find element with id " << object_id; + EXPECT_NE(actual.find(pair.first), actual.end()) + << "Cannot find element with id " << object_id; } } /* Run Unit tests using different data sets as Tests params */ INSTANTIATE_TEST_SUITE_P( - InfoTests, - InfoTest, - ::testing::Values( - // num_objects, num_ranks, num_phases - InfoTestParam(0,0,0), // empty case - InfoTestParam(2,5,1), - InfoTestParam(6,1,1) - ), - [](const testing::TestParamInfo& in_info) { - // test suffix - return std::to_string(in_info.param.num_objects) + "_" + - std::to_string(in_info.param.num_ranks) + "_" + - std::to_string(in_info.param.num_phases); - } -); + InfoTests, InfoTest, + ::testing::Values( + // num_objects, num_ranks, num_phases + InfoTestParam(0, 0, 0), // empty case + InfoTestParam(2, 5, 1), InfoTestParam(6, 1, 1)), + [](const testing::TestParamInfo& in_info) { + // test suffix + return std::to_string(in_info.param.num_objects) + "_" + + std::to_string(in_info.param.num_ranks) + "_" + + std::to_string(in_info.param.num_phases); + }); /** * Test Info::addInfo */ TEST_F(InfoTest, test_add_info) { - Info info = Info(); std::vector idx; @@ -191,16 +197,19 @@ TEST_F(InfoTest, test_add_info) { EXPECT_EQ(info.getRank(0).getPhaseWork().size(), 1); // Rank already added. Expected assertion error (DEBUG). - ASSERT_DEBUG_DEATH({ info.addInfo(object_info_map, rank); }, "Rank must not exist"); + ASSERT_DEBUG_DEATH( + { info.addInfo(object_info_map, rank); }, "Rank must not exist"); } /** * Test Info:getNumPhases with inconstent rank phases. */ TEST_F(InfoTest, test_inconsistent_number_of_phases_across_ranks_throws_error) { - Rank rank_0 = Rank(0, {{0, {}}}, {}); // Rank with 1 phase + Rank rank_0 = Rank(0, {{0, {}}}, {}); // Rank with 1 phase Rank rank_1 = Rank(1, {{0, {}}, {1, {}}}, {}); // Rank with 2 phases - Info info = Info(Generator::makeObjectInfoMap(Generator::makeObjects(10)), { {0, rank_0}, {1, rank_1} } ); + Info info = Info( + Generator::makeObjectInfoMap(Generator::makeObjects(10)), + {{0, rank_0}, {1, rank_1}}); EXPECT_THROW(info.getNumPhases(), std::runtime_error); } @@ -213,27 +222,33 @@ TEST_F(InfoTest, test_get_phase_objects) { auto objects_18_info = Generator::makeObjectInfoMap(objects_18); auto objects_20_info = Generator::makeObjectInfoMap(objects_20); - Rank rank_0 = Rank(0, { { 0, PhaseWork(0, objects_15) }, { 1, PhaseWork(1, {}) } }, {}); - Rank rank_1 = Rank(1, { { 0, PhaseWork(0, {}) }, { 1, PhaseWork(1, objects_18) } }, {}); - Rank rank_2 = Rank(2, { { 0, PhaseWork(0, objects_20) }, { 1, PhaseWork(1, {}) } }, {}); + Rank rank_0 = + Rank(0, {{0, PhaseWork(0, objects_15)}, {1, PhaseWork(1, {})}}, {}); + Rank rank_1 = + Rank(1, {{0, PhaseWork(0, {})}, {1, PhaseWork(1, objects_18)}}, {}); + Rank rank_2 = + Rank(2, {{0, PhaseWork(0, objects_20)}, {1, PhaseWork(1, {})}}, {}); - auto objects_info = std::unordered_map(); + auto objects_info = std::unordered_map(); objects_info.merge(objects_15_info); objects_info.merge(objects_18_info); objects_info.merge(objects_20_info); - Info info = Info(objects_info, { {0, rank_0}, {1, rank_1} , {2, rank_2} } ); + Info info = Info(objects_info, {{0, rank_0}, {1, rank_1}, {2, rank_2}}); auto phase_0_objects = info.getPhaseObjects(0); for (auto& [key, val] : phase_0_objects) { - fmt::print("Phase 0 has object {} at key {} with load {}\n", val.getID(), key, val.getLoad()); + fmt::print( + "Phase 0 has object {} at key {} with load {}\n", val.getID(), key, + val.getLoad()); } ASSERT_EQ(phase_0_objects.size(), 4); std::vector actual_phase_0_object_ids; - for (const auto& [key, val]: phase_0_objects) { + for (const auto& [key, val] : phase_0_objects) { actual_phase_0_object_ids.push_back(key); } - ASSERT_THAT(actual_phase_0_object_ids, ::testing::UnorderedElementsAre(0, 1, 4, 5)); + ASSERT_THAT( + actual_phase_0_object_ids, ::testing::UnorderedElementsAre(0, 1, 4, 5)); ASSERT_EQ(phase_0_objects.at(0).getLoad(), 1.5); ASSERT_EQ(phase_0_objects.at(1).getLoad(), 1.5); ASSERT_EQ(phase_0_objects.at(4).getLoad(), 2.0); @@ -241,11 +256,13 @@ TEST_F(InfoTest, test_get_phase_objects) { auto phase_1_objects = info.getPhaseObjects(1); for (auto& [key, val] : phase_1_objects) { - fmt::print("Phase 1 has object {} at key {} with load {}\n", val.getID(), key, val.getLoad()); + fmt::print( + "Phase 1 has object {} at key {} with load {}\n", val.getID(), key, + val.getLoad()); } ASSERT_EQ(phase_1_objects.size(), 2); std::vector actual_phase_1_object_ids; - for (const auto& [key, val]: phase_1_objects) { + for (const auto& [key, val] : phase_1_objects) { actual_phase_1_object_ids.push_back(key); } ASSERT_THAT(actual_phase_1_object_ids, ::testing::ElementsAre(2, 3)); @@ -265,23 +282,27 @@ TEST_F(InfoTest, test_get_max_load_after_changing_selected_phase) { auto objects_18_info = Generator::makeObjectInfoMap(objects_18); auto objects_20_info = Generator::makeObjectInfoMap(objects_20); - Rank rank_0 = Rank(0, { { 0, PhaseWork(0, objects_15) }, { 1, PhaseWork(1, {}) } }, {}); - Rank rank_1 = Rank(1, { { 0, PhaseWork(0, {}) }, { 1, PhaseWork(1, objects_18) } }, {}); - Rank rank_2 = Rank(2, { { 0, PhaseWork(0, objects_20) }, { 1, PhaseWork(1, {}) } }, {}); + Rank rank_0 = + Rank(0, {{0, PhaseWork(0, objects_15)}, {1, PhaseWork(1, {})}}, {}); + Rank rank_1 = + Rank(1, {{0, PhaseWork(0, {})}, {1, PhaseWork(1, objects_18)}}, {}); + Rank rank_2 = + Rank(2, {{0, PhaseWork(0, objects_20)}, {1, PhaseWork(1, {})}}, {}); - auto objects_info = std::unordered_map(); + auto objects_info = std::unordered_map(); objects_info.merge(objects_15_info); objects_info.merge(objects_18_info); objects_info.merge(objects_20_info); - Info info = Info(objects_info, { {0, rank_0}, {1, rank_1} , {2, rank_2} } ); + Info info = Info(objects_info, {{0, rank_0}, {1, rank_1}, {2, rank_2}}); // case selected phase is not initialized. // Initial value is memory value at time it is allocated (not allocated in constructor) try { info.getMaxLoad(); - } catch(std::exception & e) { - fmt::print("Authorized exception (unitialized selected phase): {}\n", e.what()); + } catch (std::exception& e) { + fmt::print( + "Authorized exception (unitialized selected phase): {}\n", e.what()); } info.setSelectedPhase(1); @@ -290,7 +311,7 @@ TEST_F(InfoTest, test_get_max_load_after_changing_selected_phase) { info.setSelectedPhase(0); EXPECT_EQ(info.getMaxLoad(), 2.0); - // selected phase = all phases + // selected phase = all phases info.setSelectedPhase(std::numeric_limits::max()); EXPECT_EQ(info.getMaxLoad(), 2.0); } @@ -298,10 +319,13 @@ TEST_F(InfoTest, test_get_max_load_after_changing_selected_phase) { /** * Test Info:getNumPhases with inconstent rank phases. */ -TEST_F(InfoTest, test_get_max_volume_throws_out_of_range_after_set_invalid_phase) { +TEST_F( + InfoTest, test_get_max_volume_throws_out_of_range_after_set_invalid_phase) { Rank rank_0 = Rank(0, {{0, PhaseWork()}, {1, PhaseWork()}}, {}); Rank rank_1 = Rank(1, {{0, PhaseWork()}, {1, PhaseWork()}}, {}); - Info info = Info(Generator::makeObjectInfoMap(Generator::makeObjects(10)), { {0, rank_0}, {1, rank_1} } ); + Info info = Info( + Generator::makeObjectInfoMap(Generator::makeObjects(10)), + {{0, rank_0}, {1, rank_1}}); info.setSelectedPhase(2); EXPECT_THROW(info.getMaxVolume(), std::runtime_error); } @@ -322,22 +346,25 @@ TEST_F(InfoTest, test_get_max_volume) { objects_18.at(2).addSentCommunications(4, 3.6); objects_20.at(4).addReceivedCommunications(2, 3.6); - Rank rank_0 = Rank(0, { { 0, PhaseWork(0, objects_15) }, { 1, PhaseWork(1, objects_20) } }, {}); - Rank rank_1 = Rank(1, { { 0, PhaseWork(0, {}) }, { 1, PhaseWork(1, objects_18) } }, {}); + Rank rank_0 = + Rank(0, {{0, PhaseWork(0, objects_15)}, {1, PhaseWork(1, objects_20)}}, {}); + Rank rank_1 = + Rank(1, {{0, PhaseWork(0, {})}, {1, PhaseWork(1, objects_18)}}, {}); - auto objects_info = std::unordered_map(); + auto objects_info = std::unordered_map(); objects_info.merge(objects_15_info); objects_info.merge(objects_18_info); objects_info.merge(objects_20_info); - Info info = Info(objects_info, { {0, rank_0}, {1, rank_1}} ); + Info info = Info(objects_info, {{0, rank_0}, {1, rank_1}}); // case selected phase is not initialized. // Initial value is memory value at time it is allocated (not allocated in constructor) try { info.getMaxVolume(); - } catch(std::exception & e) { - fmt::print("Authorized exception (unitialized selected phase): {}\n", e.what()); + } catch (std::exception& e) { + fmt::print( + "Authorized exception (unitialized selected phase): {}\n", e.what()); } info.setSelectedPhase(0); @@ -358,11 +385,13 @@ TEST_F(InfoTest, test_get_object_qoi) { ObjectWork object_0 = ObjectWork(0, 2.0, {}, {}, {}); Info info = Info( - Generator::makeObjectInfoMap({{ 0, object_0 }}), // 1 object - Generator::makeRanks({{ 0, object_0 }}, 1, 1) // 1 phase 1 rank + Generator::makeObjectInfoMap({{0, object_0}}), // 1 object + Generator::makeRanks({{0, object_0}}, 1, 1) // 1 phase 1 rank ); - auto qoi_list = std::vector({"load", "received_volume", "sent_volume", "max_volume", "id", "rank_id", "non-existent"}); - for (auto const& qoi: qoi_list) { + auto qoi_list = std::vector( + {"load", "received_volume", "sent_volume", "max_volume", "id", "rank_id", + "non-existent"}); + for (auto const& qoi : qoi_list) { if (qoi == "non-existent") { EXPECT_THROW(info.getObjectQoi(0, 0, qoi), std::runtime_error); } else { @@ -388,21 +417,26 @@ TEST_F(InfoTest, test_get_rank_qoi) { objects_18.at(2).addSentCommunications(4, 3.6); objects_20.at(4).addReceivedCommunications(2, 3.6); - Rank rank_0 = Rank(0, { { 0, PhaseWork(0, objects_15) }, { 1, PhaseWork(1, objects_20) } }, { {"attr1", 12}, {"attr2", "ab"} }); - Rank rank_1 = Rank(1, { { 0, PhaseWork(0, {}) }, { 1, PhaseWork(1, objects_18) } }, { {"attr1", 13}, {"attr2", "cd"} }); + Rank rank_0 = Rank( + 0, {{0, PhaseWork(0, objects_15)}, {1, PhaseWork(1, objects_20)}}, + {{"attr1", 12}, {"attr2", "ab"}}); + Rank rank_1 = Rank( + 1, {{0, PhaseWork(0, {})}, {1, PhaseWork(1, objects_18)}}, + {{"attr1", 13}, {"attr2", "cd"}}); - auto objects_info = std::unordered_map(); + auto objects_info = std::unordered_map(); objects_info.merge(objects_15_info); objects_info.merge(objects_18_info); objects_info.merge(objects_20_info); - Info info = Info(objects_info, { {0, rank_0}, {1, rank_1}} ); + Info info = Info(objects_info, {{0, rank_0}, {1, rank_1}}); // Test getRankQOIGetter - auto qoi_list = std::vector({"load", "received_volume", "sent_volume", "number_of_objects", - "number_of_migratable_objects", "migratable_load", "sentinel_load", "id", - "attr1", "attr2" }); - for (auto const& qoi: qoi_list) { + auto qoi_list = std::vector( + {"load", "received_volume", "sent_volume", "number_of_objects", + "number_of_migratable_objects", "migratable_load", "sentinel_load", "id", + "attr1", "attr2"}); + for (auto const& qoi : qoi_list) { auto qoi_getter = info.getRankQOIGetter(qoi); if (qoi == "id") { @@ -420,8 +454,10 @@ TEST_F(InfoTest, test_get_rank_qoi) { ASSERT_EQ(std::get(info.getRankAttribute(rank_0, "attr1")), 12.0); ASSERT_EQ(std::get(info.getRankAttribute(rank_1, "attr1")), 13.0); - ASSERT_EQ(std::get(info.getRankAttribute(rank_0, "attr2")), "ab"); - ASSERT_EQ(std::get(info.getRankAttribute(rank_1, "attr2")), "cd"); + ASSERT_EQ( + std::get(info.getRankAttribute(rank_0, "attr2")), "ab"); + ASSERT_EQ( + std::get(info.getRankAttribute(rank_1, "attr2")), "cd"); // Test getRankQOIAtPhase method ASSERT_EQ(info.getRankQOIAtPhase(0, 0, "sent_volume"), 2.0); @@ -436,9 +472,13 @@ TEST_F(InfoTest, test_get_rank_qoi) { ASSERT_EQ(info.getRankQOIAtPhase(0, 0, "attr1"), 12.0); } -TEST_F(InfoTest, test_convert_qoi_variant_type_to_double_throws_runtime_error_for_string) { +TEST_F( + InfoTest, + test_convert_qoi_variant_type_to_double_throws_runtime_error_for_string) { auto info = Info(); - EXPECT_THROW(info.convertQOIVariantTypeToDouble_("some_string_value"), std::runtime_error); + EXPECT_THROW( + info.convertQOIVariantTypeToDouble_("some_string_value"), + std::runtime_error); } -} // end namespace vt::tv::tests::unit +} // namespace vt::tv::tests::unit::api diff --git a/tests/unit/api/test_object_communicator.cc b/tests/unit/api/test_object_communicator.cc index 468540271..61941aab6 100644 --- a/tests/unit/api/test_object_communicator.cc +++ b/tests/unit/api/test_object_communicator.cc @@ -51,21 +51,16 @@ namespace vt::tv::tests::unit::api { /** * Provides unit tests for the vt::tv::api::ObjectCommunicator class */ -class ObjectCommunicatorTest :public ::testing::Test { - public: - ObjectCommunicator comm_0 = ObjectCommunicator(0); - - ObjectCommunicator comm_1 = ObjectCommunicator( - 1, - // recv - { - { 1, 25.5 }, { 2, 12.0 } - }, - // sent - { - { 1, 29.5 }, { 2, 10.2 }, { 3, 10.3 } - } - ); +class ObjectCommunicatorTest : public ::testing::Test { +public: + ObjectCommunicator comm_0 = ObjectCommunicator(0); + + ObjectCommunicator comm_1 = ObjectCommunicator( + 1, + // recv + {{1, 25.5}, {2, 12.0}}, + // sent + {{1, 29.5}, {2, 10.2}, {3, 10.3}}); }; /** @@ -116,7 +111,10 @@ TEST_F(ObjectCommunicatorTest, test_initial_state) { } TEST_F(ObjectCommunicatorTest, test_serialization) { - BasicSerializer>> s = BasicSerializer>>(); + BasicSerializer< + std::variant>> + s = BasicSerializer< + std::variant>>(); comm_1.serialize(s); EXPECT_EQ(s.items.size(), 3); // object_id_, received_, sent_ @@ -124,17 +122,27 @@ TEST_F(ObjectCommunicatorTest, test_serialization) { auto actual_object_id = std::get(s.items[0]); EXPECT_EQ(actual_object_id, comm_1.getObjectId()); // object id - std::multimap actual_received = std::get>(s.items[1]); - std::multimap actual_sent = std::get>(s.items[2]); + std::multimap actual_received = + std::get>(s.items[1]); + std::multimap actual_sent = + std::get>(s.items[2]); bool any_failure = false; for (auto const& [object_id, received_volume] : comm_1.getReceived()) { if (actual_received.find(object_id) == actual_received.cend()) { - fmt::print("Missing received volume {} from object {} in serialized communicator data", received_volume, object_id); + fmt::print( + "Missing received volume {} from object {} in serialized communicator " + "data", + received_volume, object_id); any_failure = true; ADD_FAILURE(); - } else if (actual_received.count(object_id) != comm_1.getReceived().count(object_id)) { - fmt::print("Different count of received volume from object {} in serialized communicator data", object_id); + } else if ( + actual_received.count(object_id) != + comm_1.getReceived().count(object_id)) { + fmt::print( + "Different count of received volume from object {} in serialized " + "communicator data", + object_id); any_failure = true; ADD_FAILURE(); } @@ -150,18 +158,27 @@ TEST_F(ObjectCommunicatorTest, test_serialization) { if (!found) { any_failure = true; - fmt::print("Missing received volume {} from object {} in serialized communicator data", received_volume, object_id); + fmt::print( + "Missing received volume {} from object {} in serialized communicator " + "data", + received_volume, object_id); ADD_FAILURE(); } } for (auto const& [object_id, sent_volume] : comm_1.getSent()) { if (actual_sent.find(object_id) == actual_sent.cend()) { - fmt::print("Missing sent volume {} from object {} in serialized communicator data", sent_volume, object_id); + fmt::print( + "Missing sent volume {} from object {} in serialized communicator data", + sent_volume, object_id); any_failure = true; ADD_FAILURE(); - } else if (actual_sent.count(object_id) != comm_1.getSent().count(object_id)) { - fmt::print("Different count of sent volume from object {} in serialized communicator data", object_id); + } else if ( + actual_sent.count(object_id) != comm_1.getSent().count(object_id)) { + fmt::print( + "Different count of sent volume from object {} in serialized " + "communicator data", + object_id); any_failure = true; ADD_FAILURE(); } @@ -177,7 +194,10 @@ TEST_F(ObjectCommunicatorTest, test_serialization) { if (!found) { any_failure = true; - fmt::print("Missing received volume {} from object {} in serialized communicator data", sent_volume, object_id); + fmt::print( + "Missing received volume {} from object {} in serialized communicator " + "data", + sent_volume, object_id); ADD_FAILURE(); } } @@ -187,4 +207,4 @@ TEST_F(ObjectCommunicatorTest, test_serialization) { } } -} // end namespace vt::tv::tests::unit +} // namespace vt::tv::tests::unit::api diff --git a/tests/unit/api/test_object_info.cc b/tests/unit/api/test_object_info.cc index 927dd24b3..b46c068c3 100644 --- a/tests/unit/api/test_object_info.cc +++ b/tests/unit/api/test_object_info.cc @@ -50,21 +50,19 @@ namespace vt::tv::tests::unit::api { /** * Provides unit tests for the vt::tv::api::ObjectInfo class */ -class ObjectInfoTest :public ::testing::Test { - public: - ObjectInfo object_0 = ObjectInfo( - 6, // id - 2, // home - false, // migratable - std::vector({ 0, 1, 2}) - ); +class ObjectInfoTest : public ::testing::Test { +public: + ObjectInfo object_0 = ObjectInfo( + 6, // id + 2, // home + false, // migratable + std::vector({0, 1, 2})); - ObjectInfo object_1 = ObjectInfo( - 7, // id - 1, // home - true, // migratable - std::vector({ 3, 5, 6}) - ); + ObjectInfo object_1 = ObjectInfo( + 7, // id + 1, // home + true, // migratable + std::vector({3, 5, 6})); }; /** @@ -98,4 +96,4 @@ TEST_F(ObjectInfoTest, test_initial_state) { EXPECT_FALSE(object_0.getIsObjGroup()); } -} // end namespace vt::tv::tests::unit +} // namespace vt::tv::tests::unit::api diff --git a/tests/unit/api/test_object_work.cc b/tests/unit/api/test_object_work.cc index a796be52d..311058927 100644 --- a/tests/unit/api/test_object_work.cc +++ b/tests/unit/api/test_object_work.cc @@ -51,15 +51,11 @@ namespace vt::tv::tests::unit::api { /** * Provides unit tests for the vt::tv::api::ObjectWork class */ -class ObjectWorkTest :public ::testing::Test { - public: - ObjectWork object_0 = ObjectWork( - 12, - 10.0, - {{ 3, 12.0 }}, - Generator::makeQOIVariants(3, "user_", "_value"), - Generator::makeQOIVariants(2, "attr_", "_value") - ); +class ObjectWorkTest : public ::testing::Test { +public: + ObjectWork object_0 = ObjectWork( + 12, 10.0, {{3, 12.0}}, Generator::makeQOIVariants(3, "user_", "_value"), + Generator::makeQOIVariants(2, "attr_", "_value")); }; TEST_F(ObjectWorkTest, test_empty_constructor) { @@ -78,13 +74,23 @@ TEST_F(ObjectWorkTest, test_std_constructor_and_getters) { EXPECT_EQ(object_0.getSubphaseLoads().at(3), static_cast(12.0)); EXPECT_EQ(object_0.getUserDefined().size(), 3); - EXPECT_EQ(object_0.getUserDefined().at("user_0"), static_cast("user_0_value")); - EXPECT_EQ(object_0.getUserDefined().at("user_1"), static_cast("user_1_value")); - EXPECT_EQ(object_0.getUserDefined().at("user_2"), static_cast("user_2_value")); + EXPECT_EQ( + object_0.getUserDefined().at("user_0"), + static_cast("user_0_value")); + EXPECT_EQ( + object_0.getUserDefined().at("user_1"), + static_cast("user_1_value")); + EXPECT_EQ( + object_0.getUserDefined().at("user_2"), + static_cast("user_2_value")); EXPECT_EQ(object_0.getAttributes().size(), 2); - EXPECT_EQ(object_0.getAttributes().at("attr_0"), static_cast("attr_0_value")); - EXPECT_EQ(object_0.getAttributes().at("attr_1"), static_cast("attr_1_value")); + EXPECT_EQ( + object_0.getAttributes().at("attr_0"), + static_cast("attr_0_value")); + EXPECT_EQ( + object_0.getAttributes().at("attr_1"), + static_cast("attr_1_value")); EXPECT_EQ(object_0.getReceivedVolume(), 0.0); EXPECT_EQ(object_0.getSentVolume(), 0.0); @@ -106,4 +112,4 @@ TEST_F(ObjectWorkTest, test_add_and_get_sent_volumes) { EXPECT_EQ(object_0.getSentVolume(), 15.85); } -} // end namespace vt::tv::tests::unit +} // namespace vt::tv::tests::unit::api diff --git a/tests/unit/api/test_phase_work.cc b/tests/unit/api/test_phase_work.cc index f4336b348..8bfb7676e 100644 --- a/tests/unit/api/test_phase_work.cc +++ b/tests/unit/api/test_phase_work.cc @@ -41,7 +41,6 @@ //@HEADER */ - #include #include "../util.h" @@ -56,11 +55,11 @@ using ObjectWorkMap = std::unordered_map; * Provides unit tests for the vt::tv::api::PhaseWork class */ class PhaseWorkTest : public ::testing::Test { - public: +public: PhaseWorkTest() { objects_0 = Generator::makeObjects(10); phase_0 = PhaseWork( - 11, // phase id + 11, // phase id objects_0 // map of ObjectWork ); } @@ -76,11 +75,10 @@ TEST_F(PhaseWorkTest, test_initial_state) { // Assertions for phase_0 EXPECT_EQ(phase_0.getPhase(), 11); EXPECT_EQ( - phase_0.getLoad(), - std::accumulate(objects_0.begin(), objects_0.end(), 0, - [](double value, const auto& p) { - return value + p.second.getLoad(); - })); + phase_0.getLoad(), + std::accumulate( + objects_0.begin(), objects_0.end(), 0, + [](double value, const auto& p) { return value + p.second.getLoad(); })); EXPECT_EQ(phase_0.getMaxVolume(), 0.0); } @@ -96,7 +94,8 @@ TEST_F(PhaseWorkTest, test_communications) { ObjectCommunicator communicator = ObjectCommunicator(3); // object id (2) != communicator object id (3). Expected assertion error (DEBUG). - ASSERT_DEBUG_DEATH({ phase_0.setCommunications(object_id, communicator); }, ""); + ASSERT_DEBUG_DEATH( + { phase_0.setCommunications(object_id, communicator); }, ""); object_id = 3; phase_0.setCommunications(object_id, communicator); @@ -123,7 +122,8 @@ TEST_F(PhaseWorkTest, test_communications) { * Test PhaseWork::serialize correctly serialize PhaseWork instance members */ TEST_F(PhaseWorkTest, test_serialization) { - BasicSerializer> s = BasicSerializer>(); + BasicSerializer> s = + BasicSerializer>(); phase_0.serialize(s); EXPECT_EQ(s.items.size(), 2); @@ -139,4 +139,4 @@ TEST_F(PhaseWorkTest, test_serialization) { } } -} // namespace vt::tv::tests::unit::api +} // namespace vt::tv::tests::unit::api diff --git a/tests/unit/api/test_rank.cc b/tests/unit/api/test_rank.cc index 80d4611f9..f2e641449 100644 --- a/tests/unit/api/test_rank.cc +++ b/tests/unit/api/test_rank.cc @@ -50,31 +50,30 @@ namespace vt::tv::tests::unit::api { /** * Provides unit tests for the vt::tv::api::Rank class */ -class RankTest :public ::testing::Test { - public: +class RankTest : public ::testing::Test { +public: + // 6 objects with 2.0 load + std::unordered_map objects = + Generator::makeObjects(6, 2.0); - // 6 objects with 2.0 load - std::unordered_map objects = Generator::makeObjects(6, 2.0); + // 3 phases (2 objects per phase) (default load = 2.0 per object) + std::unordered_map objects_0 = { + {0, objects.at(0)}, {1, objects.at(1)}}; + std::unordered_map objects_1 = { + {2, objects.at(2)}, {3, objects.at(3)}}; + std::unordered_map objects_2 = { + {4, objects.at(4)}, {5, objects.at(5)}}; + PhaseWork phase_0 = Generator::makePhase(0, objects_0); + PhaseWork phase_1 = Generator::makePhase(1, objects_1); + PhaseWork phase_2 = Generator::makePhase(2, objects_2); - // 3 phases (2 objects per phase) (default load = 2.0 per object) - std::unordered_map objects_0 = { { 0, objects.at(0) }, { 1, objects.at(1) } }; - std::unordered_map objects_1 = { { 2, objects.at(2) }, { 3, objects.at(3) } }; - std::unordered_map objects_2 = { { 4, objects.at(4) }, { 5, objects.at(5) } }; - PhaseWork phase_0 = Generator::makePhase(0, objects_0); - PhaseWork phase_1 = Generator::makePhase(1, objects_1); - PhaseWork phase_2 = Generator::makePhase(2, objects_2); - - // rank - std::unordered_map phase_info_0 = { - { 0, phase_0 }, - { 1, phase_1 }, - { 2, phase_2 }, - }; - Rank rank_0 = Rank( - 2, - phase_info_0, - Generator::makeQOIVariants(10) - ); + // rank + std::unordered_map phase_info_0 = { + {0, phase_0}, + {1, phase_1}, + {2, phase_2}, + }; + Rank rank_0 = Rank(2, phase_info_0, Generator::makeQOIVariants(10)); }; /** @@ -101,7 +100,9 @@ TEST_F(RankTest, test_initial_state) { */ TEST_F(RankTest, test_communications_and_get_max_volume) { // object 0 not in phase 2 objects must fire an error - ASSERT_THROW({ rank_0.addObjectReceivedCommunicationAtPhase(2, 0, 3, 12.0); }, std::out_of_range); + ASSERT_THROW( + { rank_0.addObjectReceivedCommunicationAtPhase(2, 0, 3, 12.0); }, + std::out_of_range); // Phase 1, object 2 receives from object 4 (phase 2) a 12.0 load. rank_0.addObjectReceivedCommunicationAtPhase(1, 2, 4, 3.5); @@ -115,4 +116,4 @@ TEST_F(RankTest, test_communications_and_get_max_volume) { ASSERT_EQ(rank_0.getPhaseWork().at(0).getMaxVolume(), 7.0); } -} // end namespace vt::tv::tests::unit +} // namespace vt::tv::tests::unit::api diff --git a/tests/unit/deps/test_vtk.cc b/tests/unit/deps/test_vtk.cc index 022fc8c69..82da71f6c 100644 --- a/tests/unit/deps/test_vtk.cc +++ b/tests/unit/deps/test_vtk.cc @@ -61,8 +61,7 @@ namespace vt::tv::tests::unit::deps { /** * Provides unit tests for the Vtk functions called from vt-tv */ -class VtkTest :public ::testing::Test { - +class VtkTest : public ::testing::Test { void SetUp() override { // This test is not testing vt-tv src. // That's why it is skipped. But it might be useful locally. @@ -122,8 +121,9 @@ TEST_F(VtkTest, test_vtk_screenshot_example) { w2i->Update(); vtkNew writer; - writer->SetFileName(fmt::format("{}/output/tests/vtk_example_screenshot.png", SRC_DIR).c_str()); - writer->SetInputConnection(w2i->GetOutputPort()); + writer->SetFileName( + fmt::format("{}/output/tests/vtk_example_screenshot.png", SRC_DIR).c_str()); + writer->SetInputConnection(windowToImageFilter->GetOutputPort()); } -} // end namespace vt::tv::tests::unit +} // namespace vt::tv::tests::unit::deps \ No newline at end of file diff --git a/tests/unit/generator.h b/tests/unit/generator.h index cfd8b3af5..0b9442c51 100644 --- a/tests/unit/generator.h +++ b/tests/unit/generator.h @@ -61,19 +61,22 @@ namespace vt::tv::tests::unit { * unit tests classes related to the api to generate some data in memory. */ class Generator { - public: +public: /** * Make a map of new objects */ static const std::unordered_map makeObjects( - const int num_objects = rand() % 10, TimeType load = 2.0, int first_id = 0) { + const int num_objects = rand() % 10, TimeType load = 2.0, + int first_id = 0) { auto object_work_map = std::unordered_map(); - for (auto object_id = first_id; object_id < first_id + num_objects; object_id++) { + for (auto object_id = first_id; object_id < first_id + num_objects; + object_id++) { // Make some ObjectWork instance auto subphase_loads = std::unordered_map(); auto user_defined = makeQOIVariants(5, "user_"); auto attributes = makeQOIVariants(5, "attr_"); - auto object_work = ObjectWork(object_id, load, subphase_loads, user_defined, attributes); + auto object_work = + ObjectWork(object_id, load, subphase_loads, user_defined, attributes); object_work_map.insert(std::make_pair(object_id, object_work)); } return object_work_map; @@ -82,8 +85,8 @@ class Generator { /** * Make a new phase */ - static const PhaseWork makePhase(PhaseType phase_id, - std::unordered_map objects) { + static const PhaseWork makePhase( + PhaseType phase_id, std::unordered_map objects) { PhaseWork phase = PhaseWork(phase_id, objects); return phase; } @@ -92,11 +95,12 @@ class Generator { * Make a map of new phases */ static const std::unordered_map makePhases( - std::unordered_map objects, - const uint64_t num_phases = rand() % 10) { + std::unordered_map objects, + const uint64_t num_phases = rand() % 10) { auto phase_info_map = std::unordered_map(); for (uint64_t phase_id = 0; phase_id < num_phases; phase_id++) { - phase_info_map.insert(std::make_pair(phase_id, makePhase(phase_id, objects))); + phase_info_map.insert( + std::make_pair(phase_id, makePhase(phase_id, objects))); } return phase_info_map; } @@ -105,8 +109,8 @@ class Generator { * Make a map of new ranks */ static const std::unordered_map makeRanks( - std::unordered_map objects, int16_t num_ranks = rand() % 10, - uint64_t num_phases = rand() % 10) { + std::unordered_map objects, + int16_t num_ranks = rand() % 10, uint64_t num_phases = rand() % 10) { auto rank_map = std::unordered_map(); for (NodeType rank_id = 0; rank_id < num_ranks; rank_id++) { auto rank = Rank(rank_id, makePhases(objects, num_phases)); @@ -119,7 +123,8 @@ class Generator { * Make a map object info from an object map */ static std::unordered_map makeObjectInfoMap( - const std::unordered_map object_work_map, bool migratable = true) { + const std::unordered_map object_work_map, + bool migratable = true) { auto object_info_map = std::unordered_map(); std::vector idx; for (auto& it : object_work_map) { @@ -130,11 +135,12 @@ class Generator { } static std::unordered_map makeQOIVariants( - uint64_t num = rand() % 10, std::string key_prefix = "attr_", - std::string value_suffix = "_value") { + uint64_t num = rand() % 10, std::string key_prefix = "attr_", + std::string value_suffix = "_value") { auto qoi_map = std::unordered_map(); for (size_t i = 0; i < num; i++) { - qoi_map[key_prefix + std::to_string(i)] = key_prefix + std::to_string(i) + value_suffix; + qoi_map[key_prefix + std::to_string(i)] = + key_prefix + std::to_string(i) + value_suffix; } return qoi_map; } @@ -142,8 +148,9 @@ class Generator { /** * Make an Info instance */ - static const Info makeInfo(int num_objects = rand() % 100, int num_ranks = rand() % 10, - uint64_t num_phases = 10) { + static const Info makeInfo( + int num_objects = rand() % 100, int num_ranks = rand() % 10, + uint64_t num_phases = 10) { auto objects = makeObjects(num_objects); auto ranks = makeRanks(objects, num_ranks, num_phases); return Info(makeObjectInfoMap(objects), ranks); @@ -152,8 +159,8 @@ class Generator { static Info loadInfoFromConfig(YAML::Node config) { using JSONReader = ::vt::tv::utility::JSONReader; - std::string input_dir = - Util::resolveDir(SRC_DIR, config["input"]["directory"].as(), true); + std::string input_dir = Util::resolveDir( + SRC_DIR, config["input"]["directory"].as(), true); int64_t n_ranks = config["input"]["n_ranks"].as(); Info info = Info(); @@ -163,12 +170,12 @@ class Generator { const int threads = VT_TV_N_THREADS; #else const int threads = 2; -#endif // VT_TV_N_THREADS +#endif // VT_TV_N_THREADS omp_set_num_threads(threads); fmt::print("vt-tv: Using {} threads\n", threads); #pragma omp parallel for #endif -#endif // VT_TV_OPENMP_ENABLED +#endif // VT_TV_OPENMP_ENABLED for (int64_t rank = 0; rank < n_ranks; rank++) { fmt::print("Reading file for rank {}\n", rank); diff --git a/tests/unit/main.cc b/tests/unit/main.cc index 31ada23ae..07d525951 100644 --- a/tests/unit/main.cc +++ b/tests/unit/main.cc @@ -45,7 +45,7 @@ extern "C" { void __ubsan_on_report() { - FAIL() << "Encountered an undefined behavior sanitizer error"; + FAIL() << "Encountered an undefined behavior sanitizer error"; } void __asan_on_error() { @@ -53,13 +53,11 @@ void __asan_on_error() { } } -int main(int argc, char **argv) { +int main(int argc, char** argv) { ::testing::InitGoogleTest(&argc, argv); int ret = 0; - { - ret = RUN_ALL_TESTS(); - } + { ret = RUN_ALL_TESTS(); } return ret; } diff --git a/tests/unit/render/test_parse_render.cc b/tests/unit/render/test_parse_render.cc index 5e2f1d68b..a8c085767 100644 --- a/tests/unit/render/test_parse_render.cc +++ b/tests/unit/render/test_parse_render.cc @@ -64,61 +64,88 @@ using Util = vt::tv::tests::unit::Util; /** * Provides unit tests for the vt::tv::utility::ParseRender class to test with config file input */ -class ParseRenderTest :public ::testing::TestWithParam { }; +class ParseRenderTest : public ::testing::TestWithParam { + void SetUp() override { + // Make the output directory for these tests + std::filesystem::create_directory(fmt::format("{}/output", SRC_DIR)); + std::filesystem::create_directory(fmt::format("{}/output/tests", SRC_DIR)); + } +}; /** * Test ParseRender:parseAndRender correcty run the different configuration files */ -TEST_P(ParseRenderTest, test_parse_config_and_render_no_png) { - std::string const & config_file = GetParam(); - auto parse_render = ParseRender(fmt::format("{}/tests/config/{}", SRC_DIR, config_file)); - YAML::Node config = YAML::LoadFile(fmt::format("{}/tests/config/{}", SRC_DIR, config_file)); - Info info = Generator::loadInfoFromConfig(config); - - auto output_dir = Util::resolveDir(SRC_DIR, config["output"]["directory"].as(), true); - std::filesystem::create_directories(output_dir); - - auto output_file_stem = config["output"]["file_stem"].as(); - - // generate file containing object jitter dimensions for output trace and input usage - // this is needed for image and mesh tests +TEST_P(ParseRenderTest, test_parse_config_and_render_output) { + std::string const& config_file = GetParam(); + auto parse_render = + ParseRender(fmt::format("{}/tests/config/{}", SRC_DIR, config_file)); ASSERT_NO_THROW(parse_render.parseAndRender()); - for (uint64_t i = 0; i(), true); - fmt::print("----- Test phase {} -----\n", i); + std::string output_file_stem = + config["output"]["file_stem"].as(); + for (uint64_t i = 0; i < info.getNumPhases(); i++) { // 1. test files exist: rank mesh, object mesh, png - ASSERT_TRUE( - std::filesystem::exists(rank_mesh_file) - ) << fmt::format("Error: rank mesh not generated at {}", rank_mesh_file); - ASSERT_TRUE( - std::filesystem::exists(object_mesh_file) - ) << fmt::format("Error: object mesh not generated at {}", object_mesh_file); - ASSERT_FALSE( - std::filesystem::exists(png_file) - ) << fmt::format("Error: PNG image not generated at {}", png_file); - - } // end phases loop + auto rank_mesh_file = + fmt::format("{}{}_rank_mesh_{}.vtp", output_dir, output_file_stem, i); + auto object_mesh_file = + fmt::format("{}{}_object_mesh_{}.vtp", output_dir, output_file_stem, i); + auto png_file = fmt::format("{}{}{}.png", output_dir, output_file_stem, i); + ASSERT_TRUE(std::filesystem::exists(rank_mesh_file)) + << fmt::format("Error: rank mesh not generated at {}", object_mesh_file); + ASSERT_TRUE(std::filesystem::exists(object_mesh_file)) << fmt::format( + "Error: object mesh not generated at {}", object_mesh_file); + ASSERT_TRUE(std::filesystem::exists(png_file)) + << fmt::format("Error: PNG image not generated at {}", png_file); + + // 2. test PNG with tolerance + auto expected_png_file = fmt::format( + "{}/tests/expected/{}/{}{}.png", SRC_DIR, output_file_stem, + output_file_stem, i); + std::vector cmd_vars = { + fmt::format("ACTUAL={}", png_file), + fmt::format("EXPECTED={}", expected_png_file), + "TOLERANCE=2.0", + }; + auto cmd = fmt::format( + "{} {}/tests/test_image.sh", fmt::join(cmd_vars, " "), SRC_DIR); + const auto [status, output] = Util::exec(cmd.c_str()); + fmt::print("PNG diff: {}\n", output); + ASSERT_EQ(status, EXIT_SUCCESS) << fmt::format("Error: {}", output); + + // TODO: testing mesh files cannot be a simple diff as below because each run generates some different data. + // The future test should test XML Nodes + // // 3. test vtp's file content + // // 3.1 rank mesh file + // auto expected_rank_mesh_file = fmt::format("{}/tests/expected/{}/{}_rank_mesh_{}.vtp", + // SRC_DIR, output_file_stem, output_file_stem, i); + // auto rank_mesh_content = Util::getFileContent(rank_mesh_file); + // auto expected_rank_mesh_content = Util::getFileContent(expected_rank_mesh_file); + // ASSERT_EQ(expected_rank_mesh_content, rank_mesh_content) << fmt::format("rank mesh file content differs from expected at phase {}", i); + + // // 3.2 object mesh file + // auto expected_object_mesh_file = fmt::format("{}/tests/expected/{}/{}_object_mesh_{}.vtp", + // SRC_DIR, output_file_stem, output_file_stem, i); + // auto object_mesh_content = Util::getFileContent(object_mesh_file); + // auto expected_object_mesh_content = Util::getFileContent(expected_object_mesh_file); + // ASSERT_EQ(expected_object_mesh_content, object_mesh_content) << fmt::format("object mesh file content differs from expected at phase {}", i); + } // end phases loop } - /* Run with different configuration files */ INSTANTIATE_TEST_SUITE_P( - ParseRenderTests, - ParseRenderTest, - ::testing::Values( - "conf-no-png.yaml", - "ccm-example-no-png.yaml" - ), - [](const ::testing::TestParamInfo& in_info) { - // test suffix as slug - auto suffix = std::regex_replace(in_info.param, std::regex("\\.yaml"), ""); - suffix = std::regex_replace(suffix, std::regex("-"), "_"); - return suffix; - } -); - -} // end namespace vt::tv::tests::unit + ParseRenderTests, ParseRenderTest, + ::testing::Values("conf.yaml", "ccm-example.yaml"), + [](const ::testing::TestParamInfo& in_info) { + // test suffix as slug + auto suffix = std::regex_replace(in_info.param, std::regex("\\.yaml"), ""); + suffix = std::regex_replace(suffix, std::regex("-"), "_"); + return suffix; + }); + +} // namespace vt::tv::tests::unit::render diff --git a/tests/unit/render/test_render.cc b/tests/unit/render/test_render.cc index 4d43e19bb..b85222162 100644 --- a/tests/unit/render/test_render.cc +++ b/tests/unit/render/test_render.cc @@ -70,109 +70,21 @@ using Util = vt::tv::tests::unit::Util; class RenderTest :public ::testing::TestWithParam { protected: - Render createRender(YAML::Node config, Info info, std::string &output_dir) { + Render createRender(YAML::Node config, Info info, std::string& output_dir) { // change output directory to use specific one for these tests - output_dir = Util::resolveDir(SRC_DIR, config["output"]["directory"].as(), true); - // Logic copied from the ParseRender class here - return Render( - { - config["viz"]["rank_qoi"].as(), - "", - config["viz"]["object_qoi"].as() - }, - config["viz"]["force_continuous_object_qoi"].as(), - info, - { - config["viz"]["x_ranks"].as(), - config["viz"]["y_ranks"].as(), - config["viz"]["z_ranks"].as() - }, - 0.0, - output_dir, - config["output"]["file_stem"].as(), - 1.0, - config["viz"]["save_meshes"].as(), - config["viz"]["save_pngs"].as(), - std::numeric_limits::max() - ); - } - - void printVtkPolyData(vtkPolyData *poly) { - std::string points_str = ""; - for(auto k=0; k < poly->GetNumberOfPoints(); ++k) { - double* coords = poly->GetPoint(k); - points_str = fmt::format("{} [{}] = [{}, {}, {}]\n", points_str, k, coords[0], coords[1], coords[2]); - } - - std::string point_array_str = ""; - for(auto k=0; k < poly->GetPointData()->GetNumberOfArrays(); ++k) { - auto array_name = poly->GetPointData()->GetArrayName(k); - auto array = poly->GetPointData()->GetArray(array_name); - - std::string components_str = ""; - for (auto tupleIdx = 0; tupleIdx < array->GetNumberOfTuples(); ++tupleIdx) { - for (auto compIdx = 0; compIdx < array->GetNumberOfComponents(); ++compIdx) { - components_str = fmt::format("{} [{}] = [TupleId={}, ComponentId={}]\n", components_str, k, tupleIdx, compIdx); - } - } - - point_array_str = fmt::format("{} [{}] = [Name=\"{}\", Type={}, Size={}, Components={}]\n", - point_array_str, - k, - array_name, - array->GetArrayType(), - array->GetSize(), - components_str - ); - } - - fmt::print( - R"STR( -> PointData.NumberOfArrays => {} -> NumberOfPoints => {} -> Lines.Data.Name => {} -> Points => [ - {} - ] - PointData.Arrays() => [ - {} - ] -)STR", - poly->GetPointData()->GetNumberOfArrays(), - poly->GetNumberOfPoints(), - Util::formatNullable(poly->GetLines()->GetData()->GetName()), - points_str, - point_array_str - ); - } - - void assertPolyEquals(vtkPolyData *actual, vtkPolyData *expected) { - // fmt::print("Actual vtkPolyData:\n"); - printVtkPolyData(actual); - // fmt::print("Expected vtkPolyData:\n"); - printVtkPolyData(expected); - - // Assertions required to test vt-tv meshaes - // Number of point data should be ranks - ASSERT_EQ(actual->GetPointData()->GetNumberOfArrays(), expected->GetPointData()->GetNumberOfArrays()); - - // Validate points - ASSERT_EQ(actual->GetNumberOfPoints(), expected->GetNumberOfPoints()); - - for(auto k=0; k < actual->GetNumberOfPoints(); ++k) { - double actualCoords[3]; - double expectedCoords[3]; - actual->GetPoint(k, actualCoords); - expected->GetPoint(k, expectedCoords); - - ASSERT_EQ(actualCoords[0], expectedCoords[0]) << "Invalid point X coordinate at index " << k << "diff=" << (expectedCoords[0] - actualCoords[0]); - ASSERT_EQ(actualCoords[1], expectedCoords[1]) << "Invalid point Y coordinate at index " << k << "diff=" << (expectedCoords[1] - actualCoords[1]); - ASSERT_EQ(actualCoords[2], expectedCoords[2]) << "Invalid point Z coordinate at index " << k << "diff=" << (expectedCoords[2] - actualCoords[2]); - } + output_dir = Util::resolveDir(SRC_DIR, "output/tests/render", true); - // Validate lines - ASSERT_EQ(actual->GetLines()->GetData()->GetName(), expected->GetLines()->GetData()->GetName()); - // TODO: browse lines + return Render( + {config["viz"]["rank_qoi"].as(), "", + config["viz"]["object_qoi"].as()}, + config["viz"]["force_continuous_object_qoi"].as(), info, + {config["viz"]["x_ranks"].as(), + config["viz"]["y_ranks"].as(), + config["viz"]["z_ranks"].as()}, + config["viz"]["object_jitter"].as(), output_dir, + config["output"]["file_stem"].as(), 1.0, + config["viz"]["save_meshes"].as(), false, + std::numeric_limits::max()); } }; @@ -197,115 +109,30 @@ TEST_P(RenderTest, test_render_from_config_with_png) { render.generate(font_size, win_size); // Check: that number of generated mesh files (*.vtp) correspond to the number of phases - for (uint64_t i = 0; i cmd_vars = { - fmt::format("ACTUAL={}", png_file), - fmt::format("EXPECTED={}", expected_png_file), - "TOLERANCE=0.01", - }; - auto cmd = fmt::format("{} {}/tests/test_image.sh", - fmt::join(cmd_vars, " "), - SRC_DIR - ); - const auto [status, output] = Util::exec(cmd.c_str()); - cout << output; - ASSERT_EQ(status, EXIT_SUCCESS) << output; - } else { - ADD_FAILURE() << "Cannot test png file (not generated)"; - } - - // 3. test vtp's file content - // 3.1 rank mesh file - fmt::print("Testing rank mesh file {}\n", std::filesystem::path(rank_mesh_file).filename().c_str()); - if (std::filesystem::exists(rank_mesh_file)) { - // 3.1.1 Compare raw vtp files - auto rank_mesh_content = Util::getFileContent(rank_mesh_file); - auto expected_rank_mesh_content = Util::getFileContent(expected_rank_mesh_file); - ASSERT_EQ(expected_rank_mesh_content, rank_mesh_content) << fmt::format("rank mesh file content differs from expected at phase {}", i); - - // 3.1.2 Compare polydata using vtkXMLPolyDataReader - vtkNew expected_rank_mesh_reader; - expected_rank_mesh_reader->SetFileName(expected_rank_mesh_file.c_str()); - expected_rank_mesh_reader->Update(); - vtkPolyData *expected_rank_mesh = expected_rank_mesh_reader->GetOutput(); - - vtkNew rank_mesh_reader; - rank_mesh_reader->SetFileName(rank_mesh_file.c_str()); - rank_mesh_reader->Update(); - vtkPolyData *rank_mesh = rank_mesh_reader->GetOutput(); - - this->assertPolyEquals(rank_mesh, expected_rank_mesh); - } else { - ADD_FAILURE() << "Cannot test rank mesh file (not generated)"; - } - - // 3.2 object mesh file - fmt::print("Testing object mesh file {}\n", std::filesystem::path(object_mesh_file).filename().c_str()); - if (std::filesystem::exists(object_mesh_file)) { - // 3.2.1 Compare raw vtp files - auto object_mesh_content = Util::getFileContent(object_mesh_file); - auto expected_object_mesh_content = Util::getFileContent(expected_object_mesh_file); - ASSERT_EQ(expected_object_mesh_content, object_mesh_content) << fmt::format("object mesh file content differs from expected at phase {}", i); - - // 3.2.2 Compare polydata using vtkXMLPolyDataReader - vtkNew expected_object_mesh_reader; - expected_object_mesh_reader->SetFileName(expected_object_mesh_file.c_str()); - expected_object_mesh_reader->Update(); - vtkPolyData *expected_object_mesh = expected_object_mesh_reader->GetOutput(); - - vtkNew object_mesh_reader; - object_mesh_reader->SetFileName(object_mesh_file.c_str()); - object_mesh_reader->Update(); - vtkPolyData *object_mesh = object_mesh_reader->GetOutput(); - - this->assertPolyEquals(object_mesh, expected_object_mesh); - } else { - ADD_FAILURE() << "Cannot test object mesh file (not generated)"; - } - - fmt::print("----- Finished testing phase {} -----\n", i); + std::string output_file_stem = + config["output"]["file_stem"].as(); + for (uint64_t i = 0; i < info.getNumPhases(); i++) { + auto rank_mesh_file = + fmt::format("{}{}_rank_mesh_{}.vtp", output_dir, output_file_stem, i); + auto object_mesh_file = + fmt::format("{}{}_object_mesh_{}.vtp", output_dir, output_file_stem, i); + ASSERT_TRUE(std::filesystem::exists(rank_mesh_file)) + << fmt::format("Error: rank mesh not generated at {}", object_mesh_file); + ASSERT_TRUE(std::filesystem::exists(object_mesh_file)) << fmt::format( + "Error: object mesh not generated at {}", object_mesh_file); } } /* Run with different configuration files */ INSTANTIATE_TEST_SUITE_P( - RenderTests, - RenderTest, - ::testing::Values( - "conf.yaml", - "ccm-example.yaml" - ), - [](const ::testing::TestParamInfo& in_info) { - // test suffix as slug - auto suffix = std::regex_replace(in_info.param, std::regex("\\.yaml"), ""); - suffix = std::regex_replace(suffix, std::regex("-"), "_"); - return suffix; - } -); + RenderTests, RenderTest, + ::testing::Values("conf.yaml", "ccm-example.yaml"), + [](const ::testing::TestParamInfo& in_info) { + // test suffix as slug + auto suffix = std::regex_replace(in_info.param, std::regex("\\.yaml"), ""); + suffix = std::regex_replace(suffix, std::regex("-"), "_"); + return suffix; + }); TEST_F(RenderTest, test_render_construct_from_info) { // old example2 now as a test @@ -336,4 +163,4 @@ TEST_F(RenderTest, test_render_construct_from_info) { SUCCEED(); } -} // end namespace vt::tv::tests::unit +} // namespace vt::tv::tests::unit::render diff --git a/tests/unit/render/test_standalone_app.cc b/tests/unit/render/test_standalone_app.cc index 0899e5573..c10d933b1 100644 --- a/tests/unit/render/test_standalone_app.cc +++ b/tests/unit/render/test_standalone_app.cc @@ -51,8 +51,8 @@ namespace vt::tv::tests::unit::render { * Provides unit tests for the standalone vt-tv app. * It is similar to the ParseRender tests except it runs as a separate process. */ -class StandaloneAppTest :public ::testing::TestWithParam> { - +class StandaloneAppTest + : public ::testing::TestWithParam> { void SetUp() override { // This test is not testing vt-tv src. // That's why it is skipped. But it might be useful locally. @@ -68,50 +68,59 @@ class StandaloneAppTest :public ::testing::TestWithParam(GetParam())); - int expected_phases = std::get<1>(GetParam()); + std::string config_file = + fmt::format("{}/tests/config/{}", SRC_DIR, std::get<0>(GetParam())); + int expected_phases = std::get<1>(GetParam()); - // Run vt-tv_standalone process - auto cmd = fmt::format("{}/apps/vt-tv_standalone --conf={}", BUILD_DIR, config_file); - const auto [status, output] = Util::exec(cmd.c_str()); - fmt::print(output); + // Run vt-tv_standalone process + auto cmd = + fmt::format("{}/apps/vt-tv_standalone --conf={}", BUILD_DIR, config_file); + const auto [status, output] = Util::exec(cmd.c_str()); + fmt::print(output); - // Load config for some checks - auto config = YAML::LoadFile(config_file); - std::string output_dir = Util::resolveDir(SRC_DIR, config["output"]["directory"].as(), true); - std::string output_file_stem = config["output"]["file_stem"].as(); + // Load config for some checks + auto config = YAML::LoadFile(config_file); + std::string output_dir = Util::resolveDir( + SRC_DIR, config["output"]["directory"].as(), true); + std::string output_file_stem = + config["output"]["file_stem"].as(); - if (config["viz"]["object_qoi"].as() == "shared_block_id") { - // Temporary: this case must be removed as soon as the `shared_block_id` QOI becomes supported. - EXPECT_THAT(output, ::testing::HasSubstr("Error reading the configuration file: Invalid Object QOI: shared_block_id")); - } else { - // Expect 1 output file per phase for both rank meshes and object meshes - for (int64_t i = 0; i() == "shared_block_id") { + // Temporary: this case must be removed as soon as the `shared_block_id` QOI becomes supported. + EXPECT_THAT( + output, + ::testing::HasSubstr("Error reading the configuration file: Invalid " + "Object QOI: shared_block_id")); + } else { + // Expect 1 output file per phase for both rank meshes and object meshes + for (int64_t i = 0; i < expected_phases; i++) { + ASSERT_TRUE(std::filesystem::exists( + fmt::format("{}{}_rank_mesh_{}.vtp", output_dir, output_file_stem, i))) + << fmt::format( + "Rank mesh not generated at {}{}_rank_mesh_{}.vtp", output_dir, + output_file_stem, i); + ASSERT_TRUE(std::filesystem::exists(fmt::format( + "{}{}_object_mesh_{}.vtp", output_dir, output_file_stem, i))) + << fmt::format( + "Object mesh not generated at {}{}_object_mesh_{}.vtp", output_dir, + output_file_stem, i); } + } } /* Run with different configuration files */ INSTANTIATE_TEST_SUITE_P( - StandaloneAppTests, - StandaloneAppTest, - // config file and expected number of phases - ::testing::Values( - // std::make_tuple("conf.yaml", 8), - std::make_tuple("ccm-example.yaml", 1) - ), - [](const ::testing::TestParamInfo>& in_info) { - // test suffix as slug - auto suffix = std::regex_replace(std::get<0>(in_info.param), std::regex("\\.yaml"), ""); - suffix = std::regex_replace(suffix, std::regex("-"), "_"); - return suffix; - } -); + StandaloneAppTests, StandaloneAppTest, + // config file and expected number of phases + ::testing::Values( + // std::make_tuple("conf.yaml", 8), + std::make_tuple("ccm-example.yaml", 1)), + [](const ::testing::TestParamInfo>& in_info) { + // test suffix as slug + auto suffix = + std::regex_replace(std::get<0>(in_info.param), std::regex("\\.yaml"), ""); + suffix = std::regex_replace(suffix, std::regex("-"), "_"); + return suffix; + }); -} // end namespace vt::tv::tests::unit +} // namespace vt::tv::tests::unit::render diff --git a/tests/unit/util.h b/tests/unit/util.h index b3b8a3e37..388a2d226 100644 --- a/tests/unit/util.h +++ b/tests/unit/util.h @@ -65,74 +65,69 @@ namespace vt::tv::tests::unit { * Utility methods */ class Util { - public: - - /** - * \brief Execute a command on the underlying system and returns exit code and output - * \throws {@link std::runtime_error} if an error occurs while opening the process - */ - static std::tuple exec(const char* cmd) { - std::array buffer; - std::string output; - int status = 100; - - FILE* pipe = popen(cmd, "r"); - if (!pipe) throw std::runtime_error("popen() failed!"); - try { - while (fgets(buffer.data(), static_cast(buffer.size()), pipe) != nullptr) { - output += buffer.data(); - } - } catch (...) { - status = WEXITSTATUS(pclose(pipe)); - throw; - } - status = WEXITSTATUS(pclose(pipe)); - - return std::make_tuple(status, output); +public: + /** + * \brief Execute a command on the underlying system and returns exit code and output + * \throws {@link std::runtime_error} if an error occurs while opening the process + */ + static std::tuple exec(const char* cmd) { + std::array buffer; + std::string output; + int status = 100; + + FILE* pipe = popen(cmd, "r"); + if (!pipe) + throw std::runtime_error("popen() failed!"); + try { + while (fgets(buffer.data(), static_cast(buffer.size()), pipe) != + nullptr) { + output += buffer.data(); + } + } catch (...) { + status = WEXITSTATUS(pclose(pipe)); + throw; } - - /** - * \brief Resolves a directory absolute path. - * \param[in] base_path Prepends "{base_path}/" to the path if path is relative - * \param[in] path The path as either a relative or an absolute path - * \param[in] add_trailing_sep Appends a trailing "/" char at the end of the path if not exist - */ - static std::string resolveDir(std::string base_path, std::string path, bool add_trailing_sep = false) { - std::filesystem::path abs_path(path); - // If it's a relative path, prepend the base path - if (abs_path.is_relative()) { - abs_path = std::filesystem::path(base_path) / path; - } - - auto abs_path_string = abs_path.string(); - // append an extra / if needed - if (add_trailing_sep && !abs_path_string.empty() && abs_path_string.back() != '/') { - abs_path_string += '/'; - } - - return abs_path_string; + status = WEXITSTATUS(pclose(pipe)); + + return std::make_tuple(status, output); + } + + /** + * \brief Resolves a directory absolute path. + * \param[in] base_path Prepends "{base_path}/" to the path if path is relative + * \param[in] path The path as either a relative or an absolute path + * \param[in] add_trailing_sep Appends a trailing "/" char at the end of the path if not exist + */ + static std::string resolveDir( + std::string base_path, std::string path, bool add_trailing_sep = false) { + std::filesystem::path abs_path(path); + // If it's a relative path, prepend the base path + if (abs_path.is_relative()) { + abs_path = std::filesystem::path(base_path) / path; } - /** - * \brief Reads file content and returns it as a string - */ - static std::string getFileContent(std::string filename) { - std::ifstream ifs(filename); - std::string content( (std::istreambuf_iterator(ifs) ), - (std::istreambuf_iterator() ) ); - ifs.close(); - return content; + auto abs_path_string = abs_path.string(); + // append an extra / if needed + if ( + add_trailing_sep && !abs_path_string.empty() && + abs_path_string.back() != '/') { + abs_path_string += '/'; } - /** - * \brief Formats a text with suport of null values - */ - static std::string formatNullable(const char* data) { - if (data == nullptr) { - return ""; - } - return fmt::format("{}", data); - } + return abs_path_string; + } + + /** + * \brief Reads file content and returns it as a string + */ + static std::string getFileContent(std::string filename) { + std::ifstream ifs(filename); + std::string content( + (std::istreambuf_iterator(ifs)), + (std::istreambuf_iterator())); + ifs.close(); + return content; + } }; } /* end namespace vt::tv::tests::unit */ diff --git a/tests/unit/utility/test_json_reader.cc b/tests/unit/utility/test_json_reader.cc index 345aa5285..387be7578 100644 --- a/tests/unit/utility/test_json_reader.cc +++ b/tests/unit/utility/test_json_reader.cc @@ -54,7 +54,7 @@ using JSONReader = vt::tv::utility::JSONReader; /** * Provides unit tests for the vt::tv::utility::JSONReader class */ -struct JSONReaderTest :public ::testing::Test {}; +struct JSONReaderTest : public ::testing::Test { }; TEST_F(JSONReaderTest, test_json_reader_1) { std::filesystem::path p = std::filesystem::path(SRC_DIR) / "data/lb_test_data" ; @@ -74,9 +74,8 @@ TEST_F(JSONReaderTest, test_json_reader_1) { for (auto const& [elm_id, oi] : obj_info) { fmt::print( - "elm_id={:x}, home={}, migratable={}, index_array size={}\n", - elm_id, oi.getHome(), oi.isMigratable(), oi.getIndexArray().size() - ); + "elm_id={:x}, home={}, migratable={}, index_array size={}\n", elm_id, + oi.getHome(), oi.isMigratable(), oi.getIndexArray().size()); EXPECT_EQ(elm_id, oi.getID()); fmt::print("elm_id: {}, oi.getID: {}\n", elm_id, oi.getID()); @@ -154,10 +153,12 @@ TEST_F(JSONReaderTest, test_json_reader_object_info_attributes) { EXPECT_TRUE(object_attributes.find("intSample") != object_attributes.end()); EXPECT_EQ(-100, std::get(object_attributes.at("intSample"))); - EXPECT_TRUE(object_attributes.find("doubleSample") != object_attributes.end()); + EXPECT_TRUE( + object_attributes.find("doubleSample") != object_attributes.end()); EXPECT_EQ(0, std::get(object_attributes.at("doubleSample"))); - EXPECT_TRUE(object_attributes.find("stringSample") != object_attributes.end()); + EXPECT_TRUE( + object_attributes.find("stringSample") != object_attributes.end()); EXPECT_EQ("", std::get(object_attributes.at("stringSample"))); } @@ -204,4 +205,4 @@ TEST_F(JSONReaderTest, test_json_reader_object_work_user_defined) { EXPECT_EQ(1, std::get(user_defined.at("isSample"))); } -} // end namespace vt::tv::tests::unit +} // namespace vt::tv::tests::unit::utility