Skip to content

Commit

Permalink
Merge pull request #2790 from caitlinross/plugin-engine
Browse files Browse the repository at this point in the history
Plugin engine
  • Loading branch information
Chuck Atkins authored Dec 7, 2021
2 parents bd77f20 + e74e33a commit ae491c8
Show file tree
Hide file tree
Showing 24 changed files with 1,497 additions and 1 deletion.
1 change: 1 addition & 0 deletions docs/user_guide/source/conf.py
Original file line number Diff line number Diff line change
Expand Up @@ -140,6 +140,7 @@
'engines/sst.rst',
'engines/inline.rst',
'engines/null.rst',
'engines/plugin.rst',
# 'engines/engines.rst',
# 'faq/faq.rst',
]
Expand Down
1 change: 1 addition & 0 deletions docs/user_guide/source/engines/engines.rst
Original file line number Diff line number Diff line change
Expand Up @@ -24,4 +24,5 @@ Parameters are passed at:
.. include:: dataman.rst
.. include:: inline.rst
.. include:: null.rst
.. include:: plugin.rst
.. include:: virtual_engines.rst
101 changes: 101 additions & 0 deletions docs/user_guide/source/engines/plugin.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,101 @@
*************
Plugin Engine
*************

The ``Plugin`` engine enables the ability to load an engine located in a separate library.
Your plugin class needs to inherit from the ``PluginEngineInterface`` class in the ``adios2/engine/plugin/PluginEngineInterface.h`` header.
Depending on the type of engine you want to implement, you'll need to override a number of methods that are inherited from the ``adios2::core::Engine`` class.
These are briefly described in the following table.
More detailed documentation can be found in ``adios2/core/Engine.h``.

========================= ===================== ===========================================================
**Method** **Engine Type** **Description**
========================= ===================== ===========================================================
``BeginStep()`` Read/Write Indicates the beginning of a step
``EndStep()`` Read/Write Indicates the end of a step
``CurrentStep()`` Read/Write Returns current step info
``DoClose()`` Read/Write Close a particular transport
``Init()`` Read/Write Engine initialization
``InitParameters()`` Read/Write Initialize parameters
``InitTransports()`` Read/Write Initialize transports
``PerformPuts()`` Write Execute all deferred mode ``Put``
``Flush()`` Write Flushes data and metadata to a transport
``DoPut()`` Write Implementation for ``Put``
``DoPutSync()`` Write Implementation for ``Put`` (Sync mode)
``DoPutDeferred()`` Write Implementation for ``Put`` (Deferred Mode)
``PerformGets()`` Read Execute all deferred mode ``Get``
``DoGetSync()`` Read Implementation for ``Get`` (Sync mode)
``DoGetDeferred()`` Read Implementation for ``Get`` (Deferred Mode)
========================= ===================== ===========================================================

Examples showing how to implement an engine plugin can be found in ``examples/plugins/engine``.
An example write engine is ``ExampleWritePlugin.h``, while an example read engine is in ``ExampleReadPlugin.h``.
The writer is a simple file writing engine that creates a directory (called ``ExamplePlugin`` by default) and writes variable information to vars.txt and actual data to data.txt.
The reader example reads the files output by the writer example.

In addition to implementing the methods above, you'll need to implement ``EngineCreate()`` and ``EngineDestroy`` functions so the Plugin Engine can create/destroy the engine object.
Because of C++ name mangling, you'll need to use ``extern "C"``.
Looking at ``ExampleWritePlugin.h``, this looks like:

.. code-block:: c++

extern "C" {

adios2::core::engine::ExampleWritePlugin *
EngineCreate(adios2::core::IO &io, const std::string &name,
const adios2::Mode mode, adios2::helper::Comm comm)
{
return new adios2::core::engine::ExampleWritePlugin(io, name, mode,
comm.Duplicate());
}

void EngineDestroy(adios2::core::engine::ExampleWritePlugin * obj)
{
delete obj;
}

}

To build your plugin, your CMake should look something like:

.. code-block:: cmake
find_package(ADIOS2 REQUIRED)
set(BUILD_SHARED_LIBS ON)
add_library(PluginEngineWrite
ExampleWritePlugin.cpp
)
target_link_libraries(PluginEngineWrite adios2::cxx11 adios2::core)
When using the Plugin Engine, ADIOS will check for your plugin at the path specified in the ``ADIOS2_PLUGIN_PATH`` environment variable.
If ``ADIOS2_PLUGIN_PATH`` is not set, and a path is not specified in the settings for the Plugin Engine (see below steps for using a plugin in your application), then the usual ``dlopen`` search is performed (see `dlopen man page <https://man7.org/linux/man-pages/man3/dlopen.3.html>`_).

The following steps show how to use your engine plugin in your application.
``examplePluginEngine_write.cpp`` and ``examplePluginEngine_read.cpp`` are an example of how to use the engine plugins described above.
The key steps to use your plugin are:

1. Set engine to ``Plugin``. i.e.:

.. code-block:: c++

io.SetEngine("Plugin");

2. Set ``PluginName`` (optional) and ``PluginLibrary`` (required) parameters.
If you don't set ``PluginName``, the Plugin Engine will give your plugin a default name of ``UserPlugin``.
In the write example, this looks like

.. code-block:: c++

io.SetParameters({{"PluginName", "WritePlugin"}});
io.SetParameters({{"PluginLibrary", "PluginEngineWrite"}});
// also possible to use the path here instead of setting ADIOS2_PLUGIN_PATH
// io.SetParameters({{"PluginLibrary", "/path/to/libPluginEngineWrite.so"}})

.. note::
You don't need to add the ``lib`` prefix or the shared library ending (e.g., ``.so``, ``.dll``, etc.).
ADIOS will add these when searching for your plugin library.
If you do at the prefix/suffix, ADIOS should still be able to find your plugin.


At this point you can open the engine and use it as you would any other ADIOS engine.
You also shouldn't need to make any changes to your CMake files for your application.
1 change: 1 addition & 0 deletions examples/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ add_subdirectory(hello)
add_subdirectory(query)
add_subdirectory(useCases)
add_subdirectory(inlineMWE)
add_subdirectory(plugins)

if(ADIOS2_HAVE_MPI)
add_subdirectory(heatTransfer)
Expand Down
6 changes: 6 additions & 0 deletions examples/plugins/CMakeLists.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
#------------------------------------------------------------------------------#
# Distributed under the OSI-approved Apache License, Version 2.0. See
# accompanying file Copyright.txt for details.
#------------------------------------------------------------------------------#

add_subdirectory(engine)
24 changes: 24 additions & 0 deletions examples/plugins/engine/CMakeLists.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
#------------------------------------------------------------------------------#
# Distributed under the OSI-approved Apache License, Version 2.0. See
# accompanying file Copyright.txt for details.
#------------------------------------------------------------------------------#

add_library(PluginEngineWriteExample
ExampleWritePlugin.cpp
)
target_link_libraries(PluginEngineWriteExample adios2::cxx11 adios2_core)

add_library(PluginEngineReadExample
ExampleReadPlugin.cpp
)
target_link_libraries(PluginEngineReadExample adios2::cxx11 adios2_core)

add_executable(examplePluginEngine_write
examplePluginEngine_write.cpp
)
target_link_libraries(examplePluginEngine_write adios2::cxx11)

add_executable(examplePluginEngine_read
examplePluginEngine_read.cpp
)
target_link_libraries(examplePluginEngine_read adios2::cxx11)
149 changes: 149 additions & 0 deletions examples/plugins/engine/ExampleReadPlugin.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,149 @@
/*
* Distributed under the OSI-approved Apache License, Version 2.0. See
* accompanying file Copyright.txt for details.
*
* ExampleReadPlugin.cpp
*
* Created on: Jul 5, 2021
* Author: Caitlin Ross <caitlin.ross@kitware.com>
*/

#include "ExampleReadPlugin.h"
#include "ExampleReadPlugin.tcc"

#include "adios2/helper/adiosType.h"

namespace adios2
{
namespace core
{
namespace engine
{

ExampleReadPlugin::ExampleReadPlugin(IO &io, const std::string &name,
const Mode mode, helper::Comm comm)
: PluginEngineInterface(io, name, mode, comm.Duplicate())
{
Init();
}

ExampleReadPlugin::~ExampleReadPlugin()
{
m_DataFile.close();
m_VarFile.close();
}

Dims convertStrToDims(const std::string &str)
{
Dims dims;
if (str.size() > 2)
{
auto vals = str.substr(1, str.size() - 2);
std::stringstream ss(vals);
while (ss.good())
{
std::string substr;
std::getline(ss, substr, ',');
dims.push_back(std::stoi(substr));
}
}
return dims;
}

void ExampleReadPlugin::Init()
{
std::string dir = "ExamplePlugin";
auto paramFileNameIt = m_IO.m_Parameters.find("DirName");
if (paramFileNameIt != m_IO.m_Parameters.end())
{
dir = paramFileNameIt->second;
}

std::string fileName = dir + "/data.txt";
m_DataFile.open(fileName, std::ofstream::in);
if (!m_DataFile)
{
throw std::ios_base::failure("ExampleReadPlugin: Failed to open file " +
fileName);
}

std::string varfName = dir + "/vars.txt";
m_VarFile.open(varfName, std::ofstream::in);
if (!m_VarFile)
{
throw std::ios_base::failure("ExampleReadPlugin: Failed to open file " +
varfName + ".vars");
}

// get var info
while (m_VarFile.good())
{
std::string name, typeStr, shapeStr, startStr, countStr;
std::getline(m_VarFile, name, ';');
std::getline(m_VarFile, typeStr, ';');
std::getline(m_VarFile, shapeStr, ';');
std::getline(m_VarFile, startStr, ';');
std::getline(m_VarFile, countStr);

auto shape = convertStrToDims(shapeStr);
auto start = convertStrToDims(startStr);
auto count = convertStrToDims(countStr);

const DataType type = helper::GetDataTypeFromString(typeStr);
if (type == DataType::Compound)
{
// not supported
}
#define declare_template_instantiation(T) \
else if (type == helper::GetDataType<T>()) \
{ \
AddVariable<T>(name, shape, start, count); \
}
ADIOS2_FOREACH_STDTYPE_1ARG(declare_template_instantiation)
#undef declare_template_instantiation
}
}

#define declare(T) \
void ExampleReadPlugin::DoGetSync(Variable<T> &variable, T *values) \
{ \
ReadVariable(variable, values); \
} \
void ExampleReadPlugin::DoGetDeferred(Variable<T> &variable, T *values) \
{ \
ReadVariable(variable, values); \
}
ADIOS2_FOREACH_STDTYPE_1ARG(declare)
#undef declare

StepStatus ExampleReadPlugin::BeginStep(StepMode mode,
const float timeoutSeconds)
{
return StepStatus::OK;
}

void ExampleReadPlugin::PerformGets() {}

size_t ExampleReadPlugin::CurrentStep() const { return m_CurrentStep; }

void ExampleReadPlugin::EndStep() { m_CurrentStep++; }

void ExampleReadPlugin::DoClose(const int transportIndex) {}

} // end namespace engine
} // end namespace core
} // end namespace adios2

extern "C" {

adios2::core::engine::ExampleReadPlugin *EngineCreate(adios2::core::IO &io,
const std::string &name,
const adios2::Mode mode,
adios2::helper::Comm comm)
{
return new adios2::core::engine::ExampleReadPlugin(io, name, mode,
comm.Duplicate());
}

void EngineDestroy(adios2::core::engine::ExampleReadPlugin *obj) { delete obj; }
}
92 changes: 92 additions & 0 deletions examples/plugins/engine/ExampleReadPlugin.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,92 @@
/*
* Distributed under the OSI-approved Apache License, Version 2.0. See
* accompanying file Copyright.txt for details.
*
* ExampleReadPlugin.h Simple file reading engine for reading files written by
* the ExampleWritePlugin engine. Looks for directory
* (called "ExamplePlugin" by default, but can be changed with engine parameter
* "DirName") and reads variable info from vars.txt and actual data from
* data.txt.
*
* Created on: Jul 5, 2021
* Author: Caitlin Ross <caitlin.ross@kitware.com>
*/

#ifndef EXAMPLEREADPLUGIN_H_
#define EXAMPLEREADPLUGIN_H_

#include <fstream>
#include <string>

#include "adios2/common/ADIOSMacros.h"
#include "adios2/common/ADIOSTypes.h"
#include "adios2/core/IO.h"
#include "adios2/engine/plugin/PluginEngineInterface.h"
#include "adios2/helper/adiosComm.h"
#include "adios2/helper/adiosString.h"

namespace adios2
{
namespace core
{
namespace engine
{

/** An engine interface to be used by the plugin infrastructure */
class ExampleReadPlugin : public PluginEngineInterface
{
public:
ExampleReadPlugin(IO &io, const std::string &name, const Mode openMode,
helper::Comm comm);
virtual ~ExampleReadPlugin();

/** Indicates beginning of a step **/
StepStatus BeginStep(StepMode mode = StepMode::Read,
const float timeoutSeconds = -1.0) override;

/** Indicates end of a step **/
void EndStep() override;

/** Return the current step **/
size_t CurrentStep() const override;

/** Execute deferred mode Gets **/
void PerformGets() override;

protected:
void Init() override;

#define declare(T) \
void DoGetSync(Variable<T> &variable, T *values) override; \
void DoGetDeferred(Variable<T> &variable, T *values) override;
ADIOS2_FOREACH_STDTYPE_1ARG(declare)
#undef declare

void DoClose(const int transportIndex = -1) override;

private:
std::ifstream m_DataFile;
std::ifstream m_VarFile;
size_t m_CurrentStep = 0;

template <typename T>
void AddVariable(const std::string &name, Dims shape, Dims start,
Dims count);

template <class T>
void ReadVariable(Variable<T> &variable, T *values);
};

} // end namespace engine
} // end namespace core
} // end namespace adios2

extern "C" {

adios2::core::engine::ExampleReadPlugin *
EngineCreate(adios2::core::IO &io, const std::string &name,
const adios2::Mode mode, adios2::helper::Comm comm);
void EngineDestroy(adios2::core::engine::ExampleReadPlugin *obj);
}

#endif /* EXAMPLEREADPLUGIN_H_ */
Loading

0 comments on commit ae491c8

Please sign in to comment.