From 3d35d8de96e91d70d6ed21acb08414744787c845 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Franz=20P=C3=B6schel?= Date: Thu, 19 Oct 2023 16:29:40 +0200 Subject: [PATCH] ADIOS2: Optionally write attributes only from given ranks Ref. https://github.com/ornladios/ADIOS2/issues/3846#issuecomment-1769255906 --- include/openPMD/IO/ADIOS/ADIOS2IOHandler.hpp | 6 +- src/IO/ADIOS/ADIOS2IOHandler.cpp | 68 ++++++++++++++++++-- test/ParallelIOTest.cpp | 2 + 3 files changed, 68 insertions(+), 8 deletions(-) diff --git a/include/openPMD/IO/ADIOS/ADIOS2IOHandler.hpp b/include/openPMD/IO/ADIOS/ADIOS2IOHandler.hpp index 86afffbbdf..5a8ef15b8d 100644 --- a/include/openPMD/IO/ADIOS/ADIOS2IOHandler.hpp +++ b/include/openPMD/IO/ADIOS/ADIOS2IOHandler.hpp @@ -274,6 +274,8 @@ class ADIOS2IOHandlerImpl return m_useGroupTable.value(); } + bool m_writeAttributesFromThisRank = true; + struct ParameterizedOperator { adios2::Operator op; @@ -285,7 +287,9 @@ class ADIOS2IOHandlerImpl json::TracingJSON m_config; static json::TracingJSON nullvalue; - void init(json::TracingJSON config); + template + void + init(json::TracingJSON config, Callback &&callbackWriteAttributesFromRank); template json::TracingJSON config(Key &&key, json::TracingJSON &cfg) diff --git a/src/IO/ADIOS/ADIOS2IOHandler.cpp b/src/IO/ADIOS/ADIOS2IOHandler.cpp index a9af92e4a0..e57dc986c7 100644 --- a/src/IO/ADIOS/ADIOS2IOHandler.cpp +++ b/src/IO/ADIOS/ADIOS2IOHandler.cpp @@ -79,7 +79,43 @@ ADIOS2IOHandlerImpl::ADIOS2IOHandlerImpl( , m_engineType(std::move(engineType)) , m_userSpecifiedExtension{std::move(specifiedExtension)} { - init(std::move(cfg)); + init( + std::move(cfg), + /* callbackWriteAttributesFromRank = */ + [communicator, this](nlohmann::json const &attribute_writing_ranks) { + int rank = 0; + MPI_Comm_rank(communicator, &rank); + auto throw_error = []() { + throw error::BackendConfigSchema( + {"adios2", "attribute_writing_ranks"}, + "Type must be either an integer or an array of integers."); + }; + if (attribute_writing_ranks.is_array()) + { + m_writeAttributesFromThisRank = false; + for (auto const &val : attribute_writing_ranks) + { + if (!val.is_number()) + { + throw_error(); + } + if (val.get() == rank) + { + m_writeAttributesFromThisRank = true; + break; + } + } + } + else if (attribute_writing_ranks.is_number()) + { + m_writeAttributesFromThisRank = + attribute_writing_ranks.get() == rank; + } + else + { + throw_error(); + } + }); } #endif // openPMD_HAVE_MPI @@ -94,7 +130,7 @@ ADIOS2IOHandlerImpl::ADIOS2IOHandlerImpl( , m_engineType(std::move(engineType)) , m_userSpecifiedExtension(std::move(specifiedExtension)) { - init(std::move(cfg)); + init(std::move(cfg), [](auto const &...) {}); } ADIOS2IOHandlerImpl::~ADIOS2IOHandlerImpl() @@ -135,7 +171,9 @@ ADIOS2IOHandlerImpl::~ADIOS2IOHandlerImpl() } } -void ADIOS2IOHandlerImpl::init(json::TracingJSON cfg) +template +void ADIOS2IOHandlerImpl::init( + json::TracingJSON cfg, Callback &&callbackWriteAttributesFromRank) { // allow overriding through environment variable m_engineType = @@ -181,6 +219,12 @@ void ADIOS2IOHandlerImpl::init(json::TracingJSON cfg) : ModifiableAttributes::No; } + if (m_config.json().contains("attribute_writing_ranks")) + { + callbackWriteAttributesFromRank( + m_config["attribute_writing_ranks"].json()); + } + auto engineConfig = config(ADIOS2Defaults::str_engine); if (!engineConfig.json().is_null()) { @@ -915,6 +959,10 @@ void ADIOS2IOHandlerImpl::writeDataset( void ADIOS2IOHandlerImpl::writeAttribute( Writable *writable, const Parameter ¶meters) { + if (!m_writeAttributesFromThisRank) + { + return; + } #if openPMD_HAS_ADIOS_2_9 switch (useGroupTable()) { @@ -3033,7 +3081,11 @@ namespace detail if (!initializedDefaults) { // Currently only schema 0 supported - m_IO.DefineAttribute(ADIOS2Defaults::str_adios2Schema, 0); + if (m_impl->m_writeAttributesFromThisRank) + { + m_IO.DefineAttribute( + ADIOS2Defaults::str_adios2Schema, 0); + } initializedDefaults = true; } @@ -3168,7 +3220,8 @@ namespace detail { if (writeOnly(m_mode) && !m_IO.InquireAttribute( - ADIOS2Defaults::str_usesstepsAttribute)) + ADIOS2Defaults::str_usesstepsAttribute) && + m_impl->m_writeAttributesFromThisRank) { m_IO.DefineAttribute( ADIOS2Defaults::str_usesstepsAttribute, 0); @@ -3189,7 +3242,8 @@ namespace detail */ if (calledExplicitly && writeOnly(m_mode) && !m_IO.InquireAttribute( - ADIOS2Defaults::str_usesstepsAttribute)) + ADIOS2Defaults::str_usesstepsAttribute) && + m_impl->m_writeAttributesFromThisRank) { m_IO.DefineAttribute( ADIOS2Defaults::str_usesstepsAttribute, 1); @@ -3356,7 +3410,7 @@ namespace detail case UseGroupTable::Yes: #if openPMD_HAS_ADIOS_2_9 { - if (writeOnly(m_mode)) + if (writeOnly(m_mode) && m_impl->m_writeAttributesFromThisRank) { requireActiveStep(); auto currentStepBuffered = currentStep(); diff --git a/test/ParallelIOTest.cpp b/test/ParallelIOTest.cpp index d884766f15..9b315e69a6 100644 --- a/test/ParallelIOTest.cpp +++ b/test/ParallelIOTest.cpp @@ -1174,6 +1174,8 @@ doshuffle = "BLOSC_BITSHUFFLE" std::string writeConfigBP4 = R"END( [adios2] unused = "parameter" +attribute_writing_ranks = 0 +use_group_table = true [adios2.engine] type = "bp4"